efc02f9cc3
When XCB_INPUT_TOUCH_BEGIN closes a popup, we then receive XCB_INPUT_TOUCH_END, and cannot find a target window (because it's destroyed). If we don't deliver it, we need to at least clear the stored point from QPointingDevicePrivate::activePoints. Then when we deliver the next touch press, m_fakeMouseSourcePointId also needs to be reset. It's now even more paramount that autotests (and real-world touchscreens) must never omit any active touchpoint from a touch event. If a point doesn't move, it must be included in the QTouchEvent, with Stationary state. If not, QGuiApp::processTouchEvent() could generate multiple TouchBegin events in a row, which gets other bits of logic confused, here and there. Fixes: QTBUG-94557 Fixes: QTBUG-98519 Fixes: QTBUG-102751 Fixes: QTBUG-103706 Pick-to: 6.2 6.3 5.15 Change-Id: Ia95e410a2bb8bc7784aa5d296fac2b89e53a9f55 Reviewed-by: Shawn Rutledge <shawn.rutledge@qt.io>
329 lines
9.8 KiB
C++
329 lines
9.8 KiB
C++
// Copyright (C) 2016 The Qt Company Ltd.
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
|
|
|
|
|
#include <QtTest/QTest>
|
|
#include <QtWidgets/QApplication>
|
|
#include <QtWidgets/QWidget>
|
|
#include <QtWidgets/QGestureEvent>
|
|
#include <QtGui/QScreen>
|
|
#include <QtGui/QPointingDevice>
|
|
#include <QtCore/QList>
|
|
#include <QtCore/QString>
|
|
#include <QtCore/QHash>
|
|
#include <QtCore/QDebug>
|
|
|
|
class tst_QGestureRecognizer : public QObject
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
tst_QGestureRecognizer();
|
|
|
|
private Q_SLOTS:
|
|
void initTestCase();
|
|
#ifndef QT_NO_GESTURES
|
|
void panGesture_data();
|
|
void panGesture();
|
|
void pinchGesture_data();
|
|
void pinchGesture();
|
|
void swipeGesture_data();
|
|
void swipeGesture();
|
|
void touchReplay();
|
|
#endif // !QT_NO_GESTURES
|
|
|
|
private:
|
|
const int m_fingerDistance;
|
|
QPointingDevice *m_touchDevice;
|
|
};
|
|
|
|
tst_QGestureRecognizer::tst_QGestureRecognizer()
|
|
: m_fingerDistance(qRound(QGuiApplication::primaryScreen()->physicalDotsPerInch() / 2.0))
|
|
, m_touchDevice(QTest::createTouchDevice())
|
|
{
|
|
qputenv("QT_PAN_TOUCHPOINTS", "2"); // Prevent device detection of pan touch point count.
|
|
}
|
|
|
|
void tst_QGestureRecognizer::initTestCase()
|
|
{
|
|
}
|
|
|
|
#ifndef QT_NO_GESTURES
|
|
|
|
typedef QList<Qt::GestureType> GestureTypeVector;
|
|
|
|
class TestWidget : public QWidget
|
|
{
|
|
public:
|
|
explicit TestWidget(const GestureTypeVector &gestureTypes);
|
|
|
|
bool gestureReceived(Qt::GestureType gestureType) const
|
|
{ return m_receivedGestures.value(gestureType); }
|
|
|
|
protected:
|
|
bool event(QEvent * event) override;
|
|
|
|
private:
|
|
typedef QHash<Qt::GestureType, bool> GestureTypeHash;
|
|
GestureTypeHash m_receivedGestures;
|
|
};
|
|
|
|
TestWidget::TestWidget(const GestureTypeVector &gestureTypes)
|
|
{
|
|
setAttribute(Qt::WA_AcceptTouchEvents);
|
|
|
|
foreach (Qt::GestureType gestureType, gestureTypes) {
|
|
grabGesture(gestureType);
|
|
m_receivedGestures.insert(gestureType, false);
|
|
}
|
|
|
|
const QRect geometry = QGuiApplication::primaryScreen()->availableGeometry();
|
|
const QSize size = geometry.size() / 2;
|
|
resize(size);
|
|
move(geometry.center() - QPoint(size.width() / 2, size.height() / 2));
|
|
}
|
|
|
|
bool TestWidget::event(QEvent * event)
|
|
{
|
|
switch (event->type()) {
|
|
case QEvent::Gesture: {
|
|
const QGestureEvent *gestureEvent = static_cast<QGestureEvent *>(event);
|
|
const GestureTypeHash::iterator hend = m_receivedGestures.end();
|
|
for (GestureTypeHash::iterator it = m_receivedGestures.begin(); it != hend; ++it) {
|
|
if (const QGesture *gesture = gestureEvent->gesture(it.key())) {
|
|
if (gesture->state() == Qt::GestureFinished)
|
|
it.value() = true;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
return QWidget::event(event);
|
|
}
|
|
|
|
static void pressSequence(QTest::QTouchEventWidgetSequence &sequence, QList<QPoint> &points,
|
|
QWidget *widget)
|
|
{
|
|
const int pointCount = points.size();
|
|
for (int p = 0; p < pointCount; ++p)
|
|
sequence.press(p, points.at(p), widget);
|
|
sequence.commit();
|
|
}
|
|
|
|
static void linearSequence(int n, const QPoint &delta, QTest::QTouchEventWidgetSequence &sequence,
|
|
QList<QPoint> &points, QWidget *widget)
|
|
{
|
|
const int pointCount = points.size();
|
|
for (int s = 0; s < n; ++s) {
|
|
for (int p = 0; p < pointCount; ++p) {
|
|
points[p] += delta;
|
|
sequence.move(p, points[p], widget);
|
|
}
|
|
sequence.commit();
|
|
}
|
|
}
|
|
|
|
static void releaseSequence(QTest::QTouchEventWidgetSequence &sequence, QList<QPoint> &points,
|
|
QWidget *widget)
|
|
{
|
|
const int pointCount = points.size();
|
|
for (int p = 0; p < pointCount; ++p)
|
|
sequence.release(p, points[p], widget);
|
|
sequence.commit();
|
|
}
|
|
|
|
// --- Pan
|
|
|
|
enum PanSubTest {
|
|
TwoFingerPanSubTest
|
|
};
|
|
|
|
void tst_QGestureRecognizer::panGesture_data()
|
|
{
|
|
QTest::addColumn<int>("panSubTest");
|
|
QTest::addColumn<bool>("gestureExpected");
|
|
QTest::newRow("Two finger") << int(TwoFingerPanSubTest) << true;
|
|
}
|
|
|
|
void tst_QGestureRecognizer::panGesture()
|
|
{
|
|
QFETCH(int, panSubTest);
|
|
QFETCH(bool, gestureExpected);
|
|
|
|
Q_UNUSED(panSubTest); // Single finger pan will be added later.
|
|
|
|
const int panPoints = 2;
|
|
const Qt::GestureType gestureType = Qt::PanGesture;
|
|
TestWidget widget(GestureTypeVector(1, gestureType));
|
|
widget.setWindowTitle(QTest::currentTestFunction());
|
|
widget.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&widget));
|
|
|
|
QList<QPoint> points;
|
|
for (int i = 0; i < panPoints; ++i)
|
|
points.append(QPoint(10 + i *20, 10 + i *20));
|
|
|
|
QTest::QTouchEventWidgetSequence panSequence = QTest::touchEvent(&widget, m_touchDevice);
|
|
pressSequence(panSequence, points, &widget);
|
|
linearSequence(5, QPoint(20, 20), panSequence, points, &widget);
|
|
releaseSequence(panSequence, points, &widget);
|
|
|
|
if (gestureExpected) {
|
|
QTRY_VERIFY(widget.gestureReceived(gestureType));
|
|
} else {
|
|
QCoreApplication::processEvents();
|
|
QVERIFY(!widget.gestureReceived(gestureType));
|
|
}
|
|
}
|
|
|
|
// --- Pinch
|
|
|
|
enum PinchSubTest {
|
|
StandardPinchSubTest
|
|
};
|
|
|
|
void tst_QGestureRecognizer::pinchGesture_data()
|
|
{
|
|
QTest::addColumn<int>("pinchSubTest");
|
|
QTest::addColumn<bool>("gestureExpected");
|
|
QTest::newRow("Standard") << int(StandardPinchSubTest) << true;
|
|
}
|
|
|
|
void tst_QGestureRecognizer::pinchGesture()
|
|
{
|
|
QFETCH(int, pinchSubTest);
|
|
QFETCH(bool, gestureExpected);
|
|
|
|
Q_UNUSED(pinchSubTest);
|
|
|
|
const Qt::GestureType gestureType = Qt::PinchGesture;
|
|
TestWidget widget(GestureTypeVector(1, gestureType));
|
|
widget.setWindowTitle(QTest::currentTestFunction());
|
|
widget.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&widget));
|
|
|
|
QList<QPoint> points;
|
|
points.append(widget.rect().center());
|
|
points.append(points.front() + QPoint(0, 20));
|
|
|
|
QTest::QTouchEventWidgetSequence pinchSequence = QTest::touchEvent(&widget, m_touchDevice);
|
|
pressSequence(pinchSequence, points, &widget);
|
|
|
|
for (int s = 0; s < 5; ++s) {
|
|
points[0] += QPoint(5, 30);
|
|
pinchSequence.move(0, points[0], &widget);
|
|
points[1] += QPoint(5, -30);
|
|
pinchSequence.move(1, points[1], &widget);
|
|
pinchSequence.commit();
|
|
}
|
|
|
|
releaseSequence(pinchSequence, points, &widget);
|
|
|
|
if (gestureExpected) {
|
|
QTRY_VERIFY(widget.gestureReceived(gestureType));
|
|
} else {
|
|
QCoreApplication::processEvents();
|
|
QVERIFY(!widget.gestureReceived(gestureType));
|
|
}
|
|
}
|
|
|
|
// --- Swipe
|
|
|
|
enum SwipeSubTest {
|
|
SwipeLineSubTest,
|
|
SwipeDirectionChangeSubTest,
|
|
SwipeSmallDirectionChangeSubTest
|
|
};
|
|
|
|
void tst_QGestureRecognizer::swipeGesture_data()
|
|
{
|
|
QTest::addColumn<int>("swipeSubTest");
|
|
QTest::addColumn<bool>("gestureExpected");
|
|
QTest::newRow("Line") << int(SwipeLineSubTest) << true;
|
|
QTest::newRow("DirectionChange") << int(SwipeDirectionChangeSubTest) << false;
|
|
QTest::newRow("SmallDirectionChange") << int(SwipeSmallDirectionChangeSubTest) << true;
|
|
}
|
|
|
|
void tst_QGestureRecognizer::swipeGesture()
|
|
{
|
|
enum { swipePoints = 3 };
|
|
|
|
QFETCH(int, swipeSubTest);
|
|
QFETCH(bool, gestureExpected);
|
|
|
|
const Qt::GestureType gestureType = Qt::SwipeGesture;
|
|
TestWidget widget(GestureTypeVector(1, gestureType));
|
|
widget.setWindowTitle(QTest::currentTestFunction());
|
|
widget.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&widget));
|
|
|
|
// Start a swipe sequence with 2 points (QTBUG-15768)
|
|
const QPoint fingerDistance(m_fingerDistance, m_fingerDistance);
|
|
QList<QPoint> points;
|
|
for (int i = 0; i < swipePoints - 1; ++i)
|
|
points.append(fingerDistance + i * fingerDistance);
|
|
|
|
QTest::QTouchEventWidgetSequence swipeSequence = QTest::touchEvent(&widget, m_touchDevice);
|
|
pressSequence(swipeSequence, points, &widget);
|
|
|
|
// Press point #3
|
|
points.append(points.last() + fingerDistance);
|
|
swipeSequence.stationary(0).stationary(1).press(points.size() - 1, points.last(), &widget);
|
|
swipeSequence.commit();
|
|
Q_ASSERT(points.size() == swipePoints);
|
|
|
|
// Move.
|
|
const QPoint moveDelta(60, 20);
|
|
switch (swipeSubTest) {
|
|
case SwipeLineSubTest:
|
|
linearSequence(5, moveDelta, swipeSequence, points, &widget);
|
|
break;
|
|
case SwipeDirectionChangeSubTest:
|
|
linearSequence(5, moveDelta, swipeSequence, points, &widget);
|
|
linearSequence(3, QPoint(-moveDelta.x(), moveDelta.y()), swipeSequence, points, &widget);
|
|
break;
|
|
case SwipeSmallDirectionChangeSubTest: { // QTBUG-46195, small changes in direction should not cause the gesture to be canceled.
|
|
const QPoint smallChangeMoveDelta(50, 1);
|
|
linearSequence(5, smallChangeMoveDelta, swipeSequence, points, &widget);
|
|
linearSequence(1, QPoint(smallChangeMoveDelta.x(), -3), swipeSequence, points, &widget);
|
|
linearSequence(5, smallChangeMoveDelta, swipeSequence, points, &widget);
|
|
}
|
|
break;
|
|
}
|
|
|
|
releaseSequence(swipeSequence, points, &widget);
|
|
|
|
if (gestureExpected) {
|
|
QTRY_VERIFY(widget.gestureReceived(gestureType));
|
|
} else {
|
|
QCoreApplication::processEvents();
|
|
QVERIFY(!widget.gestureReceived(gestureType));
|
|
}
|
|
}
|
|
|
|
void tst_QGestureRecognizer::touchReplay()
|
|
{
|
|
const Qt::GestureType gestureType = Qt::TapGesture;
|
|
QWidget parent;
|
|
TestWidget widget(GestureTypeVector(1, gestureType));
|
|
widget.setParent(&parent);
|
|
widget.setGeometry(0, 0, 100, 100);
|
|
parent.adjustSize();
|
|
parent.show();
|
|
QVERIFY(QTest::qWaitForWindowActive(&parent));
|
|
|
|
QWindow* windowHandle = parent.window()->windowHandle();
|
|
const QPoint globalPos = QPoint(42, 16);
|
|
QTest::touchEvent(windowHandle, m_touchDevice).press(1, globalPos);
|
|
QTest::touchEvent(windowHandle, m_touchDevice).release(1, globalPos);
|
|
|
|
QVERIFY(widget.gestureReceived(gestureType));
|
|
}
|
|
|
|
#endif // !QT_NO_GESTURES
|
|
|
|
QTEST_MAIN(tst_QGestureRecognizer)
|
|
|
|
#include "tst_qgesturerecognizer.moc"
|