// Copyright (C) 2017 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only #include #include #include #include #include #include #include #if QT_CONFIG(accessibility) #include #include #include #include #endif class tst_accessibility : public QQmlDataTest { Q_OBJECT public: tst_accessibility(); private slots: void a11y_data(); void a11y(); void override_data(); void override(); void ordering(); void actionAccessibility(); void actionAccessibilityImplicitName(); void sliderTest(); void accessibleName(); void locale(); private: QQmlEngine engine; }; #if QT_CONFIG(accessibility) static QPlatformAccessibility *platformAccessibility() { QPlatformIntegration *pfIntegration = QGuiApplicationPrivate::platformIntegration(); return pfIntegration ? pfIntegration->accessibility() : nullptr; } #endif QString adjustFileBaseName(const QString &fileBaseName) { #if !QT_CONFIG(accessibility) if (fileBaseName == QLatin1Literal("dayofweekrow") || fileBaseName == QLatin1Literal("monthgrid") || fileBaseName == QLatin1Literal("weeknumbercolumn")) return fileBaseName += QLatin1Literal("-2"); #else return fileBaseName; #endif } QQuickItem *findItem(QObject *object) { QQuickItem *item = qobject_cast(object); if (!item) { QQuickPopup *popup = qobject_cast(object); if (popup) item = popup->popupItem(); } return item; } tst_accessibility::tst_accessibility() : QQmlDataTest(QT_QMLTEST_DATADIR) { } void tst_accessibility::a11y_data() { QTest::addColumn("fileBaseName"); QTest::addColumn("role"); QTest::addColumn("text"); QTest::newRow("AbstractButton") << "abstractbutton" << QAccessible::Button << "AbstractButton"; QTest::newRow("BusyIndicator") << "busyindicator" << QAccessible::Indicator << ""; QTest::newRow("Button") << "button" << QAccessible::Button << "Button"; QTest::newRow("CheckBox") << "checkbox" << QAccessible::CheckBox << "CheckBox"; QTest::newRow("CheckDelegate") << "checkdelegate" << QAccessible::CheckBox << "CheckDelegate"; QTest::newRow("ComboBox") << "combobox" << QAccessible::ComboBox << "ComboBox"; QTest::newRow("Container") << "container" << QAccessible::NoRole << ""; QTest::newRow("Control") << "control" << QAccessible::NoRole << ""; QTest::newRow("Dial") << "dial" << QAccessible::Dial << ""; QTest::newRow("Dialog") << "dialog" << QAccessible::Dialog << "Dialog"; QTest::newRow("Drawer") << "drawer" << QAccessible::Dialog << ""; QTest::newRow("Frame") << "frame" << QAccessible::Border << ""; QTest::newRow("GroupBox") << "groupbox" << QAccessible::Grouping << "GroupBox"; QTest::newRow("ItemDelegate") << "itemdelegate" << QAccessible::ListItem << "ItemDelegate"; QTest::newRow("Label") << "label" << QAccessible::StaticText << "Label"; QTest::newRow("Menu") << "menu" << QAccessible::PopupMenu << ""; QTest::newRow("MenuItem") << "menuitem" << QAccessible::MenuItem << "MenuItem"; QTest::newRow("Page") << "page" << QAccessible::Pane << "Page"; QTest::newRow("PageIndicator") << "pageindicator" << QAccessible::Indicator << ""; QTest::newRow("Pane") << "pane" << QAccessible::Pane << ""; QTest::newRow("Popup") << "popup" << QAccessible::Dialog << ""; QTest::newRow("ProgressBar") << "progressbar" << QAccessible::ProgressBar << ""; QTest::newRow("RadioButton") << "radiobutton" << QAccessible::RadioButton << "RadioButton"; QTest::newRow("RadioDelegate") << "radiodelegate" << QAccessible::RadioButton << "RadioDelegate"; QTest::newRow("RangeSlider") << "rangeslider" << QAccessible::Slider << ""; QTest::newRow("RoundButton") << "roundbutton" << QAccessible::Button << "RoundButton"; QTest::newRow("ScrollBar") << "scrollbar" << QAccessible::ScrollBar << ""; QTest::newRow("ScrollIndicator") << "scrollindicator" << QAccessible::Indicator << ""; QTest::newRow("Slider") << "slider" << QAccessible::Slider << ""; QTest::newRow("SpinBox") << "spinbox" << QAccessible::SpinBox << ""; QTest::newRow("StackView") << "stackview" << QAccessible::LayeredPane << ""; QTest::newRow("SwipeDelegate") << "swipedelegate" << QAccessible::ListItem << "SwipeDelegate"; QTest::newRow("SwipeView") << "swipeview" << QAccessible::PageTabList << ""; QTest::newRow("Switch") << "switch" << QAccessible::CheckBox << "Switch"; QTest::newRow("SwitchDelegate") << "switchdelegate" << QAccessible::ListItem << "SwitchDelegate"; QTest::newRow("TabBar") << "tabbar" << QAccessible::PageTabList << ""; QTest::newRow("TabButton") << "tabbutton" << QAccessible::PageTab << "TabButton"; QTest::newRow("TextArea") << "textarea" << QAccessible::EditableText << ""; QTest::newRow("TextField") << "textfield" << QAccessible::EditableText << ""; QTest::newRow("ToolBar") << "toolbar" << QAccessible::ToolBar << ""; QTest::newRow("ToolButton") << "toolbutton" << QAccessible::Button << "ToolButton"; QTest::newRow("ToolTip") << "tooltip" << QAccessible::ToolTip << "ToolTip"; QTest::newRow("Tumbler") << "tumbler" << QAccessible::NoRole << ""; // TODO } void tst_accessibility::a11y() { QFETCH(QString, fileBaseName); QFETCH(QAccessible::Role, role); QFETCH(QString, text); QQmlComponent component(&engine); component.loadUrl(testFileUrl("defaults/" + adjustFileBaseName(fileBaseName) + ".qml")); QScopedPointer object(component.create()); QVERIFY2(!object.isNull(), qPrintable(component.errorString())); QQuickItem *item = findItem(object.data()); QVERIFY(item); #if QT_CONFIG(accessibility) QQuickAccessibleAttached *attached = QQuickAccessibleAttached::attachedProperties(item); if (QAccessible::isActive()) { QVERIFY(attached); } else { QVERIFY(!attached); QPlatformAccessibility *accessibility = platformAccessibility(); if (!accessibility) QSKIP("No QPlatformAccessibility available."); accessibility->setActive(true); attached = QQuickAccessibleAttached::attachedProperties(item); } QVERIFY(attached); QCOMPARE(attached->role(), role); QCOMPARE(attached->name(), text); #else Q_UNUSED(role); Q_UNUSED(text); #endif } void tst_accessibility::override_data() { QTest::addColumn("role"); QTest::newRow("AbstractButton") << QAccessible::Button; QTest::newRow("BusyIndicator") << QAccessible::Indicator; QTest::newRow("Button") << QAccessible::Button; QTest::newRow("CheckBox") << QAccessible::CheckBox; QTest::newRow("CheckDelegate") << QAccessible::CheckBox; QTest::newRow("ComboBox") << QAccessible::ComboBox; QTest::newRow("Container") << QAccessible::NoRole; QTest::newRow("Control") << QAccessible::NoRole; QTest::newRow("Dial") << QAccessible::Dial; QTest::newRow("Dialog") << QAccessible::Dialog; QTest::newRow("Drawer") << QAccessible::Dialog; QTest::newRow("Frame") << QAccessible::Border; QTest::newRow("GroupBox") << QAccessible::Grouping; QTest::newRow("ItemDelegate") << QAccessible::ListItem; QTest::newRow("Label") << QAccessible::StaticText; QTest::newRow("Menu") << QAccessible::PopupMenu; QTest::newRow("MenuItem") << QAccessible::MenuItem; QTest::newRow("Page") << QAccessible::Pane; QTest::newRow("PageIndicator") << QAccessible::Indicator; QTest::newRow("Pane") << QAccessible::Pane; QTest::newRow("Popup") << QAccessible::Dialog; QTest::newRow("ProgressBar") << QAccessible::ProgressBar; QTest::newRow("RadioButton") << QAccessible::RadioButton; QTest::newRow("RadioDelegate") << QAccessible::RadioButton; QTest::newRow("RangeSlider") << QAccessible::Slider; QTest::newRow("RoundButton") << QAccessible::Button; QTest::newRow("ScrollBar") << QAccessible::ScrollBar; QTest::newRow("ScrollIndicator") << QAccessible::Indicator; QTest::newRow("Slider") << QAccessible::Slider; QTest::newRow("SpinBox") << QAccessible::SpinBox; QTest::newRow("StackView") << QAccessible::LayeredPane; QTest::newRow("SwipeDelegate") << QAccessible::ListItem; QTest::newRow("SwipeView") << QAccessible::PageTabList; QTest::newRow("Switch") << QAccessible::CheckBox; QTest::newRow("SwitchDelegate") << QAccessible::ListItem; QTest::newRow("TabBar") << QAccessible::PageTabList; QTest::newRow("TabButton") << QAccessible::PageTab; QTest::newRow("TextArea") << QAccessible::EditableText; QTest::newRow("TextField") << QAccessible::EditableText; QTest::newRow("ToolBar") << QAccessible::ToolBar; QTest::newRow("ToolButton") << QAccessible::Button; QTest::newRow("ToolTip") << QAccessible::ToolTip; QTest::newRow("Tumbler") << QAccessible::NoRole; } void tst_accessibility::override() { QFETCH(QAccessible::Role, role); const QString name = QTest::currentDataTag(); const QString fileBaseName = name.toLower(); QQmlComponent component(&engine); component.loadUrl(testFileUrl("override/" + adjustFileBaseName(fileBaseName) + ".qml")); QScopedPointer object(component.create()); QVERIFY2(!object.isNull(), qPrintable(component.errorString())); QQuickItem *item = findItem(object.data()); QVERIFY(item); #if QT_CONFIG(accessibility) QQuickAccessibleAttached *attached = QQuickAccessibleAttached::attachedProperties(item); if (QAccessible::isActive()) { QVERIFY(attached); } else { QPlatformAccessibility *accessibility = platformAccessibility(); if (!accessibility) QSKIP("No QPlatformAccessibility available."); accessibility->setActive(true); if (!attached) attached = QQuickAccessibleAttached::attachedProperties(item); } QVERIFY(attached); QCOMPARE(attached->role(), role); QCOMPARE(attached->name(), name + "Override"); #else Q_UNUSED(role); Q_UNUSED(text); #endif } template void a11yDescendants(QAccessibleInterface *iface, Predicate pred) { for (int i = 0; i < iface->childCount(); ++i) { if (QAccessibleInterface *child = iface->child(i)) { pred(child); a11yDescendants(child, pred); } } } void tst_accessibility::ordering() { QQmlComponent component(&engine); component.loadUrl(testFileUrl("ordering/page.qml")); QScopedPointer object(component.create()); QVERIFY2(!object.isNull(), qPrintable(component.errorString())); #if QT_CONFIG(accessibility) QQuickItem *item = findItem(object.data()); QVERIFY(item); QAccessibleInterface *iface = QAccessible::queryAccessibleInterface(item); QVERIFY(iface); QStringList strings; a11yDescendants(iface, [&](QAccessibleInterface *iface) {strings << iface->text(QAccessible::Name);}); QCOMPARE(strings.join(QLatin1String(", ")), "Header, Content item 1, Content item 2, Footer"); #endif } void tst_accessibility::actionAccessibility() { #if QT_CONFIG(accessibility) if (!QAccessible::isActive()) { QPlatformAccessibility *accessibility = platformAccessibility(); if (!accessibility) QSKIP("No QPlatformAccessibility available."); accessibility->setActive(true); } QQmlComponent component(&engine); component.loadUrl(testFileUrl("actionAccessibility/button.qml")); QScopedPointer object(component.create()); QVERIFY2(!object.isNull(), qPrintable(component.errorString())); QQuickItem *item = qobject_cast(object.data()); QVERIFY(item); const QString description = "Show peaches some love"; QCOMPARE(item->property("text"), description); QAccessibleInterface *iface = QAccessible::queryAccessibleInterface(item); QVERIFY(iface); QCOMPARE(iface->text(QAccessible::Name), "Peach"); QCOMPARE(iface->text(QAccessible::Description), description); QVERIFY(iface->actionInterface()->actionNames().contains(QAccessibleActionInterface::pressAction())); iface->actionInterface()->doAction(QAccessibleActionInterface::pressAction()); QCOMPARE(item->property("pressCount").toInt(), 1); #endif } void tst_accessibility::actionAccessibilityImplicitName() { #if QT_CONFIG(accessibility) if (!QAccessible::isActive()) { QPlatformAccessibility *accessibility = platformAccessibility(); if (!accessibility) QSKIP("No QPlatformAccessibility available."); accessibility->setActive(true); } QQmlComponent component(&engine); component.loadUrl(testFileUrl("actionAccessibility/button2.qml")); QScopedPointer object(component.create()); QVERIFY2(!object.isNull(), qPrintable(component.errorString())); QQuickItem *item = qobject_cast(object.data()); QVERIFY(item); const QString description = "Show pears some love"; QCOMPARE(item->property("text"), "Pears"); QAccessibleInterface *iface = QAccessible::queryAccessibleInterface(item); QVERIFY(iface); QCOMPARE(iface->text(QAccessible::Name), "Pears"); // We get the action.text implicitly QCOMPARE(iface->text(QAccessible::Description), description); #endif } void tst_accessibility::sliderTest() { #if QT_CONFIG(accessibility) if (!QAccessible::isActive()) { QPlatformAccessibility *accessibility = platformAccessibility(); if (!accessibility) QSKIP("No QPlatformAccessibility available."); accessibility->setActive(true); } QQmlComponent component(&engine); component.loadUrl(testFileUrl("item.qml")); QScopedPointer object(component.create()); QVERIFY2(!object.isNull(), qPrintable(component.errorString())); auto root = QAccessible::queryAccessibleInterface(object.get()); QVERIFY(root); QCOMPARE(root->childCount(), 4); for (int childIndex = 0; childIndex < 4; ++childIndex) { auto item = root->child(childIndex); auto actionIface = item->actionInterface(); QVERIFY(actionIface); auto valueIface = item->valueInterface(); QVERIFY(valueIface); QVERIFY(actionIface->actionNames().contains(QAccessibleActionInterface::increaseAction())); QVERIFY(actionIface->actionNames().contains(QAccessibleActionInterface::decreaseAction())); QCOMPARE(valueIface->currentValue(), 25); QCOMPARE(valueIface->minimumValue(), 0); QCOMPARE(valueIface->maximumValue(), 100); valueIface->setCurrentValue(30); QCOMPARE(valueIface->currentValue(), 30); const auto stepSize = valueIface->minimumStepSize(); actionIface->doAction(QAccessibleActionInterface::increaseAction()); QCOMPARE(valueIface->currentValue(), 30 + stepSize.toDouble()); actionIface->doAction(QAccessibleActionInterface::decreaseAction()); QCOMPARE(valueIface->currentValue(), 30); QCOMPARE(valueIface->minimumValue(), 0); QCOMPARE(valueIface->maximumValue(), 100); } #endif } void tst_accessibility::accessibleName() { #if QT_CONFIG(accessibility) if (!QAccessible::isActive()) { QPlatformAccessibility *accessibility = platformAccessibility(); if (!accessibility) QSKIP("No QPlatformAccessibility available."); accessibility->setActive(true); } QQmlComponent component(&engine); // verify that accessible name matches the button text if none was set explicitly component.loadUrl(testFileUrl("accessibleName/button.qml")); QScopedPointer object(component.create()); QVERIFY2(!object.isNull(), qPrintable(component.errorString())); QAccessibleInterface *buttonAcc = QAccessible::queryAccessibleInterface(object.get()); QVERIFY(buttonAcc); QCOMPARE(buttonAcc->text(QAccessible::Name), "Hello world"); // verify that ampersand ('&') for mnemonic and to escape literal ampersand in button // text are not contained in accessible name component.loadUrl(testFileUrl("accessibleName/button2.qml")); QScopedPointer object2(component.create()); QVERIFY2(!object.isNull(), qPrintable(component.errorString())); QAccessibleInterface *button2Acc = QAccessible::queryAccessibleInterface(object2.get()); QVERIFY(button2Acc); QCOMPARE(button2Acc->text(QAccessible::Name), "This & that"); // verify that explicitly set accesible name is used component.loadUrl(testFileUrl("accessibleName/button3.qml")); QScopedPointer object3(component.create()); QVERIFY2(!object.isNull(), qPrintable(component.errorString())); QAccessibleInterface *button3Acc = QAccessible::queryAccessibleInterface(object3.get()); QVERIFY(button3Acc); QCOMPARE(button3Acc->text(QAccessible::Name), "Explicitly set accessible name"); #endif } void tst_accessibility::locale() { #if QT_CONFIG(accessibility) if (!QAccessible::isActive()) { QPlatformAccessibility *accessibility = platformAccessibility(); if (!accessibility) QSKIP("No QPlatformAccessibility available."); accessibility->setActive(true); } QQmlComponent component(&engine); // verify that locale is the default locale if none was set explicitly component.loadUrl(testFileUrl("locale/button.qml")); QScopedPointer object(component.create()); QVERIFY2(!object.isNull(), qPrintable(component.errorString())); QAccessibleInterface *buttonAcc = QAccessible::queryAccessibleInterface(object.get()); QVERIFY(buttonAcc); QVERIFY(buttonAcc->attributesInterface()); QVERIFY(buttonAcc->attributesInterface()->attributeKeys().contains( QAccessible::Attribute::Locale)); const QVariant localeVariant = buttonAcc->attributesInterface()->attributeValue(QAccessible::Attribute::Locale); QVERIFY(localeVariant.isValid() && localeVariant.canConvert()); QCOMPARE(localeVariant.toLocale(), QLocale()); // verify that locale is the one explicitly set for the button component.loadUrl(testFileUrl("locale/button2.qml")); QScopedPointer object2(component.create()); QVERIFY2(!object2.isNull(), qPrintable(component.errorString())); QAccessibleInterface *chineseButtonAcc = QAccessible::queryAccessibleInterface(object2.get()); QVERIFY(chineseButtonAcc); QVERIFY(chineseButtonAcc->attributesInterface()); QVERIFY(chineseButtonAcc->attributesInterface()->attributeKeys().contains( QAccessible::Attribute::Locale)); const QVariant chineseLocaleVariant = chineseButtonAcc->attributesInterface()->attributeValue(QAccessible::Attribute::Locale); QVERIFY(chineseLocaleVariant.isValid() && localeVariant.canConvert()); QCOMPARE(chineseLocaleVariant.toLocale(), QLocale("zh_CN")); #endif } QTEST_MAIN(tst_accessibility) #include "tst_accessibility.moc"