blob: 958b996bc1f6045538c4b9c6aa39087686576921 [file] [log] [blame]
/****************************************************************************
**
** Copyright (C) 2017 The Qt Company Ltd.
** Contact: http://www.qt.io/licensing/
**
** This file is part of the test suite of the Qt Toolkit.
**
** $QT_BEGIN_LICENSE:LGPL3$
** 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 http://www.qt.io/terms-conditions. For further
** information use the contact form at http://www.qt.io/contact-us.
**
** GNU Lesser General Public License Usage
** Alternatively, this file may be used under the terms of the GNU Lesser
** General Public License version 3 as published by the Free Software
** Foundation and appearing in the file LICENSE.LGPLv3 included in the
** packaging of this file. Please review the following information to
** ensure the GNU Lesser General Public License version 3 requirements
** will be met: https://www.gnu.org/licenses/lgpl.html.
**
** GNU General Public License Usage
** Alternatively, this file may be used under the terms of the GNU
** General Public License version 2.0 or later as published by the Free
** Software Foundation and appearing in the file LICENSE.GPL included in
** the packaging of this file. Please review the following information to
** ensure the GNU General Public License version 2.0 requirements will be
** met: http://www.gnu.org/licenses/gpl-2.0.html.
**
** $QT_END_LICENSE$
**
****************************************************************************/
#include <qtest.h>
#include <QtTest/QSignalSpy>
#include <QtQml/qqmlengine.h>
#include <QtQml/qqmlcomponent.h>
#include <QtQml/qqmlcontext.h>
#include <QtQuick/qquickview.h>
#include <QtQuick/private/qquickitem_p.h>
#include <QtQuickTemplates2/private/qquickcontrol_p.h>
#include <QtGui/private/qguiapplication_p.h>
#include <QtGui/qstylehints.h>
#include <QtGui/qtouchdevice.h>
#include "../shared/util.h"
#include "../shared/visualtestutil.h"
using namespace QQuickVisualTestUtil;
class tst_focus : public QQmlDataTest
{
Q_OBJECT
private slots:
void initTestCase();
void navigation_data();
void navigation();
void policy_data();
void policy();
void reason_data();
void reason();
void visualFocus();
void scope_data();
void scope();
};
void tst_focus::initTestCase()
{
QQmlDataTest::initTestCase();
}
void tst_focus::navigation_data()
{
QTest::addColumn<Qt::Key>("key");
QTest::addColumn<QString>("testFile");
QTest::addColumn<Qt::TabFocusBehavior>("behavior");
QTest::addColumn<QStringList>("order");
QTest::newRow("tab-all-controls") << Qt::Key_Tab << QString("activeFocusOnTab.qml") << Qt::TabFocusAllControls << (QStringList() << "button2" << "checkbox" << "checkbox1" << "checkbox2" << "radiobutton" << "radiobutton1" << "radiobutton2" << "rangeslider.first" << "rangeslider.second" << "slider" << "spinbox" << "switch" << "tabbutton1" << "tabbutton2" << "textfield" << "toolbutton" << "textarea" << "button1");
QTest::newRow("backtab-all-controls") << Qt::Key_Backtab << QString("activeFocusOnTab.qml") << Qt::TabFocusAllControls << (QStringList() << "textarea" << "toolbutton" << "textfield" << "tabbutton2" << "tabbutton1" << "switch" << "spinbox" << "slider" << "rangeslider.second" << "rangeslider.first" << "radiobutton2" << "radiobutton1" << "radiobutton" << "checkbox2" << "checkbox1" << "checkbox" << "button2" << "button1");
QTest::newRow("tab-text-controls") << Qt::Key_Tab << QString("activeFocusOnTab.qml") << Qt::TabFocusTextControls << (QStringList() << "spinbox" << "textfield" << "textarea");
QTest::newRow("backtab-text-controls") << Qt::Key_Backtab << QString("activeFocusOnTab.qml") << Qt::TabFocusTextControls << (QStringList() << "textarea" << "textfield" << "spinbox");
QTest::newRow("key-up") << Qt::Key_Up << QString("keyNavigation.qml") << Qt::TabFocusAllControls << (QStringList() << "textarea" << "toolbutton" << "textfield" << "tabbutton2" << "tabbutton1" << "switch" << "slider" << "rangeslider.first" << "radiobutton2" << "radiobutton1" << "radiobutton" << "checkbox2" << "checkbox1" << "checkbox" << "button2" << "button1");
QTest::newRow("key-down") << Qt::Key_Down << QString("keyNavigation.qml") << Qt::TabFocusAllControls << (QStringList() << "button2" << "checkbox" << "checkbox1" << "checkbox2" << "radiobutton" << "radiobutton1" << "radiobutton2" << "rangeslider.first" << "slider" << "switch" << "tabbutton1" << "tabbutton2" << "textfield" << "toolbutton" << "textarea" << "button1");
QTest::newRow("key-left") << Qt::Key_Left << QString("keyNavigation.qml") << Qt::TabFocusAllControls << (QStringList() << "toolbutton" << "tabbutton2" << "tabbutton1" << "switch" << "spinbox" << "radiobutton2" << "radiobutton1" << "radiobutton" << "checkbox2" << "checkbox1" << "checkbox" << "button2" << "button1");
QTest::newRow("key-right") << Qt::Key_Right << QString("keyNavigation.qml") << Qt::TabFocusAllControls << (QStringList() << "button2" << "checkbox" << "checkbox1" << "checkbox2" << "radiobutton" << "radiobutton1" << "radiobutton2" << "spinbox" << "switch" << "tabbutton1" << "tabbutton2" << "toolbutton" << "button1");
}
void tst_focus::navigation()
{
QFETCH(Qt::Key, key);
QFETCH(QString, testFile);
QFETCH(Qt::TabFocusBehavior, behavior);
QFETCH(QStringList, order);
QGuiApplication::styleHints()->setTabFocusBehavior(behavior);
QQuickView view;
view.contentItem()->setObjectName("contentItem");
view.setSource(testFileUrl(testFile));
view.show();
view.requestActivate();
QVERIFY(QTest::qWaitForWindowActive(&view));
QVERIFY(QGuiApplication::focusWindow() == &view);
for (const QString &name : qAsConst(order)) {
QKeyEvent event(QEvent::KeyPress, key, Qt::NoModifier);
QGuiApplication::sendEvent(&view, &event);
QVERIFY(event.isAccepted());
QQuickItem *item = findItem<QQuickItem>(view.rootObject(), name);
QVERIFY2(item, qPrintable(name));
QVERIFY2(item->hasActiveFocus(), qPrintable(QString("expected: '%1', actual: '%2'").arg(name).arg(view.activeFocusItem() ? view.activeFocusItem()->objectName() : "null")));
}
QGuiApplication::styleHints()->setTabFocusBehavior(Qt::TabFocusBehavior(-1));
}
void tst_focus::policy_data()
{
QTest::addColumn<QString>("name");
QTest::newRow("Control") << "Control";
QTest::newRow("ComboBox") << "ComboBox";
QTest::newRow("Button") << "Button";
QTest::newRow("Slider") << "Slider";
QTest::newRow("ScrollBar") << "ScrollBar";
}
void tst_focus::policy()
{
QFETCH(QString, name);
QQmlEngine engine;
QQmlComponent component(&engine);
component.setData(QString("import QtQuick.Controls 2.1; ApplicationWindow { width: 100; height: 100; %1 { anchors.fill: parent } }").arg(name).toUtf8(), QUrl());
QScopedPointer<QQuickApplicationWindow> window(qobject_cast<QQuickApplicationWindow *>(component.create()));
QVERIFY(window);
QQuickControl *control = qobject_cast<QQuickControl *>(window->contentItem()->childItems().first());
QVERIFY(control);
QVERIFY(!control->hasActiveFocus());
QVERIFY(!control->hasVisualFocus());
window->show();
window->requestActivate();
QVERIFY(QTest::qWaitForWindowActive(window.data()));
struct TouchDeviceDeleter
{
static inline void cleanup(QTouchDevice *device)
{
QWindowSystemInterface::unregisterTouchDevice(device);
delete device;
}
};
QScopedPointer<QTouchDevice, TouchDeviceDeleter> device(new QTouchDevice);
device->setType(QTouchDevice::TouchScreen);
QWindowSystemInterface::registerTouchDevice(device.data());
control->setFocusPolicy(Qt::NoFocus);
QCOMPARE(control->focusPolicy(), Qt::NoFocus);
// Qt::TabFocus vs. QQuickItem::activeFocusOnTab
control->setActiveFocusOnTab(true);
QCOMPARE(control->focusPolicy(), Qt::TabFocus);
control->setActiveFocusOnTab(false);
QCOMPARE(control->focusPolicy(), Qt::NoFocus);
control->setFocusPolicy(Qt::TabFocus);
QCOMPARE(control->focusPolicy(), Qt::TabFocus);
QCOMPARE(control->activeFocusOnTab(), true);
// Qt::TabFocus
QGuiApplication::styleHints()->setTabFocusBehavior(Qt::TabFocusAllControls);
QTest::keyClick(window.data(), Qt::Key_Tab);
QVERIFY(control->hasActiveFocus());
QVERIFY(control->hasVisualFocus());
QGuiApplication::styleHints()->setTabFocusBehavior(Qt::TabFocusBehavior(-1));
// reset
control->setFocus(false);
QVERIFY(!control->hasActiveFocus());
// Qt::ClickFocus (mouse)
control->setFocusPolicy(Qt::NoFocus);
control->setAcceptedMouseButtons(Qt::LeftButton);
QTest::mouseClick(window.data(), Qt::LeftButton, Qt::NoModifier, QPoint(control->width() / 2, control->height() / 2));
QVERIFY(!control->hasActiveFocus());
QVERIFY(!control->hasVisualFocus());
control->setFocusPolicy(Qt::ClickFocus);
QCOMPARE(control->focusPolicy(), Qt::ClickFocus);
QTest::mouseClick(window.data(), Qt::LeftButton, Qt::NoModifier, QPoint(control->width() / 2, control->height() / 2));
QVERIFY(control->hasActiveFocus());
QVERIFY(!control->hasVisualFocus());
// reset
control->setFocus(false);
QVERIFY(!control->hasActiveFocus());
// Qt::ClickFocus (touch)
control->setFocusPolicy(Qt::NoFocus);
QTest::touchEvent(window.data(), device.data()).press(0, QPoint(control->width() / 2, control->height() / 2));
QTest::touchEvent(window.data(), device.data()).release(0, QPoint(control->width() / 2, control->height() / 2));
QVERIFY(!control->hasActiveFocus());
QVERIFY(!control->hasVisualFocus());
control->setFocusPolicy(Qt::ClickFocus);
QCOMPARE(control->focusPolicy(), Qt::ClickFocus);
QTest::touchEvent(window.data(), device.data()).press(0, QPoint(control->width() / 2, control->height() / 2));
QTest::touchEvent(window.data(), device.data()).release(0, QPoint(control->width() / 2, control->height() / 2));
QVERIFY(control->hasActiveFocus());
QVERIFY(!control->hasVisualFocus());
// reset
control->setFocus(false);
QVERIFY(!control->hasActiveFocus());
// Qt::WheelFocus
QWheelEvent wheelEvent(QPoint(control->width() / 2, control->height() / 2), 10, Qt::NoButton, Qt::NoModifier);
QGuiApplication::sendEvent(control, &wheelEvent);
QVERIFY(!control->hasActiveFocus());
QVERIFY(!control->hasVisualFocus());
control->setFocusPolicy(Qt::WheelFocus);
QCOMPARE(control->focusPolicy(), Qt::WheelFocus);
QGuiApplication::sendEvent(control, &wheelEvent);
QVERIFY(control->hasActiveFocus());
QVERIFY(!control->hasVisualFocus());
}
void tst_focus::reason_data()
{
QTest::addColumn<QString>("name");
QTest::newRow("Control") << "Control";
QTest::newRow("TextField") << "TextField";
QTest::newRow("TextArea") << "TextArea";
QTest::newRow("SpinBox") << "SpinBox";
QTest::newRow("ComboBox") << "ComboBox";
}
void tst_focus::reason()
{
QFETCH(QString, name);
QQmlEngine engine;
QQmlComponent component(&engine);
component.setData(QString("import QtQuick.Controls 2.1; ApplicationWindow { width: 100; height: 100; %1 { anchors.fill: parent } }").arg(name).toUtf8(), QUrl());
QScopedPointer<QQuickApplicationWindow> window(qobject_cast<QQuickApplicationWindow *>(component.create()));
QVERIFY(window.data());
QQuickItem *control = window->contentItem()->childItems().first();
QVERIFY(control);
window->show();
window->requestActivate();
QVERIFY(QTest::qWaitForWindowActive(window.data()));
QCOMPARE(control->property("focusReason").toInt(), int(Qt::OtherFocusReason));
control->forceActiveFocus(Qt::MouseFocusReason);
QVERIFY(control->hasActiveFocus());
QCOMPARE(control->property("focusReason").toInt(), int(Qt::MouseFocusReason));
QEXPECT_FAIL("TextArea", "TODO: TextArea::visualFocus?", Continue);
QEXPECT_FAIL("TextField", "TODO: TextField::visualFocus?", Continue);
QCOMPARE(control->property("visualFocus"), QVariant(false));
window->contentItem()->setFocus(false, Qt::TabFocusReason);
QVERIFY(!control->hasActiveFocus());
QCOMPARE(control->property("focusReason").toInt(), int(Qt::TabFocusReason));
QEXPECT_FAIL("TextArea", "", Continue);
QEXPECT_FAIL("TextField", "", Continue);
QCOMPARE(control->property("visualFocus"), QVariant(false));
control->forceActiveFocus(Qt::TabFocusReason);
QVERIFY(control->hasActiveFocus());
QCOMPARE(control->property("focusReason").toInt(), int(Qt::TabFocusReason));
QEXPECT_FAIL("TextArea", "", Continue);
QEXPECT_FAIL("TextField", "", Continue);
QCOMPARE(control->property("visualFocus"), QVariant(true));
}
void tst_focus::visualFocus()
{
QQuickView view;
view.setSource(testFileUrl("visualFocus.qml"));
view.show();
view.requestActivate();
QVERIFY(QTest::qWaitForWindowActive(&view));
QQuickItem *column = view.rootObject();
QVERIFY(column);
QCOMPARE(column->childItems().count(), 2);
QQuickControl *button = qobject_cast<QQuickControl *>(column->childItems().first());
QVERIFY(button);
QQuickItem *textfield = column->childItems().last();
QVERIFY(textfield);
button->forceActiveFocus(Qt::TabFocusReason);
QVERIFY(button->hasActiveFocus());
QVERIFY(button->hasVisualFocus());
QVERIFY(button->property("showFocus").toBool());
QTest::mouseClick(&view, Qt::LeftButton, Qt::NoModifier, QPoint(textfield->x() + textfield->width() / 2, textfield->y() + textfield->height() / 2));
QVERIFY(!button->hasActiveFocus());
QVERIFY(!button->hasVisualFocus());
QVERIFY(!button->property("showFocus").toBool());
}
void tst_focus::scope_data()
{
QTest::addColumn<QString>("name");
QTest::newRow("Frame") << "Frame";
QTest::newRow("GroupBox") << "Frame";
QTest::newRow("Page") << "Page";
QTest::newRow("Pane") << "Pane";
QTest::newRow("StackView") << "StackView";
}
void tst_focus::scope()
{
QFETCH(QString, name);
QQmlEngine engine;
QQmlComponent component(&engine);
component.setData(QString("import QtQuick 2.9; import QtQuick.Controls 2.2; ApplicationWindow { property alias child: child; width: 100; height: 100; %1 { anchors.fill: parent; Item { id: child; width: 10; height: 10 } } }").arg(name).toUtf8(), QUrl());
QScopedPointer<QQuickApplicationWindow> window(qobject_cast<QQuickApplicationWindow *>(component.create()));
QVERIFY2(window, qPrintable(component.errorString()));
QQuickControl *control = qobject_cast<QQuickControl *>(window->contentItem()->childItems().first());
QVERIFY(control);
control->setFocusPolicy(Qt::WheelFocus);
control->setAcceptedMouseButtons(Qt::LeftButton);
QQuickItem *child = window->property("child").value<QQuickItem *>();
QVERIFY(child);
window->show();
window->requestActivate();
QVERIFY(QTest::qWaitForWindowActive(window.data()));
struct TouchDeviceDeleter
{
static inline void cleanup(QTouchDevice *device)
{
QWindowSystemInterface::unregisterTouchDevice(device);
delete device;
}
};
QScopedPointer<QTouchDevice, TouchDeviceDeleter> device(new QTouchDevice);
device->setType(QTouchDevice::TouchScreen);
QWindowSystemInterface::registerTouchDevice(device.data());
child->forceActiveFocus();
QVERIFY(child->hasActiveFocus());
QVERIFY(control->hasActiveFocus());
// Qt::ClickFocus (mouse)
QTest::mouseClick(window.data(), Qt::LeftButton, Qt::NoModifier, QPoint(control->width() / 2, control->height() / 2));
QVERIFY(!child->hasActiveFocus());
QVERIFY(control->hasActiveFocus());
// reset
child->forceActiveFocus();
QVERIFY(child->hasActiveFocus());
QVERIFY(control->hasActiveFocus());
// Qt::ClickFocus (touch)
QTest::touchEvent(window.data(), device.data()).press(0, QPoint(control->width() / 2, control->height() / 2));
QTest::touchEvent(window.data(), device.data()).release(0, QPoint(control->width() / 2, control->height() / 2));
QVERIFY(!child->hasActiveFocus());
QVERIFY(control->hasActiveFocus());
// reset
child->forceActiveFocus();
QVERIFY(child->hasActiveFocus());
QVERIFY(control->hasActiveFocus());
// Qt::WheelFocus
QWheelEvent wheelEvent(QPoint(control->width() / 2, control->height() / 2), 10, Qt::NoButton, Qt::NoModifier);
QGuiApplication::sendEvent(control, &wheelEvent);
QVERIFY(!child->hasActiveFocus());
QVERIFY(control->hasActiveFocus());
}
QTEST_MAIN(tst_focus)
#include "tst_focus.moc"