Make QtQuickTest::mouseEvent use QTest::mouseX
Fixes QQuickItem::isUnderMouse returning wrong information when moving the mouse cursor with QtQuickTest::mouseX Also changes TestCase.mouseDrag to actually resemble more what real life does, i.e. send mouse moves with the same localPos if the item has already moved and update tst_drag.qml accordingly Change-Id: I80e4ab097da90d21ba987466c1b82467755a6b56 Reviewed-by: Shawn Rutledge <shawn.rutledge@qt.io>
This commit is contained in:
parent
457c5bcb14
commit
4c46dce8fd
|
@ -1321,15 +1321,27 @@ Item {
|
||||||
if (ddy < (util.dragThreshold + 1))
|
if (ddy < (util.dragThreshold + 1))
|
||||||
ddy = 0
|
ddy = 0
|
||||||
|
|
||||||
|
var originalX = item.x;
|
||||||
|
var originalY = item.y;
|
||||||
|
|
||||||
mousePress(item, x, y, button, modifiers, delay)
|
mousePress(item, x, y, button, modifiers, delay)
|
||||||
//trigger dragging
|
|
||||||
mouseMove(item, x + util.dragThreshold + 1, y + util.dragThreshold + 1, moveDelay, button)
|
// trigger dragging, this doesn't actually move the item yet
|
||||||
|
var triggerDragXPos = x + Math.min(util.dragThreshold + 1, dx);
|
||||||
|
var triggerDragYPos = y + Math.min(util.dragThreshold + 1, dy);
|
||||||
|
mouseMove(item, triggerDragXPos, triggerDragYPos, moveDelay, button)
|
||||||
|
|
||||||
if (ddx > 0 || ddy > 0) {
|
if (ddx > 0 || ddy > 0) {
|
||||||
mouseMove(item, x + ddx, y + ddy, moveDelay, button)
|
// move the item by ddx, ddy
|
||||||
mouseMove(item, x + 2*ddx, y + 2*ddy, moveDelay, button)
|
mouseMove(item, triggerDragXPos + ddx, triggerDragYPos + ddy, moveDelay, button)
|
||||||
|
|
||||||
|
// move the item by ddx, ddy again
|
||||||
|
// need to account for whether the previous move actually moved the item or not
|
||||||
|
mouseMove(item, triggerDragXPos + 2*ddx - (item.x - originalX), triggerDragYPos + 2*ddy - (item.y - originalY), moveDelay, button)
|
||||||
}
|
}
|
||||||
mouseMove(item, x + dx, y + dy, moveDelay, button)
|
// Release, causes a final move
|
||||||
mouseRelease(item, x + dx, y + dy, button, modifiers, delay)
|
// need to account for whether the previous moves actually moved the item or not
|
||||||
|
mouseRelease(item, x + dx - (item.x - originalX), y + dy - (item.y - originalY), button, modifiers, delay)
|
||||||
}
|
}
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
|
|
|
@ -39,6 +39,7 @@
|
||||||
|
|
||||||
#include "quicktestevent_p.h"
|
#include "quicktestevent_p.h"
|
||||||
#include <QtTest/qtestkeyboard.h>
|
#include <QtTest/qtestkeyboard.h>
|
||||||
|
#include <QtTest/qtestmouse.h>
|
||||||
#include <QtQml/qqml.h>
|
#include <QtQml/qqml.h>
|
||||||
#include <QtQuick/qquickitem.h>
|
#include <QtQuick/qquickitem.h>
|
||||||
#include <QtQuick/qquickwindow.h>
|
#include <QtQuick/qquickwindow.h>
|
||||||
|
@ -125,8 +126,6 @@ namespace QtQuickTest
|
||||||
{
|
{
|
||||||
enum MouseAction { MousePress, MouseRelease, MouseClick, MouseDoubleClick, MouseMove, MouseDoubleClickSequence };
|
enum MouseAction { MousePress, MouseRelease, MouseClick, MouseDoubleClick, MouseMove, MouseDoubleClickSequence };
|
||||||
|
|
||||||
int lastMouseTimestamp = 0;
|
|
||||||
|
|
||||||
static void mouseEvent(MouseAction action, QWindow *window,
|
static void mouseEvent(MouseAction action, QWindow *window,
|
||||||
QObject *item, Qt::MouseButton button,
|
QObject *item, Qt::MouseButton button,
|
||||||
Qt::KeyboardModifiers stateKey, const QPointF &_pos, int delay=-1)
|
Qt::KeyboardModifiers stateKey, const QPointF &_pos, int delay=-1)
|
||||||
|
@ -138,7 +137,7 @@ namespace QtQuickTest
|
||||||
delay = QTest::defaultMouseDelay();
|
delay = QTest::defaultMouseDelay();
|
||||||
if (delay > 0) {
|
if (delay > 0) {
|
||||||
QTest::qWait(delay);
|
QTest::qWait(delay);
|
||||||
lastMouseTimestamp += delay;
|
QTest::lastMouseTimestamp += delay;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (action == MouseClick) {
|
if (action == MouseClick) {
|
||||||
|
@ -165,37 +164,32 @@ namespace QtQuickTest
|
||||||
|
|
||||||
stateKey &= static_cast<unsigned int>(Qt::KeyboardModifierMask);
|
stateKey &= static_cast<unsigned int>(Qt::KeyboardModifierMask);
|
||||||
|
|
||||||
QMouseEvent me(QEvent::User, QPoint(), Qt::LeftButton, button, stateKey);
|
if (action == MouseDoubleClick) {
|
||||||
|
// the QWindow variant of QTest::mouseEvent doesn't have MouseDoubleClick as event
|
||||||
|
// its MouseDClick is actually like the MouseDoubleClickSequence here
|
||||||
|
QMouseEvent me(QEvent::MouseButtonDblClick, pos, window->mapToGlobal(pos), button, button, stateKey);
|
||||||
|
me.setTimestamp(++QTest::lastMouseTimestamp);
|
||||||
|
QSpontaneKeyEvent::setSpontaneous(&me);
|
||||||
|
if (!qApp->notify(window, &me)) {
|
||||||
|
QWARN("Mouse event \"MouseDoubleClick\" not accepted by receiving window");
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
switch (action)
|
switch (action)
|
||||||
{
|
{
|
||||||
case MousePress:
|
case MousePress:
|
||||||
me = QMouseEvent(QEvent::MouseButtonPress, pos, window->mapToGlobal(pos), button, button, stateKey);
|
QTest::mousePress(window, button, stateKey, pos, /*delay*/ 0);
|
||||||
me.setTimestamp(++lastMouseTimestamp);
|
|
||||||
break;
|
break;
|
||||||
case MouseRelease:
|
case MouseRelease:
|
||||||
me = QMouseEvent(QEvent::MouseButtonRelease, pos, window->mapToGlobal(pos), button, 0, stateKey);
|
QTest::mouseRelease(window, button, stateKey, pos, /*delay*/ 0);
|
||||||
me.setTimestamp(++lastMouseTimestamp);
|
|
||||||
lastMouseTimestamp += 500; // avoid double clicks being generated
|
|
||||||
break;
|
|
||||||
case MouseDoubleClick:
|
|
||||||
me = QMouseEvent(QEvent::MouseButtonDblClick, pos, window->mapToGlobal(pos), button, button, stateKey);
|
|
||||||
me.setTimestamp(++lastMouseTimestamp);
|
|
||||||
break;
|
break;
|
||||||
case MouseMove:
|
case MouseMove:
|
||||||
// with move event the button is NoButton, but 'buttons' holds the currently pressed buttons
|
QTest::mouseMove(window, pos, /*delay*/ 0);
|
||||||
me = QMouseEvent(QEvent::MouseMove, pos, window->mapToGlobal(pos), Qt::NoButton, button, stateKey);
|
|
||||||
me.setTimestamp(++lastMouseTimestamp);
|
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
QTEST_ASSERT(false);
|
QTEST_ASSERT(false);
|
||||||
}
|
}
|
||||||
QSpontaneKeyEvent::setSpontaneous(&me);
|
|
||||||
if (!qApp->notify(window, &me)) {
|
|
||||||
static const char *mouseActionNames[] =
|
|
||||||
{ "MousePress", "MouseRelease", "MouseClick", "MouseDoubleClick", "MouseMove", "MouseDoubleClickSequence" };
|
|
||||||
QString warning = QString::fromLatin1("Mouse event \"%1\" not accepted by receiving window");
|
|
||||||
QWARN(warning.arg(QString::fromLatin1(mouseActionNames[static_cast<int>(action)])).toLatin1().data());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#if QT_CONFIG(wheelevent)
|
#if QT_CONFIG(wheelevent)
|
||||||
|
|
|
@ -72,12 +72,14 @@ Rectangle{
|
||||||
height:100
|
height:100
|
||||||
color: "red"
|
color: "red"
|
||||||
property bool updatePositionWhileDragging: false
|
property bool updatePositionWhileDragging: false
|
||||||
property var posX: 0
|
property double posX: 0
|
||||||
property var posY: 0
|
property double posY: 0
|
||||||
|
|
||||||
function reset() {
|
function reset(mouseX, mouseY) {
|
||||||
fakeHandle.x = 0
|
posX = mouseX;
|
||||||
fakeHandle.y = 0
|
posY = mouseY;
|
||||||
|
fakeHandle.x = mouseX;
|
||||||
|
fakeHandle.y = mouseY;
|
||||||
spyX.clear()
|
spyX.clear()
|
||||||
spyY.clear()
|
spyY.clear()
|
||||||
}
|
}
|
||||||
|
@ -116,52 +118,56 @@ Rectangle{
|
||||||
TestCase {
|
TestCase {
|
||||||
name:"mouserelease"
|
name:"mouserelease"
|
||||||
when:windowShown
|
when:windowShown
|
||||||
function test_mouseDrag() {
|
|
||||||
mouseDrag(container, 10, 10, util.dragThreshold * 2, util.dragThreshold * 3);
|
function test_mouseDrag_data() {
|
||||||
compare(container.x, util.dragThreshold - 1);
|
return [
|
||||||
compare(container.y, util.dragThreshold * 2 - 1);
|
{ tag: "short", dx: 20, dy: 30 },
|
||||||
|
{ tag: "long", dx: 70, dy: 60 },
|
||||||
|
{ tag: "longshort", dx: 70, dy: 20 },
|
||||||
|
{ tag: "shortlong", dx: 20, dy: 70 }
|
||||||
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
function test_doSomethingWhileDragging() {
|
function test_mouseDrag(data) {
|
||||||
container2.updatePositionWhileDragging = false
|
container.x = 0;
|
||||||
// dx and dy are superior to 3 times util.dragThreshold.
|
container.y = 0;
|
||||||
// but here the dragging does not update posX and posY
|
mouseDrag(container, 10, 10, data.dx, data.dy);
|
||||||
// posX and posY are only updated on mouseRelease
|
compare(container.x, data.dx - util.dragThreshold - 1);
|
||||||
container2.reset()
|
compare(container.y, data.dy - util.dragThreshold - 1);
|
||||||
mouseDrag(container2, container2.x + 10, container2.y + 10, 10*util.dragThreshold, 10*util.dragThreshold);
|
}
|
||||||
compare(spyX.count, 1)
|
|
||||||
compare(spyY.count, 1)
|
|
||||||
|
|
||||||
container2.updatePositionWhileDragging = true
|
function test_doSomethingInsteadOfDragging_data() {
|
||||||
// dx and dy are superior to 3 times util.dragThreshold.
|
return [
|
||||||
// 3 intermediate mouseMove when dragging
|
{ tag: "short", updatePositionWhileDragging: false, dx: 2*util.dragThreshold, dy: 2*util.dragThreshold },
|
||||||
container2.reset()
|
{ tag: "long", updatePositionWhileDragging: false, dx: 10*util.dragThreshold, dy: 10*util.dragThreshold },
|
||||||
mouseDrag(container2, container2.x + 10, container2.y + 10, 10*util.dragThreshold, 10*util.dragThreshold);
|
{ tag: "nothing_short", updatePositionWhileDragging: false, dx: 0, dy: 2*util.dragThreshold },
|
||||||
compare(spyX.count, 3)
|
{ tag: "long_nothing", updatePositionWhileDragging: false, dx: 10*util.dragThreshold, dy: 0 },
|
||||||
compare(spyY.count, 3)
|
{ tag: "short_update", updatePositionWhileDragging: true, dx: 2*util.dragThreshold, dy: 2*util.dragThreshold },
|
||||||
|
{ tag: "long_update", updatePositionWhileDragging: true, dx: 10*util.dragThreshold, dy: 10*util.dragThreshold },
|
||||||
|
{ tag: "nothing_short_up", updatePositionWhileDragging: true, dx: 0, dy: 2*util.dragThreshold },
|
||||||
|
{ tag: "long_nothing_up", updatePositionWhileDragging: true, dx: 10*util.dragThreshold, dy: 0 },
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
// dx and dy are inferior to 3 times util.dragThreshold.
|
function test_doSomethingInsteadOfDragging(data) {
|
||||||
// No intermediate mouseMove when dragging, only one mouseMove
|
var expectedSpyCountX;
|
||||||
container2.reset()
|
var expectedSpyCountY;
|
||||||
mouseDrag(container2, container2.x + 10, container2.y + 10, 2*util.dragThreshold, 2*util.dragThreshold);
|
|
||||||
compare(spyX.count, 1)
|
|
||||||
compare(spyY.count, 1)
|
|
||||||
|
|
||||||
// dx is superior to 3 times util.dragThreshold.
|
if (!data.updatePositionWhileDragging) {
|
||||||
// 3 intermediate mouseMove when dragging on x axis
|
expectedSpyCountX = data.dx > util.dragThreshold ? 1 : 0;
|
||||||
// no move on the y axis
|
expectedSpyCountY = data.dy > util.dragThreshold ? 1 : 0;
|
||||||
container2.reset()
|
} else {
|
||||||
mouseDrag(container2, container2.x + 10, container2.y + 10, 10*util.dragThreshold, 0);
|
expectedSpyCountX = data.dx > util.dragThreshold * 3 ? 3 :
|
||||||
compare(spyX.count, 3)
|
(data.dx > util.dragThreshold ? 1 : 0);
|
||||||
compare(spyY.count, 0)
|
expectedSpyCountY = data.dy > util.dragThreshold * 3 ? 3 :
|
||||||
|
(data.dy > util.dragThreshold ? 1 : 0);
|
||||||
|
}
|
||||||
|
|
||||||
// dy is inferior to 3 times util.dragThreshold.
|
container2.updatePositionWhileDragging = data.updatePositionWhileDragging;
|
||||||
// No intermediate mouseMove when dragging, only one mouseMove on y axis
|
container2.reset(container2.x + 10, container2.y + 10);
|
||||||
// no move on the x axis
|
mouseDrag(container2, container2.x + 10, container2.y + 10, data.dx, data.dy);
|
||||||
container2.reset()
|
compare(spyX.count, expectedSpyCountX)
|
||||||
mouseDrag(container2, container2.x + 10, container2.y + 10, 0, 2*util.dragThreshold);
|
compare(spyY.count, expectedSpyCountY)
|
||||||
compare(spyX.count, 0)
|
|
||||||
compare(spyY.count, 1)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -59,9 +59,10 @@ Rectangle {
|
||||||
|
|
||||||
Window {
|
Window {
|
||||||
id: sub
|
id: sub
|
||||||
visible: true
|
visible: false
|
||||||
property bool clicked: false
|
property bool clicked: false
|
||||||
MouseArea {
|
MouseArea {
|
||||||
|
id: subMouseArea
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
onClicked: sub.clicked = true
|
onClicked: sub.clicked = true
|
||||||
}
|
}
|
||||||
|
@ -101,6 +102,8 @@ Rectangle {
|
||||||
}
|
}
|
||||||
|
|
||||||
function test_mouse_click_subwindow() {
|
function test_mouse_click_subwindow() {
|
||||||
|
sub.visible = true;
|
||||||
|
waitForRendering(subMouseArea); // Since we don't have a waitForWindowShown
|
||||||
mouseClick(sub)
|
mouseClick(sub)
|
||||||
tryCompare(sub, "clicked", true, 10000)
|
tryCompare(sub, "clicked", true, 10000)
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
TEMPLATE = subdirs
|
TEMPLATE = subdirs
|
||||||
SUBDIRS = \
|
SUBDIRS = \
|
||||||
signalspy
|
signalspy \
|
||||||
|
testevent
|
||||||
|
|
|
@ -0,0 +1,8 @@
|
||||||
|
CONFIG += testcase
|
||||||
|
TARGET = tst_testevent
|
||||||
|
macos:CONFIG -= app_bundle
|
||||||
|
|
||||||
|
SOURCES += tst_testevent.cpp
|
||||||
|
QT += quick testlib qmltest-private
|
||||||
|
|
||||||
|
include (../../shared/util.pri)
|
|
@ -0,0 +1,81 @@
|
||||||
|
/****************************************************************************
|
||||||
|
**
|
||||||
|
** Copyright (C) 2016 The Qt Company Ltd.
|
||||||
|
** Contact: https://www.qt.io/licensing/
|
||||||
|
**
|
||||||
|
** This file is part of the test suite of the Qt Toolkit.
|
||||||
|
**
|
||||||
|
** $QT_BEGIN_LICENSE:GPL-EXCEPT$
|
||||||
|
** 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 The Qt Company. For licensing terms
|
||||||
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
||||||
|
** information use the contact form at https://www.qt.io/contact-us.
|
||||||
|
**
|
||||||
|
** GNU General Public License Usage
|
||||||
|
** Alternatively, this file may be used under the terms of the GNU
|
||||||
|
** General Public License version 3 as published by the Free Software
|
||||||
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
||||||
|
** included in the packaging of this file. Please review the following
|
||||||
|
** information to ensure the GNU General Public License requirements will
|
||||||
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
||||||
|
**
|
||||||
|
** $QT_END_LICENSE$
|
||||||
|
**
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
#include <qtest.h>
|
||||||
|
|
||||||
|
#include <QtQuick/qquickitem.h>
|
||||||
|
#include <QtQuick/qquickwindow.h>
|
||||||
|
#include "QtQuickTest/private/quicktestevent_p.h"
|
||||||
|
|
||||||
|
class tst_testevent : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
|
||||||
|
private slots:
|
||||||
|
void moveSetsQuickItemIsUnderMouse();
|
||||||
|
};
|
||||||
|
|
||||||
|
void tst_testevent::moveSetsQuickItemIsUnderMouse()
|
||||||
|
{
|
||||||
|
QQuickWindow w;
|
||||||
|
w.resize(400, 400);
|
||||||
|
w.show();
|
||||||
|
w.requestActivate();
|
||||||
|
QTest::qWaitForWindowActive(&w);
|
||||||
|
|
||||||
|
QTRY_COMPARE(QGuiApplication::focusWindow(), &w);
|
||||||
|
|
||||||
|
QQuickItem *item = new QQuickItem(w.contentItem());
|
||||||
|
item->setX(0);
|
||||||
|
item->setY(0);
|
||||||
|
item->setWidth(100);
|
||||||
|
item->setHeight(100);
|
||||||
|
|
||||||
|
QQuickItem *item2 = new QQuickItem(w.contentItem());
|
||||||
|
item2->setX(100);
|
||||||
|
item2->setY(100);
|
||||||
|
item2->setWidth(100);
|
||||||
|
item2->setHeight(100);
|
||||||
|
|
||||||
|
QuickTestEvent te;
|
||||||
|
te.mouseMove(w.contentItem(), 10, 10, /*delay*/ -1, Qt::NoButton);
|
||||||
|
|
||||||
|
QVERIFY(item->isUnderMouse());
|
||||||
|
QVERIFY(!item2->isUnderMouse());
|
||||||
|
|
||||||
|
te.mouseMove(w.contentItem(), 110, 110, /*delay*/ -1, Qt::NoButton);
|
||||||
|
|
||||||
|
QVERIFY(!item->isUnderMouse());
|
||||||
|
QVERIFY(item2->isUnderMouse());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
QTEST_MAIN(tst_testevent)
|
||||||
|
|
||||||
|
#include "tst_testevent.moc"
|
Loading…
Reference in New Issue