1053 lines
35 KiB
C++
1053 lines
35 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 <qtest.h>
|
|
#include <qtesttouch.h>
|
|
#include <QtTest/QSignalSpy>
|
|
#include <QtTest/private/qtesthelpers_p.h>
|
|
#include <QtQml/qqmlcomponent.h>
|
|
#include <QtQml/qqmlcontext.h>
|
|
#include <QtQuick/qquickview.h>
|
|
#include <QtQuick/qquickitem.h>
|
|
#include <QtQuick/private/qquickitem_p.h>
|
|
#include <QtQuick/private/qquickmousearea_p.h>
|
|
#include <QtQuickTemplates2/private/qquickbutton_p.h>
|
|
#include <QtQuickTestUtils/private/qmlutils_p.h>
|
|
#include <QtGui/QWindow>
|
|
#include <QtGui/QScreen>
|
|
#include <QtGui/QImage>
|
|
#include <QtCore/QDebug>
|
|
#include <QtQml/qqmlengine.h>
|
|
|
|
#include <QtCore/QLoggingCategory>
|
|
#include <QtGui/qstylehints.h>
|
|
#include <QtWidgets/QBoxLayout>
|
|
#include <QtWidgets/QLabel>
|
|
#include <QtWidgets/private/qapplication_p.h>
|
|
|
|
#include <QtQuickWidgets/QQuickWidget>
|
|
|
|
#if QT_CONFIG(graphicsview)
|
|
# include <QtWidgets/QGraphicsView>
|
|
# include <QtWidgets/QGraphicsProxyWidget>
|
|
#endif
|
|
Q_LOGGING_CATEGORY(lcTests, "qt.quick.tests")
|
|
|
|
class MouseRecordingQQWidget : public QQuickWidget
|
|
{
|
|
public:
|
|
explicit MouseRecordingQQWidget(QWidget *parent = nullptr) : QQuickWidget(parent) {
|
|
setAttribute(Qt::WA_AcceptTouchEvents);
|
|
}
|
|
|
|
protected:
|
|
void mousePressEvent(QMouseEvent *event) override {
|
|
qCDebug(lcTests) << event;
|
|
m_mouseEvents << event->source();
|
|
QQuickWidget::mousePressEvent(event);
|
|
}
|
|
void mouseMoveEvent(QMouseEvent *event) override {
|
|
qCDebug(lcTests) << event;
|
|
m_mouseEvents << event->source();
|
|
QQuickWidget::mouseMoveEvent(event);
|
|
}
|
|
void mouseReleaseEvent(QMouseEvent *event) override {
|
|
qCDebug(lcTests) << event;
|
|
m_mouseEvents << event->source();
|
|
QQuickWidget::mouseReleaseEvent(event);
|
|
}
|
|
|
|
public:
|
|
QList<Qt::MouseEventSource> m_mouseEvents;
|
|
};
|
|
|
|
class MouseRecordingItem : public QQuickItem
|
|
{
|
|
public:
|
|
MouseRecordingItem(bool acceptTouch, bool acceptTouchPress, QQuickItem *parent = nullptr)
|
|
: QQuickItem(parent)
|
|
, m_acceptTouchPress(acceptTouchPress)
|
|
{
|
|
setSize(QSizeF(300, 300));
|
|
setAcceptedMouseButtons(Qt::LeftButton);
|
|
setAcceptTouchEvents(acceptTouch);
|
|
}
|
|
|
|
protected:
|
|
void touchEvent(QTouchEvent* event) override {
|
|
event->setAccepted(m_acceptTouchPress);
|
|
m_touchEvents << event->type();
|
|
qCDebug(lcTests) << "accepted?" << event->isAccepted() << event;
|
|
}
|
|
void mousePressEvent(QMouseEvent *event) override {
|
|
qCDebug(lcTests) << event;
|
|
m_mouseEvents << event->source();
|
|
}
|
|
void mouseMoveEvent(QMouseEvent *event) override {
|
|
qCDebug(lcTests) << event;
|
|
m_mouseEvents << event->source();
|
|
}
|
|
void mouseReleaseEvent(QMouseEvent *event) override {
|
|
qCDebug(lcTests) << event;
|
|
m_mouseEvents << event->source();
|
|
}
|
|
|
|
public:
|
|
QList<Qt::MouseEventSource> m_mouseEvents;
|
|
QList<QEvent::Type> m_touchEvents;
|
|
|
|
private:
|
|
bool m_acceptTouchPress;
|
|
};
|
|
|
|
class tst_qquickwidget : public QQmlDataTest
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
tst_qquickwidget();
|
|
|
|
private slots:
|
|
void showHide();
|
|
void reparentAfterShow();
|
|
void changeGeometry();
|
|
void resizemodeitem();
|
|
void layoutSizeChange();
|
|
void errors();
|
|
void engine();
|
|
void readback();
|
|
void renderingSignals();
|
|
void grab();
|
|
void grabBeforeShow();
|
|
void reparentToNewWindow();
|
|
void nullEngine();
|
|
void keyEvents();
|
|
void shortcuts();
|
|
void enterLeave();
|
|
void mouseEventWindowPos();
|
|
void synthMouseFromTouch_data();
|
|
void synthMouseFromTouch();
|
|
void touchTapMouseArea();
|
|
void touchTapButton();
|
|
void touchMultipleWidgets();
|
|
void tabKey();
|
|
void resizeOverlay();
|
|
void controls();
|
|
void focusOnClick();
|
|
#if QT_CONFIG(graphicsview)
|
|
void focusOnClickInProxyWidget();
|
|
#endif
|
|
void focusPreserved();
|
|
void accessibilityHandlesViewChange();
|
|
void cleanupRhi();
|
|
|
|
private:
|
|
QPointingDevice *device = QTest::createTouchDevice();
|
|
const QRect m_availableGeometry = QGuiApplication::primaryScreen()->availableGeometry();
|
|
};
|
|
|
|
tst_qquickwidget::tst_qquickwidget()
|
|
: QQmlDataTest(QT_QMLTEST_DATADIR)
|
|
{
|
|
}
|
|
|
|
void tst_qquickwidget::showHide()
|
|
{
|
|
QWidget window;
|
|
|
|
QQuickWidget *childView = new QQuickWidget(&window);
|
|
childView->setSource(testFileUrl("rectangle.qml"));
|
|
|
|
window.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&window));
|
|
QVERIFY(!childView->quickWindow()->isVisible()); // this window is always not visible see QTBUG-65761
|
|
QVERIFY(childView->quickWindow()->visibility() != QWindow::Hidden);
|
|
|
|
window.hide();
|
|
QVERIFY(!childView->quickWindow()->isVisible());
|
|
QCOMPARE(childView->quickWindow()->visibility(), QWindow::Hidden);
|
|
}
|
|
|
|
void tst_qquickwidget::reparentAfterShow()
|
|
{
|
|
QWidget window;
|
|
|
|
QQuickWidget *childView = new QQuickWidget(&window);
|
|
childView->setSource(testFileUrl("rectangle.qml"));
|
|
window.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&window));
|
|
|
|
QScopedPointer<QQuickWidget> toplevelView(new QQuickWidget);
|
|
toplevelView->setParent(&window);
|
|
toplevelView->setSource(testFileUrl("rectangle.qml"));
|
|
toplevelView->show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&window));
|
|
}
|
|
|
|
void tst_qquickwidget::changeGeometry()
|
|
{
|
|
QWidget window;
|
|
|
|
QQuickWidget *childView = new QQuickWidget(&window);
|
|
childView->setSource(testFileUrl("rectangle.qml"));
|
|
|
|
window.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&window));
|
|
|
|
childView->setGeometry(100,100,100,100);
|
|
}
|
|
|
|
void tst_qquickwidget::resizemodeitem()
|
|
{
|
|
QWidget window;
|
|
window.setGeometry(m_availableGeometry.left(), m_availableGeometry.top(), 400, 400);
|
|
|
|
QScopedPointer<QQuickWidget> view(new QQuickWidget);
|
|
view->setParent(&window);
|
|
view->setResizeMode(QQuickWidget::SizeRootObjectToView);
|
|
QCOMPARE(QSize(0,0), view->initialSize());
|
|
view->setSource(testFileUrl("resizemodeitem.qml"));
|
|
QQuickItem* item = qobject_cast<QQuickItem*>(view->rootObject());
|
|
QVERIFY(item);
|
|
window.show();
|
|
|
|
view->showNormal();
|
|
// initial size from root object
|
|
QCOMPARE(item->width(), 200.0);
|
|
QCOMPARE(item->height(), 200.0);
|
|
QCOMPARE(view->size(), QSize(200, 200));
|
|
QCOMPARE(view->size(), view->sizeHint());
|
|
QCOMPARE(view->size(), view->initialSize());
|
|
|
|
// size update from view
|
|
view->resize(QSize(80,100));
|
|
|
|
QTRY_COMPARE(item->width(), 80.0);
|
|
QCOMPARE(item->height(), 100.0);
|
|
QCOMPARE(view->size(), QSize(80, 100));
|
|
QCOMPARE(view->size(), view->sizeHint());
|
|
|
|
view->setResizeMode(QQuickWidget::SizeViewToRootObject);
|
|
|
|
// size update from view disabled
|
|
view->resize(QSize(60,80));
|
|
QCOMPARE(item->width(), 80.0);
|
|
QCOMPARE(item->height(), 100.0);
|
|
QTRY_COMPARE(view->size(), QSize(60, 80));
|
|
|
|
// size update from root object
|
|
item->setWidth(250);
|
|
item->setHeight(350);
|
|
QCOMPARE(item->width(), 250.0);
|
|
QCOMPARE(item->height(), 350.0);
|
|
QTRY_COMPARE(view->size(), QSize(250, 350));
|
|
QCOMPARE(view->size(), QSize(250, 350));
|
|
QCOMPARE(view->size(), view->sizeHint());
|
|
|
|
// reset window
|
|
window.hide();
|
|
view.reset(new QQuickWidget(&window));
|
|
view->setResizeMode(QQuickWidget::SizeViewToRootObject);
|
|
view->setSource(testFileUrl("resizemodeitem.qml"));
|
|
item = qobject_cast<QQuickItem*>(view->rootObject());
|
|
QVERIFY(item);
|
|
window.show();
|
|
|
|
view->showNormal();
|
|
|
|
// initial size for root object
|
|
QCOMPARE(item->width(), 200.0);
|
|
QCOMPARE(item->height(), 200.0);
|
|
QCOMPARE(view->size(), view->sizeHint());
|
|
QCOMPARE(view->size(), view->initialSize());
|
|
|
|
// size update from root object
|
|
item->setWidth(80);
|
|
item->setHeight(100);
|
|
QCOMPARE(item->width(), 80.0);
|
|
QCOMPARE(item->height(), 100.0);
|
|
QTRY_COMPARE(view->size(), QSize(80, 100));
|
|
QCOMPARE(view->size(), view->sizeHint());
|
|
|
|
// size update from root object disabled
|
|
view->setResizeMode(QQuickWidget::SizeRootObjectToView);
|
|
item->setWidth(60);
|
|
item->setHeight(80);
|
|
QCOMPARE(view->width(), 80);
|
|
QCOMPARE(view->height(), 100);
|
|
QCOMPARE(QSize(item->width(), item->height()), view->sizeHint());
|
|
|
|
// size update from view
|
|
view->resize(QSize(200,300));
|
|
QTRY_COMPARE(item->width(), 200.0);
|
|
QCOMPARE(item->height(), 300.0);
|
|
QCOMPARE(view->size(), QSize(200, 300));
|
|
QCOMPARE(view->size(), view->sizeHint());
|
|
|
|
window.hide();
|
|
|
|
// if we set a specific size for the view then it should keep that size
|
|
// for SizeRootObjectToView mode.
|
|
view.reset(new QQuickWidget(&window));
|
|
view->resize(300, 300);
|
|
view->setResizeMode(QQuickWidget::SizeRootObjectToView);
|
|
QCOMPARE(QSize(0,0), view->initialSize());
|
|
view->setSource(testFileUrl("resizemodeitem.qml"));
|
|
view->resize(300, 300);
|
|
item = qobject_cast<QQuickItem*>(view->rootObject());
|
|
QVERIFY(item);
|
|
window.show();
|
|
|
|
view->showNormal();
|
|
|
|
// initial size from root object
|
|
QCOMPARE(item->width(), 300.0);
|
|
QCOMPARE(item->height(), 300.0);
|
|
QTRY_COMPARE(view->size(), QSize(300, 300));
|
|
QCOMPARE(view->size(), view->sizeHint());
|
|
QCOMPARE(view->initialSize(), QSize(200, 200)); // initial object size
|
|
}
|
|
|
|
void tst_qquickwidget::layoutSizeChange()
|
|
{
|
|
QWidget window;
|
|
window.resize(400, 400);
|
|
|
|
QVBoxLayout *layout = new QVBoxLayout(&window);
|
|
layout->setContentsMargins(0,0,0,0);
|
|
layout->setSpacing(0);
|
|
QScopedPointer<QQuickWidget> view(new QQuickWidget);
|
|
layout->addWidget(view.data());
|
|
QLabel *label = new QLabel("Label");
|
|
layout->addWidget(label);
|
|
layout->addStretch(1);
|
|
|
|
|
|
view->resize(300,300);
|
|
view->setResizeMode(QQuickWidget::SizeViewToRootObject);
|
|
QCOMPARE(QSize(0,0), view->initialSize());
|
|
view->setSource(testFileUrl("rectangle.qml"));
|
|
QQuickItem* item = qobject_cast<QQuickItem*>(view->rootObject());
|
|
QVERIFY(item);
|
|
QCOMPARE(item->height(), 200.0);
|
|
window.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&window, 5000));
|
|
QTRY_COMPARE(view->height(), 200);
|
|
QTRY_COMPARE(label->y(), 200);
|
|
|
|
item->setSize(QSizeF(100,100));
|
|
QCOMPARE(item->height(), 100.0);
|
|
QTRY_COMPARE(view->height(), 100);
|
|
QTRY_COMPARE(label->y(), 100);
|
|
}
|
|
|
|
void tst_qquickwidget::errors()
|
|
{
|
|
QQuickWidget *view = new QQuickWidget;
|
|
QScopedPointer<QQuickWidget> cleanupView(view);
|
|
QVERIFY(view->errors().isEmpty()); // don't crash
|
|
|
|
QQmlTestMessageHandler messageHandler;
|
|
view->setSource(testFileUrl("error1.qml"));
|
|
QCOMPARE(view->status(), QQuickWidget::Error);
|
|
QCOMPARE(view->errors().size(), 1);
|
|
}
|
|
|
|
void tst_qquickwidget::engine()
|
|
{
|
|
QScopedPointer<QQmlEngine> engine(new QQmlEngine);
|
|
QScopedPointer<QQuickWidget> view(new QQuickWidget(engine.data(), nullptr));
|
|
QScopedPointer<QQuickWidget> view2(new QQuickWidget(view->engine(), nullptr));
|
|
|
|
QVERIFY(view->engine());
|
|
QVERIFY(view2->engine());
|
|
QCOMPARE(view->engine(), view2->engine());
|
|
}
|
|
|
|
void tst_qquickwidget::readback()
|
|
{
|
|
QScopedPointer<QQuickWidget> view(new QQuickWidget);
|
|
view->setSource(testFileUrl("rectangle.qml"));
|
|
|
|
view->show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(view.data()));
|
|
|
|
QImage img = view->grabFramebuffer();
|
|
QVERIFY(!img.isNull());
|
|
QCOMPARE(img.width(), qCeil(view->width() * view->devicePixelRatio()));
|
|
QCOMPARE(img.height(), qCeil(view->height() * view->devicePixelRatio()));
|
|
|
|
QRgb pix = img.pixel(5, 5);
|
|
QCOMPARE(pix, qRgb(255, 0, 0));
|
|
}
|
|
|
|
void tst_qquickwidget::renderingSignals()
|
|
{
|
|
QQuickWidget widget;
|
|
QQuickWindow *window = widget.quickWindow();
|
|
QVERIFY(window);
|
|
|
|
QSignalSpy beforeRenderingSpy(window, &QQuickWindow::beforeRendering);
|
|
QSignalSpy beforeSyncSpy(window, &QQuickWindow::beforeSynchronizing);
|
|
QSignalSpy afterRenderingSpy(window, &QQuickWindow::afterRendering);
|
|
|
|
QVERIFY(beforeRenderingSpy.isValid());
|
|
QVERIFY(beforeSyncSpy.isValid());
|
|
QVERIFY(afterRenderingSpy.isValid());
|
|
|
|
QCOMPARE(beforeRenderingSpy.size(), 0);
|
|
QCOMPARE(beforeSyncSpy.size(), 0);
|
|
QCOMPARE(afterRenderingSpy.size(), 0);
|
|
|
|
widget.setSource(testFileUrl("rectangle.qml"));
|
|
|
|
QCOMPARE(beforeRenderingSpy.size(), 0);
|
|
QCOMPARE(beforeSyncSpy.size(), 0);
|
|
QCOMPARE(afterRenderingSpy.size(), 0);
|
|
|
|
widget.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&widget));
|
|
|
|
QTRY_VERIFY(beforeRenderingSpy.size() > 0);
|
|
QTRY_VERIFY(beforeSyncSpy.size() > 0);
|
|
QTRY_VERIFY(afterRenderingSpy.size() > 0);
|
|
}
|
|
|
|
void tst_qquickwidget::grab()
|
|
{
|
|
QQuickWidget view;
|
|
view.setSource(testFileUrl("rectangle.qml"));
|
|
QPixmap pixmap = view.grab();
|
|
QRgb pixel = pixmap.toImage().pixel(5, 5);
|
|
QCOMPARE(pixel, qRgb(255, 0, 0));
|
|
}
|
|
|
|
// QTBUG-49929, verify that Qt Designer grabbing the contents before drag
|
|
// does not crash due to missing GL contexts or similar.
|
|
void tst_qquickwidget::grabBeforeShow()
|
|
{
|
|
QQuickWidget widget;
|
|
QVERIFY(!widget.grab().isNull());
|
|
}
|
|
|
|
void tst_qquickwidget::reparentToNewWindow()
|
|
{
|
|
#ifdef Q_OS_ANDROID
|
|
QSKIP("This test crashes on Android (see QTBUG-100173)");
|
|
#endif
|
|
QWidget window1;
|
|
QWidget window2;
|
|
|
|
QQuickWidget *qqw = new QQuickWidget(&window1);
|
|
qqw->setSource(testFileUrl("rectangle.qml"));
|
|
window1.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&window1));
|
|
window2.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&window2));
|
|
|
|
qqw->setParent(&window2);
|
|
|
|
QSignalSpy afterRenderingSpy(qqw->quickWindow(), &QQuickWindow::afterRendering);
|
|
qqw->show();
|
|
|
|
QTRY_VERIFY(afterRenderingSpy.size() > 0);
|
|
|
|
QImage img = qqw->grabFramebuffer();
|
|
|
|
QCOMPARE(img.pixel(5, 5), qRgb(255, 0, 0));
|
|
}
|
|
|
|
void tst_qquickwidget::nullEngine()
|
|
{
|
|
QQuickWidget widget;
|
|
// Default should have no errors, even with a null qml engine
|
|
QVERIFY(widget.errors().isEmpty());
|
|
QCOMPARE(widget.status(), QQuickWidget::Null);
|
|
|
|
// A QML engine should be created lazily.
|
|
QVERIFY(widget.rootContext());
|
|
QVERIFY(widget.engine());
|
|
}
|
|
|
|
class KeyHandlingWidget : public QQuickWidget
|
|
{
|
|
public:
|
|
void keyPressEvent(QKeyEvent *e) override {
|
|
if (e->key() == Qt::Key_A)
|
|
ok = true;
|
|
}
|
|
|
|
bool ok = false;
|
|
};
|
|
|
|
void tst_qquickwidget::keyEvents()
|
|
{
|
|
// A QQuickWidget should behave like a normal widget when it comes to event handling.
|
|
// Verify that key events actually reach the widget. (QTBUG-45757)
|
|
KeyHandlingWidget widget;
|
|
widget.setSource(testFileUrl("rectangle.qml"));
|
|
widget.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(widget.window()));
|
|
|
|
// Note: send the event to the QWindow, not the QWidget, in order
|
|
// to simulate the full event processing chain.
|
|
QTest::keyClick(widget.window()->windowHandle(), Qt::Key_A);
|
|
|
|
QTRY_VERIFY(widget.ok);
|
|
}
|
|
|
|
class ShortcutEventFilter : public QObject
|
|
{
|
|
public:
|
|
bool eventFilter(QObject *obj, QEvent *e) override {
|
|
if (e->type() == QEvent::ShortcutOverride)
|
|
shortcutOk = true;
|
|
|
|
return QObject::eventFilter(obj, e);
|
|
}
|
|
|
|
bool shortcutOk = false;
|
|
};
|
|
|
|
void tst_qquickwidget::shortcuts()
|
|
{
|
|
// Verify that ShortcutOverride events do not get lost. (QTBUG-60988)
|
|
KeyHandlingWidget widget;
|
|
widget.setSource(testFileUrl("rectangle.qml"));
|
|
widget.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(widget.window()));
|
|
|
|
// Send to the widget, verify that the QQuickWindow sees it.
|
|
|
|
ShortcutEventFilter filter;
|
|
widget.quickWindow()->installEventFilter(&filter);
|
|
|
|
QKeyEvent e(QEvent::ShortcutOverride, Qt::Key_A, Qt::ControlModifier);
|
|
QCoreApplication::sendEvent(&widget, &e);
|
|
|
|
QTRY_VERIFY(filter.shortcutOk);
|
|
}
|
|
|
|
void tst_qquickwidget::enterLeave()
|
|
{
|
|
#ifdef Q_OS_ANDROID
|
|
QSKIP("Android has no cursor");
|
|
#endif
|
|
QQuickWidget view;
|
|
view.setSource(testFileUrl("enterleave.qml"));
|
|
|
|
// Ensure the cursor is away from the window first
|
|
const auto outside = m_availableGeometry.topLeft() + QPoint(50, 50);
|
|
QCursor::setPos(outside);
|
|
QTRY_VERIFY(QCursor::pos() == outside);
|
|
|
|
view.move(m_availableGeometry.topLeft() + QPoint(100, 100));
|
|
view.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&view));
|
|
QQuickItem *rootItem = view.rootObject();
|
|
QVERIFY(rootItem);
|
|
const QPoint frameOffset = view.geometry().topLeft() - view.frameGeometry().topLeft();
|
|
|
|
QTRY_VERIFY(!rootItem->property("hasMouse").toBool());
|
|
// Check the enter
|
|
QCursor::setPos(view.pos() + QPoint(50, 50) + frameOffset);
|
|
QTRY_VERIFY(rootItem->property("hasMouse").toBool());
|
|
// Now check the leave
|
|
QCursor::setPos(outside);
|
|
QTRY_VERIFY(!rootItem->property("hasMouse").toBool());
|
|
}
|
|
|
|
void tst_qquickwidget::mouseEventWindowPos()
|
|
{
|
|
QWidget widget;
|
|
widget.resize(100, 100);
|
|
QQuickWidget *quick = new QQuickWidget(&widget);
|
|
quick->setSource(testFileUrl("mouse.qml"));
|
|
quick->move(50, 50);
|
|
widget.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&widget));
|
|
QQuickItem *rootItem = quick->rootObject();
|
|
QVERIFY(rootItem);
|
|
|
|
QVERIFY(!rootItem->property("wasClicked").toBool());
|
|
QVERIFY(!rootItem->property("wasDoubleClicked").toBool());
|
|
// Moving an item under the mouse cursor will trigger a mouse move event.
|
|
// The above quick->move() will trigger a mouse move event on macOS.
|
|
// Discard that in order to get a clean slate for the actual tests.
|
|
rootItem->setProperty("wasMoved", QVariant(false));
|
|
|
|
QWindow *window = widget.windowHandle();
|
|
QVERIFY(window);
|
|
|
|
QTest::mouseMove(window, QPoint(60, 60));
|
|
QTest::mouseClick(window, Qt::LeftButton, Qt::KeyboardModifiers(), QPoint(60, 60));
|
|
QTRY_VERIFY(rootItem->property("wasClicked").toBool());
|
|
QTest::mouseDClick(window, Qt::LeftButton, Qt::KeyboardModifiers(), QPoint(60, 60));
|
|
QTRY_VERIFY(rootItem->property("wasDoubleClicked").toBool());
|
|
QTest::mouseMove(window, QPoint(70, 70));
|
|
QTRY_VERIFY(rootItem->property("wasMoved").toBool());
|
|
}
|
|
|
|
void tst_qquickwidget::synthMouseFromTouch_data()
|
|
{
|
|
QTest::addColumn<bool>("synthMouse"); // AA_SynthesizeMouseForUnhandledTouchEvents
|
|
QTest::addColumn<bool>("acceptTouch"); // QQuickItem::touchEvent: setAccepted()
|
|
|
|
QTest::newRow("no synth, accept") << false << true; // suitable for touch-capable UIs
|
|
QTest::newRow("no synth, don't accept") << false << false;
|
|
QTest::newRow("synth and accept") << true << true;
|
|
QTest::newRow("synth, don't accept") << true << false; // the default
|
|
}
|
|
|
|
void tst_qquickwidget::synthMouseFromTouch()
|
|
{
|
|
QFETCH(bool, synthMouse);
|
|
QFETCH(bool, acceptTouch);
|
|
|
|
QCoreApplication::setAttribute(Qt::AA_SynthesizeMouseForUnhandledTouchEvents, synthMouse);
|
|
QWidget window;
|
|
window.setAttribute(Qt::WA_AcceptTouchEvents);
|
|
QScopedPointer<MouseRecordingQQWidget> childView(new MouseRecordingQQWidget(&window));
|
|
MouseRecordingItem *item = new MouseRecordingItem(!synthMouse, acceptTouch, nullptr);
|
|
childView->setContent(QUrl(), nullptr, item);
|
|
window.resize(300, 300);
|
|
childView->resize(300, 300);
|
|
window.show();
|
|
QVERIFY(QTest::qWaitForWindowActive(&window));
|
|
QVERIFY(!childView->quickWindow()->isVisible()); // this window is always not visible see QTBUG-65761
|
|
QVERIFY(item->isVisible());
|
|
|
|
QPoint p1 = QPoint(20, 20);
|
|
QPoint p2 = QPoint(30, 30);
|
|
QTest::touchEvent(&window, device).press(0, p1, &window);
|
|
QTest::touchEvent(&window, device).move(0, p2, &window);
|
|
QTest::touchEvent(&window, device).release(0, p2, &window);
|
|
|
|
qCDebug(lcTests) << item->m_touchEvents << item->m_mouseEvents;
|
|
QCOMPARE(item->m_touchEvents.size(), synthMouse ? 0 : (acceptTouch ? 3 : 1));
|
|
QCOMPARE(item->m_mouseEvents.size(), synthMouse ? 3 : 0);
|
|
QCOMPARE(childView->m_mouseEvents.size(), 0);
|
|
for (const auto &ev : item->m_mouseEvents)
|
|
QCOMPARE(ev, Qt::MouseEventSynthesizedByQt);
|
|
}
|
|
|
|
void tst_qquickwidget::touchTapMouseArea()
|
|
{
|
|
QWidget window;
|
|
window.resize(100, 100);
|
|
window.setObjectName("window widget");
|
|
window.setAttribute(Qt::WA_AcceptTouchEvents);
|
|
QVERIFY(QCoreApplication::testAttribute(Qt::AA_SynthesizeMouseForUnhandledTouchEvents));
|
|
QQuickWidget *quick = new QQuickWidget(&window);
|
|
quick->setSource(testFileUrl("mouse.qml"));
|
|
quick->move(50, 50);
|
|
quick->setObjectName("quick widget");
|
|
window.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&window));
|
|
QQuickItem *rootItem = quick->rootObject();
|
|
QVERIFY(rootItem);
|
|
QQuickMouseArea *ma = rootItem->findChild<QQuickMouseArea *>();
|
|
QVERIFY(ma);
|
|
|
|
QPoint p1 = QPoint(70, 70);
|
|
QTest::touchEvent(&window, device).press(0, p1, &window);
|
|
QTRY_COMPARE(ma->isPressed(), true);
|
|
QTest::touchEvent(&window, device).move(0, p1, &window);
|
|
QTest::touchEvent(&window, device).release(0, p1, &window);
|
|
QTRY_COMPARE(ma->isPressed(), false);
|
|
QVERIFY(rootItem->property("wasClicked").toBool());
|
|
}
|
|
|
|
void tst_qquickwidget::touchTapButton()
|
|
{
|
|
QWidget window;
|
|
QQuickWidget *quick = new QQuickWidget;
|
|
quick->setSource(testFileUrl("button.qml"));
|
|
|
|
QHBoxLayout hbox;
|
|
hbox.addWidget(quick);
|
|
window.setLayout(&hbox);
|
|
window.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&window));
|
|
|
|
QQuickItem *rootItem = quick->rootObject();
|
|
QVERIFY(rootItem);
|
|
QQuickButton *button = rootItem->findChild<QQuickButton *>("button");
|
|
QVERIFY(button);
|
|
|
|
const QPoint point = quick->mapTo(&window, button->mapToScene(button->boundingRect().center()).toPoint());
|
|
QTest::touchEvent(&window, device).press(0, point, &window).commit();
|
|
QTRY_VERIFY(rootItem->property("wasPressed").toBool());
|
|
QTest::touchEvent(&window, device).release(0, point, &window).commit();
|
|
QTRY_VERIFY(rootItem->property("wasReleased").toBool());
|
|
QTRY_VERIFY(rootItem->property("wasClicked").toBool());
|
|
}
|
|
|
|
void tst_qquickwidget::touchMultipleWidgets()
|
|
{
|
|
QWidget window;
|
|
QQuickWidget *leftQuick = new QQuickWidget;
|
|
leftQuick->setSource(testFileUrl("button.qml"));
|
|
QQuickWidget *rightQuick = new QQuickWidget;
|
|
rightQuick->setSource(testFileUrl("button.qml"));
|
|
|
|
QHBoxLayout hbox;
|
|
hbox.addWidget(leftQuick);
|
|
hbox.addWidget(rightQuick);
|
|
window.setLayout(&hbox);
|
|
window.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&window));
|
|
|
|
QQuickItem *leftRootItem = leftQuick->rootObject();
|
|
QQuickItem *rightRootItem = rightQuick->rootObject();
|
|
QVERIFY(leftRootItem);
|
|
QVERIFY(rightRootItem);
|
|
QQuickButton *leftButton = leftRootItem->findChild<QQuickButton *>("button");
|
|
QQuickButton *rightButton = rightRootItem->findChild<QQuickButton *>("button");
|
|
QVERIFY(leftButton);
|
|
QVERIFY(rightButton);
|
|
|
|
const QPoint leftPoint = leftQuick->mapTo(&window, leftButton->mapToScene(
|
|
leftButton->boundingRect().center()).toPoint());
|
|
const QPoint rightPoint = rightQuick->mapTo(&window, rightButton->mapToScene(
|
|
rightButton->boundingRect().center()).toPoint());
|
|
QTest::touchEvent(&window, device).press(0, leftPoint, &window).commit();
|
|
QTRY_VERIFY(leftRootItem->property("wasPressed").toBool());
|
|
QTest::touchEvent(&window, device).press(1, rightPoint, &window).commit();
|
|
QTRY_VERIFY(rightRootItem->property("wasPressed").toBool());
|
|
QTest::touchEvent(&window, device).release(1, rightPoint, &window).commit();
|
|
QTRY_VERIFY(rightRootItem->property("wasReleased").toBool());
|
|
QVERIFY(rightRootItem->property("wasClicked").toBool());
|
|
QTest::touchEvent(&window, device).release(0, leftPoint, &window).commit();
|
|
QTRY_VERIFY(leftRootItem->property("wasReleased").toBool());
|
|
QVERIFY(leftRootItem->property("wasClicked").toBool());
|
|
}
|
|
|
|
void tst_qquickwidget::tabKey()
|
|
{
|
|
if (QGuiApplication::styleHints()->tabFocusBehavior() != Qt::TabFocusAllControls)
|
|
QSKIP("This function doesn't support NOT iterating all.");
|
|
QWidget window1;
|
|
QQuickWidget *qqw = new QQuickWidget(&window1);
|
|
qqw->setSource(testFileUrl("activeFocusOnTab.qml"));
|
|
QQuickWidget *qqw2 = new QQuickWidget(&window1);
|
|
qqw2->setSource(testFileUrl("noActiveFocusOnTab.qml"));
|
|
qqw2->move(100, 0);
|
|
window1.show();
|
|
qqw->setFocus();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&window1, 5000));
|
|
QVERIFY(qqw->hasFocus());
|
|
QQuickItem *item = qobject_cast<QQuickItem *>(qqw->rootObject());
|
|
QQuickItem *topItem = item->findChild<QQuickItem *>("topRect");
|
|
QQuickItem *middleItem = item->findChild<QQuickItem *>("middleRect");
|
|
QQuickItem *bottomItem = item->findChild<QQuickItem *>("bottomRect");
|
|
topItem->forceActiveFocus();
|
|
QVERIFY(topItem->property("activeFocus").toBool());
|
|
QTest::keyClick(qqw, Qt::Key_Tab);
|
|
QTRY_VERIFY(middleItem->property("activeFocus").toBool());
|
|
QTest::keyClick(qqw, Qt::Key_Tab);
|
|
QTRY_VERIFY(bottomItem->property("activeFocus").toBool());
|
|
QTest::keyClick(qqw, Qt::Key_Backtab);
|
|
QTRY_VERIFY(middleItem->property("activeFocus").toBool());
|
|
|
|
qqw2->setFocus();
|
|
QQuickItem *item2 = qobject_cast<QQuickItem *>(qqw2->rootObject());
|
|
QQuickItem *topItem2 = item2->findChild<QQuickItem *>("topRect2");
|
|
QTRY_VERIFY(qqw2->hasFocus());
|
|
QVERIFY(topItem2->property("activeFocus").toBool());
|
|
QTest::keyClick(qqw2, Qt::Key_Tab);
|
|
QTRY_VERIFY(qqw->hasFocus());
|
|
QVERIFY(middleItem->property("activeFocus").toBool());
|
|
}
|
|
|
|
class Overlay : public QQuickItem, public QQuickItemChangeListener
|
|
{
|
|
Q_OBJECT
|
|
|
|
public:
|
|
Overlay() = default;
|
|
|
|
~Overlay()
|
|
{
|
|
QQuickItemPrivate::get(parentItem())->removeItemChangeListener(this, QQuickItemPrivate::Geometry);
|
|
}
|
|
|
|
// componentCompleted() is too early to add the listener, as parentItem()
|
|
// is still null by that stage, so we use this function instead.
|
|
void startListening()
|
|
{
|
|
QQuickItemPrivate::get(parentItem())->addItemChangeListener(this, QQuickItemPrivate::Geometry);
|
|
}
|
|
|
|
private:
|
|
void itemGeometryChanged(QQuickItem *, QQuickGeometryChange, const QRectF &/*oldGeometry*/) override
|
|
{
|
|
auto window = QQuickItemPrivate::get(this)->window;
|
|
if (!window)
|
|
return;
|
|
|
|
setSize(window->size());
|
|
}
|
|
};
|
|
|
|
// Test that an item that resizes itself based on the window size can use a
|
|
// Geometry item change listener to respond to changes in size. This is a
|
|
// simplified test to mimic a use case involving Overlay from Qt Quick Controls 2.
|
|
void tst_qquickwidget::resizeOverlay()
|
|
{
|
|
QWidget widget;
|
|
auto contentVerticalLayout = new QVBoxLayout(&widget);
|
|
contentVerticalLayout->setContentsMargins(0, 0, 0, 0);
|
|
|
|
qmlRegisterType<Overlay>("Test", 1, 0, "Overlay");
|
|
|
|
auto quickWidget = new QQuickWidget(testFileUrl("resizeOverlay.qml"), &widget);
|
|
QCOMPARE(quickWidget->status(), QQuickWidget::Ready);
|
|
quickWidget->setResizeMode(QQuickWidget::SizeRootObjectToView);
|
|
contentVerticalLayout->addWidget(quickWidget);
|
|
|
|
auto rootItem = qobject_cast<QQuickItem*>(quickWidget->rootObject());
|
|
QVERIFY(rootItem);
|
|
|
|
auto overlay = rootItem->property("overlay").value<Overlay*>();
|
|
QVERIFY(overlay);
|
|
QVERIFY(overlay->parentItem());
|
|
overlay->startListening();
|
|
|
|
widget.resize(200, 200);
|
|
QTestPrivate::androidCompatibleShow(&widget);
|
|
QCOMPARE(rootItem->width(), 200);
|
|
QCOMPARE(rootItem->height(), 200);
|
|
QCOMPARE(overlay->width(), rootItem->width());
|
|
QCOMPARE(overlay->height(), rootItem->height());
|
|
|
|
widget.resize(300, 300);
|
|
QCOMPARE(rootItem->width(), 300);
|
|
QCOMPARE(rootItem->height(), 300);
|
|
QCOMPARE(overlay->width(), rootItem->width());
|
|
QCOMPARE(overlay->height(), rootItem->height());
|
|
}
|
|
|
|
void tst_qquickwidget::controls()
|
|
{
|
|
// Smoke test for having some basic Quick Controls in a scene in a QQuickWidget.
|
|
QWidget widget;
|
|
QVBoxLayout *contentVerticalLayout = new QVBoxLayout(&widget);
|
|
QQuickWidget *quickWidget = new QQuickWidget(testFileUrl("controls.qml"), &widget);
|
|
QCOMPARE(quickWidget->status(), QQuickWidget::Ready);
|
|
quickWidget->setResizeMode(QQuickWidget::SizeRootObjectToView);
|
|
contentVerticalLayout->addWidget(quickWidget);
|
|
|
|
widget.resize(400, 400);
|
|
widget.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&widget));
|
|
|
|
QQuickItem *rootItem = qobject_cast<QQuickItem *>(quickWidget->rootObject());
|
|
QVERIFY(rootItem);
|
|
QCOMPARE(rootItem->size(), quickWidget->size());
|
|
QSize oldSize = quickWidget->size();
|
|
|
|
// Verify that QTBUG-95937 no longer occurs. (on Windows with the default
|
|
// native windows style this used to assert in debug builds)
|
|
widget.resize(300, 300);
|
|
QTRY_VERIFY(quickWidget->width() < oldSize.width());
|
|
QTRY_COMPARE(rootItem->size(), quickWidget->size());
|
|
|
|
widget.hide();
|
|
widget.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&widget));
|
|
}
|
|
|
|
void tst_qquickwidget::focusOnClick()
|
|
{
|
|
QQuickWidget quick;
|
|
quick.setSource(testFileUrl("FocusOnClick.qml"));
|
|
quick.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&quick));
|
|
QQuickItem *rootItem = quick.rootObject();
|
|
QVERIFY(rootItem);
|
|
QWindow *window = quick.windowHandle();
|
|
QVERIFY(window);
|
|
|
|
QQuickItem *text1 = rootItem->findChild<QQuickItem *>("text1");
|
|
QVERIFY(text1);
|
|
QQuickItem *text2 = rootItem->findChild<QQuickItem *>("text2");
|
|
QVERIFY(text2);
|
|
|
|
QTest::mouseClick(window, Qt::LeftButton, Qt::KeyboardModifiers(), QPoint(75, 25));
|
|
QTRY_VERIFY(text1->hasActiveFocus());
|
|
QVERIFY(!text2->hasActiveFocus());
|
|
|
|
QTest::mouseClick(window, Qt::LeftButton, Qt::KeyboardModifiers(), QPoint(75, 75));
|
|
QTRY_VERIFY(text2->hasActiveFocus());
|
|
QVERIFY(!text1->hasActiveFocus());
|
|
|
|
}
|
|
|
|
#if QT_CONFIG(graphicsview)
|
|
void tst_qquickwidget::focusOnClickInProxyWidget()
|
|
{
|
|
QGraphicsScene scene(0,0,400,400);
|
|
|
|
QGraphicsView view1(&scene);
|
|
view1.setFrameStyle(QFrame::NoFrame);
|
|
view1.resize(400,400);
|
|
QTestPrivate::androidCompatibleShow(&view1);
|
|
|
|
|
|
|
|
QQuickWidget* quick = new QQuickWidget(testFileUrl("FocusOnClick.qml"));
|
|
quick->resize(300,100);
|
|
quick->setAttribute(Qt::WA_AcceptTouchEvents);
|
|
QGraphicsProxyWidget* proxy = scene.addWidget(quick);
|
|
proxy->setAcceptTouchEvents(true);
|
|
|
|
// QTRY_VERIFY(quick->rootObject());
|
|
QQuickItem *rootItem = quick->rootObject();
|
|
QVERIFY(rootItem);
|
|
QQuickItem *text1 = rootItem->findChild<QQuickItem *>("text1");
|
|
QVERIFY(text1);
|
|
QQuickItem *text2 = rootItem->findChild<QQuickItem *>("text2");
|
|
QVERIFY(text2);
|
|
|
|
QVERIFY(QTest::qWaitForWindowExposed(&view1));
|
|
QWindow *window1 = view1.windowHandle();
|
|
QVERIFY(window1);
|
|
|
|
// Click in the QGraphicsView, outside the QuickWidget
|
|
QTest::mouseClick(window1, Qt::LeftButton, Qt::KeyboardModifiers(), QPoint(300, 300));
|
|
QTRY_VERIFY(!text1->hasActiveFocus());
|
|
QTRY_VERIFY(!text2->hasActiveFocus());
|
|
|
|
// Click on text1
|
|
QTest::mouseClick(window1, Qt::LeftButton, Qt::KeyboardModifiers(), QPoint(75, 25));
|
|
QTRY_VERIFY(text1->hasActiveFocus());
|
|
QVERIFY(!text2->hasActiveFocus());
|
|
|
|
|
|
// Now create a second view and repeat, in order to verify that we handle one QQuickItem being in multiple windows
|
|
QGraphicsView view2(&scene);
|
|
view2.resize(400,400);
|
|
QTestPrivate::androidCompatibleShow(&view2);
|
|
|
|
QVERIFY(QTest::qWaitForWindowExposed(&view2));
|
|
QWindow *window2 = view2.windowHandle();
|
|
QVERIFY(window2);
|
|
|
|
QTest::mouseClick(window2, Qt::LeftButton, Qt::KeyboardModifiers(), QPoint(300, 300));
|
|
QTRY_VERIFY(!text1->hasActiveFocus());
|
|
QTRY_VERIFY(!text2->hasActiveFocus());
|
|
|
|
QTest::mouseClick(window2, Qt::LeftButton, Qt::KeyboardModifiers(), QPoint(75, 25));
|
|
QTRY_VERIFY(text1->hasActiveFocus());
|
|
QVERIFY(!text2->hasActiveFocus());
|
|
}
|
|
#endif
|
|
|
|
void tst_qquickwidget::focusPreserved()
|
|
{
|
|
if (!QGuiApplicationPrivate::platformIntegration()->hasCapability(QPlatformIntegration::WindowActivation))
|
|
QSKIP("Window Activation is not supported.");
|
|
if (QGuiApplication::platformName() == "android")
|
|
QSKIP("Test doesn't exit cleanly on Android and generates many warnings - QTBUG-112696");
|
|
|
|
QScopedPointer<QWidget> widget(new QWidget());
|
|
QScopedPointer<QQuickWidget> quick(new QQuickWidget());
|
|
QQuickItem *root = new QQuickItem(); // will be owned by quick after setContent
|
|
QScopedPointer<QQuickItem> content(new QQuickItem());
|
|
content->setActiveFocusOnTab(true);
|
|
content->setFocus(true);
|
|
quick->setFocusPolicy(Qt::StrongFocus);
|
|
quick->setContent(QUrl(), nullptr, root);
|
|
root->setFlag(QQuickItem::ItemHasContents);
|
|
content->setParentItem(root);
|
|
|
|
quick->setGeometry(0, 0, 200, 200);
|
|
quick->show();
|
|
quick->setFocus();
|
|
quick->activateWindow();
|
|
QVERIFY(QTest::qWaitForWindowExposed(quick.get()));
|
|
QTRY_VERIFY(quick->hasFocus());
|
|
QTRY_VERIFY(content->hasFocus());
|
|
QTRY_VERIFY(content->hasActiveFocus());
|
|
|
|
widget->show();
|
|
widget->setFocus();
|
|
widget->activateWindow();
|
|
QVERIFY(QTest::qWaitForWindowExposed(widget.get()));
|
|
QTRY_VERIFY(widget->hasFocus());
|
|
|
|
quick->setParent(widget.get());
|
|
|
|
quick->show();
|
|
quick->setFocus();
|
|
quick->activateWindow();
|
|
QTRY_VERIFY(quick->hasFocus());
|
|
QTRY_VERIFY(content->hasFocus());
|
|
QTRY_VERIFY(content->hasActiveFocus());
|
|
}
|
|
|
|
/*
|
|
Reparenting the QQuickWidget recreates the offscreen QQuickWindow.
|
|
Since the accessible interface that is cached for the QQuickWidget dispatches
|
|
all calls to the offscreen QQuickWindow, it must fix itself when the offscreen
|
|
view changes. QTBUG-108226
|
|
*/
|
|
void tst_qquickwidget::accessibilityHandlesViewChange()
|
|
{
|
|
if (QGuiApplication::platformName() == "offscreen")
|
|
QSKIP("Doesn't test anything on offscreen platform.");
|
|
if (QGuiApplication::platformName() == "android")
|
|
QSKIP("Test doesn't exit cleanly on Android and generates many warnings - QTBUG-112696");
|
|
|
|
QWidget window;
|
|
|
|
QPointer<QQuickWindow> backingScene;
|
|
|
|
QQuickWidget *childView = new QQuickWidget(&window);
|
|
childView->setSource(testFileUrl("rectangle.qml"));
|
|
window.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&window));
|
|
backingScene = childView->quickWindow();
|
|
QVERIFY(backingScene);
|
|
|
|
QAccessibleInterface *iface = QAccessible::queryAccessibleInterface(childView);
|
|
QVERIFY(iface);
|
|
(void)iface->child(0);
|
|
|
|
std::unique_ptr<QQuickWidget> quickWidget(childView);
|
|
childView->setParent(nullptr);
|
|
childView->show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(childView));
|
|
QVERIFY(!backingScene); // the old QQuickWindow should be gone now
|
|
QVERIFY(childView->quickWindow()); // long live the new QQuickWindow
|
|
|
|
iface = QAccessible::queryAccessibleInterface(childView);
|
|
QVERIFY(iface);
|
|
// this would crash if QAccessibleQuickWidget hadn't repaired itself to
|
|
// delegate calls to the new (or at least not the old, destroyed) QQuickWindow.
|
|
(void)iface->child(0);
|
|
}
|
|
|
|
class CreateDestroyWidget : public QWidget
|
|
{
|
|
public:
|
|
using QWidget::create;
|
|
using QWidget::destroy;
|
|
};
|
|
|
|
void tst_qquickwidget::cleanupRhi()
|
|
{
|
|
CreateDestroyWidget topLevel;
|
|
QQuickWidget quickWidget(&topLevel);
|
|
quickWidget.setSource(testFileUrl("rectangle.qml"));
|
|
topLevel.show();
|
|
QVERIFY(QTest::qWaitForWindowExposed(&topLevel));
|
|
|
|
topLevel.destroy();
|
|
topLevel.create();
|
|
}
|
|
|
|
QTEST_MAIN(tst_qquickwidget)
|
|
|
|
#include "tst_qquickwidget.moc"
|