1 /****************************************************************************
3 ** Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies).
4 ** All rights reserved.
5 ** Contact: Nokia Corporation (qt-info@nokia.com)
7 ** This file is part of the test suite of the Qt Toolkit.
9 ** $QT_BEGIN_LICENSE:LGPL$
10 ** GNU Lesser General Public License Usage
11 ** This file may be used under the terms of the GNU Lesser General Public
12 ** License version 2.1 as published by the Free Software Foundation and
13 ** appearing in the file LICENSE.LGPL included in the packaging of this
14 ** file. Please review the following information to ensure the GNU Lesser
15 ** General Public License version 2.1 requirements will be met:
16 ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
18 ** In addition, as a special exception, Nokia gives you certain additional
19 ** rights. These rights are described in the Nokia Qt LGPL Exception
20 ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
22 ** GNU General Public License Usage
23 ** Alternatively, this file may be used under the terms of the GNU General
24 ** Public License version 3.0 as published by the Free Software Foundation
25 ** and appearing in the file LICENSE.GPL included in the packaging of this
26 ** file. Please review the following information to ensure the GNU General
27 ** Public License version 3.0 requirements will be met:
28 ** http://www.gnu.org/copyleft/gpl.html.
31 ** Alternatively, this file may be used in accordance with the terms and
32 ** conditions contained in a signed written agreement between you and Nokia.
40 ****************************************************************************/
42 #include "quicktest.h"
43 #include "quicktestresult_p.h"
44 #include <QtTest/qtestsystem.h>
45 #include "qtestoptions_p.h"
46 #include <QApplication>
47 #include <QtDeclarative/qdeclarative.h>
48 #include <QtQuick1/qdeclarativeview.h>
49 #include <QtDeclarative/qdeclarativeengine.h>
50 #include <QtDeclarative/qdeclarativecontext.h>
51 #if defined(QML_VERSION) && QML_VERSION >= 0x020000
52 #include <QtDeclarative/qquickview.h>
53 #define QUICK_TEST_SCENEGRAPH 1
55 #include <QtDeclarative/qjsvalue.h>
56 #include <QtDeclarative/qjsengine.h>
57 #include <QtGui/qopengl.h>
58 #include <QtCore/qurl.h>
59 #include <QtCore/qfileinfo.h>
60 #include <QtCore/qdir.h>
61 #include <QtCore/qdiriterator.h>
62 #include <QtCore/qfile.h>
63 #include <QtCore/qdebug.h>
64 #include <QtCore/qeventloop.h>
65 #include <QtGui/qtextdocument.h>
67 #include <QtGui/QGuiApplication>
68 #include <QtCore/QTranslator>
72 class QTestRootObject : public QObject
75 Q_PROPERTY(bool windowShown READ windowShown NOTIFY windowShownChanged)
76 Q_PROPERTY(bool hasTestCase READ hasTestCase WRITE setHasTestCase NOTIFY hasTestCaseChanged)
78 QTestRootObject(QObject *parent = 0)
79 : QObject(parent), hasQuit(false), m_windowShown(false), m_hasTestCase(false) {}
82 bool hasTestCase() const { return m_hasTestCase; }
83 void setHasTestCase(bool value) { m_hasTestCase = value; emit hasTestCaseChanged(); }
85 bool windowShown() const { return m_windowShown; }
86 void setWindowShown(bool value) { m_windowShown = value; emit windowShownChanged(); }
89 void windowShownChanged();
90 void hasTestCaseChanged();
93 void quit() { hasQuit = true; }
96 bool m_windowShown : 1;
97 bool m_hasTestCase :1;
100 static inline QString stripQuotes(const QString &s)
102 if (s.length() >= 2 && s.startsWith(QLatin1Char('"')) && s.endsWith(QLatin1Char('"')))
103 return s.mid(1, s.length() - 2);
108 int quick_test_main(int argc, char **argv, const char *name, quick_test_viewport_create createViewport, const char *sourceDir)
110 QGuiApplication* app = 0;
111 if (!QCoreApplication::instance()) {
112 app = new QGuiApplication(argc, argv);
115 // Look for QML-specific command-line options.
116 // -import dir Specify an import directory.
117 // -input dir Specify the input directory for test cases.
118 // -qtquick1 Run with QtQuick 1 rather than QtQuick 2.
119 // -translation file Specify the translation file.
122 QString translationFile;
123 bool qtQuick2 = true;
126 while (index < argc) {
127 if (strcmp(argv[index], "-import") == 0 && (index + 1) < argc) {
128 imports += stripQuotes(QString::fromLocal8Bit(argv[index + 1]));
130 } else if (strcmp(argv[index], "-input") == 0 && (index + 1) < argc) {
131 testPath = stripQuotes(QString::fromLocal8Bit(argv[index + 1]));
133 } else if (strcmp(argv[index], "-opengl") == 0) {
135 } else if (strcmp(argv[index], "-qtquick1") == 0) {
138 } else if (strcmp(argv[index], "-translation") == 0 && (index + 1) < argc) {
139 translationFile = stripQuotes(QString::fromLocal8Bit(argv[index + 1]));
141 } else if (outargc != index) {
142 argv[outargc++] = argv[index++];
151 // Parse the command-line arguments.
152 QuickTestResult::parseArgs(argc, argv);
153 QuickTestResult::setProgramName(name);
155 QTranslator translator;
156 if (!translationFile.isEmpty()) {
157 if (translator.load(translationFile)) {
158 app->installTranslator(&translator);
160 qWarning() << "Could not load the translation file" << translationFile;
164 // Determine where to look for the test data.
165 if (testPath.isEmpty() && sourceDir)
166 testPath = QString::fromLocal8Bit(sourceDir);
167 if (testPath.isEmpty())
168 testPath = QLatin1String(".");
172 if (testPath.endsWith(QLatin1String(".qml")) && QFileInfo(testPath).isFile()) {
175 // Scan the test data directory recursively, looking for "tst_*.qml" files.
177 filters += QLatin1String("tst_*.qml");
178 QDirIterator iter(testPath, filters, QDir::Files,
179 QDirIterator::Subdirectories |
180 QDirIterator::FollowSymlinks);
181 while (iter.hasNext())
182 files += iter.next();
186 // Bail out if we didn't find any test cases.
187 if (files.isEmpty()) {
188 qWarning() << argv[0] << ": could not find any test cases under"
194 // Scan through all of the "tst_*.qml" files and run each of them
195 // in turn with a QDeclarativeView.
196 #ifdef QUICK_TEST_SCENEGRAPH
199 QTestRootObject rootobj;
200 QEventLoop eventLoop;
201 QObject::connect(view.engine(), SIGNAL(quit()),
202 &rootobj, SLOT(quit()));
203 QObject::connect(view.engine(), SIGNAL(quit()),
204 &eventLoop, SLOT(quit()));
205 view.rootContext()->setContextProperty
206 (QLatin1String("qtest"), &rootobj);
207 foreach (QString path, imports)
208 view.engine()->addImportPath(path);
210 foreach (QString file, files) {
215 rootobj.setHasTestCase(false);
216 rootobj.setWindowShown(false);
217 rootobj.hasQuit = false;
218 QString path = fi.absoluteFilePath();
219 if (path.startsWith(QLatin1String(":/")))
220 view.setSource(QUrl(QLatin1String("qrc:") + path.mid(2)));
222 view.setSource(QUrl::fromLocalFile(path));
224 if (QTest::printAvailableFunctions)
226 if (view.status() == QQuickView::Error) {
227 // Error compiling the test - flag failure in the log and continue.
228 QList<QDeclarativeError> errors = view.errors();
229 QuickTestResult results;
230 results.setTestCaseName(fi.baseName());
231 results.startLogging();
232 results.setFunctionName(QLatin1String("compile"));
233 results.setFunctionType(QuickTestResult::Func);
234 results.fail(errors.at(0).description(),
235 errors.at(0).url().toString(),
236 errors.at(0).line());
237 results.finishTestFunction();
238 results.setFunctionName(QString());
239 results.setFunctionType(QuickTestResult::NoWhere);
240 results.stopLogging();
243 if (!rootobj.hasQuit) {
244 // If the test already quit, then it was performed
245 // synchronously during setSource(). Otherwise it is
246 // an asynchronous test and we need to show the window
247 // and wait for the quit indication.
249 QTest::qWaitForWindowShown(&view);
250 rootobj.setWindowShown(true);
251 if (!rootobj.hasQuit && rootobj.hasTestCase())
258 foreach (QString file, files) {
262 QDeclarativeView view;
263 QTestRootObject rootobj;
264 QEventLoop eventLoop;
265 QObject::connect(view.engine(), SIGNAL(quit()),
266 &rootobj, SLOT(quit()));
267 QObject::connect(view.engine(), SIGNAL(quit()),
268 &eventLoop, SLOT(quit()));
270 view.setViewport((*createViewport)());
271 view.rootContext()->setContextProperty
272 (QLatin1String("qtest"), &rootobj);
273 foreach (QString path, imports)
274 view.engine()->addImportPath(path);
275 QString path = fi.absoluteFilePath();
276 if (path.startsWith(QLatin1String(":/")))
277 view.setSource(QUrl(QLatin1String("qrc:") + path.mid(2)));
279 view.setSource(QUrl::fromLocalFile(path));
280 if (QTest::printAvailableFunctions)
282 if (view.status() == QDeclarativeView::Error) {
283 // Error compiling the test - flag failure in the log and continue.
284 QList<QDeclarativeError> errors = view.errors();
285 QuickTestResult results;
286 results.setTestCaseName(fi.baseName());
287 results.startLogging();
288 results.setFunctionName(QLatin1String("compile"));
289 results.setFunctionType(QuickTestResult::Func);
290 results.fail(errors.at(0).description(),
291 errors.at(0).url().toString(),
292 errors.at(0).line());
293 results.finishTestFunction();
294 results.setFunctionName(QString());
295 results.setFunctionType(QuickTestResult::NoWhere);
296 results.stopLogging();
299 if (!rootobj.hasQuit) {
300 // If the test already quit, then it was performed
301 // synchronously during setSource(). Otherwise it is
302 // an asynchronous test and we need to show the window
303 // and wait for the quit indication.
305 QTest::qWaitForWindowShown(&view);
306 rootobj.setWindowShown(true);
307 if (!rootobj.hasQuit)
313 // Flush the current logging stream.
314 QuickTestResult::setProgramName(0);
317 // Return the number of failures as the exit code.
318 return QuickTestResult::exitCode();
323 #include "quicktest.moc"