1// Copyright (C) 2021 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only
3
4#include "visualtestutils_p.h"
5
6#include <QtCore/QCoreApplication>
7#include <QtCore/private/qvariantanimation_p.h>
8#include <QtCore/QDebug>
9#include <QtQuick/QQuickItem>
10#if QT_CONFIG(quick_itemview)
11#include <QtQuick/private/qquickitemview_p.h>
12#endif
13#include <QtQuickTest/QtQuickTest>
14#include <QtQuickTestUtils/private/viewtestutils_p.h>
15
16QT_BEGIN_NAMESPACE
17
18QQuickItem *QQuickVisualTestUtils::findVisibleChild(QQuickItem *parent, const QString &objectName)
19{
20 QQuickItem *item = nullptr;
21 QList<QQuickItem*> items = parent->findChildren<QQuickItem*>(aName: objectName);
22 for (int i = 0; i < items.size(); ++i) {
23 if (items.at(i)->isVisible() && !QQuickItemPrivate::get(item: items.at(i))->culled) {
24 item = items.at(i);
25 break;
26 }
27 }
28 return item;
29}
30
31void QQuickVisualTestUtils::dumpTree(QQuickItem *parent, int depth)
32{
33 static QString padding = QStringLiteral(" ");
34 for (int i = 0; i < parent->childItems().size(); ++i) {
35 QQuickItem *item = qobject_cast<QQuickItem*>(o: parent->childItems().at(i));
36 if (!item)
37 continue;
38 qDebug() << padding.left(n: depth*2) << item;
39 dumpTree(parent: item, depth: depth+1);
40 }
41}
42
43void QQuickVisualTestUtils::moveMouseAway(QQuickWindow *window)
44{
45#if QT_CONFIG(cursor) // Get the cursor out of the way.
46 // Using "bottomRight() + QPoint(100, 100)" was causing issues on Ubuntu,
47 // where the window was positioned at the bottom right corner of the window
48 // (even after centering the window on the screen), so we use another position.
49 QCursor::setPos(window->frameGeometry().bottomLeft() + QPoint(-10, 10));
50#endif
51
52 // make sure hover events from QQuickDeliveryAgentPrivate::flushFrameSynchronousEvents()
53 // do not interfere with the tests
54 QEvent leave(QEvent::Leave);
55 QCoreApplication::sendEvent(receiver: window, event: &leave);
56}
57
58void QQuickVisualTestUtils::centerOnScreen(QQuickWindow *window)
59{
60 QQuickViewTestUtils::centerOnScreen(window);
61}
62
63QPoint QQuickVisualTestUtils::lerpPoints(const QPoint &point1, const QPoint &point2, qreal t)
64{
65 return QPoint(_q_interpolate(f: point1.x(), t: point2.x(), progress: t), _q_interpolate(f: point1.y(), t: point2.y(), progress: t));
66};
67
68/*!
69 \internal
70
71 Convenience class to linearly interpolate between two pointer move points.
72
73 \code
74 PointLerper pointLerper(window);
75 // Lerps from {0, 0} to {15, 15}.
76 pointLerper.move(15, 15);
77 QVERIFY(parentButton->isHovered());
78
79 // Lerps from {15, 15} to {25, 25}.
80 pointLerper.move(25, 25);
81 QVERIFY(childButton->isHovered());
82 \endcode
83*/
84QQuickVisualTestUtils::PointLerper::PointLerper(QQuickWindow *window, const QPointingDevice *pointingDevice)
85 : mWindow(window)
86 , mPointingDevice(pointingDevice)
87{
88}
89
90/*!
91 \internal
92
93 Moves from the last pos (or {0, 0} if there have been no calls
94 to this function yet) to \a pos using linear interpolation
95 over 10 (default value) steps with 1 ms (default value) delays
96 between each step.
97*/
98void QQuickVisualTestUtils::PointLerper::move(const QPoint &pos, int steps, int delayInMilliseconds)
99{
100 forEachStep(steps, func: [&](qreal progress) {
101 QQuickTest::pointerMove(dev: mPointingDevice, window: mWindow, pointId: 0, p: lerpPoints(point1: mFrom, point2: pos, t: progress));
102 QTest::qWait(ms: delayInMilliseconds);
103 });
104 mFrom = pos;
105};
106
107void QQuickVisualTestUtils::PointLerper::move(int x, int y, int steps, int delayInMilliseconds)
108{
109 move(pos: QPoint(x, y), steps, delayInMilliseconds);
110};
111
112/*!
113 \internal
114
115 Returns \c true if \c {item->isVisible()} returns \c true, and
116 the item is not culled.
117*/
118bool QQuickVisualTestUtils::isDelegateVisible(QQuickItem *item)
119{
120 return item->isVisible() && !QQuickItemPrivate::get(item)->culled;
121}
122
123/*!
124 \internal
125
126 Compares \a ia with \a ib, returning \c true if the images are equal.
127 If they are not equal, \c false is returned and \a errorMessage is set.
128
129 A custom compare function to avoid issues such as:
130 When running on native Nvidia graphics cards on linux, the
131 distance field glyph pixels have a measurable, but not visible
132 pixel error. This was GT-216 with the ubuntu "nvidia-319" driver package.
133 llvmpipe does not show the same issue.
134*/
135bool QQuickVisualTestUtils::compareImages(const QImage &ia, const QImage &ib, QString *errorMessage)
136{
137 if (ia.size() != ib.size()) {
138 QDebug(errorMessage) << "Images are of different size:" << ia.size() << ib.size()
139 << "DPR:" << ia.devicePixelRatio() << ib.devicePixelRatio();
140 return false;
141 }
142 if (ia.format() != ib.format()) {
143 QDebug(errorMessage) << "Images are of different formats:" << ia.format() << ib.format();
144 return false;
145 }
146
147 int w = ia.width();
148 int h = ia.height();
149 const int tolerance = 5;
150 for (int y=0; y<h; ++y) {
151 const uint *as= (const uint *) ia.constScanLine(y);
152 const uint *bs= (const uint *) ib.constScanLine(y);
153 for (int x=0; x<w; ++x) {
154 uint a = as[x];
155 uint b = bs[x];
156
157 // No tolerance for error in the alpha.
158 if ((a & 0xff000000) != (b & 0xff000000)
159 || qAbs(t: qRed(rgb: a) - qRed(rgb: b)) > tolerance
160 || qAbs(t: qGreen(rgb: a) - qGreen(rgb: b)) > tolerance
161 || qAbs(t: qBlue(rgb: a) - qBlue(rgb: b)) > tolerance) {
162 QDebug(errorMessage) << "Mismatch at:" << x << y << ':'
163 << Qt::hex << Qt::showbase << a << b;
164 return false;
165 }
166 }
167 }
168 return true;
169}
170
171#if QT_CONFIG(quick_itemview)
172/*!
173 \internal
174
175 Finds the delegate at \c index belonging to \c itemView, using the given \c flags.
176
177 If the view needs to be polished, the function will wait for it to be done before continuing,
178 and returns \c nullptr if the polish didn't happen.
179*/
180QQuickItem *QQuickVisualTestUtils::findViewDelegateItem(QQuickItemView *itemView, int index, FindViewDelegateItemFlags flags)
181{
182 if (QQuickTest::qIsPolishScheduled(item: itemView)) {
183 if (!QQuickTest::qWaitForPolish(item: itemView)) {
184 qWarning() << "failed to polish" << itemView;
185 return nullptr;
186 }
187 }
188
189 // Do this after the polish, just in case the count changes after a polish...
190 if (index <= -1 || index >= itemView->count()) {
191 qWarning() << "index" << index << "is out of bounds for" << itemView;
192 return nullptr;
193 }
194
195 if (flags.testFlag(flag: FindViewDelegateItemFlag::PositionViewAtIndex))
196 itemView->positionViewAtIndex(index, mode: QQuickItemView::Center);
197
198 return itemView->itemAtIndex(index);
199}
200#endif
201
202QQuickVisualTestUtils::QQuickApplicationHelper::QQuickApplicationHelper(QQmlDataTest *testCase,
203 const QString &testFilePath, const QVariantMap &initialProperties, const QStringList &qmlImportPaths)
204{
205 for (const auto &path : qmlImportPaths)
206 engine.addImportPath(dir: path);
207
208 QQmlComponent component(&engine);
209
210 component.loadUrl(url: testCase->testFileUrl(fileName: testFilePath));
211 QVERIFY2(component.isReady(), qPrintable(component.errorString()));
212 QObject *rootObject = component.createWithInitialProperties(initialProperties);
213 cleanup.reset(other: rootObject);
214 if (component.isError() || !rootObject) {
215 errorMessage = QString::fromUtf8(utf8: "Failed to create window: %1").arg(a: component.errorString()).toUtf8();
216 return;
217 }
218
219 window = qobject_cast<QQuickWindow*>(object: rootObject);
220 if (!window) {
221 errorMessage = QString::fromUtf8(utf8: "Root object %1 must be a QQuickWindow subclass").arg(a: QDebug::toString(object: window)).toUtf8();
222 return;
223 }
224
225 if (window->isVisible()) {
226 errorMessage = QString::fromUtf8(utf8: "Expected window not to be visible, but it is").toUtf8();
227 return;
228 }
229
230 ready = true;
231}
232
233QQuickVisualTestUtils::MnemonicKeySimulator::MnemonicKeySimulator(QWindow *window)
234 : m_window(window), m_modifiers(Qt::NoModifier)
235{
236}
237
238void QQuickVisualTestUtils::MnemonicKeySimulator::press(Qt::Key key)
239{
240 // QTest::keyPress() but not generating the press event for the modifier key.
241 if (key == Qt::Key_Alt)
242 m_modifiers |= Qt::AltModifier;
243 QTest::simulateEvent(window: m_window, press: true, code: key, modifier: m_modifiers, text: QString(), repeat: false);
244}
245
246void QQuickVisualTestUtils::MnemonicKeySimulator::release(Qt::Key key)
247{
248 // QTest::keyRelease() but not generating the release event for the modifier key.
249 if (key == Qt::Key_Alt)
250 m_modifiers &= ~Qt::AltModifier;
251 QTest::simulateEvent(window: m_window, press: false, code: key, modifier: m_modifiers, text: QString(), repeat: false);
252}
253
254void QQuickVisualTestUtils::MnemonicKeySimulator::click(Qt::Key key)
255{
256 press(key);
257 release(key);
258}
259
260QPoint QQuickVisualTestUtils::mapCenterToWindow(const QQuickItem *item)
261{
262 return item->mapToScene(point: QPointF(item->width() / 2, item->height() / 2)).toPoint();
263}
264
265QPoint QQuickVisualTestUtils::mapToWindow(const QQuickItem *item, qreal relativeX, qreal relativeY)
266{
267 return item->mapToScene(point: QPointF(relativeX, relativeY)).toPoint();
268}
269
270QPoint QQuickVisualTestUtils::mapToWindow(const QQuickItem *item, const QPointF &relativePos)
271{
272 return mapToWindow(item, relativeX: relativePos.x(), relativeY: relativePos.y());
273}
274
275QT_END_NAMESPACE
276
277#include "moc_visualtestutils_p.cpp"
278

Provided by KDAB

Privacy Policy
Learn Advanced QML with KDAB
Find out more

source code of qtdeclarative/src/quicktestutils/quick/visualtestutils.cpp