/**************************************************************************** ** ** Copyright (C) 2012 Nokia Corporation and/or its subsidiary(-ies). ** All rights reserved. ** Contact: Nokia Corporation (qt-info@nokia.com) ** ** This file is part of the test suite of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** GNU Lesser General Public License Usage ** This file may be used under the terms of the GNU Lesser General Public ** License version 2.1 as published by the Free Software Foundation and ** appearing in the file LICENSE.LGPL included in the packaging of this ** file. Please review the following information to ensure the GNU Lesser ** General Public License version 2.1 requirements will be met: ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Nokia gives you certain additional ** rights. These rights are described in the Nokia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU General ** Public License version 3.0 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 3.0 requirements will be met: ** http://www.gnu.org/copyleft/gpl.html. ** ** Other Usage ** Alternatively, this file may be used in accordance with the terms and ** conditions contained in a signed written agreement between you and Nokia. ** ** ** ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include #include #include #include #include #include #include #include #include class tst_qdeclarativefocusscope : public QObject { Q_OBJECT public: tst_qdeclarativefocusscope() {} template T *findItem(QGraphicsObject *parent, const QString &id); private slots: void basic(); void nested(); void noFocus(); void textEdit(); void forceFocus(); void noParentFocus(); void signalEmission(); void qtBug13380(); void forceActiveFocus(); }; /* Find an item with the specified id. */ template T *tst_qdeclarativefocusscope::findItem(QGraphicsObject *parent, const QString &objectName) { const QMetaObject &mo = T::staticMetaObject; QList children = parent->childItems(); for (int i = 0; i < children.count(); ++i) { QDeclarativeItem *item = qobject_cast(children.at(i)->toGraphicsObject()); if (item) { if (mo.cast(item) && (objectName.isEmpty() || item->objectName() == objectName)) { return static_cast(item); } item = findItem(item, objectName); if (item) return static_cast(item); } } return 0; } void tst_qdeclarativefocusscope::basic() { QDeclarativeView *view = new QDeclarativeView; view->setSource(QUrl::fromLocalFile(SRCDIR "/data/test.qml")); QDeclarative1FocusScope *item0 = findItem(view->rootObject(), QLatin1String("item0")); QDeclarative1Rectangle *item1 = findItem(view->rootObject(), QLatin1String("item1")); QDeclarative1Rectangle *item2 = findItem(view->rootObject(), QLatin1String("item2")); QDeclarative1Rectangle *item3 = findItem(view->rootObject(), QLatin1String("item3")); QVERIFY(item0 != 0); QVERIFY(item1 != 0); QVERIFY(item2 != 0); QVERIFY(item3 != 0); view->show(); qApp->setActiveWindow(view); QTest::qWaitForWindowShown(view); QTRY_VERIFY(view->hasFocus()); QVERIFY(view->scene()->hasFocus()); QVERIFY(item0->hasActiveFocus() == true); QVERIFY(item1->hasActiveFocus() == true); QVERIFY(item2->hasActiveFocus() == false); QVERIFY(item3->hasActiveFocus() == false); QTest::keyClick(view, Qt::Key_Right); QVERIFY(item0->hasActiveFocus() == true); QVERIFY(item1->hasActiveFocus() == false); QVERIFY(item2->hasActiveFocus() == true); QVERIFY(item3->hasActiveFocus() == false); QTest::keyClick(view, Qt::Key_Down); QVERIFY(item0->hasActiveFocus() == false); QVERIFY(item1->hasActiveFocus() == false); QVERIFY(item2->hasActiveFocus() == false); QVERIFY(item3->hasActiveFocus() == true); delete view; } void tst_qdeclarativefocusscope::nested() { QDeclarativeView *view = new QDeclarativeView; view->setSource(QUrl::fromLocalFile(SRCDIR "/data/test2.qml")); QDeclarative1FocusScope *item1 = findItem(view->rootObject(), QLatin1String("item1")); QDeclarative1FocusScope *item2 = findItem(view->rootObject(), QLatin1String("item2")); QDeclarative1FocusScope *item3 = findItem(view->rootObject(), QLatin1String("item3")); QDeclarative1FocusScope *item4 = findItem(view->rootObject(), QLatin1String("item4")); QDeclarative1FocusScope *item5 = findItem(view->rootObject(), QLatin1String("item5")); QVERIFY(item1 != 0); QVERIFY(item2 != 0); QVERIFY(item3 != 0); QVERIFY(item4 != 0); QVERIFY(item5 != 0); view->show(); qApp->setActiveWindow(view); QTest::qWaitForWindowShown(view); QTRY_VERIFY(view->hasFocus()); QVERIFY(view->scene()->hasFocus()); QVERIFY(item1->hasActiveFocus() == true); QVERIFY(item2->hasActiveFocus() == true); QVERIFY(item3->hasActiveFocus() == true); QVERIFY(item4->hasActiveFocus() == true); QVERIFY(item5->hasActiveFocus() == true); delete view; } void tst_qdeclarativefocusscope::noFocus() { QDeclarativeView *view = new QDeclarativeView; view->setSource(QUrl::fromLocalFile(SRCDIR "/data/test4.qml")); QDeclarative1Rectangle *item0 = findItem(view->rootObject(), QLatin1String("item0")); QDeclarative1Rectangle *item1 = findItem(view->rootObject(), QLatin1String("item1")); QDeclarative1Rectangle *item2 = findItem(view->rootObject(), QLatin1String("item2")); QDeclarative1Rectangle *item3 = findItem(view->rootObject(), QLatin1String("item3")); QVERIFY(item0 != 0); QVERIFY(item1 != 0); QVERIFY(item2 != 0); QVERIFY(item3 != 0); view->show(); qApp->setActiveWindow(view); QTest::qWaitForWindowShown(view); QTRY_VERIFY(view->hasFocus()); QVERIFY(view->scene()->hasFocus()); QVERIFY(item0->hasActiveFocus() == false); QVERIFY(item1->hasActiveFocus() == false); QVERIFY(item2->hasActiveFocus() == false); QVERIFY(item3->hasActiveFocus() == false); QTest::keyClick(view, Qt::Key_Right); QVERIFY(item0->hasActiveFocus() == false); QVERIFY(item1->hasActiveFocus() == false); QVERIFY(item2->hasActiveFocus() == false); QVERIFY(item3->hasActiveFocus() == false); QTest::keyClick(view, Qt::Key_Down); QVERIFY(item0->hasActiveFocus() == false); QVERIFY(item1->hasActiveFocus() == false); QVERIFY(item2->hasActiveFocus() == false); QVERIFY(item3->hasActiveFocus() == false); delete view; } void tst_qdeclarativefocusscope::textEdit() { QDeclarativeView *view = new QDeclarativeView; view->setSource(QUrl::fromLocalFile(SRCDIR "/data/test5.qml")); QDeclarative1FocusScope *item0 = findItem(view->rootObject(), QLatin1String("item0")); QDeclarative1TextEdit *item1 = findItem(view->rootObject(), QLatin1String("item1")); QDeclarative1Rectangle *item2 = findItem(view->rootObject(), QLatin1String("item2")); QDeclarative1TextEdit *item3 = findItem(view->rootObject(), QLatin1String("item3")); QVERIFY(item0 != 0); QVERIFY(item1 != 0); QVERIFY(item2 != 0); QVERIFY(item3 != 0); view->show(); qApp->setActiveWindow(view); QTest::qWaitForWindowShown(view); QTRY_VERIFY(view->hasFocus()); QVERIFY(view->scene()->hasFocus()); QVERIFY(item0->hasActiveFocus() == true); QVERIFY(item1->hasActiveFocus() == true); QVERIFY(item2->hasActiveFocus() == false); QVERIFY(item3->hasActiveFocus() == false); QTest::keyClick(view, Qt::Key_Right); QVERIFY(item0->hasActiveFocus() == true); QVERIFY(item1->hasActiveFocus() == true); QVERIFY(item2->hasActiveFocus() == false); QVERIFY(item3->hasActiveFocus() == false); QTest::keyClick(view, Qt::Key_Right); QTest::keyClick(view, Qt::Key_Right); QTest::keyClick(view, Qt::Key_Right); QTest::keyClick(view, Qt::Key_Right); QTest::keyClick(view, Qt::Key_Right); QVERIFY(item0->hasActiveFocus() == true); QVERIFY(item1->hasActiveFocus() == false); QVERIFY(item2->hasActiveFocus() == true); QVERIFY(item3->hasActiveFocus() == false); QTest::keyClick(view, Qt::Key_Down); QVERIFY(item0->hasActiveFocus() == false); QVERIFY(item1->hasActiveFocus() == false); QVERIFY(item2->hasActiveFocus() == false); QVERIFY(item3->hasActiveFocus() == true); delete view; } void tst_qdeclarativefocusscope::forceFocus() { QDeclarativeView *view = new QDeclarativeView; view->setSource(QUrl::fromLocalFile(SRCDIR "/data/forcefocus.qml")); QDeclarative1FocusScope *item0 = findItem(view->rootObject(), QLatin1String("item0")); QDeclarative1Rectangle *item1 = findItem(view->rootObject(), QLatin1String("item1")); QDeclarative1Rectangle *item2 = findItem(view->rootObject(), QLatin1String("item2")); QDeclarative1FocusScope *item3 = findItem(view->rootObject(), QLatin1String("item3")); QDeclarative1Rectangle *item4 = findItem(view->rootObject(), QLatin1String("item4")); QDeclarative1Rectangle *item5 = findItem(view->rootObject(), QLatin1String("item5")); QVERIFY(item0 != 0); QVERIFY(item1 != 0); QVERIFY(item2 != 0); QVERIFY(item3 != 0); QVERIFY(item4 != 0); QVERIFY(item5 != 0); view->show(); qApp->setActiveWindow(view); QTest::qWaitForWindowShown(view); QTRY_VERIFY(view->hasFocus()); QVERIFY(view->scene()->hasFocus()); QVERIFY(item0->hasActiveFocus() == true); QVERIFY(item1->hasActiveFocus() == true); QVERIFY(item2->hasActiveFocus() == false); QVERIFY(item3->hasActiveFocus() == false); QVERIFY(item4->hasActiveFocus() == false); QVERIFY(item5->hasActiveFocus() == false); QTest::keyClick(view, Qt::Key_4); QVERIFY(item0->hasActiveFocus() == true); QVERIFY(item1->hasActiveFocus() == true); QVERIFY(item2->hasActiveFocus() == false); QVERIFY(item3->hasActiveFocus() == false); QVERIFY(item4->hasActiveFocus() == false); QVERIFY(item5->hasActiveFocus() == false); QTest::keyClick(view, Qt::Key_5); QVERIFY(item0->hasActiveFocus() == false); QVERIFY(item1->hasActiveFocus() == false); QVERIFY(item2->hasActiveFocus() == false); QVERIFY(item3->hasActiveFocus() == true); QVERIFY(item4->hasActiveFocus() == false); QVERIFY(item5->hasActiveFocus() == true); delete view; } void tst_qdeclarativefocusscope::noParentFocus() { QDeclarativeView *view = new QDeclarativeView; view->setSource(QUrl::fromLocalFile(SRCDIR "/data/chain.qml")); QVERIFY(view->rootObject()); QVERIFY(view->rootObject()->property("focus1") == false); QVERIFY(view->rootObject()->property("focus2") == false); QVERIFY(view->rootObject()->property("focus3") == true); QVERIFY(view->rootObject()->property("focus4") == true); QVERIFY(view->rootObject()->property("focus5") == true); delete view; } void tst_qdeclarativefocusscope::signalEmission() { QDeclarativeView *view = new QDeclarativeView; view->setSource(QUrl::fromLocalFile(SRCDIR "/data/signalEmission.qml")); QDeclarative1Rectangle *item1 = findItem(view->rootObject(), QLatin1String("item1")); QDeclarative1Rectangle *item2 = findItem(view->rootObject(), QLatin1String("item2")); QDeclarative1Rectangle *item3 = findItem(view->rootObject(), QLatin1String("item3")); QDeclarative1Rectangle *item4 = findItem(view->rootObject(), QLatin1String("item4")); QVERIFY(item1 != 0); QVERIFY(item2 != 0); QVERIFY(item3 != 0); QVERIFY(item4 != 0); view->show(); qApp->setActiveWindow(view); QTest::qWaitForWindowShown(view); QVariant blue(QColor("blue")); QVariant red(QColor("red")); QTRY_VERIFY(view->hasFocus()); QVERIFY(view->scene()->hasFocus()); item1->setFocus(true); QCOMPARE(item1->property("color"), red); QCOMPARE(item2->property("color"), blue); QCOMPARE(item3->property("color"), blue); QCOMPARE(item4->property("color"), blue); item2->setFocus(true); QCOMPARE(item1->property("color"), blue); QCOMPARE(item2->property("color"), red); QCOMPARE(item3->property("color"), blue); QCOMPARE(item4->property("color"), blue); item3->setFocus(true); QCOMPARE(item1->property("color"), blue); QCOMPARE(item2->property("color"), red); QCOMPARE(item3->property("color"), red); QCOMPARE(item4->property("color"), blue); item4->setFocus(true); QCOMPARE(item1->property("color"), blue); QCOMPARE(item2->property("color"), red); QCOMPARE(item3->property("color"), blue); QCOMPARE(item4->property("color"), red); item4->setFocus(false); QCOMPARE(item1->property("color"), blue); QCOMPARE(item2->property("color"), red); QCOMPARE(item3->property("color"), blue); QCOMPARE(item4->property("color"), blue); delete view; } void tst_qdeclarativefocusscope::qtBug13380() { QDeclarativeView *view = new QDeclarativeView; view->setSource(QUrl::fromLocalFile(SRCDIR "/data/qtBug13380.qml")); view->show(); QVERIFY(view->rootObject()); qApp->setActiveWindow(view); QTest::qWaitForWindowShown(view); QTRY_VERIFY(view->hasFocus()); QVERIFY(view->scene()->hasFocus()); QVERIFY(view->rootObject()->property("noFocus").toBool()); view->rootObject()->setProperty("showRect", true); QVERIFY(view->rootObject()->property("noFocus").toBool()); delete view; } void tst_qdeclarativefocusscope::forceActiveFocus() { QDeclarativeView *view = new QDeclarativeView; view->setSource(QUrl::fromLocalFile(SRCDIR "/data/forceActiveFocus.qml")); QGraphicsObject *rootObject = view->rootObject(); QVERIFY(rootObject); QDeclarativeItem *scope = findItem(rootObject, QLatin1String("scope")); QDeclarativeItem *itemA1 = findItem(rootObject, QLatin1String("item-a1")); QDeclarativeItem *scopeA = findItem(rootObject, QLatin1String("scope-a")); QDeclarativeItem *itemA2 = findItem(rootObject, QLatin1String("item-a2")); QDeclarativeItem *itemB1 = findItem(rootObject, QLatin1String("item-b1")); QDeclarativeItem *scopeB = findItem(rootObject, QLatin1String("scope-b")); QDeclarativeItem *itemB2 = findItem(rootObject, QLatin1String("item-b2")); QVERIFY(scope); QVERIFY(itemA1); QVERIFY(scopeA); QVERIFY(itemA2); QVERIFY(itemB1); QVERIFY(scopeB); QVERIFY(itemB2); QSignalSpy rootSpy(rootObject, SIGNAL(activeFocusChanged(bool))); QSignalSpy scopeSpy(scope, SIGNAL(activeFocusChanged(bool))); QSignalSpy scopeASpy(scopeA, SIGNAL(activeFocusChanged(bool))); QSignalSpy scopeBSpy(scopeB, SIGNAL(activeFocusChanged(bool))); // First, walk the focus from item-a1 down to item-a2 and back again itemA1->forceActiveFocus(); QVERIFY(itemA1->hasActiveFocus()); QCOMPARE(rootSpy.count(), 1); QCOMPARE(scopeSpy.count(), 1); scopeA->forceActiveFocus(); QVERIFY(!itemA1->hasActiveFocus()); QVERIFY(scopeA->hasActiveFocus()); QCOMPARE(scopeASpy.count(), 1); QCOMPARE(rootSpy.count(), 1); QCOMPARE(scopeSpy.count(), 1); itemA2->forceActiveFocus(); QVERIFY(!itemA1->hasActiveFocus()); QVERIFY(itemA2->hasActiveFocus()); QVERIFY(scopeA->hasActiveFocus()); QCOMPARE(scopeASpy.count(), 1); QCOMPARE(rootSpy.count(), 1); QCOMPARE(scopeSpy.count(), 1); scopeA->forceActiveFocus(); QVERIFY(!itemA1->hasActiveFocus()); QVERIFY(itemA2->hasActiveFocus()); QVERIFY(scopeA->hasActiveFocus()); QCOMPARE(scopeASpy.count(), 1); QCOMPARE(rootSpy.count(), 1); QCOMPARE(scopeSpy.count(), 1); itemA1->forceActiveFocus(); QVERIFY(itemA1->hasActiveFocus()); QVERIFY(!scopeA->hasActiveFocus()); QVERIFY(!itemA2->hasActiveFocus()); QCOMPARE(scopeASpy.count(), 2); QCOMPARE(rootSpy.count(), 1); QCOMPARE(scopeSpy.count(), 1); // Then jump back and forth between branch 'a' and 'b' itemB1->forceActiveFocus(); QVERIFY(itemB1->hasActiveFocus()); QCOMPARE(rootSpy.count(), 1); QCOMPARE(scopeSpy.count(), 1); scopeA->forceActiveFocus(); QVERIFY(!itemA1->hasActiveFocus()); QVERIFY(!itemB1->hasActiveFocus()); QVERIFY(scopeA->hasActiveFocus()); QCOMPARE(scopeASpy.count(), 3); QCOMPARE(rootSpy.count(), 1); QCOMPARE(scopeSpy.count(), 1); scopeB->forceActiveFocus(); QVERIFY(!scopeA->hasActiveFocus()); QVERIFY(!itemB1->hasActiveFocus()); QVERIFY(scopeB->hasActiveFocus()); QCOMPARE(scopeASpy.count(), 4); QCOMPARE(scopeBSpy.count(), 1); QCOMPARE(rootSpy.count(), 1); QCOMPARE(scopeSpy.count(), 1); itemA2->forceActiveFocus(); QVERIFY(!scopeB->hasActiveFocus()); QVERIFY(itemA2->hasActiveFocus()); QCOMPARE(scopeASpy.count(), 5); QCOMPARE(scopeBSpy.count(), 2); QCOMPARE(rootSpy.count(), 1); QCOMPARE(scopeSpy.count(), 1); itemB2->forceActiveFocus(); QVERIFY(!itemA2->hasActiveFocus()); QVERIFY(itemB2->hasActiveFocus()); QCOMPARE(scopeASpy.count(), 6); QCOMPARE(scopeBSpy.count(), 3); QCOMPARE(rootSpy.count(), 1); QCOMPARE(scopeSpy.count(), 1); delete view; } QTEST_MAIN(tst_qdeclarativefocusscope) #include "tst_qdeclarativefocusscope.moc"