diff options
author | Lars Knoll <lars.knoll@nokia.com> | 2009-03-23 09:18:55 (GMT) |
---|---|---|
committer | Simon Hausmann <simon.hausmann@nokia.com> | 2009-03-23 09:18:55 (GMT) |
commit | e5fcad302d86d316390c6b0f62759a067313e8a9 (patch) | |
tree | c2afbf6f1066b6ce261f14341cf6d310e5595bc1 /tests/auto/qaccessibility_mac/tst_qaccessibility_mac.cpp | |
download | Qt-e5fcad302d86d316390c6b0f62759a067313e8a9.zip Qt-e5fcad302d86d316390c6b0f62759a067313e8a9.tar.gz Qt-e5fcad302d86d316390c6b0f62759a067313e8a9.tar.bz2 |
Long live Qt 4.5!
Diffstat (limited to 'tests/auto/qaccessibility_mac/tst_qaccessibility_mac.cpp')
-rw-r--r-- | tests/auto/qaccessibility_mac/tst_qaccessibility_mac.cpp | 1960 |
1 files changed, 1960 insertions, 0 deletions
diff --git a/tests/auto/qaccessibility_mac/tst_qaccessibility_mac.cpp b/tests/auto/qaccessibility_mac/tst_qaccessibility_mac.cpp new file mode 100644 index 0000000..b0b8c8b --- /dev/null +++ b/tests/auto/qaccessibility_mac/tst_qaccessibility_mac.cpp @@ -0,0 +1,1960 @@ +/**************************************************************************** +** +** Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies). +** Contact: Qt Software Information (qt-info@nokia.com) +** +** This file is part of the test suite of the Qt Toolkit. +** +** $QT_BEGIN_LICENSE:LGPL$ +** No Commercial Usage +** This file contains pre-release code and may not be distributed. +** You may use this file in accordance with the terms and conditions +** contained in the either Technology Preview License Agreement or the +** Beta Release License Agreement. +** +** GNU Lesser General Public License Usage +** Alternatively, 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.0, 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. +** +** If you are unsure which license is appropriate for your use, please +** contact the sales department at qt-sales@nokia.com. +** $QT_END_LICENSE$ +** +****************************************************************************/ +#include <QtTest/QtTest> + +#if defined(Q_WS_MAC) && !defined (QT_MAC_USE_COCOA) + +#include <private/qt_mac_p.h> +#undef verify // yes, lets reserve the word "verify" + +#include <QApplication> +#include <QDebug> +#include <QTimer> +#include <QString> +#include <QFile> +#include <QVariant> +#include <QPushButton> +#include <QToolBar> +#include <QSlider> +#include <QListWidget> +#include <QTableWidget> +#include <QScrollArea> +#include <QLabel> +#include <QScrollBar> +#include <QTextEdit> +#include <QAccessibleInterface> +#include <QAccessible> +#include <QPluginLoader> +#include <private/qaccessible_mac_p.h> +#include <quiloader.h> + +#include <sys/types.h> // for getpid() +#include <unistd.h> + +Q_DECLARE_METATYPE(AXUIElementRef); + +typedef QCFType<CFArrayRef> QCFArrayRef; + +class tst_accessibility_mac : public QObject +{ +Q_OBJECT +public slots: + void printInfo(); + void testForm(); + void testButtons(); + void testLineEdit(); + void testLabel(); + void testGroups(); + void testTabWidget(); + void testTabBar(); + void testComboBox(); + void testDeleteWidget(); + void testDeleteWidgets(); + void testMultipleWindows(); + void testHiddenWidgets(); + void testActions(); + void testChangeState(); + void testSlider(); + void testScrollArea(); + void testListView(); + void testTableView(); + void testScrollBar(); + void testSplitter(); + void testTextEdit(); + void testItemViewsWithoutModel(); +private slots: + void testQAElement(); + void testQAInterface(); + + // ui tests load an .ui file. + void uitests_data(); + void uitests(); + + void tests_data(); + void tests(); +private: + void runTest(const QString &testSlot); +}; + +/* + VERIFYs that there is no error and prints an error message if there is. +*/ +void testError(AXError error, const QString &text) +{ + if (error) + qDebug() << "Error" << error << text; + QVERIFY(error == 0); +} + +/* + Prints an CFArray holding CFStrings. +*/ +void printCFStringArray(CFArrayRef array, const QString &title) +{ + const int numElements = CFArrayGetCount(array); + qDebug() << "num" << title << " " << numElements; + + for (int i = 0; i < numElements; ++i) { + CFStringRef str = (CFStringRef)CFArrayGetValueAtIndex(array, i); + qDebug() << QCFString::toQString(str); + } +} + +QStringList toQStringList(const CFArrayRef array) +{ + const int numElements = CFArrayGetCount(array); + QStringList qtStrings; + + for (int i = 0; i < numElements; ++i) { + CFStringRef str = (CFStringRef)CFArrayGetValueAtIndex(array, i); + qtStrings.append(QCFString::toQString(str)); + } + + return qtStrings; +} + +QVariant AXValueToQVariant(AXValueRef value) +{ + QVariant var; + const AXValueType type = AXValueGetType(value); + switch (type) { + case kAXValueCGPointType : { + CGPoint point; + if (AXValueGetValue(value, type, &point)) + var = QPointF(point.x, point.y); + } break; + case kAXValueCGSizeType : { + CGSize size; + if (AXValueGetValue(value, type, &size)) + var = QSizeF(size.width, size.height); + } break; + case kAXValueCGRectType : { + CGRect rect; + if (AXValueGetValue(value, type, &rect)) + var = QRectF(rect.origin.x, rect.origin.y, rect.size.width, rect.size.height); + } break; + case kAXValueCFRangeType : +#if (MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_4) + case kAXValueAXErrorType : +#endif + case kAXValueIllegalType : + default: + qDebug() << "Illegal/Unsuported AXValue:" << type; + break; + }; + return var; +} + +/* + Converts a CFTypeRef to a QVariant, for certain selected types. Prints + an error message and returns QVariant() if the type is not supported. +*/ +QVariant CFTypeToQVariant(CFTypeRef value) +{ + QVariant var; + if (value == 0) + return var; + const uint typeID = CFGetTypeID(value); + if (typeID == CFStringGetTypeID()) { + var.setValue(QCFString::toQString((CFStringRef)value)); + } else if (typeID == CFBooleanGetTypeID()) { + var.setValue((bool)CFBooleanGetValue((CFBooleanRef)value)); + } else if (typeID == AXUIElementGetTypeID()) { + var.setValue((AXUIElementRef)value); + } else if (typeID == AXValueGetTypeID()) { + var = AXValueToQVariant((AXValueRef)value); + } else if (typeID == CFNumberGetTypeID()) { + CFNumberRef number = (CFNumberRef)value; + if (CFNumberGetType(number) != kCFNumberSInt32Type) + qDebug() << "unsupported number type" << CFNumberGetType(number); + int theNumber; + CFNumberGetValue(number, kCFNumberSInt32Type, &theNumber); + var.setValue(theNumber); + } else if (typeID == CFArrayGetTypeID()) { + CFArrayRef cfarray = static_cast<CFArrayRef>(value); + QVariantList list; + CFIndex size = CFArrayGetCount(cfarray); + for (CFIndex i = 0; i < size; ++i) + list << CFTypeToQVariant(CFArrayGetValueAtIndex(cfarray, i)); + var.setValue(list); + } else { + QCFString str = CFCopyTypeIDDescription(typeID); + qDebug() << "Unknown CFType: " << typeID << (QString)str; + } + return var; +} + +/* + Tests if a given attribute is supported by an element. Expects either + no error or error -25205 (Not supported). Causes a test failure + on other error values. +*/ +bool supportsAttribute(AXUIElementRef element, CFStringRef attribute) +{ + CFArrayRef array; + AXError err = AXUIElementCopyAttributeNames(element, &array); + if (err) { + testError(err, QLatin1String("unexpected error when testing for supported attribute") + QCFString::toQString(attribute)); + return false; + } + CFRange range; + range.location = 0; + range.length = CFArrayGetCount(array); + return CFArrayContainsValue(array, range, attribute); +} + +/* + Returns the accessibility attribute specified with attribute in a QVariant +*/ +QVariant attribute(AXUIElementRef element, CFStringRef attribute) +{ + CFTypeRef value = 0; + AXError err = AXUIElementCopyAttributeValue(element, attribute, &value); + + testError(err, QString("Error getting element attribute ") + QCFString::toQString(attribute)); + + if (err) + return QVariant(); + + return CFTypeToQVariant(value); +} + +/* + Returns the title for an element. +*/ +QString title(AXUIElementRef element) +{ + return attribute(element, kAXTitleAttribute).toString(); +} + +/* + Returns the role for an element. +*/ +QString role(AXUIElementRef element) +{ + return attribute(element, kAXRoleAttribute).toString(); +} + +/* + Returns the subrole for an element. +*/ +QString subrole(AXUIElementRef element) +{ + return attribute(element, kAXSubroleAttribute).toString(); +} + +/* + Returns the role description for an element. +*/ +QString roleDescription(AXUIElementRef element) +{ + return attribute(element, kAXRoleDescriptionAttribute).toString(); +} + +/* + Returns the enabled attribute for an element. +*/ +bool enabled(AXUIElementRef element) +{ + return attribute(element, kAXEnabledAttribute).toBool(); +} + +/* + Returns the value attribute for an element as an QVariant. +*/ +QVariant value(AXUIElementRef element) +{ + return attribute(element, kAXValueAttribute); +} + +QVariant value(QAElement element) +{ + return value(element.element()); +} + +/* + Returns the description attribute for an element as an QVariant. +*/ +QVariant description(AXUIElementRef element) +{ + return attribute(element, kAXDescriptionAttribute); +} + +/* + Returns the value attribute for an element as an bool. +*/ +bool boolValue(AXUIElementRef element) +{ + return attribute(element, kAXValueAttribute).toBool(); +} + +/* + Returns the parent for an element +*/ +AXUIElementRef parent(AXUIElementRef element) +{ + return attribute(element, kAXParentAttribute).value<AXUIElementRef>(); +} + +/* + Returns the (top-level) window(not a sheet or a drawer) for an element +*/ +AXUIElementRef window(AXUIElementRef element) +{ + return attribute(element, kAXWindowAttribute).value<AXUIElementRef>(); +} + +/* + Returns the (top-level) UI element(can also be a sheet or drawer) for an element +*/ +#if (MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_4) +AXUIElementRef topLevelUIElement(AXUIElementRef element) +{ + return attribute(element, kAXTopLevelUIElementAttribute).value<AXUIElementRef>(); +} +#endif + +/* + Returns thie size of the element. +*/ +QSizeF size(AXUIElementRef element) +{ + return attribute(element, kAXSizeAttribute).value<QSizeF>(); +} + +/* + Returns the position of the element. +*/ +QPointF position(AXUIElementRef element) +{ + return attribute(element, kAXPositionAttribute).value<QPointF>(); +} + +/* + Returns the rect of the element. +*/ +QRectF rect(AXUIElementRef element) +{ + return QRectF(position(element), size(element)); +} + +bool above(AXUIElementRef a, AXUIElementRef b) +{ + return (position(a).y() + size(a).height() <= position(b).y()); +} + +bool contains(AXUIElementRef a, AXUIElementRef b) +{ + return rect(a).contains(rect(b)); +} + +QList<AXUIElementRef> tabs(AXUIElementRef element) +{ + CFTypeRef value; + AXError err = AXUIElementCopyAttributeValue(element, kAXTabsAttribute, &value); + if (err) + return QList<AXUIElementRef>(); + + CFArrayRef array = (CFArrayRef)value; + QList<AXUIElementRef> elements; + const int count = CFArrayGetCount(array); + for (int i = 0; i < count; ++i) + elements.append((AXUIElementRef)CFArrayGetValueAtIndex(array, i)); + + return elements; +} + +QList<AXUIElementRef> elementListAttribute(AXUIElementRef element, CFStringRef attributeName) +{ + QList<AXUIElementRef> elementList; + QVariantList variants = attribute(element, attributeName).value<QVariantList>(); + foreach(QVariant variant, variants) + elementList.append(variant.value<AXUIElementRef>()); + return elementList; +} + +AXUIElementRef elementAttribute(AXUIElementRef element, CFStringRef attributeName) +{ + return attribute(element, attributeName).value<AXUIElementRef>(); +} + +QString stringAttribute(AXUIElementRef element, CFStringRef attributeName) +{ + return attribute(element, attributeName).value<QString>(); +} + + +/* + Returns the UIElement at the given position. +*/ +AXUIElementRef childAtPoint(QPointF position) +{ + AXUIElementRef element = 0; + const AXError err = AXUIElementCopyElementAtPosition(AXUIElementCreateApplication(getpid()), position.x(), position.y(), &element); + if (err) { + qDebug() << "Error getting element at " << position; + return 0; + } + + return element; +} + +/* + Returns a QStringList containing the names of the actions the ui element supports +*/ +QStringList actionNames(AXUIElementRef element) +{ + CFArrayRef cfStrings; + const AXError err = AXUIElementCopyActionNames(element, &cfStrings); + testError(err, "Unable to get action names"); + return toQStringList(cfStrings); +} + +bool supportsAction(const AXUIElementRef element, const QString &actionName) +{ + const QStringList actions = actionNames(element); + return actions.contains(actionName); +} + +bool performAction(const AXUIElementRef element, const QString &actionName) +{ + const AXError err = AXUIElementPerformAction(element, QCFString(actionName)); + return (err == 0); +} + +/* + Om 10.4 and up, verifyes the AXRoleDescription attribute for an element, + on 10.3 and below this test always passes. + + The reason for this is that the HICopyAccessibilityRoleDescription call + used to implement this functionality was introduced in 10.4. +*/ +#if (MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_4) + #define VERIFY_ROLE_DESCRIPTION(ELEMENT, TEXT) \ + QCOMPARE(roleDescription(ELEMENT), QString(TEXT)) +#else + #define VERIFY_ROLE_DESCRIPTION(ELEMENT, TEXT) QVERIFY(true) +#endif + + +CFArrayRef childrenArray(AXUIElementRef element) +{ + CFTypeRef value = 0; + AXError err = AXUIElementCopyAttributeValue(element, kAXChildrenAttribute, &value); + if (!err && CFGetTypeID(value) == CFArrayGetTypeID()) { + return (CFArrayRef)value; + } + + return CFArrayCreate(0,0,0,0); +} + +/* + Gest the child count from an element. +*/ +int numChildren(AXUIElementRef element) +{ + return CFArrayGetCount(childrenArray(element)); +} + +/* + Gets the child with index childIndex from element. Returns 0 if not found. +*/ +AXUIElementRef child(AXUIElementRef element, int childIndex) +{ + CFArrayRef children = childrenArray(element); + if (childIndex >= CFArrayGetCount(children)) + return 0; + + const void *data = CFArrayGetValueAtIndex(children, childIndex); + return (AXUIElementRef)data; +} + +/* + Gets the child titled childTitle from element. Returns 0 if not found. +*/ +AXUIElementRef childByTitle(AXUIElementRef element, const QString &childTitle) +{ + CFArrayRef children = childrenArray(element); + const int numChildren = CFArrayGetCount(children); + for (int i = 0; i < numChildren; ++i) { + const AXUIElementRef childElement = (AXUIElementRef)CFArrayGetValueAtIndex(children, i); + // Test for support for title attribute before getting it to avoid test fail. + if (supportsAttribute(childElement, kAXTitleAttribute) && title(childElement) == childTitle) + return childElement; + } + return 0; +} + +/* + Gets the child with the given value from element. Returns 0 if not found. +*/ +AXUIElementRef childByValue(AXUIElementRef element, const QVariant &testValue) +{ + CFArrayRef children = childrenArray(element); + const int numChildren = CFArrayGetCount(children); + for (int i = 0; i < numChildren; ++i) { + const AXUIElementRef childElement = (AXUIElementRef)CFArrayGetValueAtIndex(children, i); + // Test for support for value attribute before getting it to avoid test fail. + if (supportsAttribute(childElement, kAXValueAttribute) && value(childElement) == testValue) + return childElement; + } + return 0; +} + +/* + Gets the child by role from element. Returns 0 if not found. +*/ +AXUIElementRef childByRole(AXUIElementRef element, const QString &macRole) +{ + CFArrayRef children = childrenArray(element); + const int numChildren = CFArrayGetCount(children); + for (int i = 0; i < numChildren; ++i) { + const AXUIElementRef childElement = (AXUIElementRef)CFArrayGetValueAtIndex(children, i); + if (role(childElement) == macRole) + return childElement; + } + return 0; +} + +void printTypeForAttribute(AXUIElementRef element, CFStringRef attribute) +{ + CFTypeRef value = 0; + AXError err = AXUIElementCopyAttributeValue(element, attribute, &value); + if (!err) { + qDebug() << "type id" << CFGetTypeID(value); + QCFString str = CFCopyTypeIDDescription(CFGetTypeID(value)); + qDebug() << (QString)str; + } else { + qDebug() << "Attribute Get error" << endl; + } +} + +int indent = 0; +QString space() +{ + QString space; + for (int i = 0; i < indent; ++i) { + space += " "; + } + return space; +} + + +/* + Recursively prints acccesibility info for currentElement and all its children. +*/ +void printElementInfo(AXUIElementRef currentElement) +{ + if (HIObjectIsAccessibilityIgnored(AXUIElementGetHIObject(currentElement))) { + qDebug() << space() << "Ignoring element with role" << role(currentElement); + return; + } + + qDebug() << space() <<"Role" << role(currentElement); + if (supportsAttribute(currentElement, kAXTitleAttribute)) + qDebug() << space() << "Title" << title(currentElement); + else + qDebug() << space() << "Title not supported"; + + if (supportsAttribute(currentElement, kAXValueAttribute)) + qDebug() << space() << "Value" << attribute(currentElement, kAXValueAttribute); + else + qDebug() << space() << "Value not supported"; + + qDebug() << space() << "Number of children" << numChildren(currentElement); + for (int i = 0; i < numChildren(currentElement); ++i) { + AXUIElementRef childElement = child(currentElement, i); + // Skip the menu bar. + if (role(childElement) != "AXMenuBar") { + indent+= 4; + printElementInfo(childElement); + indent-= 4; + } + } + qDebug() << " "; +} + +/* + Recursively prints the child interfaces belonging to interface. +*/ + +void printChildren(const QAInterface &interface) +{ + if (interface.isValid() == false) + return; + + QList<QAInterface> children = interface.children(); + if (children.isEmpty()) + return; + + qDebug() << "## Children for" << interface; + foreach (const QAInterface &child, children) { + qDebug() << child << "index in parent" << interface.indexOfChild(child); + } + foreach (const QAInterface &child, children) { + printChildren(child); + } +} + +bool isIgnored(AXUIElementRef currentElement) +{ + return HIObjectIsAccessibilityIgnored(AXUIElementGetHIObject(currentElement)); +} + +bool equal(CFTypeRef o1, CFTypeRef o2) +{ + if (o1 == 0 || o2 == 0) + return false; + return CFEqual(o1, o2); +} + +/* + Verifies basic element info. +*/ +#define VERIFY_ELEMENT(element, _parent, _role) \ + QVERIFY(element != 0); \ + QVERIFY(role(element) == _role); \ + QVERIFY(equal(::parent(element), _parent)); +/* + Verifies that the application and the main form is there has the right info. +*/ +void testAppAndForm(AXUIElementRef application) +{ + QVERIFY(title(application) == "tst_qaccessibility_mac"); + QVERIFY(role(application) == "AXApplication"); + + AXUIElementRef form = childByTitle(application, "Form"); + VERIFY_ELEMENT(form, application, "AXWindow"); +} + +void tst_accessibility_mac::printInfo() +{ + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + printElementInfo(currentApplication); +} + +/* + Tests for form.ui +*/ +void tst_accessibility_mac::testForm() +{ + // Get reference to the current application. + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + testAppAndForm(currentApplication); + childByTitle(currentApplication, "Form"); +} + +/* + Tests for buttons.ui +*/ +void tst_accessibility_mac::testButtons() +{ + // Get reference to the current application. + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + testAppAndForm(currentApplication); + AXUIElementRef form = childByTitle(currentApplication, "Form"); + + AXUIElementRef ren = childByTitle(form, "Ren"); + VERIFY_ELEMENT(ren, form, "AXButton"); + QVERIFY(enabled(ren) == true); + VERIFY_ROLE_DESCRIPTION(ren, "button"); + + AXUIElementRef stimpy = childByTitle(form, "Stimpy"); + VERIFY_ELEMENT(stimpy, form, "AXRadioButton"); + QVERIFY(enabled(stimpy) == true); + QVERIFY(value(stimpy).toInt() == 1); // checked; + VERIFY_ROLE_DESCRIPTION(stimpy, "radio button"); + + AXUIElementRef pinky = childByTitle(form, "Pinky"); + VERIFY_ELEMENT(pinky, form, "AXCheckBox"); + QVERIFY(enabled(pinky) == false); + QVERIFY(value(pinky).toInt() == 0); // unchecked; + VERIFY_ROLE_DESCRIPTION(pinky, "check box"); + + AXUIElementRef brain = childByTitle(form, "Brain"); + VERIFY_ELEMENT(brain, form, "AXButton"); + VERIFY_ROLE_DESCRIPTION(brain, "button"); +} + +void tst_accessibility_mac::testLabel() +{ + // Get reference to the current application. + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + + testAppAndForm(currentApplication); + AXUIElementRef form = childByTitle(currentApplication, "Form"); + AXUIElementRef label = childByValue(form, "This is a Text Label"); + QVERIFY(label); + VERIFY_ELEMENT(label, form, "AXStaticText"); + VERIFY_ROLE_DESCRIPTION(label, "text"); + QCOMPARE(supportsAttribute(label, kAXDescriptionAttribute), false); +} + +/* + Tests for lineedit.ui +*/ +void tst_accessibility_mac::testLineEdit() +{ + // Get reference to the current application. + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + + testAppAndForm(currentApplication); + AXUIElementRef form = childByTitle(currentApplication, "Form"); + AXUIElementRef lineEdit = childByValue(form, "Line edit"); + VERIFY_ELEMENT(lineEdit, form, "AXTextField"); + VERIFY_ROLE_DESCRIPTION(lineEdit, "text field"); +} + +/* + Tests for groups.ui +*/ +void tst_accessibility_mac::testGroups() +{ + // Get reference to the current application. + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + + testAppAndForm(currentApplication); + AXUIElementRef form = childByTitle(currentApplication, "Form"); + + AXUIElementRef groupA = childByTitle(form, "Group A"); + VERIFY_ELEMENT(groupA, form, "AXGroup"); + AXUIElementRef button1 = childByTitle(groupA, "PushButton 1"); + VERIFY_ELEMENT(button1, groupA, "AXButton"); + VERIFY_ROLE_DESCRIPTION(groupA, "group"); + + AXUIElementRef groupB = childByTitle(form, "Group B"); + VERIFY_ELEMENT(groupB, form, "AXGroup"); + AXUIElementRef button3 = childByTitle(groupB, "PushButton 3"); + VERIFY_ELEMENT(button3, groupB, "AXButton"); +} + +/* + Tests for tabs.ui +*/ +void tst_accessibility_mac::testTabWidget() +{ + { // Test that the QTabWidget hierarchy is what we expect it to be. + QTabWidget tabWidget; + tabWidget.show(); + QAInterface interface = QAccessible::queryAccessibleInterface(&tabWidget); + tabWidget.addTab(new QPushButton("Foo"), "FooTab"); + tabWidget.addTab(new QPushButton("Bar"), "BarTab"); + QCOMPARE(interface.childCount(), 2); + const QList<QAInterface> children = interface.children(); + QVERIFY(children.at(0).object()->inherits("QStackedWidget")); + QVERIFY(children.at(1).object()->inherits("QTabBar")); + + const QList<QAInterface> tabBarChildren = children.at(1).children(); + QCOMPARE(tabBarChildren.count(), 4); + QCOMPARE(tabBarChildren.at(0).text(QAccessible::Name), QLatin1String("FooTab")); + QCOMPARE(tabBarChildren.at(1).text(QAccessible::Name), QLatin1String("BarTab")); + QCOMPARE(tabBarChildren.at(0).role(), QAccessible::PageTab); + QCOMPARE(tabBarChildren.at(1).role(), QAccessible::PageTab); + + // Check that the hierarchy manager is able to register the tab bar children. + QAccessibleHierarchyManager *manager = QAccessibleHierarchyManager::instance(); + QAInterface tabBarInterface = children.at(1); + QAElement tabBarElement = manager->registerInterface(tabBarInterface); + QCOMPARE(manager->lookup(tabBarElement).childCount(), 4); + manager->registerChildren(tabBarInterface); + QAElement tabButtonElement = manager->lookup(tabBarChildren.at(1)); + QAInterface tabButtonInterface = manager->lookup(tabButtonElement); + QCOMPARE(tabButtonInterface.text(QAccessible::Name), QLatin1String("BarTab")); + QVERIFY(isItInteresting(tabButtonInterface) == true); + } + + // Get reference to the current application. + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + + testAppAndForm(currentApplication); + const QString formTitle = "Form"; + AXUIElementRef form = childByTitle(currentApplication, formTitle); + QVERIFY(form); + + const QString tabRole = "AXTabGroup"; + AXUIElementRef tabGroup = childByRole(form, tabRole); + QVERIFY(tabGroup); + + // Test that we have three child buttons (the tab buttons + plus the contents of the first tab) + const int numChildren = ::numChildren(tabGroup); + QCOMPARE(numChildren, 3); + + const QString tab1Title = "Tab 1"; + AXUIElementRef tabButton1 = childByTitle(tabGroup, tab1Title); + QVERIFY (tabButton1); + VERIFY_ELEMENT(tabButton1, tabGroup, "AXRadioButton"); + QCOMPARE(title(tabButton1), tab1Title); + + const QString tab2Title = "Tab 2"; + const AXUIElementRef tabButton2 = childByTitle(tabGroup, tab2Title); + QVERIFY(tabButton2); + VERIFY_ELEMENT(tabButton2, tabGroup, "AXRadioButton"); + QCOMPARE(title(tabButton2), tab2Title); + + // Test that the window and top-level-ui-elment is the form. + // Window is not reported properly on 10.5 + if (QSysInfo::MacintoshVersion < QSysInfo::MV_10_5) { + QVERIFY(equal(window(tabGroup), form)); + QVERIFY(equal(window(tabButton1), form)); + } +#if (MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_4) + QVERIFY(equal(topLevelUIElement(tabGroup), form)); + QVERIFY(equal(topLevelUIElement(tabButton1), form)); +#endif + // Test the bounding rectangles for the tab group and buttons. + const QRectF groupRect(position(tabGroup), size(tabGroup)); + const QRectF tabButton1Rect(position(tabButton1), size(tabButton1)); + const QRectF tabButton2Rect(position(tabButton2), size(tabButton2)); + + QVERIFY(groupRect.isNull() == false); + QVERIFY(tabButton1Rect.isNull() == false); + QVERIFY(tabButton1Rect.isNull() == false); + + QVERIFY(groupRect.contains(tabButton1Rect)); + QVERIFY(groupRect.contains(tabButton2Rect)); + QVERIFY(tabButton2Rect.contains(tabButton1Rect) == false); + + // Test the childAtPoint event. + const AXUIElementRef childAtTab1Position = childAtPoint(position(tabButton1) + QPointF(5,5)); + QVERIFY(equal(childAtTab1Position, tabButton1)); + const AXUIElementRef childAtOtherPosition = childAtPoint(position(tabButton1) - QPointF(5,5)); + QVERIFY(equal(childAtOtherPosition, tabButton1) == false); + + // Test AXTabs attribute + QVERIFY(supportsAttribute(tabGroup, kAXTabsAttribute)); + QList<AXUIElementRef> tabElements = tabs(tabGroup); + QCOMPARE(tabElements.count(), 2); + QVERIFY(equal(tabElements.at(0), tabButton1)); + QVERIFY(equal(tabElements.at(1), tabButton2)); + + // Perform the press action on each child. + for (int i = 0; i < numChildren; ++i) { + const AXUIElementRef child = ::child(tabGroup, i); + QVERIFY(supportsAction(child, "AXPress")); + QVERIFY(performAction(child, "AXPress")); + } +} + +void tst_accessibility_mac::testTabBar() +{ + QTabBar tabBar; + tabBar.addTab("Tab A"); + tabBar.addTab("Tab B"); + tabBar.show(); + + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + AXUIElementRef window = childByRole(currentApplication, "AXWindow"); + QVERIFY(window); + + const QString tabRole = "AXTabGroup"; + AXUIElementRef tabGroup = childByRole(window, tabRole); + QVERIFY(tabGroup); + + const int numChildren = ::numChildren(tabGroup); + QCOMPARE(numChildren, 2); + + const QString tab1Title = "Tab A"; + AXUIElementRef tabButton1 = childByTitle(tabGroup, tab1Title); + QVERIFY (tabButton1); + VERIFY_ELEMENT(tabButton1, tabGroup, "AXRadioButton"); + QCOMPARE(title(tabButton1), tab1Title); + + const QString tab2Title = "Tab B"; + const AXUIElementRef tabButton2 = childByTitle(tabGroup, tab2Title); + QVERIFY(tabButton2); + VERIFY_ELEMENT(tabButton2, tabGroup, "AXRadioButton"); + QCOMPARE(title(tabButton2), tab2Title); + + // Test the childAtPoint event. + const AXUIElementRef childAtTab1Position = childAtPoint(position(tabButton1) + QPointF(5,5)); + QVERIFY(equal(childAtTab1Position, tabButton1)); + const AXUIElementRef childAtOtherPosition = childAtPoint(position(tabButton1) - QPointF(5,5)); + QVERIFY(equal(childAtOtherPosition, tabButton1) == false); + + // Test AXTabs attribute + QVERIFY(supportsAttribute(tabGroup, kAXTabsAttribute)); + QList<AXUIElementRef> tabElements = tabs(tabGroup); + QCOMPARE(tabElements.count(), 2); + QVERIFY(equal(tabElements.at(0), tabButton1)); + QVERIFY(equal(tabElements.at(1), tabButton2)); + + // Perform the press action on each child. + for (int i = 0; i < numChildren; ++i) { + const AXUIElementRef child = ::child(tabGroup, i); + QVERIFY(supportsAction(child, "AXPress")); + QVERIFY(performAction(child, "AXPress")); + } +} + +void tst_accessibility_mac::testComboBox() +{ + // Get reference to the current application. + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + + testAppAndForm(currentApplication); + const QString formTitle = "Form"; + AXUIElementRef form = childByTitle(currentApplication, formTitle); + + const QString comboBoxRole = "AXPopUpButton"; + AXUIElementRef comboBox = childByRole(form, comboBoxRole); + QVERIFY(comboBox != 0); + QVERIFY(supportsAction(comboBox, "AXPress")); + QVERIFY(performAction(comboBox, "AXPress")); +} + +void tst_accessibility_mac::testDeleteWidget() +{ + const QString buttonTitle = "Hi there"; + QWidget *form = new QWidget(0, Qt::Window); + form->setWindowTitle("Form"); + form->show(); + QPushButton *button = new QPushButton(buttonTitle, form); + button->show(); + + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + testAppAndForm(currentApplication); + AXUIElementRef formElement = childByTitle(currentApplication, "Form"); + + AXUIElementRef buttonElement = childByTitle(formElement, buttonTitle); + QVERIFY(buttonElement); + + button->hide(); + delete button; + + buttonElement = childByTitle(formElement, buttonTitle); + QVERIFY(!buttonElement); + delete form; +} + +void tst_accessibility_mac::testDeleteWidgets() +{ + const QString buttonTitle = "Hi there"; + const int repeats = 10; + + for (int i = 0; i < repeats; ++i) { + + QWidget *form = new QWidget(0, Qt::Window); + form->setWindowTitle("Form"); + form->show(); + + QPushButton *button = new QPushButton(buttonTitle, form); + button->show(); + + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + testAppAndForm(currentApplication); + AXUIElementRef formElement = childByTitle(currentApplication, "Form"); + AXUIElementRef buttonElement = childByTitle(formElement, buttonTitle); + QVERIFY(buttonElement); + delete form; + + { + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + QVERIFY(currentApplication); + AXUIElementRef formElement = childByTitle(currentApplication, "Form"); + QVERIFY(!formElement); + } + } + + for (int i = 0; i < repeats; ++i) { + QWidget *form = new QWidget(0, Qt::Window); + form->setWindowTitle("Form"); + + + new QScrollBar(form); + form->show(); + + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + testAppAndForm(currentApplication); + AXUIElementRef formElement = childByTitle(currentApplication, "Form"); + + const AXUIElementRef scrollBarElement = childByRole(formElement, "AXScrollBar"); + QVERIFY(scrollBarElement); + delete form; + + { + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + QVERIFY(currentApplication); + AXUIElementRef formElement = childByTitle(currentApplication, "Form"); + QVERIFY(!formElement); + } + } + + for (int i = 0; i < repeats; ++i) { + QWidget *form = new QWidget(0, Qt::Window); + form->setWindowTitle("Form"); + + QListWidget *listWidget = new QListWidget(form); + listWidget->addItem("Foo"); + listWidget->addItem("Bar"); + form->show(); + + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + testAppAndForm(currentApplication); + AXUIElementRef formElement = childByTitle(currentApplication, "Form"); + + const AXUIElementRef scrollAreaElement = childByRole(formElement, "AXScrollArea"); + QVERIFY(scrollAreaElement); + + const AXUIElementRef listElement = childByRole(scrollAreaElement, "AXList"); + QVERIFY(listElement); + delete form; + + { + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + QVERIFY(currentApplication); + AXUIElementRef formElement = childByTitle(currentApplication, "Form"); + QVERIFY(!formElement); + } + } + +} + +void tst_accessibility_mac::testMultipleWindows() +{ + const QString formATitle("FormA"); + const QString formBTitle("FormB"); + + // Create a window + QWidget *formA = new QWidget(0, Qt::Window); + formA->setWindowTitle(formATitle); + formA->show(); + + // Test if we can access the window + AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + AXUIElementRef formAElement = childByTitle(currentApplication, formATitle); + QVERIFY(formAElement); + + // Create another window + QWidget *formB = new QWidget(0, Qt::Window); + formB->setWindowTitle(formBTitle); + formB->show(); + + // Test if we can access both windows + formAElement = childByTitle(currentApplication, formATitle); + QVERIFY(formAElement); + + AXUIElementRef formBElement = childByTitle(currentApplication, formBTitle); + QVERIFY(formBElement); + + delete formA; +} + +void tst_accessibility_mac::testHiddenWidgets() +{ + const QString windowTitle ="a widget"; + QWidget * const window = new QWidget(0); + window->setWindowTitle(windowTitle); + window->show(); + + const AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + const AXUIElementRef windowElement = childByTitle(currentApplication, windowTitle); + QVERIFY(windowElement); + QCOMPARE(isIgnored(windowElement), false); + + const QString buttonTitle = "a button"; + QPushButton * const button = new QPushButton(window); + button->setText(buttonTitle); + button->show(); + + const AXUIElementRef buttonElement = childByTitle(windowElement, buttonTitle); + QVERIFY(buttonElement); + QCOMPARE(isIgnored(buttonElement), false); + + const QString toolbarTitle = "a toolbar"; + QToolBar * const toolbar = new QToolBar(toolbarTitle, window); + toolbar->show(); + + const AXUIElementRef toolBarElement = childByTitle(windowElement, toolbarTitle); + QVERIFY(toolBarElement == 0); + + delete window; +}; + +void tst_accessibility_mac::testActions() +{ + // create a window with a push button + const QString windowTitle ="a widget"; + QWidget * const window = new QWidget(); + window->setWindowTitle(windowTitle); + window->show(); + + const AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + const AXUIElementRef windowElement = childByTitle(currentApplication, windowTitle); + QVERIFY(windowElement); + + const QString buttonTitle = "a button"; + QPushButton * const button = new QPushButton(window); + button->setText(buttonTitle); + button->show(); + + const AXUIElementRef buttonElement = childByTitle(windowElement, buttonTitle); + QVERIFY(buttonElement); + + // Verify that the button has the Press action. + const QStringList actions = actionNames(buttonElement); + const QString pressActionName("AXPress"); + QVERIFY(actions.contains(pressActionName)); + + // Press button and check the pressed signal + QSignalSpy pressed(button, SIGNAL(pressed())); + QVERIFY(performAction(buttonElement, pressActionName)); + QCOMPARE(pressed.count(), 1); + + pressed.clear(); + QVERIFY(performAction(buttonElement, QString("does not exist"))); + QCOMPARE(pressed.count(), 0); + + delete window; +}; + +void tst_accessibility_mac::testChangeState() +{ + const QString windowTitle ="a widget"; + QWidget * const window = new QWidget(); + window->setWindowTitle(windowTitle); + window->show(); + + const AXUIElementRef applicationElement = AXUIElementCreateApplication(getpid()); + const AXUIElementRef windowElement = childByTitle(applicationElement, windowTitle); + QVERIFY(windowElement); + const int otherChildren = numChildren(windowElement); + + const QString buttonTitle = "Button"; + QPushButton * const button = new QPushButton(buttonTitle, window); + button->setText(buttonTitle); + + // Test that show/hide adds/removes the button from the hierachy. + QVERIFY(childByTitle(windowElement, buttonTitle) == 0); + QCOMPARE(numChildren(windowElement), otherChildren); + button->show(); + QVERIFY(childByTitle(windowElement, buttonTitle) != 0); + QCOMPARE(numChildren(windowElement), otherChildren + 1); + button->hide(); + QVERIFY(childByTitle(windowElement, buttonTitle) == 0); + QCOMPARE(numChildren(windowElement), otherChildren); + button->show(); + QVERIFY(childByTitle(windowElement, buttonTitle) != 0); + QCOMPARE(numChildren(windowElement), otherChildren + 1); + + // Test that hiding and showing a widget also removes and adds all its children. + { + QWidget * const parent = new QWidget(window); + const int otherChildren = numChildren(windowElement); + + QPushButton * const child = new QPushButton(parent); + const QString childButtonTitle = "child button"; + child->setText(childButtonTitle); + + parent->show(); + QVERIFY(childByTitle(windowElement, childButtonTitle) != 0); + QCOMPARE(numChildren(windowElement), otherChildren + 1); + + parent->hide(); + QVERIFY(childByTitle(windowElement, childButtonTitle) == 0); + QCOMPARE(numChildren(windowElement), otherChildren ); + + parent->show(); + QVERIFY(childByTitle(windowElement, childButtonTitle) != 0); + QCOMPARE(numChildren(windowElement), otherChildren + 1); + + delete parent; + } + + // Test that the enabled attribute is updated after a call to setEnabled. + const AXUIElementRef buttonElement = childByTitle(windowElement, buttonTitle); + QVERIFY(enabled(buttonElement)); + button->setEnabled(false); + QVERIFY(enabled(buttonElement) == false); + button->setEnabled(true); + QVERIFY(enabled(buttonElement)); + + // Test that changing the title updates the accessibility information. + const QString buttonTitle2 = "Button 2"; + button->setText(buttonTitle2); + QVERIFY(childByTitle(windowElement, buttonTitle2) != 0); + QVERIFY(childByTitle(windowElement, buttonTitle) == 0); + + delete window; +} + +void tst_accessibility_mac::testSlider() +{ + const QString windowTitle = "a widget"; + QWidget * const window = new QWidget(); + window->setWindowTitle(windowTitle); + window->show(); + + const AXUIElementRef applicationElement = AXUIElementCreateApplication(getpid()); + QVERIFY(applicationElement); + const AXUIElementRef windowElement = childByTitle(applicationElement, windowTitle); + QVERIFY(windowElement); + const int windowChildren = numChildren(windowElement); + + QSlider * const slider = new QSlider(window); + slider->show(); + const AXUIElementRef sliderElement = childByRole(windowElement, "AXSlider"); + QVERIFY(sliderElement); + + // Test that the slider and its children are removed from the hierachy when we call hide(). + QCOMPARE(numChildren(windowElement), windowChildren + 1); + slider->hide(); + QCOMPARE(numChildren(windowElement), windowChildren); + + delete slider; +} + +void tst_accessibility_mac::testScrollArea() +{ + QWidget window; + const QString windowTitle = "window"; + window.setWindowTitle(windowTitle); + window.resize(300, 300); + + QScrollArea scrollArea(&window); + scrollArea.resize(300, 300); + scrollArea.setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOn); + scrollArea.setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOn); + + QLabel label; + label.setText("Foo"); + scrollArea.setWidget(&label); + + window.show(); + + // Verify that the QAinterface returns the correct children + QAInterface interface = QAccessible::queryAccessibleInterface(&scrollArea); + QCOMPARE(interface.childCount(), 3); + + QAInterface viewport = interface.navigate(QAccessible::Child, 1); + QVERIFY(viewport.isValid()); + + QAInterface scrollBarContainer1 = interface.navigate(QAccessible::Child, 2); + QVERIFY(scrollBarContainer1.isValid()); + + QAInterface scrollBar1 = scrollBarContainer1.navigate(QAccessible::Child, 1); + + QVERIFY(scrollBar1.isValid()); + QVERIFY(scrollBar1.role() == QAccessible::ScrollBar); + + QAInterface scrollBarContainer2 = interface.navigate(QAccessible::Child, 3); + QVERIFY(scrollBarContainer1.isValid()); + + QAInterface scrollBar2 = scrollBarContainer2.navigate(QAccessible::Child, 1); + QVERIFY(scrollBar2.isValid()); + QVERIFY(scrollBar2.role() == QAccessible::ScrollBar); + + // Navigate to the scroll area from the application + const AXUIElementRef applicationElement = AXUIElementCreateApplication(getpid()); + QVERIFY(applicationElement); + const AXUIElementRef windowElement = childByTitle(applicationElement, windowTitle); + QVERIFY(windowElement); + const AXUIElementRef scrollAreaElement = childByRole(windowElement, "AXScrollArea"); + QVERIFY(scrollAreaElement); + + // Get the scroll bars + QVERIFY(supportsAttribute(scrollAreaElement, kAXHorizontalScrollBarAttribute)); + const AXUIElementRef horizontalScrollBar = elementAttribute(scrollAreaElement, kAXHorizontalScrollBarAttribute); + QVERIFY(horizontalScrollBar); + QVERIFY(role(horizontalScrollBar) == "AXScrollBar"); + QVERIFY(stringAttribute(horizontalScrollBar, kAXOrientationAttribute) == "AXHorizontalOrientation"); + + QVERIFY(supportsAttribute(scrollAreaElement, kAXVerticalScrollBarAttribute)); + const AXUIElementRef verticalScrollBar = elementAttribute(scrollAreaElement, kAXVerticalScrollBarAttribute); + QVERIFY(verticalScrollBar); + QVERIFY(role(verticalScrollBar) == "AXScrollBar"); + QVERIFY(stringAttribute(verticalScrollBar, kAXOrientationAttribute) == "AXVerticalOrientation"); + + // Get the contents and verify that we get the label. + QVERIFY(supportsAttribute(scrollAreaElement, kAXContentsAttribute)); + const QList<AXUIElementRef> contents = elementListAttribute(scrollAreaElement, kAXContentsAttribute); + QCOMPARE(contents.count(), 1); + AXUIElementRef content = contents.at(0); + QVERIFY(role(content) == "AXStaticText"); + QVERIFY(title(content) == "Foo"); + + // Turn scroll bars off + { + scrollArea.setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); + scrollArea.setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); + QVERIFY(supportsAttribute(scrollAreaElement, kAXHorizontalScrollBarAttribute) == false); + QVERIFY(supportsAttribute(scrollAreaElement, kAXVerticalScrollBarAttribute) == false); + + QVERIFY(supportsAttribute(scrollAreaElement, kAXContentsAttribute)); + const QList<AXUIElementRef> contents = elementListAttribute(scrollAreaElement, kAXContentsAttribute); + QCOMPARE(contents.count(), 1); + AXUIElementRef content = contents.at(0); + + QVERIFY(role(content) == "AXStaticText"); + QVERIFY(title(content) == "Foo"); + } + + // Turn the horizontal scrollbar on. + { + scrollArea.setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOn); + scrollArea.setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); + QVERIFY(supportsAttribute(scrollAreaElement, kAXHorizontalScrollBarAttribute) == true); + QVERIFY(supportsAttribute(scrollAreaElement, kAXVerticalScrollBarAttribute) == false); + + const AXUIElementRef horizontalScrollBar = elementAttribute(scrollAreaElement, kAXHorizontalScrollBarAttribute); + QVERIFY(horizontalScrollBar); + QVERIFY(role(horizontalScrollBar) == "AXScrollBar"); + QVERIFY(stringAttribute(horizontalScrollBar, kAXOrientationAttribute) == "AXHorizontalOrientation"); + + QVERIFY(supportsAttribute(scrollAreaElement, kAXContentsAttribute)); + const QList<AXUIElementRef> contents = elementListAttribute(scrollAreaElement, kAXContentsAttribute); + QCOMPARE(contents.count(), 1); + AXUIElementRef content = contents.at(0); + QVERIFY(role(content) == "AXStaticText"); + QVERIFY(title(content) == "Foo"); + } + + // Turn the vertical scrollbar on. + { + scrollArea.setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); + scrollArea.setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOn); + QVERIFY(supportsAttribute(scrollAreaElement, kAXHorizontalScrollBarAttribute) == false); + QVERIFY(supportsAttribute(scrollAreaElement, kAXVerticalScrollBarAttribute) == true); + + QVERIFY(supportsAttribute(scrollAreaElement, kAXVerticalScrollBarAttribute)); + const AXUIElementRef verticalScrollBar = elementAttribute(scrollAreaElement, kAXVerticalScrollBarAttribute); + QVERIFY(verticalScrollBar); + QVERIFY(role(verticalScrollBar) == "AXScrollBar"); + QVERIFY(stringAttribute(verticalScrollBar, kAXOrientationAttribute) == "AXVerticalOrientation"); + + QVERIFY(supportsAttribute(scrollAreaElement, kAXContentsAttribute)); + const QList<AXUIElementRef> contents = elementListAttribute(scrollAreaElement, kAXContentsAttribute); + QCOMPARE(contents.count(), 1); + AXUIElementRef content = contents.at(0); + QVERIFY(role(content) == "AXStaticText"); + QVERIFY(title(content) == "Foo"); + } +} + +void tst_accessibility_mac::testListView() +{ + QWidget window; + const QString windowTitle("window"); + window.setWindowTitle(windowTitle); + window.resize(300, 300); + + QListWidget *listWidget = new QListWidget(&window); + listWidget->setObjectName("listwidget"); + listWidget->addItem("A"); + listWidget->addItem("B"); + listWidget->addItem("C"); + + window.show(); + QTest::qWait(1); + + { + // Verify that QAInterface works as expected for list views + QAInterface listWidgetInterface = QAccessible::queryAccessibleInterface(listWidget); + QCOMPARE(listWidgetInterface.role(), QAccessible::Client); + QCOMPARE(listWidgetInterface.childCount(), 1); + QAInterface viewPort = listWidgetInterface.childAt(1); + QCOMPARE(viewPort.role(), QAccessible::List); + QVERIFY(viewPort.object() != 0); + QCOMPARE(viewPort.childCount(), 3); + const QList<QAInterface> rows = viewPort.children(); + QCOMPARE(rows.count(), 3); + QVERIFY(rows.at(0).object() == 0); + QCOMPARE(rows.at(0).parent().indexOfChild(rows.at(0)), 1); + QCOMPARE(rows.at(1).parent().indexOfChild(rows.at(1)), 2); + QCOMPARE(rows.at(2).parent().indexOfChild(rows.at(2)), 3); + + // test the QAInterface comparison operator + QVERIFY(rows.at(0) == rows.at(0)); + QVERIFY(rows.at(0) != rows.at(1)); + QVERIFY(rows.at(0) != viewPort); + QVERIFY(viewPort == viewPort); + QVERIFY(listWidgetInterface != viewPort); + QVERIFY(listWidgetInterface == listWidgetInterface); + + // test QAInterface::isHIView() + QVERIFY(viewPort.isHIView()); + QVERIFY(listWidgetInterface.isHIView()); + QVERIFY(rows.at(0).isHIView() == false); + } + + const AXUIElementRef applicationElement = AXUIElementCreateApplication(getpid()); + QVERIFY(applicationElement); + const AXUIElementRef windowElement = childByTitle(applicationElement, windowTitle); + QVERIFY(windowElement); + const AXUIElementRef scrollAreaElement = childByRole(windowElement, "AXScrollArea"); + QVERIFY(scrollAreaElement); + const AXUIElementRef listElement = childByRole(scrollAreaElement, "AXList"); + QVERIFY(listElement); + QVERIFY(equal(::parent(listElement), scrollAreaElement)); + + // Window is not reported properly on 10.5 + if (QSysInfo::MacintoshVersion < QSysInfo::MV_10_5) + QVERIFY(equal(::window(listElement), windowElement)); + + const AXUIElementRef A = childByTitle(listElement, "A"); + QVERIFY(A); + const AXUIElementRef B = childByTitle(listElement, "B"); + QVERIFY(B); + const AXUIElementRef C = childByTitle(listElement, "C"); + QVERIFY(C); + + QVERIFY(value(A) == "A"); + QVERIFY(equal(::parent(A), listElement)); + QVERIFY(enabled(A)); + // Window is not reported properly on 10.5 + if (QSysInfo::MacintoshVersion < QSysInfo::MV_10_5) + QVERIFY(equal(::window(A), windowElement)); + + QVERIFY(above(A, B)); + QVERIFY(!above(B, A)); + QVERIFY(above(B, C)); + QVERIFY(contains(listElement, A)); + QVERIFY(!contains(A, listElement)); + QVERIFY(contains(listElement, B)); + QVERIFY(contains(listElement, C)); +} + +void tst_accessibility_mac::testTableView() +{ + QWidget window; + const QString windowTitle("window"); + window.setWindowTitle(windowTitle); + window.resize(300, 300); + + QTableWidget *tableWidget = new QTableWidget(&window); + tableWidget->setObjectName("tablewidget"); + tableWidget->setRowCount(3); + tableWidget->setColumnCount(2); + + tableWidget->setItem(0, 0, new QTableWidgetItem("A1")); + tableWidget->setItem(0, 1, new QTableWidgetItem("A2")); + + tableWidget->setItem(1, 0, new QTableWidgetItem("B1")); + tableWidget->setItem(1, 1, new QTableWidgetItem("B2")); + + tableWidget->setItem(2, 0, new QTableWidgetItem("C1")); + tableWidget->setItem(2, 1, new QTableWidgetItem("C2")); + + window.show(); + + { + // Verify that QAInterface works as expected for table view children. + QAInterface tableWidgetInterface = QAccessible::queryAccessibleInterface(tableWidget); + QCOMPARE(tableWidgetInterface.role(), QAccessible::Client); + QCOMPARE(tableWidgetInterface.childCount(), 1); + QAInterface viewPort = tableWidgetInterface.childAt(1); + QCOMPARE(viewPort.childCount(), 4); + QCOMPARE(viewPort.role(), QAccessible::Table); + QVERIFY(viewPort.object() != 0); + const QList<QAInterface> rows = viewPort.children(); + QCOMPARE(rows.count(), 4); + QVERIFY(rows.at(0).object() == 0); + QCOMPARE(rows.at(0).parent().indexOfChild(rows.at(0)), 1); + QCOMPARE(rows.at(1).parent().indexOfChild(rows.at(1)), 2); + QCOMPARE(rows.at(2).parent().indexOfChild(rows.at(2)), 3); + + QAInterface Arow = rows.at(1); + QCOMPARE(Arow.role(), QAccessible::Row); + QAInterface Brow = rows.at(2); + + QVERIFY(Arow.name() == "1"); + QVERIFY(Brow.name() == "2"); + + QVERIFY(Arow == Arow); + QVERIFY(Brow != Arow); + QVERIFY(Arow.isHIView() == false); + QCOMPARE(Arow.childCount(), 3); + QList<QAInterface> Achildren = Arow.children(); + QCOMPARE(Achildren.count(), 3); + QAInterface A1 = Achildren.at(1); + QAInterface A2 = Achildren.at(2); + QCOMPARE(Arow.indexOfChild(A1), 2); + QCOMPARE(Arow.indexOfChild(A2), 3); + QCOMPARE(A1.role(), QAccessible::Cell); + + QList<QAInterface> Bchildren = Brow.children(); + QCOMPARE(Bchildren.count(), 3); + QAInterface B1 = Bchildren.at(1); + QAInterface B2 = Bchildren.at(2); + QVERIFY(B1.parent() == Brow); + QVERIFY(B1.parent() != Arow); + QCOMPARE(Arow.indexOfChild(B1), -1); + + QVERIFY(A1 == A1); + QVERIFY(A1 != A2); + QVERIFY(B1 != A1); + QVERIFY(B1 != A2); + QVERIFY(A1 != Arow); + QVERIFY(A1 != Brow); + QVERIFY(A1 != viewPort); + QVERIFY(A1.isHIView() == false); + + QVERIFY(B1.parent() == Brow); + QVERIFY(A1.parent() == Arow); + B1 = A1; + QVERIFY(B1.parent() == Arow); + } + + const AXUIElementRef applicationElement = AXUIElementCreateApplication(getpid()); + QVERIFY(applicationElement); + const AXUIElementRef windowElement = childByTitle(applicationElement, windowTitle); + QVERIFY(windowElement); + const AXUIElementRef scrollAreaElement = childByRole(windowElement, "AXScrollArea"); + QVERIFY(scrollAreaElement); + const AXUIElementRef tableElement = childByRole(scrollAreaElement, "AXTable"); + QVERIFY(tableElement); + + { + // Verify that QAccessibleHierarchyManager can look up table view children correctly + QAccessibleHierarchyManager *manager = QAccessibleHierarchyManager::instance(); + QAInterface tableInterface = manager->lookup(tableElement); + QVERIFY(tableInterface.isValid()); + QVERIFY(tableInterface.role() == QAccessible::Table); + + QAInterface ArowInterface = tableInterface.childAt(2); + + QVERIFY(ArowInterface.name() == "1"); + QVERIFY(manager->lookup(manager->lookup(ArowInterface)).name() == "1"); + + QCOMPARE(ArowInterface.childCount(), 3); + QAInterface A1Interface = ArowInterface.childAt(2); + + QCOMPARE(A1Interface.value(), QString("A1")); + QVERIFY(value(manager->lookup(A1Interface)).toString() == "A1"); + + QAInterface A2Interface = ArowInterface.childAt(3); + QAElement A2Element = manager->lookup(A2Interface); + QVERIFY(manager->lookup(A2Element).value() == "A2"); + QVERIFY(value(A2Element).toString() == "A2"); + + QAInterface BrowInterface = tableInterface.childAt(3); + QVERIFY(BrowInterface.value() == "2"); + QVERIFY(manager->lookup(manager->lookup(BrowInterface)).value() == "2"); + + QCOMPARE(BrowInterface.childCount(), 3); + QAInterface B1Interface = BrowInterface.childAt(2); + QVERIFY(value(manager->lookup(B1Interface)).toString() == "B1"); + + QAInterface B2Interface = BrowInterface.childAt(3); + QAElement B2Element = manager->lookup(B2Interface); + QVERIFY(manager->lookup(B2Element).value() == "B2"); + QVERIFY(value(B2Element).toString() == "B2"); + } + + + + const AXUIElementRef Arow = childByTitle(tableElement, "1"); + QVERIFY(Arow); + const AXUIElementRef Brow = childByTitle(tableElement, "2"); + QVERIFY(Brow); + const AXUIElementRef Crow = childByTitle(tableElement, "3"); + QVERIFY(Crow); + + QCOMPARE(numChildren(Arow), 3); + const AXUIElementRef A1cell = childByTitle(Arow, "A1"); + QVERIFY(A1cell); + QVERIFY(role(A1cell) == "AXTextField"); + const AXUIElementRef A2cell = childByTitle(Arow, "A2"); + QVERIFY(A2cell); + QVERIFY(equal(::parent(A2cell), Arow)); + + const AXUIElementRef B2cell = childByTitle(Brow, "B2"); + QVERIFY(B2cell); + QVERIFY(equal(::parent(B2cell), Brow)); + + { + QVERIFY(supportsAttribute(tableElement, kAXRowsAttribute)); + const QList<AXUIElementRef> rows = elementListAttribute(tableElement, kAXRowsAttribute); + QCOMPARE(rows.count(), 3); // the header is not a row + QVERIFY(value(rows.at(1)) == "2"); + QVERIFY(value(rows.at(2)) == "3"); + } + + { + QVERIFY(supportsAttribute(tableElement, kAXVisibleRowsAttribute)); + const QList<AXUIElementRef> rows = elementListAttribute(tableElement, kAXVisibleRowsAttribute); + QCOMPARE(rows.count(), 3); + QVERIFY(value(rows.at(1)) == "2"); + } + { + QVERIFY(supportsAttribute(tableElement, kAXSelectedRowsAttribute)); + const QList<AXUIElementRef> rows = elementListAttribute(tableElement, kAXSelectedRowsAttribute); + QCOMPARE(rows.count(), 0); + } + + // test row visibility + { + QTableWidget tableWidget; + tableWidget.setObjectName("tablewidget"); + tableWidget.setRowCount(1000); + tableWidget.setColumnCount(1); + + for (int i =0; i < 1000; ++i) { + tableWidget.setItem(i, 0, new QTableWidgetItem("item")); + } + tableWidget.show(); + + QAInterface tableWidgetInterface = QAccessible::queryAccessibleInterface(&tableWidget); + QAInterface viewPortInterface = tableWidgetInterface.childAt(1); + QCOMPARE(viewPortInterface.childCount(), 1001); + + QVERIFY((viewPortInterface.childAt(2).state() & QAccessible::Invisible) == false); + QVERIFY((viewPortInterface.childAt(2).state() & QAccessible::Offscreen) == false); + + QVERIFY(viewPortInterface.childAt(500).state() & QAccessible::Invisible); +// QVERIFY(viewPortInterface.childAt(500).state() & QAccessible::Offscreen); + tableWidget.hide(); + } + +// printElementInfo(tableElement); +// QTest::qWait(1000000); +} + +void tst_accessibility_mac::testScrollBar() +{ + { + QScrollBar scrollBar; + scrollBar.show(); + + QAInterface scrollBarInterface = QAccessible::queryAccessibleInterface(&scrollBar); + QVERIFY(scrollBarInterface.isValid()); + QCOMPARE(scrollBarInterface.childCount(), 5); + QCOMPARE(scrollBarInterface.indexOfChild(scrollBarInterface.childAt(1)), 1); + QCOMPARE(scrollBarInterface.indexOfChild(scrollBarInterface.childAt(2)), 2); + QCOMPARE(scrollBarInterface.indexOfChild(scrollBarInterface.childAt(5)), 5); + QCOMPARE(scrollBarInterface.indexOfChild(scrollBarInterface), -1); + } + + const AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + testAppAndForm(currentApplication); + const AXUIElementRef form = childByTitle(currentApplication, "Form"); + QVERIFY(form); + const AXUIElementRef scrollBarElement = childByRole(form, "AXScrollBar"); + QVERIFY(scrollBarElement); + QCOMPARE(attribute(scrollBarElement, kAXOrientationAttribute).toString(), QLatin1String("AXVerticalOrientation")); + + { + const AXUIElementRef lineUpElement = childByTitle(scrollBarElement, "Line up"); + QVERIFY(lineUpElement); + QCOMPARE (subrole(lineUpElement), QLatin1String("AXDecrementArrow")); + }{ + const AXUIElementRef lineDownElement = childByTitle(scrollBarElement, "Line down"); + QVERIFY(lineDownElement); + QCOMPARE (subrole(lineDownElement), QLatin1String("AXIncrementArrow")); + }{ + const AXUIElementRef pageUpElement = childByTitle(scrollBarElement, "Page up"); + QVERIFY(pageUpElement); + QCOMPARE (subrole(pageUpElement), QLatin1String("AXDecrementPage")); + }{ + const AXUIElementRef pageDownElement = childByTitle(scrollBarElement, "Page down"); + QVERIFY(pageDownElement); + QCOMPARE (subrole(pageDownElement), QLatin1String("AXIncrementPage")); + }{ + const AXUIElementRef valueIndicatorElement = childByTitle(scrollBarElement, "Position"); + QVERIFY(valueIndicatorElement); + QCOMPARE(value(valueIndicatorElement).toInt(), 50); + } +} + +void tst_accessibility_mac::testSplitter() +{ + const AXUIElementRef currentApplication = AXUIElementCreateApplication(getpid()); + testAppAndForm(currentApplication); + const AXUIElementRef form = childByTitle(currentApplication, "Form"); + QVERIFY(form); + + const AXUIElementRef splitGroupElement = childByRole(form, "AXSplitGroup"); + QVERIFY(splitGroupElement); + + for (int i = 0; i < numChildren(splitGroupElement); ++i) + QVERIFY(child(splitGroupElement, 3)); + + // Visual Order: Foo splitter Bar splitter Baz + QList<AXUIElementRef> splitterList = elementListAttribute(splitGroupElement, kAXSplittersAttribute); + QCOMPARE(splitterList.count(), 2); + foreach (AXUIElementRef splitter, splitterList) { + QCOMPARE(role(splitter), QLatin1String("AXSplitter")); + QVERIFY(supportsAttribute(splitter, kAXPreviousContentsAttribute)); + QVERIFY(supportsAttribute(splitter, kAXNextContentsAttribute)); + QCOMPARE(attribute(splitter, kAXOrientationAttribute).toString(), QLatin1String("AXVerticalOrientation")); + QList<AXUIElementRef> prevList = elementListAttribute(splitter, kAXPreviousContentsAttribute); + QCOMPARE(prevList.count(), 1); + QList<AXUIElementRef> nextList = elementListAttribute(splitter, kAXNextContentsAttribute); + QCOMPARE(nextList.count(), 1); + + // verify order + if (title(prevList.at(0)) == QLatin1String("Foo")) + QCOMPARE(title(nextList.at(0)), QLatin1String("Bar")); + else if (title(prevList.at(0)) == QLatin1String("Bar")) + QCOMPARE(title(nextList.at(0)), QLatin1String("Baz")); + else { + QFAIL("Splitter contents and handles are out of order"); + } + } +} + +void tst_accessibility_mac::testTextEdit() +{ + QWidget window; + const QString windowTitle("window"); + window.setWindowTitle(windowTitle); + window.resize(300, 300); + + QTextEdit *textEdit = new QTextEdit(&window); + textEdit->resize(300, 300); + const QString textLine("this is a line"); + textEdit->setText(textLine); + + window.show(); + + const AXUIElementRef applicationElement = AXUIElementCreateApplication(getpid()); + QVERIFY(applicationElement); + const AXUIElementRef windowElement = childByTitle(applicationElement, windowTitle); + QVERIFY(windowElement); + const AXUIElementRef scrollAreaElement = childByRole(windowElement, "AXScrollArea"); + QVERIFY(scrollAreaElement); + const AXUIElementRef textElement = childByRole(scrollAreaElement, "AXTextField"); + QVERIFY(textElement); + QVERIFY(value(textElement) == textLine); +} + +void testModelLessItemView(QAbstractItemView *itemView, const QByteArray &role) +{ + const QString windowTitle("window"); + itemView->setWindowTitle(windowTitle); + itemView->show(); + + QTest::qWait(100); +#if defined(Q_WS_X11) + qt_x11_wait_for_window_manager(w); +#endif + + QAccessibleInterface *acc = QAccessible::queryAccessibleInterface(itemView); + QVERIFY(acc->isValid()); + QCOMPARE(acc->childCount(), 1); + acc->role(0); + acc->rect(0); + + QAccessibleInterface *accViewport = 0; + int entry = acc->navigate(QAccessible::Child, 1, &accViewport); + QVERIFY(accViewport); + QCOMPARE(entry, 0); + QVERIFY(accViewport->isValid()); + QCOMPARE(accViewport->childCount(), 0); + accViewport->role(0); + accViewport->rect(0); + + delete acc; + delete accViewport; + + const AXUIElementRef applicationElement = AXUIElementCreateApplication(getpid()); + QVERIFY(applicationElement); + const AXUIElementRef windowElement = childByTitle(applicationElement, windowTitle); + QVERIFY(windowElement); + const AXUIElementRef scrollAreaElement = childByRole(windowElement, "AXScrollArea"); + QVERIFY(scrollAreaElement); + const AXUIElementRef tableElement = childByRole(scrollAreaElement, role); + QVERIFY(tableElement); + + delete itemView; +} + +void tst_accessibility_mac::testItemViewsWithoutModel() +{ + testModelLessItemView(new QListView(), "AXList"); + testModelLessItemView(new QTableView(), "AXTable"); +} + +void tst_accessibility_mac::testQAElement() +{ + { + QAElement element; + QVERIFY(element.isValid() == false); + } + + { + QAElement element(0, 0); + QVERIFY(element.isValid() == false); + } + + { + int argc = 0; + char **argv = 0; + QApplication app(argc, argv); + QWidget w; + QAElement element(reinterpret_cast<HIObjectRef>(w.winId()), 0); + QVERIFY(element.isValid() == true); + } + +} + +void tst_accessibility_mac::testQAInterface() +{ + { + QAInterface interface; + QVERIFY(interface.isValid() == false); + } + + { + QAInterface interface(0, 0); + QVERIFY(interface.isValid() == false); + } + + { + int argc = 0; + char **argv = 0; + QApplication app(argc, argv); + + { + QWidget w; + QAInterface element(QAccessible::queryAccessibleInterface(&w), 0); + QVERIFY(element.isValid() == true); + } + { + QWidget w; + QAInterface element(QAccessible::queryAccessibleInterface(&w), 100); + QVERIFY(element.isValid() == false); + } + } +} + +void tst_accessibility_mac::uitests_data() +{ + QTest::addColumn<QString>("uiFilename"); + QTest::addColumn<QString>("testSlot"); + + QTest::newRow("form") << "form.ui" << SLOT(testForm()); + QTest::newRow("buttons") << "buttons.ui" << SLOT(testButtons()); + QTest::newRow("label") << "label.ui" << SLOT(testLabel()); + QTest::newRow("line edit") << "lineedit.ui" << SLOT(testLineEdit()); + QTest::newRow("groups") << "groups.ui" << SLOT(testGroups()); + QTest::newRow("tabs") << "tabs.ui" << SLOT(testTabWidget()); + QTest::newRow("combobox") << "combobox.ui" << SLOT(testComboBox()); + QTest::newRow("scrollbar") << "scrollbar.ui" << SLOT(testScrollBar()); + QTest::newRow("splitters") << "splitters.ui" << SLOT(testSplitter()); +} + +void tst_accessibility_mac::uitests() +{ + QFETCH(QString, uiFilename); + QFETCH(QString, testSlot); + + // The Accessibility interface must be enabled to run this test. + if (!AXAPIEnabled()) + QSKIP("Accessibility not enabled. Check \"Enable access for assistive devices\" in the system preferences -> universal access to run this test.", SkipAll); + + int argc = 0; + char **argv = 0; + QApplication app(argc, argv); + + // Create and display form. + QUiLoader loader; + QFile file(":" + uiFilename); + QVERIFY(file.exists()); + file.open(QFile::ReadOnly); + QWidget *window = loader.load(&file, 0); + QVERIFY(window); + file.close(); + window->show(); + + QTimer::singleShot(50, this, qPrintable(testSlot)); + // Quit when returning to the main event loop after running tests. + QTimer::singleShot(200, &app, SLOT(quit())); + app.exec(); + delete window; +} + +void tst_accessibility_mac::tests_data() +{ + QTest::addColumn<QString>("testSlot"); + QTest::newRow("deleteWidget") << SLOT(testDeleteWidget()); + QTest::newRow("deleteWidgets") << SLOT(testDeleteWidgets()); + QTest::newRow("multipleWindows") << SLOT(testMultipleWindows()); + QTest::newRow("hiddenWidgets") << SLOT(testHiddenWidgets()); + QTest::newRow("actions") << SLOT(testActions()); + QTest::newRow("changeState") << SLOT(testChangeState()); + QTest::newRow("slider") << SLOT(testSlider()); + QTest::newRow("scrollArea") << SLOT(testScrollArea()); + QTest::newRow("listView") << SLOT(testListView()); + QTest::newRow("tableView") << SLOT(testTableView()); + QTest::newRow("textEdit") << SLOT(testTextEdit()); + QTest::newRow("ItemViews without model") << SLOT(testItemViewsWithoutModel()); + QTest::newRow("tabbar") << SLOT(testTabBar()); +} + +void tst_accessibility_mac::tests() +{ + QFETCH(QString, testSlot); + runTest(testSlot); +} + +/* + Tests show that querying the accessibility interface directly does not work. (I get a + kAXErrorAPIDisabled error, indicating that the accessible API is disabled, which it isn't.) + To work around this, we run the tests in a callback slot called from the main event loop. +*/ +void tst_accessibility_mac::runTest(const QString &testSlot) +{ + // The Accessibility interface must be enabled to run this test. + if (!AXAPIEnabled()) + QSKIP("Accessibility not enabled. Check \"Enable access for assistive devices\" in the system preferences -> universal access to run this test.", SkipAll); + + int argc = 0; + char **argv = 0; + QApplication app(argc, argv); + + QTimer::singleShot(50, this, qPrintable(testSlot)); + // Quit when returning to the main event loop after running tests. + QTimer::singleShot(200, &app, SLOT(quit())); + app.exec(); + +} + +QTEST_APPLESS_MAIN(tst_accessibility_mac) + +#else // defined(Q_WS_MAC) && !defined (QT_MAC_USE_COCOA) + +QTEST_NOOP_MAIN + +#endif + +#include "tst_qaccessibility_mac.moc" + + |