1 | /**************************************************************************** |
2 | ** |
3 | ** Copyright (C) 2016 The Qt Company Ltd. |
4 | ** Contact: https://www.qt.io/licensing/ |
5 | ** |
6 | ** This file is part of the test suite of the Qt Toolkit. |
7 | ** |
8 | ** $QT_BEGIN_LICENSE:GPL-EXCEPT$ |
9 | ** Commercial License Usage |
10 | ** Licensees holding valid commercial Qt licenses may use this file in |
11 | ** accordance with the commercial license agreement provided with the |
12 | ** Software or, alternatively, in accordance with the terms contained in |
13 | ** a written agreement between you and The Qt Company. For licensing terms |
14 | ** and conditions see https://www.qt.io/terms-conditions. For further |
15 | ** information use the contact form at https://www.qt.io/contact-us. |
16 | ** |
17 | ** GNU General Public License Usage |
18 | ** Alternatively, this file may be used under the terms of the GNU |
19 | ** General Public License version 3 as published by the Free Software |
20 | ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT |
21 | ** included in the packaging of this file. Please review the following |
22 | ** information to ensure the GNU General Public License requirements will |
23 | ** be met: https://www.gnu.org/licenses/gpl-3.0.html. |
24 | ** |
25 | ** $QT_END_LICENSE$ |
26 | ** |
27 | ****************************************************************************/ |
28 | #include <qtest.h> |
29 | #include <QTextDocument> |
30 | #include <QTcpServer> |
31 | #include <QTcpSocket> |
32 | #include <QDir> |
33 | #include <QPainter> |
34 | #include <QSignalSpy> |
35 | |
36 | #include <QtQml/qqmlengine.h> |
37 | #include <QtQml/qqmlcomponent.h> |
38 | #include <private/qquickborderimage_p.h> |
39 | #include <private/qquickimagebase_p.h> |
40 | #include <private/qquickscalegrid_p_p.h> |
41 | #include <private/qquickloader_p.h> |
42 | #include <QtQuick/qquickview.h> |
43 | #include <QtQml/qqmlcontext.h> |
44 | |
45 | #include "../../shared/testhttpserver.h" |
46 | #include "../../shared/util.h" |
47 | #include "../shared/visualtestutil.h" |
48 | |
49 | Q_LOGGING_CATEGORY(lcTests, "qt.quick.tests" ) |
50 | |
51 | Q_DECLARE_METATYPE(QQuickImageBase::Status) |
52 | |
53 | class tst_qquickborderimage : public QQmlDataTest |
54 | |
55 | { |
56 | Q_OBJECT |
57 | public: |
58 | tst_qquickborderimage(); |
59 | |
60 | private slots: |
61 | void cleanup(); |
62 | void noSource(); |
63 | void imageSource(); |
64 | void imageSource_data(); |
65 | void clearSource(); |
66 | void resized(); |
67 | void smooth(); |
68 | void mirror(); |
69 | void tileModes(); |
70 | void sciSource(); |
71 | void sciSource_data(); |
72 | void invalidSciFile(); |
73 | void validSciFiles_data(); |
74 | void validSciFiles(); |
75 | void pendingRemoteRequest(); |
76 | void pendingRemoteRequest_data(); |
77 | void statusChanges(); |
78 | void statusChanges_data(); |
79 | void sourceSizeChanges(); |
80 | void progressAndStatusChanges(); |
81 | #if QT_CONFIG(opengl) |
82 | void borderImageMesh(); |
83 | #endif |
84 | void multiFrame_data(); |
85 | void multiFrame(); |
86 | |
87 | private: |
88 | QQmlEngine engine; |
89 | }; |
90 | |
91 | void tst_qquickborderimage::cleanup() |
92 | { |
93 | QQuickWindow window; |
94 | window.releaseResources(); |
95 | engine.clearComponentCache(); |
96 | } |
97 | |
98 | tst_qquickborderimage::tst_qquickborderimage() |
99 | { |
100 | } |
101 | |
102 | void tst_qquickborderimage::noSource() |
103 | { |
104 | QString componentStr = "import QtQuick 2.0\nBorderImage { source: \"\" }" ; |
105 | QQmlComponent component(&engine); |
106 | component.setData(componentStr.toLatin1(), baseUrl: QUrl::fromLocalFile(localfile: "" )); |
107 | QQuickBorderImage *obj = qobject_cast<QQuickBorderImage*>(object: component.create()); |
108 | QVERIFY(obj != nullptr); |
109 | QCOMPARE(obj->source(), QUrl()); |
110 | QCOMPARE(obj->width(), 0.); |
111 | QCOMPARE(obj->height(), 0.); |
112 | QCOMPARE(obj->horizontalTileMode(), QQuickBorderImage::Stretch); |
113 | QCOMPARE(obj->verticalTileMode(), QQuickBorderImage::Stretch); |
114 | |
115 | delete obj; |
116 | } |
117 | |
118 | void tst_qquickborderimage::imageSource_data() |
119 | { |
120 | QTest::addColumn<QString>(name: "source" ); |
121 | QTest::addColumn<bool>(name: "remote" ); |
122 | QTest::addColumn<QString>(name: "error" ); |
123 | |
124 | QTest::newRow(dataTag: "local" ) << testFileUrl(fileName: "colors.png" ).toString() << false << "" ; |
125 | QTest::newRow(dataTag: "local not found" ) << testFileUrl(fileName: "no-such-file.png" ).toString() << false |
126 | << "<Unknown File>:2:1: QML BorderImage: Cannot open: " + testFileUrl(fileName: "no-such-file.png" ).toString(); |
127 | QTest::newRow(dataTag: "remote" ) << "/colors.png" << true << "" ; |
128 | QTest::newRow(dataTag: "remote not found" ) << "/no-such-file.png" << true |
129 | << "<Unknown File>:2:1: QML BorderImage: Error transferring {{ServerBaseUrl}}/no-such-file.png - server replied: Not found" ; |
130 | } |
131 | |
132 | void tst_qquickborderimage::imageSource() |
133 | { |
134 | QFETCH(QString, source); |
135 | QFETCH(bool, remote); |
136 | QFETCH(QString, error); |
137 | |
138 | #if defined(Q_OS_LINUX) && !defined(Q_OS_ANDROID) |
139 | if (qstrcmp(str1: QTest::currentDataTag(), str2: "remote" ) == 0 |
140 | || qstrcmp(str1: QTest::currentDataTag(), str2: "remote not found" ) == 0) { |
141 | QSKIP("Remote tests cause occasional hangs in the CI system -- QTBUG-45655" ); |
142 | } |
143 | #endif |
144 | |
145 | TestHTTPServer server; |
146 | if (remote) { |
147 | QVERIFY2(server.listen(), qPrintable(server.errorString())); |
148 | server.serveDirectory(dataDirectory()); |
149 | source = server.urlString(documentPath: source); |
150 | error.replace(QStringLiteral("{{ServerBaseUrl}}" ), after: server.baseUrl().toString()); |
151 | } |
152 | |
153 | if (!error.isEmpty()) |
154 | QTest::ignoreMessage(type: QtWarningMsg, message: error.toUtf8()); |
155 | |
156 | QString componentStr = "import QtQuick 2.0\nBorderImage { source: \"" + source + "\" }" ; |
157 | QQmlComponent component(&engine); |
158 | component.setData(componentStr.toLatin1(), baseUrl: QUrl::fromLocalFile(localfile: "" )); |
159 | QQuickBorderImage *obj = qobject_cast<QQuickBorderImage*>(object: component.create()); |
160 | QVERIFY(obj != nullptr); |
161 | |
162 | if (remote) |
163 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Loading); |
164 | |
165 | QCOMPARE(obj->source(), remote ? source : QUrl(source)); |
166 | |
167 | if (error.isEmpty()) { |
168 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Ready); |
169 | QCOMPARE(obj->width(), 120.); |
170 | QCOMPARE(obj->height(), 120.); |
171 | QCOMPARE(obj->sourceSize().width(), 120); |
172 | QCOMPARE(obj->sourceSize().height(), 120); |
173 | QCOMPARE(obj->horizontalTileMode(), QQuickBorderImage::Stretch); |
174 | QCOMPARE(obj->verticalTileMode(), QQuickBorderImage::Stretch); |
175 | } else { |
176 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Error); |
177 | } |
178 | |
179 | delete obj; |
180 | } |
181 | |
182 | void tst_qquickborderimage::clearSource() |
183 | { |
184 | QString componentStr = "import QtQuick 2.0\nBorderImage { source: srcImage }" ; |
185 | QQmlContext *ctxt = engine.rootContext(); |
186 | ctxt->setContextProperty("srcImage" , testFileUrl(fileName: "colors.png" )); |
187 | QQmlComponent component(&engine); |
188 | component.setData(componentStr.toLatin1(), baseUrl: QUrl::fromLocalFile(localfile: "" )); |
189 | QQuickBorderImage *obj = qobject_cast<QQuickBorderImage*>(object: component.create()); |
190 | QVERIFY(obj != nullptr); |
191 | QCOMPARE(obj->status(), QQuickBorderImage::Ready); |
192 | QCOMPARE(obj->width(), 120.); |
193 | QCOMPARE(obj->height(), 120.); |
194 | |
195 | ctxt->setContextProperty("srcImage" , "" ); |
196 | QVERIFY(obj->source().isEmpty()); |
197 | QCOMPARE(obj->status(), QQuickBorderImage::Null); |
198 | QCOMPARE(obj->width(), 0.); |
199 | QCOMPARE(obj->height(), 0.); |
200 | |
201 | delete obj; |
202 | } |
203 | |
204 | void tst_qquickborderimage::resized() |
205 | { |
206 | QString componentStr = "import QtQuick 2.0\nBorderImage { source: \"" + testFileUrl(fileName: "colors.png" ).toString() + "\"; width: 300; height: 300 }" ; |
207 | QQmlComponent component(&engine); |
208 | component.setData(componentStr.toLatin1(), baseUrl: QUrl::fromLocalFile(localfile: "" )); |
209 | QQuickBorderImage *obj = qobject_cast<QQuickBorderImage*>(object: component.create()); |
210 | QVERIFY(obj != nullptr); |
211 | QCOMPARE(obj->width(), 300.); |
212 | QCOMPARE(obj->height(), 300.); |
213 | QCOMPARE(obj->sourceSize().width(), 120); |
214 | QCOMPARE(obj->sourceSize().height(), 120); |
215 | QCOMPARE(obj->horizontalTileMode(), QQuickBorderImage::Stretch); |
216 | QCOMPARE(obj->verticalTileMode(), QQuickBorderImage::Stretch); |
217 | |
218 | delete obj; |
219 | } |
220 | |
221 | void tst_qquickborderimage::smooth() |
222 | { |
223 | QString componentStr = "import QtQuick 2.0\nBorderImage { source: \"" + testFile(fileName: "colors.png" ) + "\"; smooth: true; width: 300; height: 300 }" ; |
224 | QQmlComponent component(&engine); |
225 | component.setData(componentStr.toLatin1(), baseUrl: QUrl::fromLocalFile(localfile: "" )); |
226 | QQuickBorderImage *obj = qobject_cast<QQuickBorderImage*>(object: component.create()); |
227 | QVERIFY(obj != nullptr); |
228 | QCOMPARE(obj->width(), 300.); |
229 | QCOMPARE(obj->height(), 300.); |
230 | QCOMPARE(obj->smooth(), true); |
231 | QCOMPARE(obj->horizontalTileMode(), QQuickBorderImage::Stretch); |
232 | QCOMPARE(obj->verticalTileMode(), QQuickBorderImage::Stretch); |
233 | |
234 | delete obj; |
235 | } |
236 | |
237 | void tst_qquickborderimage::mirror() |
238 | { |
239 | QQuickView *window = new QQuickView; |
240 | window->setSource(testFileUrl(fileName: "mirror.qml" )); |
241 | QQuickBorderImage *image = qobject_cast<QQuickBorderImage*>(object: window->rootObject()); |
242 | QVERIFY(image != nullptr); |
243 | |
244 | QImage screenshot = window->grabWindow(); |
245 | |
246 | QImage srcPixmap(screenshot); |
247 | QTransform transform; |
248 | transform.translate(dx: image->width(), dy: 0).scale(sx: -1, sy: 1.0); |
249 | srcPixmap = srcPixmap.transformed(matrix: transform); |
250 | |
251 | image->setProperty(name: "mirror" , value: true); |
252 | screenshot = window->grabWindow(); |
253 | |
254 | window->show(); |
255 | QVERIFY(QTest::qWaitForWindowExposed(window)); |
256 | if (window->rendererInterface()->graphicsApi() == QSGRendererInterface::Software) |
257 | QSKIP("QTBUG-53823" ); |
258 | QCOMPARE(screenshot, srcPixmap); |
259 | |
260 | delete window; |
261 | } |
262 | |
263 | void tst_qquickborderimage::tileModes() |
264 | { |
265 | { |
266 | QString componentStr = "import QtQuick 2.0\nBorderImage { source: \"" + testFile(fileName: "colors.png" ) + "\"; width: 100; height: 300; horizontalTileMode: BorderImage.Repeat; verticalTileMode: BorderImage.Repeat }" ; |
267 | QQmlComponent component(&engine); |
268 | component.setData(componentStr.toLatin1(), baseUrl: QUrl::fromLocalFile(localfile: "" )); |
269 | QQuickBorderImage *obj = qobject_cast<QQuickBorderImage*>(object: component.create()); |
270 | QVERIFY(obj != nullptr); |
271 | QCOMPARE(obj->width(), 100.); |
272 | QCOMPARE(obj->height(), 300.); |
273 | QCOMPARE(obj->horizontalTileMode(), QQuickBorderImage::Repeat); |
274 | QCOMPARE(obj->verticalTileMode(), QQuickBorderImage::Repeat); |
275 | |
276 | delete obj; |
277 | } |
278 | { |
279 | QString componentStr = "import QtQuick 2.0\nBorderImage { source: \"" + testFile(fileName: "colors.png" ) + "\"; width: 300; height: 150; horizontalTileMode: BorderImage.Round; verticalTileMode: BorderImage.Round }" ; |
280 | QQmlComponent component(&engine); |
281 | component.setData(componentStr.toLatin1(), baseUrl: QUrl::fromLocalFile(localfile: "" )); |
282 | QQuickBorderImage *obj = qobject_cast<QQuickBorderImage*>(object: component.create()); |
283 | QVERIFY(obj != nullptr); |
284 | QCOMPARE(obj->width(), 300.); |
285 | QCOMPARE(obj->height(), 150.); |
286 | QCOMPARE(obj->horizontalTileMode(), QQuickBorderImage::Round); |
287 | QCOMPARE(obj->verticalTileMode(), QQuickBorderImage::Round); |
288 | |
289 | delete obj; |
290 | } |
291 | } |
292 | |
293 | void tst_qquickborderimage::sciSource() |
294 | { |
295 | QFETCH(QString, source); |
296 | QFETCH(bool, valid); |
297 | QFETCH(bool, remote); |
298 | |
299 | TestHTTPServer server; |
300 | if (remote) { |
301 | QVERIFY2(server.listen(), qPrintable(server.errorString())); |
302 | server.serveDirectory(dataDirectory()); |
303 | source = server.urlString(documentPath: source); |
304 | server.registerFileNameForContentSubstitution(fileName: QUrl(source).path()); |
305 | } |
306 | |
307 | QString componentStr = "import QtQuick 2.0\nBorderImage { source: \"" + source + "\"; width: 300; height: 300 }" ; |
308 | QQmlComponent component(&engine); |
309 | component.setData(componentStr.toLatin1(), baseUrl: QUrl::fromLocalFile(localfile: "" )); |
310 | QQuickBorderImage *obj = qobject_cast<QQuickBorderImage*>(object: component.create()); |
311 | QVERIFY(obj != nullptr); |
312 | |
313 | if (remote) |
314 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Loading); |
315 | |
316 | QCOMPARE(obj->source(), remote ? source : QUrl(source)); |
317 | QCOMPARE(obj->width(), 300.); |
318 | QCOMPARE(obj->height(), 300.); |
319 | |
320 | if (valid) { |
321 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Ready); |
322 | QCOMPARE(obj->border()->left(), 10); |
323 | QCOMPARE(obj->border()->top(), 20); |
324 | QCOMPARE(obj->border()->right(), 30); |
325 | QCOMPARE(obj->border()->bottom(), 40); |
326 | QCOMPARE(obj->horizontalTileMode(), QQuickBorderImage::Round); |
327 | QCOMPARE(obj->verticalTileMode(), QQuickBorderImage::Repeat); |
328 | } else { |
329 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Error); |
330 | } |
331 | |
332 | delete obj; |
333 | } |
334 | |
335 | void tst_qquickborderimage::sciSource_data() |
336 | { |
337 | QTest::addColumn<QString>(name: "source" ); |
338 | QTest::addColumn<bool>(name: "valid" ); |
339 | QTest::addColumn<bool>(name: "remote" ); |
340 | |
341 | QTest::newRow(dataTag: "local" ) << testFileUrl(fileName: "colors-round.sci" ).toString() << true << /*remote*/false; |
342 | QTest::newRow(dataTag: "local quoted filename" ) << testFileUrl(fileName: "colors-round-quotes.sci" ).toString() << true << /*remote*/false; |
343 | QTest::newRow(dataTag: "local not found" ) << testFileUrl(fileName: "no-such-file.sci" ).toString() << false << /*remote*/false; |
344 | QTest::newRow(dataTag: "remote" ) << "/colors-round.sci" << true << /*remote*/true; |
345 | QTest::newRow(dataTag: "remote filename quoted" ) << "/colors-round-quotes.sci" << true << /*remote*/true; |
346 | QTest::newRow(dataTag: "remote image" ) << "/colors-round-remote.sci" << true << /*remote*/true; |
347 | QTest::newRow(dataTag: "remote not found" ) << "/no-such-file.sci" << false << /*remote*/true; |
348 | } |
349 | |
350 | void tst_qquickborderimage::invalidSciFile() |
351 | { |
352 | QTest::ignoreMessage(type: QtWarningMsg, message: "QQuickGridScaledImage: Invalid tile rule specified. Using Stretch." ); // for "Roun" |
353 | QTest::ignoreMessage(type: QtWarningMsg, message: "QQuickGridScaledImage: Invalid tile rule specified. Using Stretch." ); // for "Repea" |
354 | |
355 | QString componentStr = "import QtQuick 2.0\nBorderImage { source: \"" + testFileUrl(fileName: "invalid.sci" ).toString() +"\"; width: 300; height: 300 }" ; |
356 | QQmlComponent component(&engine); |
357 | component.setData(componentStr.toLatin1(), baseUrl: QUrl::fromLocalFile(localfile: "" )); |
358 | QQuickBorderImage *obj = qobject_cast<QQuickBorderImage*>(object: component.create()); |
359 | QVERIFY(obj != nullptr); |
360 | QCOMPARE(obj->width(), 300.); |
361 | QCOMPARE(obj->height(), 300.); |
362 | QCOMPARE(obj->status(), QQuickImageBase::Error); |
363 | QCOMPARE(obj->horizontalTileMode(), QQuickBorderImage::Stretch); |
364 | QCOMPARE(obj->verticalTileMode(), QQuickBorderImage::Stretch); |
365 | |
366 | delete obj; |
367 | } |
368 | |
369 | void tst_qquickborderimage::validSciFiles_data() |
370 | { |
371 | QTest::addColumn<QString>(name: "source" ); |
372 | |
373 | QTest::newRow(dataTag: "valid1" ) << testFileUrl(fileName: "valid1.sci" ).toString(); |
374 | QTest::newRow(dataTag: "valid2" ) << testFileUrl(fileName: "valid2.sci" ).toString(); |
375 | QTest::newRow(dataTag: "valid3" ) << testFileUrl(fileName: "valid3.sci" ).toString(); |
376 | QTest::newRow(dataTag: "valid4" ) << testFileUrl(fileName: "valid4.sci" ).toString(); |
377 | } |
378 | |
379 | void tst_qquickborderimage::validSciFiles() |
380 | { |
381 | QFETCH(QString, source); |
382 | |
383 | QString componentStr = "import QtQuick 2.0\nBorderImage { source: \"" + source +"\"; width: 300; height: 300 }" ; |
384 | QQmlComponent component(&engine); |
385 | component.setData(componentStr.toLatin1(), baseUrl: QUrl::fromLocalFile(localfile: "" )); |
386 | QQuickBorderImage *obj = qobject_cast<QQuickBorderImage*>(object: component.create()); |
387 | QVERIFY(obj != nullptr); |
388 | QCOMPARE(obj->width(), 300.); |
389 | QCOMPARE(obj->height(), 300.); |
390 | QCOMPARE(obj->horizontalTileMode(), QQuickBorderImage::Round); |
391 | QCOMPARE(obj->verticalTileMode(), QQuickBorderImage::Repeat); |
392 | |
393 | delete obj; |
394 | } |
395 | |
396 | void tst_qquickborderimage::pendingRemoteRequest() |
397 | { |
398 | QFETCH(QString, source); |
399 | |
400 | QString componentStr = "import QtQuick 2.0\nBorderImage { source: \"" + source + "\" }" ; |
401 | QQmlComponent component(&engine); |
402 | component.setData(componentStr.toLatin1(), baseUrl: QUrl::fromLocalFile(localfile: "" )); |
403 | QQuickBorderImage *obj = qobject_cast<QQuickBorderImage*>(object: component.create()); |
404 | QVERIFY(obj != nullptr); |
405 | QCOMPARE(obj->status(), QQuickBorderImage::Loading); |
406 | |
407 | // verify no crash |
408 | // This will cause a delayed "QThread: Destroyed while thread is still running" warning |
409 | delete obj; |
410 | QTest::qWait(ms: 50); |
411 | } |
412 | |
413 | void tst_qquickborderimage::pendingRemoteRequest_data() |
414 | { |
415 | QTest::addColumn<QString>(name: "source" ); |
416 | |
417 | QTest::newRow(dataTag: "png file" ) << "http://localhost/none.png" ; |
418 | QTest::newRow(dataTag: "sci file" ) << "http://localhost/none.sci" ; |
419 | } |
420 | |
421 | //QTBUG-26155 |
422 | void tst_qquickborderimage::statusChanges_data() |
423 | { |
424 | QTest::addColumn<QString>(name: "source" ); |
425 | QTest::addColumn<int>(name: "emissions" ); |
426 | QTest::addColumn<bool>(name: "remote" ); |
427 | QTest::addColumn<QQuickImageBase::Status>(name: "finalStatus" ); |
428 | |
429 | QTest::newRow(dataTag: "localfile" ) << testFileUrl(fileName: "colors.png" ).toString() << 1 << false << QQuickImageBase::Ready; |
430 | QTest::newRow(dataTag: "nofile" ) << "" << 0 << false << QQuickImageBase::Null; |
431 | QTest::newRow(dataTag: "nonexistent" ) << testFileUrl(fileName: "thisfiledoesnotexist.png" ).toString() << 1 << false << QQuickImageBase::Error; |
432 | QTest::newRow(dataTag: "noprotocol" ) << QString("thisfiledoesnotexisteither.png" ) << 2 << false << QQuickImageBase::Error; |
433 | QTest::newRow(dataTag: "remote" ) << "/colors.png" << 2 << true << QQuickImageBase::Ready; |
434 | } |
435 | |
436 | void tst_qquickborderimage::statusChanges() |
437 | { |
438 | QFETCH(QString, source); |
439 | QFETCH(int, emissions); |
440 | QFETCH(bool, remote); |
441 | QFETCH(QQuickImageBase::Status, finalStatus); |
442 | |
443 | TestHTTPServer server; |
444 | if (remote) { |
445 | QVERIFY2(server.listen(), qPrintable(server.errorString())); |
446 | server.serveDirectory(dataDirectory()); |
447 | source = server.urlString(documentPath: source); |
448 | } |
449 | |
450 | QString componentStr = "import QtQuick 2.0\nBorderImage { width: 300; height: 300 }" ; |
451 | QQmlComponent component(&engine); |
452 | component.setData(componentStr.toLatin1(), baseUrl: QUrl("" )); |
453 | |
454 | QQuickBorderImage *obj = qobject_cast<QQuickBorderImage*>(object: component.create()); |
455 | qRegisterMetaType<QQuickImageBase::Status>(); |
456 | QSignalSpy spy(obj, SIGNAL(statusChanged(QQuickImageBase::Status))); |
457 | QVERIFY(obj != nullptr); |
458 | obj->setSource(source); |
459 | if (remote) |
460 | server.sendDelayedItem(); |
461 | QTRY_COMPARE(obj->status(), finalStatus); |
462 | QCOMPARE(spy.count(), emissions); |
463 | |
464 | delete obj; |
465 | } |
466 | |
467 | void tst_qquickborderimage::sourceSizeChanges() |
468 | { |
469 | TestHTTPServer server; |
470 | QVERIFY2(server.listen(), qPrintable(server.errorString())); |
471 | server.serveDirectory(dataDirectory()); |
472 | |
473 | QQmlEngine engine; |
474 | QQmlComponent component(&engine); |
475 | component.setData("import QtQuick 2.0\nBorderImage { source: srcImage }" , baseUrl: QUrl::fromLocalFile(localfile: "" )); |
476 | QTRY_VERIFY(component.isReady()); |
477 | QQmlContext *ctxt = engine.rootContext(); |
478 | ctxt->setContextProperty("srcImage" , "" ); |
479 | QQuickBorderImage *obj = qobject_cast<QQuickBorderImage*>(object: component.create()); |
480 | QVERIFY(obj != nullptr); |
481 | |
482 | QSignalSpy sourceSizeSpy(obj, SIGNAL(sourceSizeChanged())); |
483 | |
484 | // Local |
485 | ctxt->setContextProperty("srcImage" , QUrl("" )); |
486 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Null); |
487 | QTRY_COMPARE(sourceSizeSpy.count(), 0); |
488 | |
489 | ctxt->setContextProperty("srcImage" , testFileUrl(fileName: "heart200.png" )); |
490 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Ready); |
491 | QTRY_COMPARE(sourceSizeSpy.count(), 1); |
492 | |
493 | ctxt->setContextProperty("srcImage" , testFileUrl(fileName: "heart200.png" )); |
494 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Ready); |
495 | QTRY_COMPARE(sourceSizeSpy.count(), 1); |
496 | |
497 | ctxt->setContextProperty("srcImage" , testFileUrl(fileName: "heart200_copy.png" )); |
498 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Ready); |
499 | QTRY_COMPARE(sourceSizeSpy.count(), 1); |
500 | |
501 | ctxt->setContextProperty("srcImage" , testFileUrl(fileName: "colors.png" )); |
502 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Ready); |
503 | QTRY_COMPARE(sourceSizeSpy.count(), 2); |
504 | |
505 | ctxt->setContextProperty("srcImage" , QUrl("" )); |
506 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Null); |
507 | QTRY_COMPARE(sourceSizeSpy.count(), 3); |
508 | |
509 | // Remote |
510 | ctxt->setContextProperty("srcImage" , server.url(documentPath: "/heart200.png" )); |
511 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Ready); |
512 | QTRY_COMPARE(sourceSizeSpy.count(), 4); |
513 | |
514 | ctxt->setContextProperty("srcImage" , server.url(documentPath: "/heart200.png" )); |
515 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Ready); |
516 | QTRY_COMPARE(sourceSizeSpy.count(), 4); |
517 | |
518 | ctxt->setContextProperty("srcImage" , server.url(documentPath: "/heart200_copy.png" )); |
519 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Ready); |
520 | QTRY_COMPARE(sourceSizeSpy.count(), 4); |
521 | |
522 | ctxt->setContextProperty("srcImage" , server.url(documentPath: "/colors.png" )); |
523 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Ready); |
524 | QTRY_COMPARE(sourceSizeSpy.count(), 5); |
525 | |
526 | ctxt->setContextProperty("srcImage" , QUrl("" )); |
527 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Null); |
528 | QTRY_COMPARE(sourceSizeSpy.count(), 6); |
529 | |
530 | delete obj; |
531 | } |
532 | |
533 | void tst_qquickborderimage::progressAndStatusChanges() |
534 | { |
535 | TestHTTPServer server; |
536 | QVERIFY2(server.listen(), qPrintable(server.errorString())); |
537 | server.serveDirectory(dataDirectory()); |
538 | |
539 | QQmlEngine engine; |
540 | QString componentStr = "import QtQuick 2.0\nBorderImage { source: srcImage }" ; |
541 | QQmlContext *ctxt = engine.rootContext(); |
542 | ctxt->setContextProperty("srcImage" , testFileUrl(fileName: "heart200.png" )); |
543 | QQmlComponent component(&engine); |
544 | component.setData(componentStr.toLatin1(), baseUrl: QUrl::fromLocalFile(localfile: "" )); |
545 | QQuickBorderImage *obj = qobject_cast<QQuickBorderImage*>(object: component.create()); |
546 | QVERIFY(obj != nullptr); |
547 | QCOMPARE(obj->status(), QQuickBorderImage::Ready); |
548 | QTRY_COMPARE(obj->progress(), 1.0); |
549 | |
550 | qRegisterMetaType<QQuickBorderImage::Status>(); |
551 | QSignalSpy sourceSpy(obj, SIGNAL(sourceChanged(QUrl))); |
552 | QSignalSpy progressSpy(obj, SIGNAL(progressChanged(qreal))); |
553 | QSignalSpy statusSpy(obj, SIGNAL(statusChanged(QQuickImageBase::Status))); |
554 | |
555 | // Same file |
556 | ctxt->setContextProperty("srcImage" , testFileUrl(fileName: "heart200.png" )); |
557 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Ready); |
558 | QTRY_COMPARE(obj->progress(), 1.0); |
559 | QTRY_COMPARE(sourceSpy.count(), 0); |
560 | QTRY_COMPARE(progressSpy.count(), 0); |
561 | QTRY_COMPARE(statusSpy.count(), 0); |
562 | |
563 | // Loading local file |
564 | ctxt->setContextProperty("srcImage" , testFileUrl(fileName: "colors.png" )); |
565 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Ready); |
566 | QTRY_COMPARE(obj->progress(), 1.0); |
567 | QTRY_COMPARE(sourceSpy.count(), 1); |
568 | QTRY_COMPARE(progressSpy.count(), 0); |
569 | QTRY_COMPARE(statusSpy.count(), 1); |
570 | |
571 | // Loading remote file |
572 | ctxt->setContextProperty("srcImage" , server.url(documentPath: "/heart200.png" )); |
573 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Loading); |
574 | QTRY_COMPARE(obj->progress(), 0.0); |
575 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Ready); |
576 | QTRY_COMPARE(obj->progress(), 1.0); |
577 | QTRY_COMPARE(sourceSpy.count(), 2); |
578 | QTRY_VERIFY(progressSpy.count() > 1); |
579 | QTRY_COMPARE(statusSpy.count(), 3); |
580 | |
581 | ctxt->setContextProperty("srcImage" , "" ); |
582 | QTRY_COMPARE(obj->status(), QQuickBorderImage::Null); |
583 | QTRY_COMPARE(obj->progress(), 0.0); |
584 | QTRY_COMPARE(sourceSpy.count(), 3); |
585 | QTRY_VERIFY(progressSpy.count() > 2); |
586 | QTRY_COMPARE(statusSpy.count(), 4); |
587 | |
588 | delete obj; |
589 | } |
590 | #if QT_CONFIG(opengl) |
591 | void tst_qquickborderimage::borderImageMesh() |
592 | { |
593 | QQuickView *window = new QQuickView; |
594 | |
595 | window->setSource(testFileUrl(fileName: "nonmesh.qml" )); |
596 | window->show(); |
597 | QVERIFY(QTest::qWaitForWindowExposed(window)); |
598 | QImage nonmesh = window->grabWindow(); |
599 | |
600 | window->setSource(testFileUrl(fileName: "mesh.qml" )); |
601 | QImage mesh = window->grabWindow(); |
602 | |
603 | QString errorMessage; |
604 | QVERIFY2(QQuickVisualTestUtil::compareImages(mesh, nonmesh, &errorMessage), |
605 | qPrintable(errorMessage)); |
606 | } |
607 | #endif |
608 | |
609 | void tst_qquickborderimage::multiFrame_data() |
610 | { |
611 | QTest::addColumn<QString>(name: "qmlfile" ); |
612 | QTest::addColumn<bool>(name: "asynchronous" ); |
613 | |
614 | QTest::addRow(format: "default" ) << "multiframe.qml" << false; |
615 | QTest::addRow(format: "async" ) << "multiframeAsync.qml" << true; |
616 | } |
617 | |
618 | void tst_qquickborderimage::multiFrame() |
619 | { |
620 | if ((QGuiApplication::platformName() == QLatin1String("offscreen" )) |
621 | || (QGuiApplication::platformName() == QLatin1String("minimal" ))) |
622 | QSKIP("Skipping due to grabWindow not functional on offscreen/minimal platforms" ); |
623 | |
624 | QFETCH(QString, qmlfile); |
625 | QFETCH(bool, asynchronous); |
626 | Q_UNUSED(asynchronous) |
627 | |
628 | QQuickView view(testFileUrl(fileName: qmlfile)); |
629 | QQuickBorderImage *image = qobject_cast<QQuickBorderImage*>(object: view.rootObject()); |
630 | QVERIFY(image); |
631 | QSignalSpy countSpy(image, SIGNAL(frameCountChanged())); |
632 | QSignalSpy currentSpy(image, SIGNAL(currentFrameChanged())); |
633 | if (asynchronous) { |
634 | QCOMPARE(image->frameCount(), 0); |
635 | QTRY_COMPARE(image->frameCount(), 4); |
636 | QCOMPARE(countSpy.count(), 1); |
637 | } else { |
638 | QCOMPARE(image->frameCount(), 4); |
639 | } |
640 | QCOMPARE(image->currentFrame(), 0); |
641 | view.show(); |
642 | QVERIFY(QTest::qWaitForWindowExposed(&view)); |
643 | QCoreApplication::processEvents(); // Process all queued events |
644 | |
645 | QImage contents = view.grabWindow(); |
646 | if (contents.width() < 160) |
647 | QSKIP("Skipping due to grabWindow not functional" ); |
648 | |
649 | // The middle of the first frame looks blue, approximately qRgba(0x43, 0x7e, 0xd6, 0xff) |
650 | QColor color = contents.pixelColor(x: 60, y: 60); |
651 | qCDebug(lcTests) << "expected bluish color, got" << color; |
652 | QVERIFY(color.redF() < 0.75); |
653 | QVERIFY(color.greenF() < 0.75); |
654 | QVERIFY(color.blueF() > 0.75); |
655 | |
656 | image->setCurrentFrame(1); |
657 | QTRY_COMPARE(image->status(), QQuickImageBase::Ready); |
658 | QCOMPARE(currentSpy.count(), 1); |
659 | QCOMPARE(image->currentFrame(), 1); |
660 | contents = view.grabWindow(); |
661 | // The middle of the second frame looks green, approximately qRgba(0x3a, 0xd2, 0x31, 0xff) |
662 | color = contents.pixelColor(x: 60, y: 60); |
663 | qCDebug(lcTests) << "expected greenish color, got" << color; |
664 | QVERIFY(color.redF() < 0.75); |
665 | QVERIFY(color.green() > 0.75); |
666 | QVERIFY(color.blueF() < 0.75); |
667 | } |
668 | |
669 | QTEST_MAIN(tst_qquickborderimage) |
670 | |
671 | #include "tst_qquickborderimage.moc" |
672 | |