Bugfix QDesktopServices on Windows
It was doing tricks with URLs that it shouldn't be doing... including running QDir::toNativeSeparators on a URL. Task-number: QTBUG-32311 Change-Id: I5b6f640919956998c00dcf507f931045f21a9e53 Reviewed-by: Friedemann Kleint <Friedemann.Kleint@digia.com>
This commit is contained in:
parent
b4008713a2
commit
eb5749e8d9
@ -39,6 +39,7 @@
|
|||||||
**
|
**
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
|
#define QT_NO_URL_CAST_FROM_STRING
|
||||||
#include "qwindowsservices.h"
|
#include "qwindowsservices.h"
|
||||||
#include "qtwindows_additional.h"
|
#include "qtwindows_additional.h"
|
||||||
|
|
||||||
@ -55,19 +56,20 @@ QT_BEGIN_NAMESPACE
|
|||||||
|
|
||||||
enum { debug = 0 };
|
enum { debug = 0 };
|
||||||
|
|
||||||
static inline bool shellExecute(const QString &file)
|
static inline bool shellExecute(const QUrl &url)
|
||||||
{
|
{
|
||||||
#ifndef Q_OS_WINCE
|
#ifndef Q_OS_WINCE
|
||||||
const QString nativeFilePath = QDir::toNativeSeparators(file);
|
const QString nativeFilePath =
|
||||||
|
url.isLocalFile() ? QDir::toNativeSeparators(url.toLocalFile()) : url.toString(QUrl::FullyEncoded);
|
||||||
const quintptr result = (quintptr)ShellExecute(0, 0, (wchar_t*)nativeFilePath.utf16(), 0, 0, SW_SHOWNORMAL);
|
const quintptr result = (quintptr)ShellExecute(0, 0, (wchar_t*)nativeFilePath.utf16(), 0, 0, SW_SHOWNORMAL);
|
||||||
// ShellExecute returns a value greater than 32 if successful
|
// ShellExecute returns a value greater than 32 if successful
|
||||||
if (result <= 32) {
|
if (result <= 32) {
|
||||||
qWarning("ShellExecute '%s' failed (error %s).", qPrintable(file), qPrintable(QString::number(result)));
|
qWarning("ShellExecute '%s' failed (error %s).", qPrintable(url.toString()), qPrintable(QString::number(result)));
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
#else
|
#else
|
||||||
Q_UNUSED(file)
|
Q_UNUSED(url);
|
||||||
return false;
|
return false;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
@ -131,7 +133,7 @@ static inline bool launchMail(const QUrl &url)
|
|||||||
}
|
}
|
||||||
// Pass the url as the parameter. Should use QProcess::startDetached(),
|
// Pass the url as the parameter. Should use QProcess::startDetached(),
|
||||||
// but that cannot handle a Windows command line [yet].
|
// but that cannot handle a Windows command line [yet].
|
||||||
command.replace(QStringLiteral("%1"), url.toString());
|
command.replace(QStringLiteral("%1"), url.toString(QUrl::FullyEncoded));
|
||||||
if (debug)
|
if (debug)
|
||||||
qDebug() << __FUNCTION__ << "Launching" << command;
|
qDebug() << __FUNCTION__ << "Launching" << command;
|
||||||
//start the process
|
//start the process
|
||||||
@ -152,16 +154,14 @@ static inline bool launchMail(const QUrl &url)
|
|||||||
bool QWindowsServices::openUrl(const QUrl &url)
|
bool QWindowsServices::openUrl(const QUrl &url)
|
||||||
{
|
{
|
||||||
const QString scheme = url.scheme();
|
const QString scheme = url.scheme();
|
||||||
if (scheme.isEmpty())
|
|
||||||
return openDocument(url);
|
|
||||||
if (scheme == QStringLiteral("mailto") && launchMail(url))
|
if (scheme == QStringLiteral("mailto") && launchMail(url))
|
||||||
return true;
|
return true;
|
||||||
return shellExecute(QLatin1String(url.toEncoded()));
|
return shellExecute(url);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool QWindowsServices::openDocument(const QUrl &url)
|
bool QWindowsServices::openDocument(const QUrl &url)
|
||||||
{
|
{
|
||||||
return shellExecute(url.isLocalFile() ? url.toLocalFile() : url.toString());
|
return shellExecute(url);
|
||||||
}
|
}
|
||||||
|
|
||||||
QT_END_NAMESPACE
|
QT_END_NAMESPACE
|
||||||
|
@ -82,6 +82,7 @@ void tst_qdesktopservices::openUrl()
|
|||||||
QCOMPARE(QDesktopServices::openUrl(QUrl()), false);
|
QCOMPARE(QDesktopServices::openUrl(QUrl()), false);
|
||||||
#if defined(Q_OS_WIN) && !defined(Q_OS_WINCE)
|
#if defined(Q_OS_WIN) && !defined(Q_OS_WINCE)
|
||||||
// this test is only valid on windows on other systems it might mean open a new document in the application handling .file
|
// this test is only valid on windows on other systems it might mean open a new document in the application handling .file
|
||||||
|
QTest::ignoreMessage(QtWarningMsg, "ShellExecute 'file://invalid.file' failed (error 3).");
|
||||||
QCOMPARE(QDesktopServices::openUrl(QUrl("file://invalid.file")), false);
|
QCOMPARE(QDesktopServices::openUrl(QUrl("file://invalid.file")), false);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
@ -9,6 +9,7 @@ lance \
|
|||||||
network_remote_stresstest \
|
network_remote_stresstest \
|
||||||
network_stresstest \
|
network_stresstest \
|
||||||
qcursor \
|
qcursor \
|
||||||
|
qdesktopservices \
|
||||||
qdesktopwidget \
|
qdesktopwidget \
|
||||||
qgraphicsitem \
|
qgraphicsitem \
|
||||||
qgraphicsitemgroup \
|
qgraphicsitemgroup \
|
||||||
|
14
tests/manual/qdesktopservices/qdesktopservices.pro
Normal file
14
tests/manual/qdesktopservices/qdesktopservices.pro
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
QT += testlib
|
||||||
|
|
||||||
|
TARGET = tst_qdesktopservices
|
||||||
|
CONFIG += console
|
||||||
|
CONFIG -= app_bundle
|
||||||
|
|
||||||
|
TEMPLATE = app
|
||||||
|
|
||||||
|
|
||||||
|
SOURCES += tst_qdesktopservices.cpp
|
||||||
|
DEFINES += SRCDIR=\\\"$$PWD/\\\"
|
||||||
|
|
||||||
|
OTHER_FILES += \
|
||||||
|
test.txt
|
1
tests/manual/qdesktopservices/test.txt
Normal file
1
tests/manual/qdesktopservices/test.txt
Normal file
@ -0,0 +1 @@
|
|||||||
|
This is just a plain text file.
|
126
tests/manual/qdesktopservices/tst_qdesktopservices.cpp
Normal file
126
tests/manual/qdesktopservices/tst_qdesktopservices.cpp
Normal file
@ -0,0 +1,126 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
**
|
||||||
|
** Copyright (C) 2013 Intel Corporation.
|
||||||
|
** Contact: http://www.qt-project.org/legal
|
||||||
|
**
|
||||||
|
** This file is part of the test suite of the Qt Toolkit.
|
||||||
|
**
|
||||||
|
** $QT_BEGIN_LICENSE:LGPL$
|
||||||
|
** Commercial License Usage
|
||||||
|
** Licensees holding valid commercial Qt licenses may use this file in
|
||||||
|
** accordance with the commercial license agreement provided with the
|
||||||
|
** Software or, alternatively, in accordance with the terms contained in
|
||||||
|
** a written agreement between you and Digia. For licensing terms and
|
||||||
|
** conditions see http://qt.digia.com/licensing. For further information
|
||||||
|
** use the contact form at http://qt.digia.com/contact-us.
|
||||||
|
**
|
||||||
|
** GNU Lesser General Public License Usage
|
||||||
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
||||||
|
** General Public License version 2.1 as published by the Free Software
|
||||||
|
** Foundation and appearing in the file LICENSE.LGPL included in the
|
||||||
|
** packaging of this file. Please review the following information to
|
||||||
|
** ensure the GNU Lesser General Public License version 2.1 requirements
|
||||||
|
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
||||||
|
**
|
||||||
|
** In addition, as a special exception, Digia gives you certain additional
|
||||||
|
** rights. These rights are described in the Digia Qt LGPL Exception
|
||||||
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
||||||
|
**
|
||||||
|
** GNU General Public License Usage
|
||||||
|
** Alternatively, this file may be used under the terms of the GNU
|
||||||
|
** General Public License version 3.0 as published by the Free Software
|
||||||
|
** Foundation and appearing in the file LICENSE.GPL included in the
|
||||||
|
** packaging of this file. Please review the following information to
|
||||||
|
** ensure the GNU General Public License version 3.0 requirements will be
|
||||||
|
** met: http://www.gnu.org/copyleft/gpl.html.
|
||||||
|
**
|
||||||
|
**
|
||||||
|
** $QT_END_LICENSE$
|
||||||
|
**
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
#include <QString>
|
||||||
|
#include <QtTest>
|
||||||
|
#include <QCoreApplication>
|
||||||
|
#include <QDesktopServices>
|
||||||
|
|
||||||
|
class tst_QDesktopServices : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
private Q_SLOTS:
|
||||||
|
void openUrl();
|
||||||
|
void openUrl_data();
|
||||||
|
};
|
||||||
|
|
||||||
|
void tst_QDesktopServices::openUrl_data()
|
||||||
|
{
|
||||||
|
QTest::addColumn<QUrl>("data");
|
||||||
|
QTest::addColumn<QString>("message");
|
||||||
|
|
||||||
|
QUrl localFile = QUrl::fromLocalFile(QFINDTESTDATA("test.txt"));
|
||||||
|
|
||||||
|
QTest::newRow("text-file")
|
||||||
|
<< localFile
|
||||||
|
<< "This should open test.txt in a text editor";
|
||||||
|
|
||||||
|
localFile.setQuery("x=y");
|
||||||
|
QTest::newRow("text-file-with-query")
|
||||||
|
<< localFile
|
||||||
|
<< "This should open test.txt in a text editor. Queries do not usually show up.";
|
||||||
|
|
||||||
|
localFile.setQuery(QString());
|
||||||
|
localFile.setFragment("top");
|
||||||
|
QTest::newRow("text-file-with-fragment")
|
||||||
|
<< localFile
|
||||||
|
<< "This should open test.txt in a text editor. Fragments do not usually show up.";
|
||||||
|
|
||||||
|
QTest::newRow("browser-plain")
|
||||||
|
<< QUrl("http://qt-project.org")
|
||||||
|
<< "This should open http://qt-project.org in the default web browser";
|
||||||
|
|
||||||
|
QTest::newRow("search-url")
|
||||||
|
<< QUrl("http://google.com/search?q=Qt+Project")
|
||||||
|
<< "This should search \"Qt Project\" on Google";
|
||||||
|
|
||||||
|
QTest::newRow("search-url-with-space")
|
||||||
|
<< QUrl("http://google.com/search?q=Qt Project")
|
||||||
|
<< "This should search \"Qt Project\" on Google";
|
||||||
|
|
||||||
|
QTest::newRow("search-url-with-quotes")
|
||||||
|
<< QUrl("http://google.com/search?q=\"Qt+Project\"")
|
||||||
|
<< "This should search '\"Qt Project\"' on Google (including the quotes)";
|
||||||
|
|
||||||
|
QTest::newRow("search-url-with-hashtag")
|
||||||
|
<< QUrl("http://google.com/search?q=%23qtproject")
|
||||||
|
<< "This should search \"#qtproject\" on Google. The # should appear in the Google search field";
|
||||||
|
|
||||||
|
QTest::newRow("search-url-with-fragment")
|
||||||
|
<< QUrl("http://google.com/search?q=Qt+Project#top")
|
||||||
|
<< "This should search \"Qt Project\" on Google. There should be no # in the Google search field";
|
||||||
|
|
||||||
|
// see QTBUG-32311
|
||||||
|
QTest::newRow("search-url-with-slashes")
|
||||||
|
<< QUrl("http://google.com/search?q=/profile/5")
|
||||||
|
<< "This should search \"/profile/5\" on Google.";
|
||||||
|
|
||||||
|
QTest::newRow("mail")
|
||||||
|
<< QUrl("mailto:development@qt-project.org")
|
||||||
|
<< "This should open an email composer with the destination set to development@qt-project.org";
|
||||||
|
|
||||||
|
QTest::newRow("mail-subject")
|
||||||
|
<< QUrl("mailto:development@qt-project.org?subject=[Development]%20Test%20Mail")
|
||||||
|
<< "This should open an email composer and tries to set the subject";
|
||||||
|
}
|
||||||
|
|
||||||
|
void tst_QDesktopServices::openUrl()
|
||||||
|
{
|
||||||
|
QFETCH(QUrl, data);
|
||||||
|
QFETCH(QString, message);
|
||||||
|
qWarning("\n\nOpening \"%s\": %s", qPrintable(data.toString()), qPrintable(message));
|
||||||
|
QDesktopServices::openUrl(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
QTEST_MAIN(tst_QDesktopServices)
|
||||||
|
|
||||||
|
#include "tst_qdesktopservices.moc"
|
Loading…
x
Reference in New Issue
Block a user