tests/auto/qaccessibility/tst_qaccessibility.cpp
changeset 0 1918ee327afb
child 3 41300fa6a67c
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/tests/auto/qaccessibility/tst_qaccessibility.cpp	Mon Jan 11 14:00:40 2010 +0000
@@ -0,0 +1,4090 @@
+/****************************************************************************
+**
+** Copyright (C) 2009 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$
+** 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 Technology Preview License Agreement accompanying
+** this package.
+**
+** 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.1, included in the file LGPL_EXCEPTION.txt in this package.
+**
+** If you have questions regarding the use of this file, please contact
+** Nokia at qt-info@nokia.com.
+**
+**
+**
+**
+**
+**
+**
+**
+** $QT_END_LICENSE$
+**
+****************************************************************************/
+
+
+#ifdef QT3_SUPPORT
+#include <Qt3Support/Qt3Support>
+#endif
+#include <QtTest/QtTest>
+#ifndef Q_OS_WINCE
+#include "../../shared/util.h"
+#include <QtGui>
+#include <math.h>
+
+
+#include "QtTest/qtestaccessible.h"
+
+#if defined(Q_OS_WINCE)
+extern "C" bool SystemParametersInfo(UINT uiAction, UINT uiParam, PVOID pvParam, UINT fWinIni);
+#define SPI_GETPLATFORMTYPE 257
+inline bool IsValidCEPlatform() {
+    wchar_t tszPlatform[64];
+    if (SystemParametersInfo(SPI_GETPLATFORMTYPE, sizeof(tszPlatform) / sizeof(*tszPlatform), tszPlatform, 0)) {
+        QString platform = QString::fromWCharArray(tszPlatform);
+        if ((platform == QLatin1String("PocketPC")) || (platform == QLatin1String("Smartphone")))
+            return false;
+    }
+    return true;
+}
+#endif
+
+static inline bool verifyChild(QWidget *child, QAccessibleInterface *interface,
+                               int index, const QRect &domain)
+{
+    if (!child) {
+        qWarning("tst_QAccessibility::verifyChild: null pointer to child.");
+        return false;
+    }
+
+    if (!interface) {
+        qWarning("tst_QAccessibility::verifyChild: null pointer to interface.");
+        return false;
+    }
+
+    // QAccessibleInterface::childAt():
+    // Calculate global child position and check that the interface
+    // returns the correct index for that position.
+    QPoint globalChildPos = child->mapToGlobal(QPoint(0, 0));
+    int indexFromChildAt = interface->childAt(globalChildPos.x(), globalChildPos.y());
+    if (indexFromChildAt != index) {
+        qWarning("tst_QAccessibility::verifyChild (childAt()):");
+        qWarning() << "Expected:" << index;
+        qWarning() << "Actual:  " << indexFromChildAt;
+        return false;
+    }
+
+    // QAccessibleInterface::rect():
+    // Calculate global child geometry and check that the interface
+    // returns a QRect which is equal to the calculated QRect.
+    const QRect expectedGlobalRect = QRect(globalChildPos, child->size());
+    const QRect rectFromInterface = interface->rect(index);
+    if (expectedGlobalRect != rectFromInterface) {
+        qWarning("tst_QAccessibility::verifyChild (rect()):");
+        qWarning() << "Expected:" << expectedGlobalRect;
+        qWarning() << "Actual:  " << rectFromInterface;
+        return false;
+    }
+
+    // Verify that the child is within its domain.
+    if (!domain.contains(rectFromInterface)) {
+        qWarning("tst_QAccessibility::verifyChild: Child is not within its domain.");
+        return false;
+    }
+
+    // Verify that we get a valid QAccessibleInterface for the child.
+    QAccessibleInterface *childInterface = QAccessible::queryAccessibleInterface(child);
+    if (!childInterface) {
+        qWarning("tst_QAccessibility::verifyChild: Failed to retrieve interface for child.");
+        return false;
+    }
+
+    // QAccessibleInterface::indexOfChild():
+    // Verify that indexOfChild() returns an index equal to the index passed by,
+    // or -1 if child is "Self" (index == 0).
+    int indexFromIndexOfChild = interface->indexOfChild(childInterface);
+    delete childInterface;
+    int expectedIndex = index == 0 ? -1 : index;
+    if (indexFromIndexOfChild != expectedIndex) {
+        qWarning("tst_QAccessibility::verifyChild (indexOfChild()):");
+        qWarning() << "Expected:" << expectedIndex;
+        qWarning() << "Actual:  " << indexFromIndexOfChild;
+        return false;
+    }
+
+    // Navigate to child, compare its object and role with the interface from queryAccessibleInterface(child).
+    {
+        QAccessibleInterface *navigatedChildInterface = 0;
+        const int status = interface->navigate(QAccessible::Child, index, &navigatedChildInterface);
+        // We are navigating to a separate widget/interface, so status should be 0.
+        if (status != 0)
+            return false;
+
+        if (navigatedChildInterface == 0)
+            return false;
+        delete navigatedChildInterface;
+    }
+
+    return true;
+}
+
+static inline int indexOfChild(QAccessibleInterface *parentInterface, QWidget *childWidget)
+{
+    if (!parentInterface || !childWidget)
+        return -1;
+    QAccessibleInterface *childInterface = QAccessibleInterface::queryAccessibleInterface(childWidget);
+    if (!childInterface)
+        return -1;
+    int index = parentInterface->indexOfChild(childInterface);
+    delete childInterface;
+    return index;
+}
+
+#define EXPECT(cond) \
+    do { \
+        if (!errorAt && !(cond)) { \
+            errorAt = __LINE__; \
+            qWarning("level: %d, middle: %d, role: %d (%s)", treelevel, middle, iface->role(0), #cond); \
+        } \
+    } while (0)
+
+static int verifyHierarchy(QAccessibleInterface *iface)
+{
+    int errorAt = 0;
+    int entry = 0;
+    static int treelevel = 0;   // for error diagnostics
+    QAccessibleInterface *middleChild, *if2, *if3;
+    middleChild = 0;
+    ++treelevel;
+    int middle = iface->childCount()/2 + 1;
+    if (iface->childCount() >= 2) {
+        entry = iface->navigate(QAccessible::Child, middle, &middleChild);
+    }
+    for (int i = 0; i < iface->childCount() && !errorAt; ++i) {
+        entry = iface->navigate(QAccessible::Child, i + 1, &if2);
+        if (entry == 0) {
+            // navigate Ancestor...
+            QAccessibleInterface *parent = 0;
+            entry = if2->navigate(QAccessible::Ancestor, 1, &parent);
+            EXPECT(entry == 0 && iface->object() == parent->object());
+            delete parent;
+
+            // navigate Sibling...
+            if (middleChild) {
+                entry = if2->navigate(QAccessible::Sibling, middle, &if3);
+                EXPECT(entry == 0 && if3->object() == middleChild->object());
+                delete if3;
+                EXPECT(iface->indexOfChild(middleChild) == middle);
+            }
+
+            // verify children...
+            if (!errorAt)
+                errorAt = verifyHierarchy(if2);
+            delete if2;
+        } else {
+            // leaf nodes
+        }
+    }
+    delete middleChild;
+
+    --treelevel;
+    return errorAt;
+}
+
+
+//TESTED_FILES=
+
+class tst_QAccessibility : public QObject
+{
+    Q_OBJECT
+public:
+    tst_QAccessibility();
+    virtual ~tst_QAccessibility();
+
+public slots:
+    void initTestCase();
+    void cleanupTestCase();
+    void init();
+    void cleanup();
+private slots:
+    void eventTest();
+    void customWidget();
+    void deletedWidget();
+
+    void childCount();
+    void childAt(); // also indexOfChild
+    void relationTo();
+    void navigateGeometric();
+    void navigateHierarchy();
+    void navigateSlider();
+    void navigateCovered();
+    void navigateControllers();
+    void navigateLabels();
+    void text();
+    void setText();
+    void hideShowTest();
+
+    void userActionCount();
+    void actionText();
+    void doAction();
+
+    void buttonTest();
+    void sliderTest();
+    void scrollBarTest();
+    void tabTest();
+    void menuTest();
+    void spinBoxTest();
+    void doubleSpinBoxTest();
+    void textEditTest();
+    void textBrowserTest();
+    void listViewTest();
+    void mdiAreaTest();
+    void mdiSubWindowTest();
+    void lineEditTest();
+    void workspaceTest();
+    void dialogButtonBoxTest();
+    void dialTest();
+    void rubberBandTest();
+    void abstractScrollAreaTest();
+    void scrollAreaTest();
+    void tableWidgetTest();
+    void tableViewTest();
+    void calendarWidgetTest();
+    void dockWidgetTest();
+    void pushButtonTest();
+    void comboBoxTest();
+    void accessibleName();
+    void treeWidgetTest();
+    void labelTest();
+    void accelerators();
+
+private:
+    QWidget *createGUI();
+};
+
+const double Q_PI = 3.14159265358979323846;
+
+QString eventName(const int ev)
+{
+    switch(ev) {
+    case 0x0001: return "SoundPlayed";
+    case 0x0002: return "Alert";
+    case 0x0003: return "ForegroundChanged";
+    case 0x0004: return "MenuStart";
+    case 0x0005: return "MenuEnd";
+    case 0x0006: return "PopupMenuStart";
+    case 0x0007: return "PopupMenuEnd";
+    case 0x000C: return "ContextHelpStart";
+    case 0x000D: return "ContextHelpEnd";
+    case 0x000E: return "DragDropStart";
+    case 0x000F: return "DragDropEnd";
+    case 0x0010: return "DialogStart";
+    case 0x0011: return "DialogEnd";
+    case 0x0012: return "ScrollingStart";
+    case 0x0013: return "ScrollingEnd";
+    case 0x0018: return "MenuCommand";
+    case 0x8000: return "ObjectCreated";
+    case 0x8001: return "ObjectDestroyed";
+    case 0x8002: return "ObjectShow";
+    case 0x8003: return "ObjectHide";
+    case 0x8004: return "ObjectReorder";
+    case 0x8005: return "Focus";
+    case 0x8006: return "Selection";
+    case 0x8007: return "SelectionAdd";
+    case 0x8008: return "SelectionRemove";
+    case 0x8009: return "SelectionWithin";
+    case 0x800A: return "StateChanged";
+    case 0x800B: return "LocationChanged";
+    case 0x800C: return "NameChanged";
+    case 0x800D: return "DescriptionChanged";
+    case 0x800E: return "ValueChanged";
+    case 0x800F: return "ParentChanged";
+    case 0x80A0: return "HelpChanged";
+    case 0x80B0: return "DefaultActionChanged";
+    case 0x80C0: return "AcceleratorChanged";
+    default: return "Unknown Event";
+    }
+}
+
+static QString stateNames(int state)
+{
+    QString stateString;
+    if (state == 0x00000000) stateString += " Normal";
+    if (state & 0x00000001) stateString += " Unavailable";
+    if (state & 0x00000002) stateString += " Selected";
+    if (state & 0x00000004) stateString += " Focused";
+    if (state & 0x00000008) stateString += " Pressed";
+    if (state & 0x00000010) stateString += " Checked";
+    if (state & 0x00000020) stateString += " Mixed";
+    if (state & 0x00000040) stateString += " ReadOnly";
+    if (state & 0x00000080) stateString += " HotTracked";
+    if (state & 0x00000100) stateString += " DefaultButton";
+    if (state & 0x00000200) stateString += " Expanded";
+    if (state & 0x00000400) stateString += " Collapsed";
+    if (state & 0x00000800) stateString += " Busy";
+    if (state & 0x00001000) stateString += " Floating";
+    if (state & 0x00002000) stateString += " Marqueed";
+    if (state & 0x00004000) stateString += " Animated";
+    if (state & 0x00008000) stateString += " Invisible";
+    if (state & 0x00010000) stateString += " Offscreen";
+    if (state & 0x00020000) stateString += " Sizeable";
+    if (state & 0x00040000) stateString += " Moveable";
+    if (state & 0x00080000) stateString += " SelfVoicing";
+    if (state & 0x00100000) stateString += " Focusable";
+    if (state & 0x00200000) stateString += " Selectable";
+    if (state & 0x00400000) stateString += " Linked";
+    if (state & 0x00800000) stateString += " Traversed";
+    if (state & 0x01000000) stateString += " MultiSelectable";
+    if (state & 0x02000000) stateString += " ExtSelectable";
+    if (state & 0x04000000) stateString += " AlertLow";
+    if (state & 0x08000000) stateString += " AlertMedium";
+    if (state & 0x10000000) stateString += " AlertHigh";
+    if (state & 0x20000000) stateString += " Protected";
+    if (state & 0x3fffffff) stateString += " Valid";
+
+    if (stateString.isEmpty())
+        stateString = "Unknown state " + QString::number(state);
+
+    return stateString;
+}
+
+QAccessible::State state(QWidget * const widget)
+{
+    QAccessibleInterface *iface = QAccessible::queryAccessibleInterface(widget);
+    Q_ASSERT(iface);
+    QAccessible::State state = iface->state(0);
+    delete iface;
+    return state;
+}
+
+void printState(QWidget * const widget)
+{
+    qDebug() << "State for" << widget->metaObject()->className() << stateNames(state(widget));
+}
+
+void printState(QAccessibleInterface * const iface, const int child = 0)
+{
+    qDebug() << "State for" << iface->object()->metaObject()->className() << "child" << child
+             <<  iface->text(QAccessible::Name, child) << stateNames(iface->state(child));
+}
+
+
+class QtTestAccessibleWidget: public QWidget
+{
+    Q_OBJECT
+public:
+    QtTestAccessibleWidget(QWidget *parent, const char *name): QWidget(parent)
+    {
+        setObjectName(name);
+        QPalette pal;
+        pal.setColor(backgroundRole(), Qt::black);//black is beautiful
+        setPalette(pal);
+        setFixedSize(5, 5);
+    }
+};
+
+#ifdef QTEST_ACCESSIBILITY
+class QtTestAccessibleWidgetIface: public QAccessibleWidget
+{
+public:
+    QtTestAccessibleWidgetIface(QtTestAccessibleWidget *w): QAccessibleWidget(w) {}
+    QString text(Text t, int control) const
+    {
+        if (t == Help)
+            return QString::fromLatin1("Help yourself");
+        return QAccessibleWidget::text(t, control);
+    }
+    static QAccessibleInterface *ifaceFactory(const QString &key, QObject *o)
+    {
+        if (key == "QtTestAccessibleWidget")
+            return new QtTestAccessibleWidgetIface(static_cast<QtTestAccessibleWidget*>(o));
+        return 0;
+    }
+};
+#endif
+
+tst_QAccessibility::tst_QAccessibility()
+{
+}
+
+tst_QAccessibility::~tst_QAccessibility()
+{
+}
+
+void tst_QAccessibility::initTestCase()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QTestAccessibility::initialize();
+    QAccessible::installFactory(QtTestAccessibleWidgetIface::ifaceFactory);
+#endif
+}
+
+void tst_QAccessibility::cleanupTestCase()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QTestAccessibility::cleanup();
+#endif
+}
+
+void tst_QAccessibility::init()
+{
+    QTestAccessibility::clearEvents();
+}
+
+void tst_QAccessibility::cleanup()
+{
+#ifdef QTEST_ACCESSIBILITY
+    const EventList list = QTestAccessibility::events();
+    if (!list.isEmpty()) {
+        qWarning("%d accessibility event(s) were not handled in testfunction '%s':", list.count(),
+                 QString(QTest::currentTestFunction()).toAscii().constData());
+        for (int i = 0; i < list.count(); ++i)
+            qWarning(" %d: Object: %p Event: '%s' (%d) Child: %d", i + 1, list.at(i).object,
+                     eventName(list.at(i).event).toAscii().constData(), list.at(i).event, list.at(i).child);
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::eventTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QPushButton* button = new QPushButton(0);
+    button->setObjectName(QString("Olaf"));
+
+    button->show();
+    QVERIFY_EVENT(button, 0, QAccessible::ObjectShow);
+    button->setFocus(Qt::MouseFocusReason);
+    QTestAccessibility::clearEvents();
+    QTest::mouseClick(button, Qt::LeftButton, 0, QPoint(button->width()-7,button->height()-5));
+    QVERIFY_EVENT(button, 0, QAccessible::StateChanged);
+    QVERIFY_EVENT(button, 0, QAccessible::StateChanged);
+
+    button->hide();
+    QVERIFY_EVENT(button, 0, QAccessible::ObjectHide);
+
+    delete button;
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::customWidget()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QtTestAccessibleWidget* widget = new QtTestAccessibleWidget(0, "Heinz");
+
+    QAccessibleInterface *iface = QAccessible::queryAccessibleInterface(widget);
+    QVERIFY(iface != 0);
+    QVERIFY(iface->isValid());
+    QCOMPARE(iface->object(), (QObject*)widget);
+    QCOMPARE(iface->object()->objectName(), QString("Heinz"));
+    QCOMPARE(iface->text(QAccessible::Help, 0), QString("Help yourself"));
+
+    delete iface;
+    delete widget;
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::deletedWidget()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QtTestAccessibleWidget *widget = new QtTestAccessibleWidget(0, "Ralf");
+    QAccessibleInterface *iface = QAccessible::queryAccessibleInterface(widget);
+    QVERIFY(iface != 0);
+    QVERIFY(iface->isValid());
+    QCOMPARE(iface->object(), (QObject*)widget);
+
+    delete widget;
+    widget = 0;
+    QVERIFY(!iface->isValid());
+    delete iface;
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+QWidget *tst_QAccessibility::createGUI()
+{
+#if !defined(QT3_SUPPORT)
+    qWarning( "Should never get here without Qt3Support");
+    return 0;
+#else
+#  ifdef QTEST_ACCESSIBILITY
+    QWidget *toplevel = new QWidget(0, Qt::X11BypassWindowManagerHint);
+    QGridLayout *grid = new QGridLayout(toplevel, 2, 2);
+
+    // topLeft - hierarchies
+    Q3VBox *topLeft = new Q3VBox(toplevel, "topLeft");
+    topLeft->setSpacing(2);
+    grid->addWidget(topLeft, 0, 0);
+
+    Q3VButtonGroup *group1 = new Q3VButtonGroup("Title1:", topLeft, "group1");
+    /*QPushButton *pb1 = */ new QPushButton("Button&1", group1, "pb1");
+    Q3VButtonGroup *group2 = new Q3VButtonGroup("Title2:", topLeft, "group2");
+    /*QPushButton *pb2 = */ new QPushButton("Button2", group2, "pb2");
+
+    Q3WidgetStack *stack = new Q3WidgetStack(topLeft, "stack");
+    QLabel *page1 = new QLabel("Page 1", stack, "page1");
+    stack->addWidget(page1);
+    QLabel *page2 = new QLabel("Page 2", stack, "page2");
+    stack->addWidget(page2);
+    QLabel *page3 = new QLabel("Page 3", stack, "page3");
+    stack->addWidget(page3);
+
+    // topRight - controlling
+    Q3VBox *topRight= new Q3VBox(toplevel, "topRight");
+    grid->addWidget(topRight, 0, 1);
+
+    QPushButton *pbOk = new QPushButton("Ok", topRight, "pbOk" );
+    pbOk->setDefault(TRUE);
+    QSlider *slider = new QSlider(Qt::Horizontal, topRight, "slider");
+    QLCDNumber *sliderLcd = new QLCDNumber(topRight, "sliderLcd");
+    QSpinBox *spinBox = new QSpinBox(topRight, "spinBox");
+
+    connect(pbOk, SIGNAL(clicked()), toplevel, SLOT(close()) );
+    connect(slider, SIGNAL(valueChanged(int)), sliderLcd, SLOT(display(int)));
+    connect(spinBox, SIGNAL(valueChanged(int)), slider, SLOT(setValue(int)));
+
+    spinBox->setValue(50);
+
+    // bottomLeft - labeling and controlling
+    Q3HBox *bottomLeft = new Q3HBox(toplevel, "bottomLeft");
+    grid->addWidget(bottomLeft, 1, 0);
+
+    QLabel *label = new QLabel("This is a &lineedit:", bottomLeft, "label");
+    QLineEdit *lineedit = new QLineEdit(bottomLeft, "lineedit");
+    label->setBuddy(lineedit);
+    QLabel *label2 = new QLabel(bottomLeft, "label2");
+
+    connect(lineedit, SIGNAL(textChanged(const QString&)), label2, SLOT(setText(const QString&)));
+
+    Q3VButtonGroup *radiogroup = new Q3VButtonGroup("Exclusive &choices:", bottomLeft, "radiogroup");
+    QLineEdit *frequency = new QLineEdit(radiogroup, "frequency");
+    frequency->setText("100 Mhz");
+    QRadioButton *radioAM = new QRadioButton("&AM", radiogroup, "radioAM");
+    /* QRadioButton *radioFM = */ new QRadioButton("&FM", radiogroup, "radioFM");
+    /* QRadioButton *radioSW = */ new QRadioButton("&Shortwave", radiogroup, "radioSW");
+
+    // bottomRight - ### empty
+    Q3HBox *bottomRight = new Q3HBox(toplevel, "bottomRight");
+    grid->addWidget(bottomRight, 1, 1);
+
+    toplevel->adjustSize(); // sends layout and child events
+
+    // some tooltips
+    QToolTip::add(label, "A label");
+    QToolTip::add(lineedit, "A line edit");
+    // some whatsthis
+    QWhatsThis::add(label, "A label displays static text");
+    QWhatsThis::add(frequency, "You can enter a single line of text here");
+
+    radioAM->setFocus();
+    QTestAccessibility::clearEvents();
+    return toplevel;
+#  else
+    Q_ASSERT(0); // this function cannot be called without accessibility support
+    return 0;
+#  endif
+#endif // !QT3_SUPPORT
+}
+
+void tst_QAccessibility::childAt()
+{
+#if !defined(QT3_SUPPORT)
+    QSKIP("This test needs Qt3Support", SkipAll);
+#else
+#ifdef QTEST_ACCESSIBILITY
+    QWidget *toplevel = createGUI();
+    QAccessibleInterface *acc_toplevel = QAccessible::queryAccessibleInterface(toplevel);
+    QVERIFY(acc_toplevel);
+    // this is necessary to have the layout setup correctly
+    toplevel->show();
+
+    QObjectList children = toplevel->queryList("QWidget", 0, 0, 0);
+    for (int c = 1; c <= children.count(); ++c) {
+        QWidget *child = qobject_cast<QWidget*>(children.at(c-1));
+        QAccessibleInterface *acc_child = QAccessible::queryAccessibleInterface(child);
+        QVERIFY(acc_child);
+        QCOMPARE(acc_child->relationTo(0, acc_toplevel, 0) & QAccessible::HierarchyMask, QAccessible::Child);
+
+        QPoint center(child->mapToGlobal(child->rect().center()));
+        QRect childRect(child->geometry());
+        childRect.moveCenter(center);
+
+        QCOMPARE(acc_child->rect(0), childRect);
+        QCOMPARE(acc_toplevel->childAt(childRect.center().x(), childRect.center().y()), c);
+        QCOMPARE(acc_toplevel->childAt(childRect.left(), childRect.top()), c);
+        QCOMPARE(acc_toplevel->childAt(childRect.left(), childRect.bottom()), c);
+        QCOMPARE(acc_toplevel->childAt(childRect.right(), childRect.top()), c);
+        QCOMPARE(acc_toplevel->childAt(childRect.right(), childRect.bottom()), c);
+
+        QCOMPARE(acc_toplevel->indexOfChild(acc_child), c);
+        delete acc_child;
+    }
+
+    delete acc_toplevel;
+    delete toplevel;
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+#endif // !QT3_SUPPORT
+}
+
+void tst_QAccessibility::childCount()
+{
+#if !defined(QT3_SUPPORT)
+    QSKIP("This test needs Qt3Support", SkipAll);
+#else
+#ifdef QTEST_ACCESSIBILITY
+    QWidget *toplevel = createGUI();
+    QObject *topLeft = toplevel->child("topLeft");
+    QObject *topRight = toplevel->child("topRight");
+    QObject *bottomLeft = toplevel->child("bottomLeft");
+    QObject *bottomRight = toplevel->child("bottomRight");
+
+    QAccessibleInterface* acc_toplevel = QAccessible::queryAccessibleInterface(toplevel);
+    QAccessibleInterface* acc_topLeft = QAccessible::queryAccessibleInterface(topLeft);
+    QAccessibleInterface* acc_topRight = QAccessible::queryAccessibleInterface(topRight);
+    QAccessibleInterface* acc_bottomLeft = QAccessible::queryAccessibleInterface(bottomLeft);
+    QAccessibleInterface* acc_bottomRight = QAccessible::queryAccessibleInterface(bottomRight);
+
+    QVERIFY(acc_toplevel);
+    QVERIFY(acc_topLeft);
+    QVERIFY(acc_topRight);
+    QVERIFY(acc_bottomLeft);
+    QVERIFY(acc_bottomRight);
+
+    toplevel->show();
+    QCOMPARE(acc_toplevel->childCount(), toplevel->queryList("QWidget", 0, 0, 0).count());
+    QCOMPARE(acc_topLeft->childCount(), topLeft->queryList("QWidget", 0, 0, 0).count());
+    QCOMPARE(acc_topRight->childCount(), topRight->queryList("QWidget", 0, 0, 0).count());
+    QCOMPARE(acc_bottomLeft->childCount(), bottomLeft->queryList("QWidget", 0, 0, 0).count());
+    QCOMPARE(acc_bottomRight->childCount(), bottomRight->queryList("QWidget", 0, 0, 0).count());
+
+    delete acc_toplevel;
+    delete acc_topLeft;
+    delete acc_topRight;
+    delete acc_bottomLeft;
+    delete acc_bottomRight;
+    delete toplevel;
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+#endif // !QT3_SUPPORT
+}
+
+void tst_QAccessibility::relationTo()
+{
+#if !defined(QT3_SUPPORT)
+    QSKIP("This test needs Qt3Support", SkipAll);
+#else
+#ifdef QTEST_ACCESSIBILITY
+    QWidget *toplevel = createGUI();
+    toplevel->resize(400,300);
+    QObject *topLeft = toplevel->child("topLeft");
+    QObject *topRight = toplevel->child("topRight");
+    QObject *bottomLeft = toplevel->child("bottomLeft");
+    QObject *bottomRight = toplevel->child("bottomRight");
+
+    toplevel->show();
+
+    QAccessibleInterface *acc_toplevel = QAccessible::queryAccessibleInterface(toplevel);
+
+    QAccessibleInterface *acc_topLeft = QAccessible::queryAccessibleInterface(topLeft);
+    QAccessibleInterface *acc_group1 = QAccessible::queryAccessibleInterface(topLeft->child("group1"));
+    QVERIFY(topLeft->child("group1"));
+    QAccessibleInterface *acc_pb1 = QAccessible::queryAccessibleInterface(topLeft->child("group1")->child("pb1"));
+    QAccessibleInterface *acc_group2 = QAccessible::queryAccessibleInterface(topLeft->child("group2"));
+    QAccessibleInterface *acc_pb2 = 0;
+    QAccessibleInterface *acc_stack = QAccessible::queryAccessibleInterface(topLeft->child("stack"));
+    QAccessibleInterface *acc_page1 = QAccessible::queryAccessibleInterface(topLeft->child("stack")->child("page1"));
+    QAccessibleInterface *acc_page2 = QAccessible::queryAccessibleInterface(topLeft->child("stack")->child("page2"));
+    QAccessibleInterface *acc_page3 = QAccessible::queryAccessibleInterface(topLeft->child("stack")->child("page3"));
+    QAccessibleInterface *acc_topRight = QAccessible::queryAccessibleInterface(topRight);
+    QAccessibleInterface *acc_pbOk = QAccessible::queryAccessibleInterface(topRight->child("pbOk"));
+    QAccessibleInterface *acc_slider = QAccessible::queryAccessibleInterface(topRight->child("slider"));
+    QAccessibleInterface *acc_spinBox = QAccessible::queryAccessibleInterface(topRight->child("spinBox"));
+    QAccessibleInterface *acc_sliderLcd = QAccessible::queryAccessibleInterface(topRight->child("sliderLcd"));
+
+    QAccessibleInterface *acc_bottomLeft = QAccessible::queryAccessibleInterface(bottomLeft);
+    QAccessibleInterface *acc_label = QAccessible::queryAccessibleInterface(bottomLeft->child("label"));
+    QAccessibleInterface *acc_lineedit = QAccessible::queryAccessibleInterface(bottomLeft->child("lineedit"));
+    QAccessibleInterface *acc_label2 = QAccessible::queryAccessibleInterface(bottomLeft->child("label2"));
+    QAccessibleInterface *acc_radiogroup = QAccessible::queryAccessibleInterface(bottomLeft->child("radiogroup"));
+    QAccessibleInterface *acc_radioAM = QAccessible::queryAccessibleInterface(bottomLeft->child("radiogroup")->child("radioAM"));
+    QAccessibleInterface *acc_radioFM = QAccessible::queryAccessibleInterface(bottomLeft->child("radiogroup")->child("radioFM"));
+    QAccessibleInterface *acc_radioSW = QAccessible::queryAccessibleInterface(bottomLeft->child("radiogroup")->child("radioSW"));
+    QAccessibleInterface *acc_frequency = QAccessible::queryAccessibleInterface(bottomLeft->child("radiogroup")->child("frequency"));
+
+    QAccessibleInterface *acc_bottomRight = QAccessible::queryAccessibleInterface(bottomRight);
+
+    QVERIFY(acc_toplevel);
+    QVERIFY(acc_topLeft);
+    QVERIFY(acc_topRight);
+    QVERIFY(acc_bottomLeft);
+    QVERIFY(acc_bottomRight);
+    QVERIFY(acc_group1);
+    QVERIFY(acc_group2);
+    QVERIFY(acc_stack);
+    QVERIFY(acc_page1);
+    QVERIFY(acc_page2);
+    QVERIFY(acc_page3);
+    QVERIFY(acc_pbOk);
+    QVERIFY(acc_slider);
+    QVERIFY(acc_spinBox);
+    QVERIFY(acc_sliderLcd);
+    QVERIFY(acc_label);
+    QVERIFY(acc_lineedit);
+    QVERIFY(acc_radiogroup);
+    QVERIFY(acc_radioAM);
+    QVERIFY(acc_radioFM);
+    QVERIFY(acc_radioSW);
+    QVERIFY(acc_frequency);
+
+    // hierachy relations
+    QCOMPARE(acc_toplevel->relationTo(0, acc_toplevel, 0) & QAccessible::HierarchyMask,
+        QAccessible::Self);
+    QCOMPARE(acc_toplevel->relationTo(1, acc_toplevel, 0) & QAccessible::HierarchyMask,
+        QAccessible::Child);
+    QCOMPARE(acc_toplevel->relationTo(0, acc_toplevel, 1) & QAccessible::HierarchyMask,
+        QAccessible::Ancestor);
+
+    QCOMPARE(acc_toplevel->relationTo(0, acc_topLeft, 0) & QAccessible::HierarchyMask,
+        QAccessible::Ancestor);
+    QCOMPARE(acc_toplevel->relationTo(0, acc_topRight, 0) & QAccessible::HierarchyMask,
+        QAccessible::Ancestor);
+    QCOMPARE(acc_toplevel->relationTo(0, acc_bottomLeft, 0) & QAccessible::HierarchyMask,
+        QAccessible::Ancestor);
+    QCOMPARE(acc_toplevel->relationTo(0, acc_bottomRight, 0) & QAccessible::HierarchyMask,
+        QAccessible::Ancestor);
+
+    QCOMPARE(acc_toplevel->relationTo(0, acc_group1, 0) & QAccessible::HierarchyMask,
+        QAccessible::Ancestor);
+    QCOMPARE(acc_toplevel->relationTo(0, acc_page1, 0) & QAccessible::HierarchyMask,
+        QAccessible::Ancestor);
+    QCOMPARE(acc_group1->relationTo(0, acc_toplevel, 0) & QAccessible::HierarchyMask,
+        QAccessible::Descendent);
+    QCOMPARE(acc_stack->relationTo(0, acc_toplevel, 0) & QAccessible::HierarchyMask,
+        QAccessible::Descendent);
+    QCOMPARE(acc_page1->relationTo(0, acc_stack, 0) & QAccessible::HierarchyMask,
+        QAccessible::Child);
+    QCOMPARE(acc_page1->relationTo(0, acc_toplevel, 0) & QAccessible::HierarchyMask,
+        QAccessible::Descendent);
+
+    QCOMPARE(acc_topLeft->relationTo(0, acc_toplevel, 0) & QAccessible::HierarchyMask,
+        QAccessible::Child);
+    QCOMPARE(acc_topRight->relationTo(0, acc_toplevel, 0) & QAccessible::HierarchyMask,
+        QAccessible::Child);
+    QCOMPARE(acc_bottomLeft->relationTo(0, acc_toplevel, 0) & QAccessible::HierarchyMask,
+        QAccessible::Child);
+    QCOMPARE(acc_bottomRight->relationTo(0, acc_toplevel, 0) & QAccessible::HierarchyMask,
+        QAccessible::Child);
+
+    QCOMPARE(acc_topLeft->relationTo(0, acc_topRight, 0) & QAccessible::HierarchyMask,
+        QAccessible::Sibling);
+    QCOMPARE(acc_topLeft->relationTo(0, acc_bottomLeft, 0) & QAccessible::HierarchyMask,
+        QAccessible::Sibling);
+    QCOMPARE(acc_topLeft->relationTo(0, acc_bottomRight, 0) & QAccessible::HierarchyMask,
+        QAccessible::Sibling);
+
+    QCOMPARE(acc_pb1->relationTo(0, acc_pb2, 0), QAccessible::Unrelated);
+
+    // geometrical relations - only valid for siblings
+    QCOMPARE(acc_topLeft->relationTo(0, acc_topRight, 0), QAccessible::Sibling | QAccessible::Left);
+    QCOMPARE(acc_topLeft->relationTo(0, acc_bottomLeft, 0), QAccessible::Sibling | QAccessible::Up);
+    QCOMPARE(acc_topLeft->relationTo(0, acc_bottomRight, 0), QAccessible::Sibling | QAccessible::Left | QAccessible::Up);
+
+    QCOMPARE(acc_bottomRight->relationTo(0, acc_topLeft, 0), QAccessible::Sibling | QAccessible::Right | QAccessible::Down);
+    QCOMPARE(acc_bottomRight->relationTo(0, acc_topRight, 0), QAccessible::Sibling | QAccessible::Down);
+    QCOMPARE(acc_bottomRight->relationTo(0, acc_bottomLeft, 0), QAccessible::Sibling | QAccessible::Right);
+#ifdef Q_WS_MAC
+    QEXPECT_FAIL("", "Task 155501", Continue);
+#endif
+    QCOMPARE(acc_group1->relationTo(0, acc_group2, 0), QAccessible::Sibling | QAccessible::Up);
+#ifdef Q_WS_MAC
+    QEXPECT_FAIL("", "Task 155501", Continue);
+#endif
+    QCOMPARE(acc_group2->relationTo(0, acc_group1, 0), QAccessible::Sibling | QAccessible::Down);
+
+    // Covers/Covered tested in navigateCovered
+
+    // logical relations - focus
+    QCOMPARE(acc_radioAM->relationTo(0, acc_radioFM, 0) & QAccessible::FocusChild,
+        QAccessible::Unrelated);
+    QCOMPARE(acc_radioAM->relationTo(0, acc_radiogroup, 0) & QAccessible::FocusChild,
+        QAccessible::FocusChild);
+    QCOMPARE(acc_radioAM->relationTo(0, acc_bottomLeft, 0) & QAccessible::FocusChild,
+        QAccessible::FocusChild);
+    QCOMPARE(acc_radioAM->relationTo(0, acc_topLeft, 0) & QAccessible::FocusChild,
+        QAccessible::Unrelated);
+    QCOMPARE(acc_radioAM->relationTo(0, acc_toplevel, 0) & QAccessible::FocusChild,
+        QAccessible::FocusChild);
+
+    // logical relations - labels
+    QCOMPARE(acc_label->relationTo(0, acc_lineedit, 0) & QAccessible::LogicalMask,
+        QAccessible::Label);
+    QCOMPARE(acc_lineedit->relationTo(0, acc_label, 0) & QAccessible::LogicalMask,
+        QAccessible::Labelled);
+    QCOMPARE(acc_label->relationTo(0, acc_radiogroup, 0) & QAccessible::LogicalMask,
+        QAccessible::Unrelated);
+    QCOMPARE(acc_lineedit->relationTo(0, acc_lineedit, 0) & QAccessible::LogicalMask,
+        QAccessible::Unrelated);
+
+    QEXPECT_FAIL("", "Make me accessible", Continue);
+    QCOMPARE(acc_radiogroup->relationTo(0, acc_radioAM, 0) & QAccessible::LogicalMask,
+        QAccessible::Label | QAccessible::Controlled);
+    QEXPECT_FAIL("", "Make me accessible", Continue);
+    QCOMPARE(acc_radiogroup->relationTo(0, acc_radioFM, 0) & QAccessible::LogicalMask,
+        QAccessible::Label | QAccessible::Controlled);
+    QEXPECT_FAIL("", "Make me accessible", Continue);
+    QCOMPARE(acc_radiogroup->relationTo(0, acc_radioSW, 0) & QAccessible::LogicalMask,
+        QAccessible::Label | QAccessible::Controlled);
+    QCOMPARE(acc_radiogroup->relationTo(0, acc_frequency, 0) & QAccessible::LogicalMask,
+        QAccessible::Label);
+    QCOMPARE(acc_frequency->relationTo(0, acc_radiogroup, 0) & QAccessible::LogicalMask,
+        QAccessible::Labelled);
+    QCOMPARE(acc_radiogroup->relationTo(0, acc_lineedit, 0) & QAccessible::LogicalMask,
+        QAccessible::Unrelated);
+
+    // logical relations - controller
+    QCOMPARE(acc_pbOk->relationTo(0, acc_toplevel, 0) & QAccessible::LogicalMask,
+        QAccessible::Controller);
+    QCOMPARE(acc_slider->relationTo(0, acc_sliderLcd, 0) & QAccessible::LogicalMask,
+        QAccessible::Controller);
+    QCOMPARE(acc_spinBox->relationTo(0, acc_slider, 0) & QAccessible::LogicalMask,
+        QAccessible::Controller);
+    QCOMPARE(acc_lineedit->relationTo(0, acc_label2, 0) & QAccessible::LogicalMask,
+        QAccessible::Controller);
+
+    delete acc_toplevel;
+    delete acc_topLeft;
+    delete acc_group1;
+    delete acc_pb1;
+    delete acc_group2;
+    delete acc_stack;
+    delete acc_page1;
+    delete acc_page2;
+    delete acc_page3;
+    delete acc_topRight;
+    delete acc_pbOk;
+    delete acc_slider;
+    delete acc_spinBox;
+    delete acc_sliderLcd;
+    delete acc_bottomLeft;
+    delete acc_label;
+    delete acc_lineedit;
+    delete acc_label2;
+    delete acc_radiogroup;
+    delete acc_radioAM;
+    delete acc_radioFM;
+    delete acc_radioSW;
+    delete acc_frequency;
+    delete acc_bottomRight;
+
+    delete toplevel;
+
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+#endif // !QT3_SUPPORT
+}
+
+void tst_QAccessibility::navigateGeometric()
+{
+#ifdef QTEST_ACCESSIBILITY
+    {
+    static const int skip = 20; //speed the test up significantly
+    static const double step = Q_PI / 180;
+    QWidget *w = new QWidget(0);
+    w->setObjectName(QString("Josef"));
+    w->setFixedSize(400, 400);
+
+    // center widget
+    QtTestAccessibleWidget *center = new QtTestAccessibleWidget(w, "Sol");
+    center->move(200, 200);
+
+    // arrange 360 widgets around it in a circle
+    QtTestAccessibleWidget *aw = 0;
+    int i;
+    for (i = 0; i < 360; i += skip) {
+        aw = new QtTestAccessibleWidget(w, QString::number(i).toLatin1());
+        aw->move( int(200.0 + 100.0 * sin(step * (double)i)), int(200.0 + 100.0 * cos(step * (double)i)) );
+    }
+
+    aw = new QtTestAccessibleWidget(w, "Earth");
+    QAccessibleInterface *iface = QAccessible::queryAccessibleInterface(center);
+    QAccessibleInterface *target = 0;
+    QVERIFY(iface != 0);
+    QVERIFY(iface->isValid());
+
+    w->show();
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(w);
+    QTest::qWait(100);
+#endif
+
+    // let one widget rotate around center
+    for (i = 0; i < 360; i+=skip) {
+        aw->move( int(200.0 + 75.0 * sin(step * (double)i)), int(200.0 + 75.0 * cos(step * (double)i)) );
+
+        if (i < 45 || i > 315) {
+            QCOMPARE(iface->navigate(QAccessible::Down, 0, &target), 0);
+        } else if ( i < 135 ) {
+            QCOMPARE(iface->navigate(QAccessible::Right, 0, &target), 0);
+        } else if ( i < 225 ) {
+            QCOMPARE(iface->navigate(QAccessible::Up, 0, &target), 0);
+        } else {
+            QCOMPARE(iface->navigate(QAccessible::Left, 0, &target), 0);
+        }
+
+        QVERIFY(target);
+        QVERIFY(target->isValid());
+        QVERIFY(target->object());
+        QCOMPARE(target->object()->objectName(), aw->objectName());
+        delete target; target = 0;
+    }
+
+    // test invisible widget
+    target = QAccessible::queryAccessibleInterface(aw);
+    QVERIFY(!(target->state(0) & QAccessible::Invisible));
+    aw->hide();
+    QVERIFY(target->state(0) & QAccessible::Invisible);
+    delete target; target = 0;
+
+    aw->move(center->x() + 10, center->y());
+    QCOMPARE(iface->navigate(QAccessible::Right, 0, &target), 0);
+    QVERIFY(target);
+    QVERIFY(target->isValid());
+    QVERIFY(target->object());
+    QVERIFY(QString(target->object()->objectName()) != "Earth");
+    delete target; target = 0;
+
+    aw->move(center->x() - 10, center->y());
+    QCOMPARE(iface->navigate(QAccessible::Left, 0, &target), 0);
+    QVERIFY(target);
+    QVERIFY(target->isValid());
+    QVERIFY(target->object());
+    QVERIFY(QString(target->object()->objectName()) != "Earth");
+    delete target; target = 0;
+
+    aw->move(center->x(), center->y() + 10);
+    QCOMPARE(iface->navigate(QAccessible::Down, 0, &target), 0);
+    QVERIFY(target);
+    QVERIFY(target->isValid());
+    QVERIFY(target->object());
+    QVERIFY(QString(target->object()->objectName()) != "Earth");
+    delete target; target = 0;
+
+    aw->move(center->x(), center->y() - 10);
+    QCOMPARE(iface->navigate(QAccessible::Up, 0, &target), 0);
+    QVERIFY(target);
+    QVERIFY(target->isValid());
+    QVERIFY(target->object());
+    QVERIFY(QString(target->object()->objectName()) != "Earth");
+    delete target; target = 0;
+
+    delete iface;
+    delete w;
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::navigateSlider()
+{
+#ifdef QTEST_ACCESSIBILITY
+    {
+    QSlider *slider = new QSlider(0);
+    slider->setObjectName(QString("Slidy"));
+    slider->show();
+    QAccessibleInterface *iface = QAccessible::queryAccessibleInterface(slider);
+    QAccessibleInterface *target = 0;
+    QVERIFY(iface != 0);
+    QVERIFY(iface->isValid());
+    QCOMPARE(iface->childCount(), 3);
+    QCOMPARE(iface->navigate(QAccessible::Child, 1, &target), 1);
+    QVERIFY(target == 0);
+    QCOMPARE(iface->navigate(QAccessible::Child, 2, &target), 2);
+    QVERIFY(target == 0);
+    QCOMPARE(iface->navigate(QAccessible::Child, 3, &target), 3);
+    QVERIFY(target == 0);
+    QCOMPARE(iface->navigate(QAccessible::Child, 4, &target), -1);
+    QVERIFY(target == 0);
+    QCOMPARE(iface->navigate(QAccessible::Child, 0, &target), -1);
+    QVERIFY(target == 0);
+    QCOMPARE(iface->navigate(QAccessible::Child, -42, &target), -1);
+    QVERIFY(target == 0);
+
+    delete iface;
+    delete slider;
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::navigateCovered()
+{
+#ifdef QTEST_ACCESSIBILITY
+    {
+    QWidget *w = new QWidget(0);
+    w->setObjectName(QString("Harry"));
+    QWidget *w1 = new QWidget(w);
+    w1->setObjectName(QString("1"));
+    QWidget *w2 = new QWidget(w);
+    w2->setObjectName(QString("2"));
+    w->show();
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(w);
+    QTest::qWait(100);
+#endif
+
+    w->setFixedSize(6, 6);
+    w1->setFixedSize(5, 5);
+    w2->setFixedSize(5, 5);
+    w2->move(0, 0);
+    w1->raise();
+
+    QAccessibleInterface *iface1 = QAccessible::queryAccessibleInterface(w1);
+    QVERIFY(iface1 != 0);
+    QVERIFY(iface1->isValid());
+    QAccessibleInterface *iface2 = QAccessible::queryAccessibleInterface(w2);
+    QVERIFY(iface2 != 0);
+    QVERIFY(iface2->isValid());
+    QAccessibleInterface *iface3 = 0;
+
+    QCOMPARE(iface1->navigate(QAccessible::Covers, -42, &iface3), -1);
+    QVERIFY(iface3 == 0);
+    QCOMPARE(iface1->navigate(QAccessible::Covers, 0, &iface3), -1);
+    QVERIFY(iface3 == 0);
+    QCOMPARE(iface1->navigate(QAccessible::Covers, 2, &iface3), -1);
+    QVERIFY(iface3 == 0);
+
+    for (int loop = 0; loop < 2; ++loop) {
+        for (int x = 0; x < w->width(); ++x) {
+            for (int y = 0; y < w->height(); ++y) {
+                w1->move(x, y);
+                if (w1->geometry().intersects(w2->geometry())) {
+                    QVERIFY(iface1->relationTo(0, iface2, 0) & QAccessible::Covers);
+                    QVERIFY(iface2->relationTo(0, iface1, 0) & QAccessible::Covered);
+                    QCOMPARE(iface1->navigate(QAccessible::Covered, 1, &iface3), 0);
+                    QVERIFY(iface3 != 0);
+                    QVERIFY(iface3->isValid());
+                    QCOMPARE(iface3->object(), iface2->object());
+                    delete iface3; iface3 = 0;
+                    QCOMPARE(iface2->navigate(QAccessible::Covers, 1, &iface3), 0);
+                    QVERIFY(iface3 != 0);
+                    QVERIFY(iface3->isValid());
+                    QCOMPARE(iface3->object(), iface1->object());
+                    delete iface3; iface3 = 0;
+                } else {
+                    QVERIFY(!(iface1->relationTo(0, iface2, 0) & QAccessible::Covers));
+                    QVERIFY(!(iface2->relationTo(0, iface1, 0) & QAccessible::Covered));
+                    QCOMPARE(iface1->navigate(QAccessible::Covered, 1, &iface3), -1);
+                    QVERIFY(iface3 == 0);
+                    QCOMPARE(iface1->navigate(QAccessible::Covers, 1, &iface3), -1);
+                    QVERIFY(iface3 == 0);
+                    QCOMPARE(iface2->navigate(QAccessible::Covered, 1, &iface3), -1);
+                    QVERIFY(iface3 == 0);
+                    QCOMPARE(iface2->navigate(QAccessible::Covers, 1, &iface3), -1);
+                    QVERIFY(iface3 == 0);
+                }
+            }
+        }
+        if (!loop) {
+            // switch children for second loop
+            w2->raise();
+            QAccessibleInterface *temp = iface1;
+            iface1 = iface2;
+            iface2 = temp;
+        }
+    }
+    delete iface1; iface1 = 0;
+    delete iface2; iface2 = 0;
+    iface1 = QAccessible::queryAccessibleInterface(w1);
+    QVERIFY(iface1 != 0);
+    QVERIFY(iface1->isValid());
+    iface2 = QAccessible::queryAccessibleInterface(w2);
+    QVERIFY(iface2 != 0);
+    QVERIFY(iface2->isValid());
+
+    w1->move(0,0);
+    w2->move(0,0);
+    w1->raise();
+    QVERIFY(iface1->relationTo(0, iface2, 0) & QAccessible::Covers);
+    QVERIFY(iface2->relationTo(0, iface1, 0) & QAccessible::Covered);
+    QVERIFY(!(iface1->state(0) & QAccessible::Invisible));
+    w1->hide();
+    QVERIFY(iface1->state(0) & QAccessible::Invisible);
+    QVERIFY(!(iface1->relationTo(0, iface2, 0) & QAccessible::Covers));
+    QVERIFY(!(iface2->relationTo(0, iface1, 0) & QAccessible::Covered));
+    QCOMPARE(iface2->navigate(QAccessible::Covered, 1, &iface3), -1);
+    QVERIFY(iface3 == 0);
+    QCOMPARE(iface1->navigate(QAccessible::Covers, 1, &iface3), -1);
+    QVERIFY(iface3 == 0);
+
+    delete iface1; iface1 = 0;
+    delete iface2; iface2 = 0;
+    delete w;
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::navigateHierarchy()
+{
+#ifdef QTEST_ACCESSIBILITY
+    {
+    QWidget *w = new QWidget(0);
+    w->setObjectName(QString("Hans"));
+    w->show();
+    QWidget *w1 = new QWidget(w);
+    w1->setObjectName(QString("1"));
+    w1->show();
+    QWidget *w2 = new QWidget(w);
+    w2->setObjectName(QString("2"));
+    w2->show();
+    QWidget *w3 = new QWidget(w);
+    w3->setObjectName(QString("3"));
+    w3->show();
+    QWidget *w31 = new QWidget(w3);
+    w31->setObjectName(QString("31"));
+    w31->show();
+
+    QAccessibleInterface *target = 0;
+    QAccessibleInterface *iface = QAccessible::queryAccessibleInterface(w);
+    QVERIFY(iface != 0);
+    QVERIFY(iface->isValid());
+
+    QCOMPARE(iface->navigate(QAccessible::Sibling, -42, &target), -1);
+    QVERIFY(target == 0);
+    QCOMPARE(iface->navigate(QAccessible::Sibling, 42, &target), -1);
+    QVERIFY(target == 0);
+    QCOMPARE(iface->navigate(QAccessible::Child, 15, &target), -1);
+    QVERIFY(target == 0);
+    QCOMPARE(iface->navigate(QAccessible::Child, 0, &target), -1);
+    QVERIFY(target == 0);
+    QCOMPARE(iface->navigate(QAccessible::Child, 1, &target), 0);
+    QVERIFY(target != 0);
+    QVERIFY(target->isValid());
+    QCOMPARE(target->object(), (QObject*)w1);
+    delete iface; iface = 0;
+
+    QCOMPARE(target->navigate(QAccessible::Sibling, 0, &iface), -1);
+    QVERIFY(iface == 0);
+    QCOMPARE(target->navigate(QAccessible::Sibling, 42, &iface), -1);
+    QVERIFY(iface == 0);
+    QCOMPARE(target->navigate(QAccessible::Sibling, -42, &iface), -1);
+    QVERIFY(iface == 0);
+    QCOMPARE(target->navigate(QAccessible::Sibling, 2, &iface), 0);
+    QVERIFY(iface != 0);
+    QVERIFY(iface->isValid());
+    QCOMPARE(iface->object(), (QObject*)w2);
+    delete target; target = 0;
+    QCOMPARE(iface->navigate(QAccessible::Sibling, 3, &target), 0);
+    QVERIFY(target != 0);
+    QVERIFY(target->isValid());
+    QCOMPARE(target->object(), (QObject*)w3);
+    delete iface; iface = 0;
+
+    QCOMPARE(target->navigate(QAccessible::Child, 1, &iface), 0);
+    QVERIFY(iface != 0);
+    QVERIFY(iface->isValid());
+    QCOMPARE(iface->object(), (QObject*)w31);
+    delete target; target = 0;
+
+    QCOMPARE(iface->navigate(QAccessible::Sibling, -1, &target), -1);
+    QVERIFY(target == 0);
+    QCOMPARE(iface->navigate(QAccessible::Sibling, 0, &target), -1);
+    QVERIFY(target == 0);
+    QCOMPARE(iface->navigate(QAccessible::Sibling, 1, &target), 0);
+    QVERIFY(target != 0);
+    QVERIFY(target->isValid());
+    QCOMPARE(target->object(), (QObject*)w31);
+    delete iface; iface = 0;
+
+    QCOMPARE(target->navigate(QAccessible::Ancestor, 42, &iface), -1);
+    QVERIFY(iface == 0);
+    QCOMPARE(target->navigate(QAccessible::Ancestor, -1, &iface), -1);
+    QVERIFY(iface == 0);
+    QCOMPARE(target->navigate(QAccessible::Ancestor, 0, &iface), -1);
+    QVERIFY(iface == 0);
+    QCOMPARE(target->navigate(QAccessible::Ancestor, 1, &iface), 0);
+    QVERIFY(iface != 0);
+    QVERIFY(iface->isValid());
+    QCOMPARE(iface->object(), (QObject*)w3);
+    delete iface; iface = 0;
+    QCOMPARE(target->navigate(QAccessible::Ancestor, 2, &iface), 0);
+    QVERIFY(iface != 0);
+    QVERIFY(iface->isValid());
+    QCOMPARE(iface->object(), (QObject*)w);
+    delete iface; iface = 0;
+    QCOMPARE(target->navigate(QAccessible::Ancestor, 3, &iface), 0);
+    QVERIFY(iface != 0);
+    QVERIFY(iface->isValid());
+    QCOMPARE(iface->object(), (QObject*)qApp);
+    delete iface; iface = 0;
+    delete target; target = 0;
+
+    delete w;
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+#define QSETCOMPARE(thetypename, elements, otherelements) \
+    QCOMPARE((QSet<thetypename>() << elements), (QSet<thetypename>() << otherelements))
+
+void tst_QAccessibility::navigateControllers()
+{
+#if !defined(QT3_SUPPORT)
+    QSKIP("This test needs Qt3Support", SkipAll);
+#else
+#ifdef QTEST_ACCESSIBILITY
+    {
+    Q3VBox vbox;
+    QSlider     slider(&vbox);
+    QSpinBox    spinBox(&vbox);
+    QLCDNumber  lcd1(&vbox);
+    QLCDNumber  lcd2(&vbox);
+    QLabel      label(&vbox);
+    vbox.show();
+
+    slider.setObjectName("slider");
+    spinBox.setObjectName("spinBox");
+    lcd1.setObjectName("lcd1");
+    lcd2.setObjectName("lcd2");
+    label.setObjectName("label");
+
+    QTestAccessibility::clearEvents();
+
+    connect(&slider, SIGNAL(valueChanged(int)), &lcd1, SLOT(display(int)));
+    connect(&slider, SIGNAL(valueChanged(int)), &lcd2, SLOT(display(int)));
+    connect(&spinBox, SIGNAL(valueChanged(int)), &lcd2, SLOT(display(int)));
+    connect(&spinBox, SIGNAL(valueChanged(int)), &lcd1, SLOT(display(int)));
+    connect(&spinBox, SIGNAL(valueChanged(const QString&)), &label, SLOT(setText(const QString&)));
+
+    QAccessibleInterface *acc_slider = QAccessible::queryAccessibleInterface(&slider);
+    QAccessibleInterface *acc_spinBox = QAccessible::queryAccessibleInterface(&spinBox);
+    QAccessibleInterface *acc_lcd1 = QAccessible::queryAccessibleInterface(&lcd1);
+    QAccessibleInterface *acc_lcd2 = QAccessible::queryAccessibleInterface(&lcd2);
+    QAccessibleInterface *acc_label = QAccessible::queryAccessibleInterface(&label);
+
+    QVERIFY(acc_slider->relationTo(0, acc_lcd1, 0) & QAccessible::Controller);
+    QVERIFY(acc_slider->relationTo(0, acc_lcd2, 0) & QAccessible::Controller);
+    QVERIFY(acc_spinBox->relationTo(0, acc_lcd1, 0) & QAccessible::Controller);
+    QVERIFY(acc_spinBox->relationTo(0, acc_lcd2, 0) & QAccessible::Controller);
+    QVERIFY(acc_spinBox->relationTo(0, acc_label, 0) & QAccessible::Controller);
+
+    QAccessibleInterface *acc_target1, *acc_target2, *acc_target3;
+    // from controller
+    QCOMPARE(acc_slider->navigate(QAccessible::Controlled, 0, &acc_target1), -1);
+    QVERIFY(!acc_target1);
+    QCOMPARE(acc_slider->navigate(QAccessible::Controlled, 1, &acc_target1), 0);
+    QCOMPARE(acc_slider->navigate(QAccessible::Controlled, 2, &acc_target2), 0);
+    QSETCOMPARE(QObject*, acc_lcd1->object()    <<  acc_lcd2->object(),
+                          acc_target1->object() <<  acc_target2->object());
+    delete acc_target1;
+    delete acc_target2;
+
+    QCOMPARE(acc_slider->navigate(QAccessible::Controlled, 3, &acc_target1), -1);
+    QVERIFY(!acc_target1);
+
+    QCOMPARE(acc_spinBox->navigate(QAccessible::Controlled, 0, &acc_target1), -1);
+    QVERIFY(!acc_target1);
+    QCOMPARE(acc_spinBox->navigate(QAccessible::Controlled, 1, &acc_target1), 0);
+    QCOMPARE(acc_spinBox->navigate(QAccessible::Controlled, 2, &acc_target2), 0);
+    QCOMPARE(acc_spinBox->navigate(QAccessible::Controlled, 3, &acc_target3), 0);
+    QSETCOMPARE(QObject*, acc_lcd1->object()    <<  acc_lcd2->object()    << acc_label->object(),
+                          acc_target1->object() <<  acc_target2->object() << acc_target3->object());
+    delete acc_target1;
+    delete acc_target2;
+    delete acc_target3;
+
+    QCOMPARE(acc_spinBox->navigate(QAccessible::Controlled, 4, &acc_target1), -1);
+    QVERIFY(!acc_target1);
+
+    // to controller
+    QCOMPARE(acc_lcd1->navigate(QAccessible::Controller, 0, &acc_target1), -1);
+    QVERIFY(!acc_target1);
+    QCOMPARE(acc_lcd1->navigate(QAccessible::Controller, 1, &acc_target1), 0);
+    QCOMPARE(acc_lcd1->navigate(QAccessible::Controller, 2, &acc_target2), 0);
+    QSETCOMPARE(QObject*, acc_slider->object()  <<  acc_spinBox->object(),
+                          acc_target1->object() <<  acc_target2->object());
+    delete acc_target1;
+    delete acc_target2;
+    QCOMPARE(acc_lcd1->navigate(QAccessible::Controller, 3, &acc_target1), -1);
+    QVERIFY(!acc_target1);
+
+    delete acc_label;
+    delete acc_lcd2;
+    delete acc_lcd1;
+    delete acc_spinBox;
+    delete acc_slider;
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+#endif // !QT3_SUPPORT
+}
+
+void tst_QAccessibility::navigateLabels()
+{
+#if !defined(QT3_SUPPORT)
+    QSKIP("This test needs Qt3Support", SkipAll);
+#else
+#ifdef QTEST_ACCESSIBILITY
+    {
+    Q3VBox vbox;
+    Q3HBox hbox(&vbox);
+
+    QLabel      label(&hbox);
+    label.setText("This is a lineedit:");
+    QLineEdit   lineedit(&hbox);
+    label.setBuddy(&lineedit);
+
+    Q3VButtonGroup groupbox(&vbox);
+    groupbox.setTitle("Be my children!");
+    QRadioButton radio(&groupbox);
+    QLabel      label2(&groupbox);
+    label2.setText("Another lineedit:");
+    QLineEdit   lineedit2(&groupbox);
+    label2.setBuddy(&lineedit2);
+    Q3GroupBox groupbox2(&groupbox);
+    groupbox2.setTitle("Some grand-children");
+    QLineEdit   grandchild(&groupbox2);
+
+    Q3GroupBox border(&vbox);
+    QLineEdit   lineedit3(&border);
+    vbox.show();
+    QTestAccessibility::clearEvents();
+
+    QAccessibleInterface *acc_label = QAccessible::queryAccessibleInterface(&label);
+    QAccessibleInterface *acc_lineedit = QAccessible::queryAccessibleInterface(&lineedit);
+    QAccessibleInterface *acc_groupbox = QAccessible::queryAccessibleInterface(&groupbox);
+    QAccessibleInterface *acc_radio = QAccessible::queryAccessibleInterface(&radio);
+    QAccessibleInterface *acc_label2 = QAccessible::queryAccessibleInterface(&label2);
+    QAccessibleInterface *acc_lineedit2 = QAccessible::queryAccessibleInterface(&lineedit2);
+    QAccessibleInterface *acc_groupbox2 = QAccessible::queryAccessibleInterface(&groupbox2);
+    QAccessibleInterface *acc_grandchild = QAccessible::queryAccessibleInterface(&grandchild);
+    QAccessibleInterface *acc_border = QAccessible::queryAccessibleInterface(&border);
+    QAccessibleInterface *acc_lineedit3 = QAccessible::queryAccessibleInterface(&lineedit3);
+
+    QVERIFY(acc_label->relationTo(0, acc_lineedit,0) & QAccessible::Label);
+    QVERIFY(acc_groupbox->relationTo(0, acc_radio,0) & QAccessible::Label);
+    QVERIFY(acc_groupbox->relationTo(0, acc_lineedit2,0) & QAccessible::Label);
+    QVERIFY(acc_groupbox->relationTo(0, acc_groupbox2,0) & QAccessible::Label);
+    QVERIFY(acc_groupbox2->relationTo(0, acc_grandchild,0) & QAccessible::Label);
+    QVERIFY(!(acc_border->relationTo(0, acc_lineedit3,0) & QAccessible::Label));
+
+    QAccessibleInterface *acc_target;
+    // from label
+    QCOMPARE(acc_label->navigate(QAccessible::Labelled, 0, &acc_target), -1);
+    QVERIFY(!acc_target);
+    QCOMPARE(acc_label->navigate(QAccessible::Labelled, 1, &acc_target), 0);
+    QVERIFY(acc_target->object() == acc_lineedit->object());
+    delete acc_target; acc_target = 0;
+    QCOMPARE(acc_label->navigate(QAccessible::Labelled, 2, &acc_target), -1);
+    QVERIFY(!acc_target);
+
+    QCOMPARE(acc_groupbox->navigate(QAccessible::Labelled, 0, &acc_target), -1);
+    QVERIFY(!acc_target);
+    QCOMPARE(acc_groupbox->navigate(QAccessible::Labelled, 1, &acc_target), 0);
+    QVERIFY(acc_target->object() == acc_radio->object());
+    delete acc_target; acc_target = 0;
+    QCOMPARE(acc_groupbox->navigate(QAccessible::Labelled, 2, &acc_target), 0);
+    QVERIFY(acc_target->object() == acc_label2->object());
+    delete acc_target; acc_target = 0;
+    QCOMPARE(acc_groupbox->navigate(QAccessible::Labelled, 3, &acc_target), 0);
+    QVERIFY(acc_target->object() == acc_lineedit2->object());
+    delete acc_target; acc_target = 0;
+    QCOMPARE(acc_groupbox->navigate(QAccessible::Labelled, 4, &acc_target), 0);
+    QVERIFY(acc_target->object() == acc_groupbox2->object());
+    delete acc_target; acc_target = 0;
+    QCOMPARE(acc_groupbox->navigate(QAccessible::Labelled, 5, &acc_target), -1);
+    QVERIFY(!acc_target);
+
+    QCOMPARE(acc_border->navigate(QAccessible::Labelled, 0, &acc_target), -1);
+    QVERIFY(!acc_target);
+    QCOMPARE(acc_border->navigate(QAccessible::Labelled, 1, &acc_target), -1);
+    QVERIFY(!acc_target);
+
+    // to label
+    QCOMPARE(acc_lineedit->navigate(QAccessible::Label, 0, &acc_target), -1);
+    QVERIFY(!acc_target);
+    QCOMPARE(acc_lineedit->navigate(QAccessible::Label, 1, &acc_target), 0);
+    QVERIFY(acc_target->object() == acc_label->object());
+    delete acc_target; acc_target = 0;
+    QCOMPARE(acc_lineedit->navigate(QAccessible::Label, 2, &acc_target), -1);
+    QVERIFY(!acc_target);
+
+    QCOMPARE(acc_radio->navigate(QAccessible::Label, 0, &acc_target), -1);
+    QVERIFY(!acc_target);
+    QCOMPARE(acc_radio->navigate(QAccessible::Label, 1, &acc_target), 0);
+    QVERIFY(acc_target->object() == acc_groupbox->object());
+    delete acc_target; acc_target = 0;
+    QCOMPARE(acc_radio->navigate(QAccessible::Label, 2, &acc_target), -1);
+    QVERIFY(!acc_target);
+
+    QCOMPARE(acc_lineedit2->navigate(QAccessible::Label, 1, &acc_target), 0);
+    QVERIFY(acc_target->object() == acc_label2->object());
+    delete acc_target; acc_target = 0;
+    QCOMPARE(acc_lineedit2->navigate(QAccessible::Label, 2, &acc_target), 0);
+    QVERIFY(acc_target->object() == acc_groupbox->object());
+    delete acc_target; acc_target = 0;
+    QCOMPARE(acc_lineedit2->navigate(QAccessible::Label, 3, &acc_target), -1);
+    QVERIFY(!acc_target);
+
+    QCOMPARE(acc_grandchild->navigate(QAccessible::Label, 1, &acc_target), 0);
+    QVERIFY(acc_target->object() == acc_groupbox2->object());
+    delete acc_target; acc_target = 0;
+    QCOMPARE(acc_grandchild->navigate(QAccessible::Label, 2, &acc_target), -1);
+    QVERIFY(!acc_target);
+    QCOMPARE(acc_grandchild->navigate(QAccessible::Label, 3, &acc_target), -1);
+    QVERIFY(!acc_target);
+
+    delete acc_label;
+    delete acc_lineedit;
+    delete acc_groupbox;
+    delete acc_radio;
+    delete acc_label2;
+    delete acc_lineedit2;
+    delete acc_groupbox2;
+    delete acc_grandchild;
+    delete acc_border;
+    delete acc_lineedit3;
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+#endif // !QT3_SUPPORT
+}
+
+static QWidget *createWidgets()
+{
+    QWidget *w = new QWidget();
+
+    QHBoxLayout *box = new QHBoxLayout(w);
+
+    int i = 0;
+    box->addWidget(new QComboBox(w));
+    box->addWidget(new QPushButton(QString::fromAscii("widget text %1").arg(i++), w));
+    box->addWidget(new QHeaderView(Qt::Vertical, w));
+    box->addWidget(new QTreeView(w));
+    box->addWidget(new QTreeWidget(w));
+    box->addWidget(new QListView(w));
+    box->addWidget(new QListWidget(w));
+    box->addWidget(new QTableView(w));
+    box->addWidget(new QTableWidget(w));
+    box->addWidget(new QCalendarWidget(w));
+    box->addWidget(new QDialogButtonBox(w));
+    box->addWidget(new QGroupBox(QString::fromAscii("widget text %1").arg(i++), w));
+    box->addWidget(new QFrame(w));
+    box->addWidget(new QLineEdit(QString::fromAscii("widget text %1").arg(i++), w));
+    box->addWidget(new QProgressBar(w));
+    box->addWidget(new QTabWidget(w));
+    box->addWidget(new QCheckBox(QString::fromAscii("widget text %1").arg(i++), w));
+    box->addWidget(new QRadioButton(QString::fromAscii("widget text %1").arg(i++), w));
+    box->addWidget(new QDial(w));
+    box->addWidget(new QScrollBar(w));
+    box->addWidget(new QSlider(w));
+    box->addWidget(new QDateTimeEdit(w));
+    box->addWidget(new QDoubleSpinBox(w));
+    box->addWidget(new QSpinBox(w));
+    box->addWidget(new QLabel(QString::fromAscii("widget text %1").arg(i++), w));
+    box->addWidget(new QLCDNumber(w));
+    box->addWidget(new QStackedWidget(w));
+    box->addWidget(new QToolBox(w));
+    box->addWidget(new QLabel(QString::fromAscii("widget text %1").arg(i++), w));
+    box->addWidget(new QTextEdit(QString::fromAscii("widget text %1").arg(i++), w));
+
+    /* Not in the list
+     * QAbstractItemView, QGraphicsView, QScrollArea,
+     * QToolButton, QDockWidget, QFocusFrame, QMainWindow, QMenu, QMenuBar, QSizeGrip, QSplashScreen, QSplitterHandle,
+     * QStatusBar, QSvgWidget, QTabBar, QToolBar, QWorkspace, QSplitter
+     */
+    return w;
+}
+
+void tst_QAccessibility::accessibleName()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QWidget *toplevel = createWidgets();
+    toplevel->show();
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(toplevel);
+    QTest::qWait(100);
+#endif
+    QLayout *lout = toplevel->layout();
+    for (int i = 0; i < lout->count(); i++) {
+        QLayoutItem *item = lout->itemAt(i);
+        QWidget *child = item->widget();
+
+        QString name = tr("Widget Name %1").arg(i);
+        child->setAccessibleName(name);
+        QAccessibleInterface *acc = QAccessible::queryAccessibleInterface(child);
+        QCOMPARE(acc->text(QAccessible::Name, 0), name);
+
+        QString desc = tr("Widget Description %1").arg(i);
+        child->setAccessibleDescription(desc);
+        QCOMPARE(acc->text(QAccessible::Description, 0), desc);
+
+    }
+
+    delete toplevel;
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::text()
+{
+#if !defined(QT3_SUPPORT)
+    QSKIP("This test needs Qt3Support", SkipAll);
+#else
+#ifdef QTEST_ACCESSIBILITY
+    QWidget *toplevel = createGUI();
+    toplevel->show();
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(toplevel);
+    QTest::qWait(100);
+#endif
+    QObject *topLeft = toplevel->child("topLeft");
+    QObject *topRight = toplevel->child("topRight");
+    QObject *bottomLeft = toplevel->child("bottomLeft");
+
+    QAccessibleInterface *acc_pb1 = QAccessible::queryAccessibleInterface(topLeft->child("pb1"));
+
+    QAccessibleInterface *acc_pbOk = QAccessible::queryAccessibleInterface(topRight->child("pbOk"));
+    QAccessibleInterface *acc_slider = QAccessible::queryAccessibleInterface(topRight->child("slider"));
+    QAccessibleInterface *acc_spinBox = QAccessible::queryAccessibleInterface(topRight->child("spinBox"));
+    QAccessibleInterface *acc_sliderLcd = QAccessible::queryAccessibleInterface(topRight->child("sliderLcd"));
+
+    QAccessibleInterface *acc_label = QAccessible::queryAccessibleInterface(bottomLeft->child("label"));
+    QAccessibleInterface *acc_lineedit = QAccessible::queryAccessibleInterface(bottomLeft->child("lineedit"));
+    QAccessibleInterface *acc_radiogroup = QAccessible::queryAccessibleInterface(bottomLeft->child("radiogroup"));
+    QVERIFY(bottomLeft->child("radiogroup"));
+    QAccessibleInterface *acc_radioAM = QAccessible::queryAccessibleInterface(bottomLeft->child("radiogroup")->child("radioAM"));
+    QAccessibleInterface *acc_frequency = QAccessible::queryAccessibleInterface(bottomLeft->child("radiogroup")->child("frequency"));
+
+    QVERIFY(acc_pb1);
+
+    QVERIFY(acc_pbOk);
+    QVERIFY(acc_slider);
+    QVERIFY(acc_spinBox);
+    QVERIFY(acc_sliderLcd);
+
+    QVERIFY(acc_label);
+    QVERIFY(acc_lineedit);
+    QVERIFY(acc_radiogroup);
+    QVERIFY(acc_radioAM);
+    QVERIFY(acc_frequency);
+
+    QVERIFY(acc_label->relationTo(0, acc_lineedit, 0) & QAccessible::Label);
+    QVERIFY(acc_radiogroup->relationTo(0, acc_frequency, 0) & QAccessible::Label);
+    QVERIFY(acc_slider->relationTo(0, acc_sliderLcd, 0) & QAccessible::Controller);
+    QVERIFY(acc_spinBox->relationTo(0, acc_slider, 0) & QAccessible::Controller);
+
+    // Name
+    QCOMPARE(acc_lineedit->text(QAccessible::Name, 0), acc_label->text(QAccessible::Name,0));
+    QCOMPARE(acc_frequency->text(QAccessible::Name, 0), acc_radiogroup->text(QAccessible::Name,0));
+    QCOMPARE(acc_sliderLcd->text(QAccessible::Name, 0), acc_slider->text(QAccessible::Value,0));
+    QCOMPARE(acc_pbOk->text(QAccessible::Name, 0), QString("Ok"));
+    QCOMPARE(acc_radioAM->text(QAccessible::Name, 0), QString("AM"));
+    QCOMPARE(acc_pb1->text(QAccessible::Name, 0), QString("Button1"));
+
+    // Description
+    QString desc = qobject_cast<QWidget*>(acc_label->object())->toolTip();
+    QVERIFY(!desc.isEmpty());
+    QCOMPARE(acc_label->text(QAccessible::Description, 0), desc);
+    desc = qobject_cast<QWidget*>(acc_lineedit->object())->toolTip();
+    QVERIFY(!desc.isEmpty());
+    QCOMPARE(acc_lineedit->text(QAccessible::Description, 0), desc);
+
+    // Help
+    QString help = qobject_cast<QWidget*>(acc_label->object())->whatsThis();
+    QVERIFY(!help.isEmpty());
+    QCOMPARE(acc_label->text(QAccessible::Help, 0), help);
+    help = qobject_cast<QWidget*>(acc_frequency->object())->whatsThis();
+    QVERIFY(!help.isEmpty());
+    QCOMPARE(acc_frequency->text(QAccessible::Help, 0), help);
+
+    // Value
+    QString value = acc_frequency->object()->property("text").toString();
+    QVERIFY(!value.isEmpty());
+    QCOMPARE(acc_frequency->text(QAccessible::Value, 0), value);
+    value = acc_slider->object()->property("value").toString();
+    QVERIFY(!value.isEmpty());
+    QCOMPARE(acc_slider->text(QAccessible::Value, 0), value);
+    QCOMPARE(acc_spinBox->text(QAccessible::Value, 0), value);
+
+    // Accelerator
+    QCOMPARE(acc_pbOk->text(QAccessible::Accelerator, 0), Q3Accel::keyToString(Qt::Key_Enter));
+    QCOMPARE(acc_pb1->text(QAccessible::Accelerator, 0), Q3Accel::keyToString(Qt::ALT + Qt::Key_1));
+    QCOMPARE(acc_lineedit->text(QAccessible::Accelerator, 0), Q3Accel::keyToString(Qt::ALT) + "L");
+    QCOMPARE(acc_frequency->text(QAccessible::Accelerator, 0), Q3Accel::keyToString(Qt::ALT) + "C");
+
+    delete acc_pb1;
+    delete acc_pbOk;
+    delete acc_slider;
+    delete acc_spinBox;
+    delete acc_sliderLcd;
+
+    delete acc_label;
+    delete acc_lineedit;
+    delete acc_radiogroup;
+        delete acc_radioAM;
+        delete acc_frequency;
+
+    delete toplevel;
+    QTestAccessibility::clearEvents();
+
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+#endif // !QT3_SUPPORT
+}
+
+void tst_QAccessibility::setText()
+{
+#if !defined(QT3_SUPPORT)
+    QSKIP("This test needs Qt3Support", SkipAll);
+#else
+#ifdef QTEST_ACCESSIBILITY
+    QWidget *toplevel = createGUI();
+    toplevel->show();
+    QObject *bottomLeft = toplevel->findChild<QObject *>("bottomLeft");
+
+    QAccessibleInterface *acc_lineedit = QAccessible::queryAccessibleInterface(bottomLeft->findChild<QLineEdit *>("lineedit"));
+    // Value, read-write
+    QString txt = acc_lineedit->text(QAccessible::Value, 0);
+    QVERIFY(txt.isEmpty());
+    txt = QLatin1String("Writable");
+    acc_lineedit->setText(QAccessible::Value, 0, txt);
+    QCOMPARE(acc_lineedit->text(QAccessible::Value, 0), txt);
+
+    // Description, read-only
+    txt = acc_lineedit->text(QAccessible::Description, 0);
+    QVERIFY(!txt.isEmpty());
+    acc_lineedit->setText(QAccessible::Description, 0, QLatin1String(""));
+    QCOMPARE(acc_lineedit->text(QAccessible::Description, 0), txt);
+
+    QVERIFY(acc_lineedit);
+
+    delete acc_lineedit;
+    delete toplevel;
+    QTestAccessibility::clearEvents();
+
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+#endif //QT3_SUPPORT
+}
+
+void tst_QAccessibility::hideShowTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QWidget * const window = new QWidget();
+    QWidget * const child = new QWidget(window);
+
+    QVERIFY(state(window) & QAccessible::Invisible);
+    QVERIFY(state(child)  & QAccessible::Invisible);
+
+    QTestAccessibility::clearEvents();
+
+    // show() and veryfy that both window and child are not invisible and get ObjectShow events.
+    window->show();
+    QVERIFY(state(window) ^ QAccessible::Invisible);
+    QVERIFY(state(child)  ^ QAccessible::Invisible);
+    QVERIFY(QTestAccessibility::events().contains(QTestAccessibilityEvent(window, 0, QAccessible::ObjectShow)));
+    QVERIFY(QTestAccessibility::events().contains(QTestAccessibilityEvent(child, 0, QAccessible::ObjectShow)));
+    QTestAccessibility::clearEvents();
+
+    // hide() and veryfy that both window and child are invisible and get ObjectHide events.
+    window->hide();
+    QVERIFY(state(window) & QAccessible::Invisible);
+    QVERIFY(state(child)  & QAccessible::Invisible);
+    QVERIFY(QTestAccessibility::events().contains(QTestAccessibilityEvent(window, 0, QAccessible::ObjectHide)));
+    QVERIFY(QTestAccessibility::events().contains(QTestAccessibilityEvent(child, 0, QAccessible::ObjectHide)));
+    QTestAccessibility::clearEvents();
+
+    delete window;
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::userActionCount()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QWidget widget;
+
+    QAccessibleInterface *test = QAccessible::queryAccessibleInterface(&widget);
+    QVERIFY(test);
+    QVERIFY(test->isValid());
+    QCOMPARE(test->userActionCount(0), 0);
+    QCOMPARE(test->userActionCount(1), 0);
+    QCOMPARE(test->userActionCount(-1), 0);
+    delete test; test = 0;
+
+    QFrame frame;
+
+    test = QAccessible::queryAccessibleInterface(&frame);
+    QVERIFY(test);
+    QVERIFY(test->isValid());
+    QCOMPARE(test->userActionCount(0), 0);
+    QCOMPARE(test->userActionCount(1), 0);
+    QCOMPARE(test->userActionCount(-1), 0);
+    delete test; test = 0;
+
+    QLineEdit lineEdit;
+
+    test = QAccessible::queryAccessibleInterface(&lineEdit);
+    QVERIFY(test);
+    QVERIFY(test->isValid());
+    QCOMPARE(test->userActionCount(0), 0);
+    QCOMPARE(test->userActionCount(1), 0);
+    QCOMPARE(test->userActionCount(-1), 0);
+    delete test; test = 0;
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::actionText()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QWidget widget;
+    widget.show();
+
+    QAccessibleInterface *test = QAccessible::queryAccessibleInterface(&widget);
+    QVERIFY(test);
+    QVERIFY(test->isValid());
+
+    QCOMPARE(test->actionText(1, QAccessible::Name, 0), QString());
+    QCOMPARE(test->actionText(0, QAccessible::Name, 1), QString());
+    QCOMPARE(test->actionText(1, QAccessible::Name, 1), QString());
+    QCOMPARE(test->actionText(QAccessible::SetFocus, QAccessible::Name, -1), QString());
+
+    QCOMPARE(test->actionText(QAccessible::DefaultAction, QAccessible::Name, 0), QString("SetFocus"));
+    QCOMPARE(test->actionText(QAccessible::SetFocus, QAccessible::Name, 0), QString("SetFocus"));
+
+    delete test; test = 0;
+
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::doAction()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QSKIP("TODO: Implement me", SkipAll);
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::buttonTest()
+{
+//#ifdef QTEST_ACCESSIBILITY
+#if 0
+    QAccessibleInterface *test = 0;
+    Q3VBox vbox;
+
+    // Standard push button
+    QPushButton pushButton("Ok", &vbox);
+
+    // toggle push button
+    QPushButton togglepush("Toggle", &vbox);
+    togglepush.setToggleButton(TRUE);
+
+    // push button with a menu
+    QPushButton menuButton("Menu", &vbox);
+    Q3PopupMenu buttonMenu(&menuButton);
+    buttonMenu.insertItem("Some item");
+    menuButton.setPopup(&buttonMenu);
+
+    // standard checkbox
+    QCheckBox checkBox("Check me!", &vbox);
+
+    // tristate checkbox
+    QCheckBox tristate("Tristate!", &vbox);
+    tristate.setTristate(TRUE);
+
+    // radiobutton
+    QRadioButton radio("Radio me!", &vbox);
+
+    // standard toolbutton
+    QToolButton toolbutton(&vbox);
+    toolbutton.setText("Tool");
+    toolbutton.setMinimumSize(20,20);
+
+    // standard toolbutton
+    QToolButton toggletool(&vbox);
+    toggletool.setToggleButton(TRUE);
+    toggletool.setText("Toggle");
+    toggletool.setMinimumSize(20,20);
+
+    // menu toolbutton
+    QToolButton menuToolButton(&vbox);
+    menuToolButton.setText("Menu Tool");
+    Q3PopupMenu toolMenu(&menuToolButton);
+    toolMenu.insertItem("Some item");
+    menuToolButton.setPopup(&toolMenu);
+    menuToolButton.setMinimumSize(20,20);
+
+    // splitted menu toolbutton
+    QToolButton splitToolButton(&vbox);
+    splitToolButton.setTextLabel("Split Tool");
+    Q3PopupMenu splitMenu(&splitToolButton);
+    splitMenu.insertItem("Some item");
+    splitToolButton.setPopup(&splitMenu);
+    splitToolButton.setPopupDelay(0);
+    splitToolButton.setMinimumSize(20,20);
+
+    // test push button
+    QVERIFY(QAccessible::queryAccessibleInterface(&pushButton, &test));
+    QCOMPARE(test->role(0), QAccessible::PushButton);
+    QCOMPARE(test->defaultAction(0), QAccessible::Press);
+    QCOMPARE(test->actionText(test->defaultAction(0), QAccessible::Name, 0), QString("Press"));
+    QCOMPARE(test->state(0), (int)QAccessible::Normal);
+    pushButton.setDown(TRUE);
+    QCOMPARE(test->state(0), (int)QAccessible::Pressed);
+    QVERIFY(test->doAction(QAccessible::Press, 0));
+    QTest::qWait(500);
+    QCOMPARE(test->state(0), (int)QAccessible::Normal);
+    test->release();
+
+    // test toggle push button
+    QVERIFY(QAccessible::queryAccessibleInterface(&togglepush, &test));
+    QCOMPARE(test->role(0), QAccessible::CheckBox);
+    QCOMPARE(test->defaultAction(0), QAccessible::Press);
+    QCOMPARE(test->actionText(test->defaultAction(0), QAccessible::Name, 0), QString("Check"));
+    QCOMPARE(test->state(0), (int)QAccessible::Normal);
+    QVERIFY(test->doAction(QAccessible::Press, 0));
+    QTest::qWait(500);
+    QCOMPARE(test->actionText(test->defaultAction(0), QAccessible::Name, 0), QString("Uncheck"));
+    QCOMPARE(test->state(0), (int)QAccessible::Checked);
+    test->release();
+
+    // test menu push button
+    QVERIFY(QAccessible::queryAccessibleInterface(&menuButton, &test));
+    QCOMPARE(test->role(0), QAccessible::ButtonMenu);
+    QCOMPARE(test->defaultAction(0), QAccessible::Press);
+    QCOMPARE(test->actionText(test->defaultAction(0), QAccessible::Name, 0), QString("Open"));
+    QCOMPARE(test->state(0), (int)QAccessible::HasPopup);
+    test->release();
+
+    // test check box
+    QVERIFY(QAccessible::queryAccessibleInterface(&checkBox, &test));
+    QCOMPARE(test->role(0), QAccessible::CheckBox);
+    QCOMPARE(test->defaultAction(0), QAccessible::Press);
+    QCOMPARE(test->actionText(test->defaultAction(0), QAccessible::Name, 0), QString("Check"));
+    QCOMPARE(test->state(0), (int)QAccessible::Normal);
+    QVERIFY(test->doAction(QAccessible::Press, 0));
+    QTest::qWait(500);
+    QCOMPARE(test->actionText(test->defaultAction(0), QAccessible::Name, 0), QString("Uncheck"));
+    QCOMPARE(test->state(0), (int)QAccessible::Checked);
+    test->release();
+
+    // test tristate check box
+    QVERIFY(QAccessible::queryAccessibleInterface(&tristate, &test));
+    QCOMPARE(test->role(0), QAccessible::CheckBox);
+    QCOMPARE(test->defaultAction(0), QAccessible::Press);
+    QCOMPARE(test->actionText(test->defaultAction(0), QAccessible::Name, 0), QString("Toggle"));
+    QCOMPARE(test->state(0), (int)QAccessible::Normal);
+    QVERIFY(test->doAction(QAccessible::Press, 0));
+    QTest::qWait(500);
+    QCOMPARE(test->actionText(test->defaultAction(0), QAccessible::Name, 0), QString("Check"));
+    QCOMPARE(test->state(0), (int)QAccessible::Mixed);
+    QVERIFY(test->doAction(QAccessible::Press, 0));
+    QTest::qWait(500);
+    QCOMPARE(test->actionText(test->defaultAction(0), QAccessible::Name, 0), QString("Uncheck"));
+    QCOMPARE(test->state(0), (int)QAccessible::Checked);
+    test->release();
+
+    // test radiobutton
+    QVERIFY(QAccessible::queryAccessibleInterface(&radio, &test));
+    QCOMPARE(test->role(0), QAccessible::RadioButton);
+    QCOMPARE(test->defaultAction(0), QAccessible::Press);
+    QCOMPARE(test->actionText(test->defaultAction(0), QAccessible::Name, 0), QString("Check"));
+    QCOMPARE(test->state(0), (int)QAccessible::Normal);
+    QVERIFY(test->doAction(QAccessible::Press, 0));
+    QTest::qWait(500);
+    QCOMPARE(test->actionText(test->defaultAction(0), QAccessible::Name, 0), QString("Check"));
+    QCOMPARE(test->state(0), (int)QAccessible::Checked);
+    test->release();
+
+    // test standard toolbutton
+    QVERIFY(QAccessible::queryAccessibleInterface(&toolbutton, &test));
+    QCOMPARE(test->role(0), QAccessible::PushButton);
+    QCOMPARE(test->defaultAction(0), QAccessible::Press);
+    QCOMPARE(test->actionText(test->defaultAction(0), QAccessible::Name, 0), QString("Press"));
+    QCOMPARE(test->state(0), (int)QAccessible::Normal);
+    test->release();
+
+    // toggle tool button
+    QVERIFY(QAccessible::queryAccessibleInterface(&toggletool, &test));
+    QCOMPARE(test->role(0), QAccessible::CheckBox);
+    QCOMPARE(test->defaultAction(0), QAccessible::Press);
+    QCOMPARE(test->actionText(test->defaultAction(0), QAccessible::Name, 0), QString("Check"));
+    QCOMPARE(test->state(0), (int)QAccessible::Normal);
+    QVERIFY(test->doAction(QAccessible::Press, 0));
+    QTest::qWait(500);
+    QCOMPARE(test->actionText(test->defaultAction(0), QAccessible::Name, 0), QString("Uncheck"));
+    QCOMPARE(test->state(0), (int)QAccessible::Checked);
+    test->release();
+
+    // test menu toolbutton
+    QVERIFY(QAccessible::queryAccessibleInterface(&menuToolButton, &test));
+    QCOMPARE(test->role(0), QAccessible::ButtonMenu);
+    QCOMPARE(test->defaultAction(0), 1);
+    QCOMPARE(test->actionText(test->defaultAction(0), QAccessible::Name, 0), QString("Open"));
+    QCOMPARE(test->state(0), (int)QAccessible::HasPopup);
+    QCOMPARE(test->actionCount(0), 1);
+    QCOMPARE(test->actionText(QAccessible::Press, QAccessible::Name, 0), QString("Press"));
+    test->release();
+
+    // test splitted menu toolbutton
+    QVERIFY(QAccessible::queryAccessibleInterface(&splitToolButton, &test));
+    QCOMPARE(test->childCount(), 2);
+    QCOMPARE(test->role(0), QAccessible::ButtonDropDown);
+    QCOMPARE(test->role(1), QAccessible::PushButton);
+    QCOMPARE(test->role(2), QAccessible::ButtonMenu);
+    QCOMPARE(test->defaultAction(0), QAccessible::Press);
+    QCOMPARE(test->defaultAction(1), QAccessible::Press);
+    QCOMPARE(test->defaultAction(2), QAccessible::Press);
+    QCOMPARE(test->actionText(test->defaultAction(0), QAccessible::Name, 0), QString("Press"));
+    QCOMPARE(test->state(0), (int)QAccessible::HasPopup);
+    QCOMPARE(test->actionCount(0), 1);
+    QCOMPARE(test->actionText(1, QAccessible::Name, 0), QString("Open"));
+    QCOMPARE(test->actionText(test->defaultAction(1), QAccessible::Name, 1), QString("Press"));
+    QCOMPARE(test->state(1), (int)QAccessible::Normal);
+    QCOMPARE(test->actionText(test->defaultAction(2), QAccessible::Name, 2), QString("Open"));
+    QCOMPARE(test->state(2), (int)QAccessible::HasPopup);
+    test->release();
+
+    QTestAccessibility::clearEvents();
+
+#else
+//    QSKIP("Test needs accessibility support.", SkipAll);
+    QSKIP("No action interface in Qt 4 yet.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::sliderTest()
+{
+#if !defined(QT3_SUPPORT)
+    QSKIP("This test needs Qt3Support", SkipAll);
+#else
+#ifdef QTEST_ACCESSIBILITY
+    QAccessibleInterface *test = 0;
+    Q3VBox vbox;
+    QLabel labelHorizontal("Horizontal", &vbox);
+    QSlider sliderHorizontal(Qt::Horizontal, &vbox);
+    labelHorizontal.setBuddy(&sliderHorizontal);
+
+    QLabel labelVertical("Vertical", &vbox);
+    QSlider sliderVertical(Qt::Vertical, &vbox);
+    labelVertical.setBuddy(&sliderVertical);
+    vbox.show();
+
+    // test horizontal slider
+    test = QAccessible::queryAccessibleInterface(&sliderHorizontal);
+    QVERIFY(test);
+    QCOMPARE(test->childCount(), 3);
+    QCOMPARE(test->role(0), QAccessible::Slider);
+    QCOMPARE(test->role(1), QAccessible::PushButton);
+    QCOMPARE(test->role(2), QAccessible::Indicator);
+    QCOMPARE(test->role(3), QAccessible::PushButton);
+
+    QCOMPARE(test->text(QAccessible::Name, 0), labelHorizontal.text());
+    QCOMPARE(test->text(QAccessible::Name, 1), QSlider::tr("Page left"));
+    QCOMPARE(test->text(QAccessible::Name, 2), QSlider::tr("Position"));
+    QCOMPARE(test->text(QAccessible::Name, 3), QSlider::tr("Page right"));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(sliderHorizontal.value()));
+    QCOMPARE(test->text(QAccessible::Value, 1), QString());
+    QCOMPARE(test->text(QAccessible::Value, 2), QString::number(sliderHorizontal.value()));
+    QCOMPARE(test->text(QAccessible::Value, 3), QString());
+// Skip acton tests.
+#if 0
+    QCOMPARE(test->defaultAction(0), QAccessible::SetFocus);
+    QCOMPARE(test->defaultAction(1), QAccessible::Press);
+    QCOMPARE(test->defaultAction(2), QAccessible::NoAction);
+    QCOMPARE(test->defaultAction(3), QAccessible::Press);
+    QCOMPARE(test->actionText(QAccessible::SetFocus, QAccessible::Name, 0), QSlider::tr("Set Focus"));
+    QCOMPARE(test->actionText(QAccessible::Press, QAccessible::Name, 1), QSlider::tr("Press"));
+    QCOMPARE(test->actionText(QAccessible::Increase, QAccessible::Name, 2), QSlider::tr("Increase"));
+    QCOMPARE(test->actionText(QAccessible::Decrease, QAccessible::Name, 2), QSlider::tr("Decrease"));
+    QCOMPARE(test->actionText(QAccessible::Press, QAccessible::Name, 3), QSlider::tr("Press"));
+    QVERIFY(test->doAction(QAccessible::Press, 3));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(sliderHorizontal.pageStep()));
+    QVERIFY(test->doAction(QAccessible::Press, 3));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(2*sliderHorizontal.pageStep()));
+    QVERIFY(test->doAction(QAccessible::Press, 1));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(sliderHorizontal.pageStep()));
+    QVERIFY(test->doAction(QAccessible::Press, 1));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(0));
+    QVERIFY(test->doAction(QAccessible::Increase, 2));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(sliderHorizontal.lineStep()));
+    QVERIFY(test->doAction(QAccessible::Increase, 2));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(2*sliderHorizontal.lineStep()));
+    QVERIFY(test->doAction(QAccessible::Decrease, 2));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(sliderHorizontal.lineStep()));
+    QVERIFY(test->doAction(QAccessible::Decrease, 2));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(0));
+#endif
+    delete test;
+
+    // test vertical slider
+    test = QAccessible::queryAccessibleInterface(&sliderVertical);
+    QVERIFY(test);
+    QCOMPARE(test->childCount(), 3);
+    QCOMPARE(test->role(0), QAccessible::Slider);
+    QCOMPARE(test->role(1), QAccessible::PushButton);
+    QCOMPARE(test->role(2), QAccessible::Indicator);
+    QCOMPARE(test->role(3), QAccessible::PushButton);
+
+    QCOMPARE(test->text(QAccessible::Name, 0), labelVertical.text());
+    QCOMPARE(test->text(QAccessible::Name, 1), QSlider::tr("Page up"));
+    QCOMPARE(test->text(QAccessible::Name, 2), QSlider::tr("Position"));
+    QCOMPARE(test->text(QAccessible::Name, 3), QSlider::tr("Page down"));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(sliderVertical.value()));
+    QCOMPARE(test->text(QAccessible::Value, 1), QString());
+    QCOMPARE(test->text(QAccessible::Value, 2), QString::number(sliderVertical.value()));
+    QCOMPARE(test->text(QAccessible::Value, 3), QString());
+// Skip acton tests.
+#if 0
+    QCOMPARE(test->defaultAction(0), QAccessible::SetFocus);
+    QCOMPARE(test->defaultAction(1), QAccessible::Press);
+    QCOMPARE(test->defaultAction(2), QAccessible::NoAction);
+    QCOMPARE(test->defaultAction(3), QAccessible::Press);
+    QCOMPARE(test->actionText(QAccessible::SetFocus, QAccessible::Name, 0), QSlider::tr("Set Focus"));
+    QCOMPARE(test->actionText(QAccessible::Press, QAccessible::Name, 1), QSlider::tr("Press"));
+    QCOMPARE(test->actionText(QAccessible::Increase, QAccessible::Name, 2), QSlider::tr("Increase"));
+    QCOMPARE(test->actionText(QAccessible::Decrease, QAccessible::Name, 2), QSlider::tr("Decrease"));
+    QCOMPARE(test->actionText(QAccessible::Press, QAccessible::Name, 3), QSlider::tr("Press"));
+    QVERIFY(test->doAction(QAccessible::Press, 3));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(sliderVertical.pageStep()));
+    QVERIFY(test->doAction(QAccessible::Press, 3));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(2*sliderVertical.pageStep()));
+    QVERIFY(test->doAction(QAccessible::Press, 1));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(sliderVertical.pageStep()));
+    QVERIFY(test->doAction(QAccessible::Press, 1));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(0));
+    QVERIFY(test->doAction(QAccessible::Increase, 2));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(sliderVertical.lineStep()));
+    QVERIFY(test->doAction(QAccessible::Increase, 2));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(2*sliderVertical.lineStep()));
+    QVERIFY(test->doAction(QAccessible::Decrease, 2));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(sliderVertical.lineStep()));
+    QVERIFY(test->doAction(QAccessible::Decrease, 2));
+    QCOMPARE(test->text(QAccessible::Value, 0), QString::number(0));
+#endif
+    delete test;
+
+    // Test that when we hide() a slider, the PageLeft, Indicator, and PageRight also gets the
+    // Invisible state bit set.
+    enum SubControls { PageLeft = 1, Position = 2, PageRight = 3 };
+
+    QSlider *slider  = new QSlider();
+    QAccessibleInterface * const sliderInterface = QAccessible::queryAccessibleInterface(slider);
+    QVERIFY(sliderInterface);
+
+    QVERIFY(sliderInterface->state(0)         & QAccessible::Invisible);
+    QVERIFY(sliderInterface->state(PageLeft)  & QAccessible::Invisible);
+    QVERIFY(sliderInterface->state(Position)  & QAccessible::Invisible);
+    QVERIFY(sliderInterface->state(PageRight) & QAccessible::Invisible);
+
+    slider->show();
+    QVERIFY(sliderInterface->state(0)         ^ QAccessible::Invisible);
+    QVERIFY(sliderInterface->state(PageLeft)  ^ QAccessible::Invisible);
+    QVERIFY(sliderInterface->state(Position)  ^ QAccessible::Invisible);
+    QVERIFY(sliderInterface->state(PageRight) ^ QAccessible::Invisible);
+    QVERIFY(QTestAccessibility::events().contains(QTestAccessibilityEvent(slider, 0, QAccessible::ObjectShow)));
+    QTestAccessibility::clearEvents();
+
+    slider->hide();
+    QVERIFY(sliderInterface->state(0)         & QAccessible::Invisible);
+    QVERIFY(sliderInterface->state(PageLeft)  & QAccessible::Invisible);
+    QVERIFY(sliderInterface->state(Position)  & QAccessible::Invisible);
+    QVERIFY(sliderInterface->state(PageRight) & QAccessible::Invisible);
+    QVERIFY(QTestAccessibility::events().contains(QTestAccessibilityEvent(slider, 0, QAccessible::ObjectHide)));
+    QTestAccessibility::clearEvents();
+
+    // Test that the left/right subcontrols are set to unavailable when the slider is at the minimum/maximum.
+    slider->show();
+    slider->setMinimum(0);
+    slider->setMaximum(100);
+
+    slider->setValue(50);
+    QVERIFY(sliderInterface->state(PageLeft)  ^ QAccessible::Unavailable);
+    QVERIFY(sliderInterface->state(Position)  ^ QAccessible::Unavailable);
+    QVERIFY(sliderInterface->state(PageRight) ^ QAccessible::Unavailable);
+
+    slider->setValue(0);
+    QVERIFY(sliderInterface->state(PageLeft)  & QAccessible::Unavailable);
+    QVERIFY(sliderInterface->state(Position)  ^ QAccessible::Unavailable);
+    QVERIFY(sliderInterface->state(PageRight) ^ QAccessible::Unavailable);
+
+    slider->setValue(100);
+    QVERIFY(sliderInterface->state(PageLeft)  ^ QAccessible::Unavailable);
+    QVERIFY(sliderInterface->state(Position)  ^ QAccessible::Unavailable);
+    QVERIFY(sliderInterface->state(PageRight) & QAccessible::Unavailable);
+
+    delete sliderInterface;
+    delete slider;
+
+    // Test that the rects are ok.
+    {
+        QSlider *slider  = new QSlider(Qt::Horizontal);
+        slider->show();
+#if defined(Q_WS_X11)
+        qt_x11_wait_for_window_manager(slider);
+        QTest::qWait(100);
+#endif
+        QAccessibleInterface * const sliderInterface = QAccessible::queryAccessibleInterface(slider);
+        QVERIFY(sliderInterface);
+
+        slider->setMinimum(0);
+        slider->setMaximum(100);
+        slider->setValue(50);
+
+        const QRect sliderRect = sliderInterface->rect(0);
+        QVERIFY(sliderRect.isValid());
+
+        // Verify that the sub-control rects are valid and inside the slider rect.
+        for (int i = PageLeft; i <= PageRight; ++i) {
+            const QRect testRect = sliderInterface->rect(i);
+            QVERIFY(testRect.isValid());
+            QVERIFY(sliderRect.contains(testRect));
+        }
+
+        delete slider;
+        delete sliderInterface;
+    }
+
+
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+#endif //!QT3_SUPPORT
+}
+
+void tst_QAccessibility::scrollBarTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    // Test that when we hide() a slider, the PageLeft, Indicator, and PageRight also gets the
+    // Invisible state bit set.
+    enum SubControls { LineUp = 1,
+        PageUp = 2,
+        Position = 3,
+        PageDown = 4,
+        LineDown = 5
+ };
+
+    QScrollBar *scrollBar  = new QScrollBar();
+    QAccessibleInterface * const scrollBarInterface = QAccessible::queryAccessibleInterface(scrollBar);
+    QVERIFY(scrollBarInterface);
+
+    QVERIFY(scrollBarInterface->state(0)         & QAccessible::Invisible);
+    QVERIFY(scrollBarInterface->state(PageUp)    & QAccessible::Invisible);
+    QVERIFY(scrollBarInterface->state(Position)  & QAccessible::Invisible);
+    QVERIFY(scrollBarInterface->state(PageDown)  & QAccessible::Invisible);
+
+    scrollBar->show();
+    QVERIFY(scrollBarInterface->state(0)         ^ QAccessible::Invisible);
+    QVERIFY(scrollBarInterface->state(PageUp)    ^ QAccessible::Invisible);
+    QVERIFY(scrollBarInterface->state(Position)  ^ QAccessible::Invisible);
+    QVERIFY(scrollBarInterface->state(PageDown)  ^ QAccessible::Invisible);
+    QVERIFY(QTestAccessibility::events().contains(QTestAccessibilityEvent(scrollBar, 0, QAccessible::ObjectShow)));
+    QTestAccessibility::clearEvents();
+
+    scrollBar->hide();
+    QVERIFY(scrollBarInterface->state(0)         & QAccessible::Invisible);
+    QVERIFY(scrollBarInterface->state(PageUp)    & QAccessible::Invisible);
+    QVERIFY(scrollBarInterface->state(Position)  & QAccessible::Invisible);
+    QVERIFY(scrollBarInterface->state(PageDown)  & QAccessible::Invisible);
+    QVERIFY(QTestAccessibility::events().contains(QTestAccessibilityEvent(scrollBar, 0, QAccessible::ObjectHide)));
+    QTestAccessibility::clearEvents();
+
+    // Test that the left/right subcontrols are set to unavailable when the scrollBar is at the minimum/maximum.
+    scrollBar->show();
+    scrollBar->setMinimum(0);
+    scrollBar->setMaximum(100);
+
+    scrollBar->setValue(50);
+    QVERIFY(scrollBarInterface->state(PageUp)    ^ QAccessible::Unavailable);
+    QVERIFY(scrollBarInterface->state(Position)  ^ QAccessible::Unavailable);
+    QVERIFY(scrollBarInterface->state(PageDown)  ^ QAccessible::Unavailable);
+
+    scrollBar->setValue(0);
+    QVERIFY(scrollBarInterface->state(PageUp)    & QAccessible::Unavailable);
+    QVERIFY(scrollBarInterface->state(Position)  ^ QAccessible::Unavailable);
+    QVERIFY(scrollBarInterface->state(PageDown)  ^ QAccessible::Unavailable);
+
+    scrollBar->setValue(100);
+    QVERIFY(scrollBarInterface->state(PageUp)   ^ QAccessible::Unavailable);
+    QVERIFY(scrollBarInterface->state(Position) ^ QAccessible::Unavailable);
+    QVERIFY(scrollBarInterface->state(PageDown) & QAccessible::Unavailable);
+
+    delete scrollBarInterface;
+    delete scrollBar;
+
+    // Test that the rects are ok.
+    {
+        QScrollBar *scrollBar  = new QScrollBar(Qt::Horizontal);
+        scrollBar->resize(200, 50);
+        scrollBar->show();
+#if defined(Q_WS_X11)
+        qt_x11_wait_for_window_manager(scrollBar);
+        QTest::qWait(100);
+#endif
+        QAccessibleInterface * const scrollBarInterface = QAccessible::queryAccessibleInterface(scrollBar);
+        QVERIFY(scrollBarInterface);
+
+        scrollBar->setMinimum(0);
+        scrollBar->setMaximum(100);
+        scrollBar->setValue(50);
+
+        QApplication::processEvents();
+
+        const QRect scrollBarRect = scrollBarInterface->rect(0);
+        QVERIFY(scrollBarRect.isValid());
+
+        // Verify that the sub-control rects are valid and inside the scrollBar rect.
+        for (int i = LineUp; i <= LineDown; ++i) {
+            const QRect testRect = scrollBarInterface->rect(i);
+            QVERIFY(testRect.isValid());
+            QVERIFY(scrollBarRect.contains(testRect));
+        }
+        delete scrollBarInterface;
+        delete scrollBar;
+    }
+
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+
+}
+
+void tst_QAccessibility::tabTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QTabBar *tabBar = new QTabBar();
+    tabBar->show();
+
+    QAccessibleInterface * const interface = QAccessible::queryAccessibleInterface(tabBar);
+    QVERIFY(interface);
+    QCOMPARE(interface->childCount(), 2);
+    interface->doAction(QAccessible::Press, 1);
+    interface->doAction(QAccessible::Press, 2);
+
+    // Test that the Invisible bit for the navigation buttons gets set
+    // and cleared correctly.
+    QVERIFY(interface->state(1) & QAccessible::Invisible);
+
+    const int lots = 10;
+    for (int i = 0; i < lots; ++i)
+        tabBar->addTab("Foo");
+
+    QVERIFY((interface->state(1) & QAccessible::Invisible) == false);
+    tabBar->hide();
+    QVERIFY(interface->state(1) & QAccessible::Invisible);
+
+    tabBar->show();
+    tabBar->setCurrentIndex(0);
+
+    // Test that sending a focus action to a tab does not select it.
+    interface->doAction(QAccessible::Focus, 2, QVariantList());
+    QCOMPARE(tabBar->currentIndex(), 0);
+
+    // Test that sending a press action to a tab selects it.
+    interface->doAction(QAccessible::Press, 2, QVariantList());
+    QCOMPARE(tabBar->currentIndex(), 1);
+
+    delete tabBar;
+    delete interface;
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::menuTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    {
+    QMainWindow mw;
+    mw.resize(300, 200);
+    QMenu *file = mw.menuBar()->addMenu("&File");
+    QMenu *fileNew = file->addMenu("&New...");
+    fileNew->menuAction()->setShortcut(tr("Ctrl+N"));
+    fileNew->addAction("Text file");
+    fileNew->addAction("Image file");
+    file->addAction("&Open")->setShortcut(tr("Ctrl+O"));
+    file->addAction("&Save")->setShortcut(tr("Ctrl+S"));
+    file->addSeparator();
+    file->addAction("E&xit")->setShortcut(tr("Alt+F4"));
+
+    QMenu *edit = mw.menuBar()->addMenu("&Edit");
+    edit->addAction("&Undo")->setShortcut(tr("Ctrl+Z"));
+    edit->addAction("&Redo")->setShortcut(tr("Ctrl+Y"));
+    edit->addSeparator();
+    edit->addAction("Cu&t")->setShortcut(tr("Ctrl+X"));
+    edit->addAction("&Copy")->setShortcut(tr("Ctrl+C"));
+    edit->addAction("&Paste")->setShortcut(tr("Ctrl+V"));
+    edit->addAction("&Delete")->setShortcut(tr("Del"));
+    edit->addSeparator();
+    edit->addAction("Pr&operties");
+
+    mw.menuBar()->addSeparator();
+
+    QMenu *help = mw.menuBar()->addMenu("&Help");
+    help->addAction("&Contents");
+    help->addAction("&About");
+
+    mw.menuBar()->addAction("Action!");
+
+    mw.show(); // triggers layout
+    QTest::qWait(100);
+
+    QAccessibleInterface *interface = QAccessible::queryAccessibleInterface(mw.menuBar());
+
+    QCOMPARE(verifyHierarchy(interface),  0);
+
+    QVERIFY(interface);
+    QCOMPARE(interface->childCount(), 5);
+    QCOMPARE(interface->role(0), QAccessible::MenuBar);
+    QCOMPARE(interface->role(1), QAccessible::MenuItem);
+    QCOMPARE(interface->role(2), QAccessible::MenuItem);
+    QCOMPARE(interface->role(3), QAccessible::Separator);
+    QCOMPARE(interface->role(4), QAccessible::MenuItem);
+    QCOMPARE(interface->role(5), QAccessible::MenuItem);
+#ifndef Q_WS_MAC
+#ifdef Q_OS_WINCE
+    if (!IsValidCEPlatform()) {
+        QSKIP("Tests do not work on Mobile platforms due to native menus", SkipAll);
+    }
+#endif
+    QCOMPARE(mw.mapFromGlobal(interface->rect(0).topLeft()), mw.menuBar()->geometry().topLeft());
+    QCOMPARE(interface->rect(0).size(), mw.menuBar()->size());
+
+    QVERIFY(interface->rect(0).contains(interface->rect(1)));
+    QVERIFY(interface->rect(0).contains(interface->rect(2)));
+    // QVERIFY(interface->rect(0).contains(interface->rect(3))); //separator might be invisible
+    QVERIFY(interface->rect(0).contains(interface->rect(4)));
+    QVERIFY(interface->rect(0).contains(interface->rect(5)));
+#endif
+
+    QCOMPARE(interface->text(QAccessible::Name, 1), QString("File"));
+    QCOMPARE(interface->text(QAccessible::Name, 2), QString("Edit"));
+    QCOMPARE(interface->text(QAccessible::Name, 3), QString());
+    QCOMPARE(interface->text(QAccessible::Name, 4), QString("Help"));
+    QCOMPARE(interface->text(QAccessible::Name, 5), QString("Action!"));
+
+// TODO: Currently not working, task to fix is #100019.
+#ifndef Q_OS_MAC
+    QCOMPARE(interface->text(QAccessible::Accelerator, 1), tr("Alt+F"));
+    QCOMPARE(interface->text(QAccessible::Accelerator, 2), tr("Alt+E"));
+    QCOMPARE(interface->text(QAccessible::Accelerator, 4), tr("Alt+H"));
+    QCOMPARE(interface->text(QAccessible::Accelerator, 3), QString());
+    QCOMPARE(interface->text(QAccessible::Accelerator, 4), tr("Alt+H"));
+    QCOMPARE(interface->text(QAccessible::Accelerator, 5), QString());
+#endif
+
+    QCOMPARE(interface->actionText(QAccessible::DefaultAction, QAccessible::Name, 1), QString("Open"));
+    QCOMPARE(interface->actionText(QAccessible::DefaultAction, QAccessible::Name, 2), QString("Open"));
+    QCOMPARE(interface->actionText(QAccessible::DefaultAction, QAccessible::Name, 3), QString());
+    QCOMPARE(interface->actionText(QAccessible::DefaultAction, QAccessible::Name, 4), QString("Open"));
+    QCOMPARE(interface->actionText(QAccessible::DefaultAction, QAccessible::Name, 5), QString("Execute"));
+
+    bool menuFade = qApp->isEffectEnabled(Qt::UI_FadeMenu);
+    int menuFadeDelay = 300;
+    interface->doAction(QAccessible::DefaultAction, 1);
+    if(menuFade)
+        QTest::qWait(menuFadeDelay);
+    QVERIFY(file->isVisible() && !edit->isVisible() && !help->isVisible());
+    interface->doAction(QAccessible::DefaultAction, 2);
+    if(menuFade)
+        QTest::qWait(menuFadeDelay);
+    QVERIFY(!file->isVisible() && edit->isVisible() && !help->isVisible());
+    interface->doAction(QAccessible::DefaultAction, 3);
+    if(menuFade)
+        QTest::qWait(menuFadeDelay);
+    QVERIFY(!file->isVisible() && !edit->isVisible() && !help->isVisible());
+    interface->doAction(QAccessible::DefaultAction, 4);
+    if(menuFade)
+        QTest::qWait(menuFadeDelay);
+    QVERIFY(!file->isVisible() && !edit->isVisible() && help->isVisible());
+    interface->doAction(QAccessible::DefaultAction, 5);
+    if(menuFade)
+        QTest::qWait(menuFadeDelay);
+    QVERIFY(!file->isVisible() && !edit->isVisible() && !help->isVisible());
+
+    interface->doAction(QAccessible::DefaultAction, 1);
+    delete interface;
+    interface = QAccessible::queryAccessibleInterface(file);
+    QCOMPARE(interface->childCount(), 5);
+    QCOMPARE(interface->role(0), QAccessible::PopupMenu);
+    QCOMPARE(interface->role(1), QAccessible::MenuItem);
+    QCOMPARE(interface->role(2), QAccessible::MenuItem);
+    QCOMPARE(interface->role(3), QAccessible::MenuItem);
+    QCOMPARE(interface->role(4), QAccessible::Separator);
+    QCOMPARE(interface->role(5), QAccessible::MenuItem);
+    QCOMPARE(interface->actionText(QAccessible::DefaultAction, QAccessible::Name, 1), QString("Open"));
+    QCOMPARE(interface->actionText(QAccessible::DefaultAction, QAccessible::Name, 2), QString("Execute"));
+    QCOMPARE(interface->actionText(QAccessible::DefaultAction, QAccessible::Name, 3), QString("Execute"));
+    QCOMPARE(interface->actionText(QAccessible::DefaultAction, QAccessible::Name, 4), QString());
+    QCOMPARE(interface->actionText(QAccessible::DefaultAction, QAccessible::Name, 5), QString("Execute"));
+
+    QAccessibleInterface *iface = 0;
+    QAccessibleInterface *iface2 = 0;
+
+    // traverse siblings with navigate(Sibling, ...)
+    int entry = interface->navigate(QAccessible::Child, 1, &iface);
+    QCOMPARE(entry, 0);
+    QVERIFY(iface);
+    QCOMPARE(iface->role(0), QAccessible::MenuItem);
+
+    QAccessible::Role fileRoles[5] = {
+        QAccessible::MenuItem,
+        QAccessible::MenuItem,
+        QAccessible::MenuItem,
+        QAccessible::Separator,
+        QAccessible::MenuItem
+    };
+    for (int child = 0; child < 5; ++child) {
+        entry = iface->navigate(QAccessible::Sibling, child + 1, &iface2);
+        QCOMPARE(entry, 0);
+        QVERIFY(iface2);
+        QCOMPARE(iface2->role(0), fileRoles[child]);
+        delete iface2;
+    }
+    delete iface;
+
+    // traverse menu items with navigate(Down, ...)
+    entry = interface->navigate(QAccessible::Child, 1, &iface);
+    QCOMPARE(entry, 0);
+    QVERIFY(iface);
+    QCOMPARE(iface->role(0), QAccessible::MenuItem);
+
+    for (int child = 0; child < 4; ++child) {
+        entry = iface->navigate(QAccessible::Down, 1, &iface2);
+        delete iface;
+        iface = iface2;
+        QCOMPARE(entry, 0);
+        QVERIFY(iface);
+        QCOMPARE(iface->role(0), fileRoles[child + 1]);
+    }
+    delete iface;
+
+    // traverse menu items with navigate(Up, ...)
+    entry = interface->navigate(QAccessible::Child, interface->childCount(), &iface);
+    QCOMPARE(entry, 0);
+    QVERIFY(iface);
+    QCOMPARE(iface->role(0), QAccessible::MenuItem);
+
+    for (int child = 3; child >= 0; --child) {
+        entry = iface->navigate(QAccessible::Up, 1, &iface2);
+        delete iface;
+        iface = iface2;
+        QCOMPARE(entry, 0);
+        QVERIFY(iface);
+        QCOMPARE(iface->role(0), fileRoles[child]);
+    }
+    delete iface;
+
+    // "New" item
+    entry = interface->navigate(QAccessible::Child, 1, &iface);
+    QCOMPARE(entry, 0);
+    QVERIFY(iface);
+    QCOMPARE(iface->role(0), QAccessible::MenuItem);
+
+    // "New" menu
+    entry = iface->navigate(QAccessible::Child, 1, &iface2);
+    delete iface;
+    iface = iface2;
+    QCOMPARE(entry, 0);
+    QVERIFY(iface);
+    QCOMPARE(iface->role(0), QAccessible::PopupMenu);
+
+    // "Text file" menu item
+    entry = iface->navigate(QAccessible::Child, 1, &iface2);
+    delete iface;
+    iface = iface2;
+    QCOMPARE(entry, 0);
+    QVERIFY(iface);
+    QCOMPARE(iface->role(0), QAccessible::MenuItem);
+
+    // Traverse to the menubar.
+    QAccessibleInterface *ifaceMenuBar = 0;
+    entry = iface->navigate(QAccessible::Ancestor, 5, &ifaceMenuBar);
+    QCOMPARE(ifaceMenuBar->role(0), QAccessible::MenuBar);
+    delete ifaceMenuBar;
+
+    delete iface;
+
+    // move mouse pointer away, since that might influence the
+    // subsequent tests
+    QTest::mouseMove(&mw, QPoint(-1, -1));
+    QTest::qWait(100);
+    if (menuFade)
+        QTest::qWait(menuFadeDelay);
+    interface->doAction(QAccessible::DefaultAction, 1);
+    QTestEventLoop::instance().enterLoop(2);
+
+    QVERIFY(file->isVisible());
+    QVERIFY(fileNew->isVisible());
+    QVERIFY(!edit->isVisible());
+    QVERIFY(!help->isVisible());
+
+    QTestAccessibility::clearEvents();
+    mw.hide();
+
+
+    // Do not crash if the menu don't have a parent
+    QMenu *menu = new QMenu;
+    menu->addAction(QLatin1String("one"));
+    menu->addAction(QLatin1String("two"));
+    menu->addAction(QLatin1String("three"));
+    iface = QAccessible::queryAccessibleInterface(menu);
+    QCOMPARE(iface->navigate(QAccessible::Ancestor, 1, &iface2), 0);
+    QCOMPARE(iface2->role(0), QAccessible::Application);
+    // caused a *crash*
+    iface2->state(0);
+    delete iface2;
+    delete iface;
+    delete menu;
+
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs Qt >= 0x040000 and accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::spinBoxTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QSpinBox * const spinBox = new QSpinBox();
+    spinBox->show();
+
+    QAccessibleInterface * const interface = QAccessible::queryAccessibleInterface(spinBox);
+    QVERIFY(interface);
+
+    const QRect widgetRect = spinBox->geometry();
+    const QRect accessibleRect = interface->rect(0);
+    QCOMPARE(accessibleRect, widgetRect);
+
+    // Test that we get valid rects for all the spinbox child interfaces.
+    const int numChildren = interface->childCount();
+    for (int i = 1; i <= numChildren; ++i) {
+        const QRect childRect = interface->rect(i);
+        QVERIFY(childRect.isNull() == false);
+    }
+
+    spinBox->setFocus();
+    QTestAccessibility::clearEvents();
+    QTest::keyPress(spinBox, Qt::Key_Up);
+    QTest::qWait(200);
+    EventList events = QTestAccessibility::events();
+    QTestAccessibilityEvent expectedEvent(spinBox, 0, (int)QAccessible::ValueChanged);
+    QVERIFY(events.contains(expectedEvent));
+    delete spinBox;
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::doubleSpinBoxTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QDoubleSpinBox *doubleSpinBox = new QDoubleSpinBox;
+    doubleSpinBox->show();
+
+    QAccessibleInterface *interface = QAccessible::queryAccessibleInterface(doubleSpinBox);
+    QVERIFY(interface);
+
+    const QRect widgetRect = doubleSpinBox->geometry();
+    const QRect accessibleRect = interface->rect(0);
+    QCOMPARE(accessibleRect, widgetRect);
+
+    // Test that we get valid rects for all the spinbox child interfaces.
+    const int numChildren = interface->childCount();
+    for (int i = 1; i <= numChildren; ++i) {
+        const QRect childRect = interface->rect(i);
+        QVERIFY(childRect.isValid());
+    }
+
+    delete doubleSpinBox;
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::textEditTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    {
+    QTextEdit edit;
+    QString text = "hello world\nhow are you today?\n";
+    edit.setText(text);
+    edit.show();
+
+    QAccessibleInterface *iface = QAccessible::queryAccessibleInterface(&edit);
+    QCOMPARE(iface->text(QAccessible::Value, 0), text);
+    QCOMPARE(iface->childCount(), 6);
+    QCOMPARE(iface->text(QAccessible::Value, 4), QString("hello world"));
+    QCOMPARE(iface->text(QAccessible::Value, 5), QString("how are you today?"));
+    QCOMPARE(iface->text(QAccessible::Value, 6), QString());
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::textBrowserTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    {
+    QTextBrowser textBrowser;
+    QString text = QLatin1String("Hello world\nhow are you today?\n");
+    textBrowser.setText(text);
+    textBrowser.show();
+
+    QAccessibleInterface *interface = QAccessible::queryAccessibleInterface(&textBrowser);
+    QVERIFY(interface);
+    QCOMPARE(interface->role(0), QAccessible::StaticText);
+    QCOMPARE(interface->text(QAccessible::Value, 0), text);
+    QCOMPARE(interface->childCount(), 6);
+    QCOMPARE(interface->text(QAccessible::Value, 4), QString("Hello world"));
+    QCOMPARE(interface->text(QAccessible::Value, 5), QString("how are you today?"));
+    QCOMPARE(interface->text(QAccessible::Value, 6), QString());
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::listViewTest()
+{
+#if 1 //def QTEST_ACCESSIBILITY
+    {
+        QListView listView;
+        QAccessibleInterface *iface = QAccessible::queryAccessibleInterface(&listView);
+        QVERIFY(iface);
+        QCOMPARE(iface->childCount(), 1);
+        delete iface;
+    }
+    {
+    QListWidget listView;
+    listView.addItem(tr("A"));
+    listView.addItem(tr("B"));
+    listView.addItem(tr("C"));
+    listView.resize(400,400);
+    listView.show();
+    QTest::qWait(1); // Need this for indexOfchild to work.
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(&listView);
+    QTest::qWait(100);
+#endif
+
+    QAccessibleInterface *iface = QAccessible::queryAccessibleInterface(&listView);
+    QCOMPARE((int)iface->role(0), (int)QAccessible::Client);
+    QCOMPARE((int)iface->role(1), (int)QAccessible::List);
+    QCOMPARE(iface->childCount(), 1);
+    QAccessibleInterface *child;
+    iface->navigate(QAccessible::Child, 1, &child);
+    delete iface;
+    iface = child;
+    QCOMPARE(iface->text(QAccessible::Name, 1), QString("A"));
+    QCOMPARE(iface->text(QAccessible::Name, 2), QString("B"));
+    QCOMPARE(iface->text(QAccessible::Name, 3), QString("C"));
+
+    QCOMPARE(iface->childCount(), 3);
+
+    QAccessibleInterface *childA = 0;
+    QCOMPARE(iface->navigate(QAccessible::Child, 1, &childA), 0);
+    QVERIFY(childA);
+    QCOMPARE(iface->indexOfChild(childA), 1);
+    QCOMPARE(childA->text(QAccessible::Name, 1), QString("A"));
+    delete childA;
+
+    QAccessibleInterface *childB = 0;
+    QCOMPARE(iface->navigate(QAccessible::Child, 2, &childB), 0);
+    QVERIFY(childB);
+    QCOMPARE(iface->indexOfChild(childB), 2);
+    QCOMPARE(childB->text(QAccessible::Name, 1), QString("B"));
+    delete childB;
+
+    QAccessibleInterface *childC = 0;
+    QCOMPARE(iface->navigate(QAccessible::Child, 3, &childC), 0);
+    QVERIFY(childC);
+    QCOMPARE(iface->indexOfChild(childC), 3);
+    QCOMPARE(childC->text(QAccessible::Name, 1), QString("C"));
+    delete childC;
+    QTestAccessibility::clearEvents();
+
+    // Check for events
+    QTest::mouseClick(listView.viewport(), Qt::LeftButton, 0, listView.visualItemRect(listView.item(1)).center());
+    QTest::mouseClick(listView.viewport(), Qt::LeftButton, 0, listView.visualItemRect(listView.item(2)).center());
+    QVERIFY(QTestAccessibility::events().contains(QTestAccessibilityEvent(listView.viewport(), 2, QAccessible::Selection)));
+    QVERIFY(QTestAccessibility::events().contains(QTestAccessibilityEvent(listView.viewport(), 3, QAccessible::Selection)));
+    delete iface;
+
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+
+void tst_QAccessibility::mdiAreaTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    {
+    QMdiArea mdiArea;
+    mdiArea.resize(400,300);
+    mdiArea.show();
+    const int subWindowCount = 3;
+    for (int i = 0; i < subWindowCount; ++i)
+        mdiArea.addSubWindow(new QWidget, Qt::Dialog)->show();
+
+    QList<QMdiSubWindow *> subWindows = mdiArea.subWindowList();
+    QCOMPARE(subWindows.count(), subWindowCount);
+
+    QAccessibleInterface *interface = QAccessible::queryAccessibleInterface(&mdiArea);
+    QVERIFY(interface);
+    QCOMPARE(interface->childCount(), subWindowCount);
+
+    // Right, right, right, ...
+    for (int i = 0; i < subWindowCount; ++i) {
+        QAccessibleInterface *destination = 0;
+        int index = interface->navigate(QAccessible::Right, i + 1, &destination);
+        if (i == subWindowCount - 1) {
+            QVERIFY(!destination);
+            QCOMPARE(index, -1);
+        } else {
+            QVERIFY(destination);
+            QCOMPARE(index, 0);
+            QCOMPARE(destination->object(), (QObject*)subWindows.at(i + 1));
+            delete destination;
+        }
+    }
+
+    // Left, left, left, ...
+    for (int i = subWindowCount; i > 0; --i) {
+        QAccessibleInterface *destination = 0;
+        int index = interface->navigate(QAccessible::Left, i, &destination);
+        if (i == 1) {
+            QVERIFY(!destination);
+            QCOMPARE(index, -1);
+        } else {
+            QVERIFY(destination);
+            QCOMPARE(index, 0);
+            QCOMPARE(destination->object(), (QObject*)subWindows.at(i - 2));
+            delete destination;
+        }
+    }
+    // ### Add test for Up and Down.
+
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::mdiSubWindowTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    {
+    QMdiArea mdiArea;
+    mdiArea.show();
+    qApp->setActiveWindow(&mdiArea);
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(&mdiArea);
+    QTest::qWait(150);
+#endif
+
+    bool isSubWindowsPlacedNextToEachOther = false;
+    const int subWindowCount =  5;
+    for (int i = 0; i < subWindowCount; ++i) {
+        QMdiSubWindow *window = mdiArea.addSubWindow(new QPushButton("QAccessibilityTest"));
+        window->show();
+        // Parts of this test requires that the sub windows are placed next
+        // to each other. In order to achieve that QMdiArea must have
+        // a width which is larger than subWindow->width() * subWindowCount.
+        if (i == 0) {
+            int minimumWidth = window->width() * subWindowCount + 20;
+            mdiArea.resize(mdiArea.size().expandedTo(QSize(minimumWidth, 0)));
+#if defined(Q_WS_X11)
+            qt_x11_wait_for_window_manager(&mdiArea);
+            QTest::qWait(100);
+#endif
+            if (mdiArea.width() >= minimumWidth)
+                isSubWindowsPlacedNextToEachOther = true;
+        }
+    }
+
+    QList<QMdiSubWindow *> subWindows = mdiArea.subWindowList();
+    QCOMPARE(subWindows.count(), subWindowCount);
+
+    QMdiSubWindow *testWindow = subWindows.at(3);
+    QVERIFY(testWindow);
+    QAccessibleInterface *interface = QAccessible::queryAccessibleInterface(testWindow);
+
+    // childCount
+    QVERIFY(interface);
+    QCOMPARE(interface->childCount(), 1);
+
+    // setText / text
+    QCOMPARE(interface->text(QAccessible::Name, 0), QString());
+    QCOMPARE(interface->text(QAccessible::Name, 1), QString());
+    testWindow->setWindowTitle(QLatin1String("ReplaceMe"));
+    QCOMPARE(interface->text(QAccessible::Name, 0), QLatin1String("ReplaceMe"));
+    QCOMPARE(interface->text(QAccessible::Name, 1), QLatin1String("ReplaceMe"));
+    interface->setText(QAccessible::Name, 0, QLatin1String("TitleSetOnWindow"));
+    QCOMPARE(interface->text(QAccessible::Name, 0), QLatin1String("TitleSetOnWindow"));
+    interface->setText(QAccessible::Name, 1, QLatin1String("TitleSetOnChild"));
+    QCOMPARE(interface->text(QAccessible::Name, 0), QLatin1String("TitleSetOnChild"));
+
+    mdiArea.setActiveSubWindow(testWindow);
+
+    // state
+    QAccessible::State state = QAccessible::Normal | QAccessible::Focusable | QAccessible::Focused
+                               | QAccessible::Movable | QAccessible::Sizeable;
+    QCOMPARE(interface->state(0), state);
+    const QRect originalGeometry = testWindow->geometry();
+    testWindow->showMaximized();
+    state &= ~QAccessible::Sizeable;
+    state &= ~QAccessible::Movable;
+    QCOMPARE(interface->state(0), state);
+    testWindow->showNormal();
+    testWindow->move(-10, 0);
+    QVERIFY(interface->state(0) & QAccessible::Offscreen);
+    testWindow->setVisible(false);
+    QVERIFY(interface->state(0) & QAccessible::Invisible);
+    testWindow->setVisible(true);
+    testWindow->setEnabled(false);
+    QVERIFY(interface->state(0) & QAccessible::Unavailable);
+    testWindow->setEnabled(true);
+    qApp->setActiveWindow(&mdiArea);
+    mdiArea.setActiveSubWindow(testWindow);
+    testWindow->setFocus();
+    QVERIFY(testWindow->isAncestorOf(qApp->focusWidget()));
+    QVERIFY(interface->state(0) & QAccessible::Focused);
+    testWindow->setGeometry(originalGeometry);
+
+    if (isSubWindowsPlacedNextToEachOther) {
+        // This part of the test can only be run if the sub windows are
+        // placed next to each other.
+        QAccessibleInterface *destination = 0;
+        QCOMPARE(interface->navigate(QAccessible::Child, 1, &destination), 0);
+        QVERIFY(destination);
+        QCOMPARE(destination->object(), (QObject*)testWindow->widget());
+        delete destination;
+        QCOMPARE(interface->navigate(QAccessible::Left, 0, &destination), 0);
+        QVERIFY(destination);
+        QCOMPARE(destination->object(), (QObject*)subWindows.at(2));
+        delete destination;
+        QCOMPARE(interface->navigate(QAccessible::Right, 0, &destination), 0);
+        QVERIFY(destination);
+        QCOMPARE(destination->object(), (QObject*)subWindows.at(4));
+        delete destination;
+    }
+
+    // rect
+    const QPoint globalPos = testWindow->mapToGlobal(QPoint(0, 0));
+    QCOMPARE(interface->rect(0), QRect(globalPos, testWindow->size()));
+    testWindow->hide();
+    QCOMPARE(interface->rect(0), QRect());
+    QCOMPARE(interface->rect(1), QRect());
+    testWindow->showMinimized();
+    QCOMPARE(interface->rect(1), QRect());
+    testWindow->showNormal();
+    testWindow->widget()->hide();
+    QCOMPARE(interface->rect(1), QRect());
+    testWindow->widget()->show();
+    const QRect widgetGeometry = testWindow->contentsRect();
+    const QPoint globalWidgetPos = QPoint(globalPos.x() + widgetGeometry.x(),
+                                          globalPos.y() + widgetGeometry.y());
+    QCOMPARE(interface->rect(1), QRect(globalWidgetPos, widgetGeometry.size()));
+
+    // childAt
+    QCOMPARE(interface->childAt(-10, 0), -1);
+    QCOMPARE(interface->childAt(globalPos.x(), globalPos.y()), 0);
+    QCOMPARE(interface->childAt(globalWidgetPos.x(), globalWidgetPos.y()), 1);
+    testWindow->widget()->hide();
+    QCOMPARE(interface->childAt(globalWidgetPos.x(), globalWidgetPos.y()), 0);
+
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::lineEditTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QLineEdit *le = new QLineEdit;
+    QAccessibleInterface *iface = QAccessible::queryAccessibleInterface(le);
+    QVERIFY(iface);
+    le->show();
+
+    QApplication::processEvents();
+    QCOMPARE(iface->childCount(), 0);
+    QVERIFY(iface->state(0) & QAccessible::Sizeable);
+    QVERIFY(iface->state(0) & QAccessible::Movable);
+    QCOMPARE(bool(iface->state(0) & QAccessible::Focusable), le->isActiveWindow());
+    QVERIFY(iface->state(0) & QAccessible::Selectable);
+    QVERIFY(iface->state(0) & QAccessible::HasPopup);
+    QCOMPARE(bool(iface->state(0) & QAccessible::Focused), le->hasFocus());
+
+    QString secret(QLatin1String("secret"));
+    le->setText(secret);
+    le->setEchoMode(QLineEdit::Normal);
+    QVERIFY(!(iface->state(0) & QAccessible::Protected));
+    QCOMPARE(iface->text(QAccessible::Value, 0), secret);
+    le->setEchoMode(QLineEdit::NoEcho);
+    QVERIFY(iface->state(0) & QAccessible::Protected);
+    QVERIFY(iface->text(QAccessible::Value, 0).isEmpty());
+    le->setEchoMode(QLineEdit::Password);
+    QVERIFY(iface->state(0) & QAccessible::Protected);
+    QVERIFY(iface->text(QAccessible::Value, 0).isEmpty());
+    le->setEchoMode(QLineEdit::PasswordEchoOnEdit);
+    QVERIFY(iface->state(0) & QAccessible::Protected);
+    QVERIFY(iface->text(QAccessible::Value, 0).isEmpty());
+    le->setEchoMode(QLineEdit::Normal);
+    QVERIFY(!(iface->state(0) & QAccessible::Protected));
+    QCOMPARE(iface->text(QAccessible::Value, 0), secret);
+
+    QWidget *toplevel = new QWidget;
+    le->setParent(toplevel);
+    toplevel->show();
+    QApplication::processEvents();
+    QVERIFY(!(iface->state(0) & QAccessible::Sizeable));
+    QVERIFY(!(iface->state(0) & QAccessible::Movable));
+    QCOMPARE(bool(iface->state(0) & QAccessible::Focusable), le->isActiveWindow());
+    QVERIFY(iface->state(0) & QAccessible::Selectable);
+    QVERIFY(iface->state(0) & QAccessible::HasPopup);
+    QCOMPARE(bool(iface->state(0) & QAccessible::Focused), le->hasFocus());
+
+    QLineEdit *le2 = new QLineEdit(toplevel);
+    le2->show();
+    QTest::qWait(100);
+    le2->activateWindow();
+    QTest::qWait(100);
+    le->setFocus(Qt::TabFocusReason);
+    QTestAccessibility::clearEvents();
+    le2->setFocus(Qt::TabFocusReason);
+    QTRY_VERIFY(QTestAccessibility::events().contains(QTestAccessibilityEvent(le2, 0, QAccessible::Focus)));
+    delete iface;
+    delete le;
+    delete le2;
+    delete toplevel;
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::workspaceTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    {
+    QWorkspace workspace;
+    workspace.resize(400,300);
+    workspace.show();
+    const int subWindowCount =  3;
+    for (int i = 0; i < subWindowCount; ++i) {
+        QWidget *window = workspace.addWindow(new QWidget);
+        if (i > 0)
+            window->move(window->x() + 1, window->y());
+        window->show();
+        window->resize(70, window->height());
+    }
+
+    QWidgetList subWindows = workspace.windowList();
+    QCOMPARE(subWindows.count(), subWindowCount);
+
+    QAccessibleInterface *interface = QAccessible::queryAccessibleInterface(&workspace);
+    QVERIFY(interface);
+    QCOMPARE(interface->childCount(), subWindowCount);
+
+    // Right, right, right, ...
+    for (int i = 0; i < subWindowCount; ++i) {
+        QAccessibleInterface *destination = 0;
+        int index = interface->navigate(QAccessible::Right, i + 1, &destination);
+        if (i == subWindowCount - 1) {
+            QVERIFY(!destination);
+            QCOMPARE(index, -1);
+        } else {
+            QVERIFY(destination);
+            QCOMPARE(index, 0);
+            QCOMPARE(destination->object(), (QObject*)subWindows.at(i + 1));
+            delete destination;
+        }
+    }
+
+    // Left, left, left, ...
+    for (int i = subWindowCount; i > 0; --i) {
+        QAccessibleInterface *destination = 0;
+        int index = interface->navigate(QAccessible::Left, i, &destination);
+        if (i == 1) {
+            QVERIFY(!destination);
+            QCOMPARE(index, -1);
+        } else {
+            QVERIFY(destination);
+            QCOMPARE(index, 0);
+            QCOMPARE(destination->object(), (QObject*)subWindows.at(i - 2));
+            delete destination;
+        }
+    }
+    // ### Add test for Up and Down.
+
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::dialogButtonBoxTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    {
+    QDialogButtonBox box(QDialogButtonBox::Reset |
+                         QDialogButtonBox::Help |
+                         QDialogButtonBox::Ok, Qt::Horizontal);
+
+
+    QAccessibleInterface *iface = QAccessible::queryAccessibleInterface(&box);
+    QVERIFY(iface);
+    box.show();
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(&box);
+    QTest::qWait(100);
+#endif
+
+    QApplication::processEvents();
+    QCOMPARE(iface->childCount(), 3);
+    QCOMPARE(iface->role(0), QAccessible::Grouping);
+    QCOMPARE(iface->role(1), QAccessible::PushButton);
+    QCOMPARE(iface->role(2), QAccessible::PushButton);
+    QCOMPARE(iface->role(3), QAccessible::PushButton);
+    QStringList actualOrder;
+    QAccessibleInterface *child;
+    QAccessibleInterface *leftmost;
+    iface->navigate(QAccessible::Child, 1, &child);
+    // first find the leftmost button
+    while (child->navigate(QAccessible::Left, 1, &leftmost) != -1) {
+        delete child;
+        child = leftmost;
+    }
+    leftmost = child;
+
+    // then traverse from left to right to find the correct order of the buttons
+    int right = 0;
+    while (right != -1) {
+        actualOrder << leftmost->text(QAccessible::Name, 0);
+        right = leftmost->navigate(QAccessible::Right, 1, &child);
+        delete leftmost;
+        leftmost = child;
+    }
+
+    QStringList expectedOrder;
+    QDialogButtonBox::ButtonLayout btnlout =
+        QDialogButtonBox::ButtonLayout(QApplication::style()->styleHint(QStyle::SH_DialogButtonLayout));
+    switch (btnlout) {
+    case QDialogButtonBox::WinLayout:
+        expectedOrder << QDialogButtonBox::tr("Reset")
+                      << QDialogButtonBox::tr("OK")
+                      << QDialogButtonBox::tr("Help");
+        break;
+    case QDialogButtonBox::GnomeLayout:
+    case QDialogButtonBox::KdeLayout:
+    case QDialogButtonBox::MacLayout:
+        expectedOrder << QDialogButtonBox::tr("Help")
+                      << QDialogButtonBox::tr("Reset")
+                      << QDialogButtonBox::tr("OK");
+        break;
+    }
+    QCOMPARE(actualOrder, expectedOrder);
+    delete iface;
+    QApplication::processEvents();
+    QTestAccessibility::clearEvents();
+    }
+
+    {
+    QDialogButtonBox box(QDialogButtonBox::Reset |
+                         QDialogButtonBox::Help |
+                         QDialogButtonBox::Ok, Qt::Horizontal);
+
+
+    // Test up and down navigation
+    QAccessibleInterface *iface = QAccessible::queryAccessibleInterface(&box);
+    QVERIFY(iface);
+    box.setOrientation(Qt::Vertical);
+    box.show();
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(&box);
+    QTest::qWait(100);
+#endif
+
+    QApplication::processEvents();
+    QAccessibleInterface *child;
+    QStringList actualOrder;
+    iface->navigate(QAccessible::Child, 1, &child);
+    // first find the topmost button
+    QAccessibleInterface *other;
+    while (child->navigate(QAccessible::Up, 1, &other) != -1) {
+        delete child;
+        child = other;
+    }
+    other = child;
+
+    // then traverse from top to bottom to find the correct order of the buttons
+    actualOrder.clear();
+    int right = 0;
+    while (right != -1) {
+        actualOrder << other->text(QAccessible::Name, 0);
+        right = other->navigate(QAccessible::Down, 1, &child);
+        delete other;
+        other = child;
+    }
+
+    QStringList expectedOrder;
+    expectedOrder << QDialogButtonBox::tr("OK")
+                  << QDialogButtonBox::tr("Reset")
+                  << QDialogButtonBox::tr("Help");
+
+    QCOMPARE(actualOrder, expectedOrder);
+    delete iface;
+    QApplication::processEvents();
+
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::dialTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    {
+    QDial dial;
+    dial.setValue(20);
+    QCOMPARE(dial.value(), 20);
+    dial.show();
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(&dial);
+    QTest::qWait(100);
+#endif
+
+    QAccessibleInterface *interface = QAccessible::queryAccessibleInterface(&dial);
+    QVERIFY(interface);
+
+    // Child count; 1 = SpeedoMeter, 2 = SliderHandle.
+    QCOMPARE(interface->childCount(), 2);
+
+    QCOMPARE(interface->role(0), QAccessible::Dial);
+    QCOMPARE(interface->role(1), QAccessible::Slider);
+    QCOMPARE(interface->role(2), QAccessible::Indicator);
+
+    QCOMPARE(interface->text(QAccessible::Value, 0), QString::number(dial.value()));
+    QCOMPARE(interface->text(QAccessible::Value, 1), QString::number(dial.value()));
+    QCOMPARE(interface->text(QAccessible::Value, 2), QString::number(dial.value()));
+    QCOMPARE(interface->text(QAccessible::Name, 0), QLatin1String("QDial"));
+    QCOMPARE(interface->text(QAccessible::Name, 1), QLatin1String("SpeedoMeter"));
+    QCOMPARE(interface->text(QAccessible::Name, 2), QLatin1String("SliderHandle"));
+    QCOMPARE(interface->text(QAccessible::Name, 3), QLatin1String(""));
+
+    QCOMPARE(interface->state(1), interface->state(0));
+    QCOMPARE(interface->state(2), interface->state(0) | QAccessible::HotTracked);
+
+    // Rect
+    QCOMPARE(interface->rect(0), dial.geometry());
+    QVERIFY(interface->rect(1).isValid());
+    QVERIFY(dial.geometry().contains(interface->rect(1)));
+    QVERIFY(interface->rect(2).isValid());
+    QVERIFY(interface->rect(1).contains(interface->rect(2)));
+    QVERIFY(!interface->rect(3).isValid());
+
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::rubberBandTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QRubberBand rubberBand(QRubberBand::Rectangle);
+    QAccessibleInterface *interface = QAccessible::queryAccessibleInterface(&rubberBand);
+    QVERIFY(interface);
+    QCOMPARE(interface->role(0), QAccessible::Border);
+    delete interface;
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::abstractScrollAreaTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    {
+    QAbstractScrollArea abstractScrollArea;
+
+    QAccessibleInterface *interface = QAccessible::queryAccessibleInterface(&abstractScrollArea);
+    QVERIFY(interface);
+    QVERIFY(!interface->rect(0).isValid());
+    QVERIFY(!interface->rect(1).isValid());
+    QCOMPARE(interface->childAt(200, 200), -1);
+
+    abstractScrollArea.resize(400, 400);
+    abstractScrollArea.show();
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(&abstractScrollArea);
+    QTest::qWait(100);
+#endif
+    const QRect globalGeometry = QRect(abstractScrollArea.mapToGlobal(QPoint(0, 0)),
+                                       abstractScrollArea.size());
+
+    // Viewport.
+    QCOMPARE(interface->childCount(), 1);
+    QWidget *viewport = abstractScrollArea.viewport();
+    QVERIFY(viewport);
+    QVERIFY(verifyChild(viewport, interface, 1, globalGeometry));
+
+    // Horizontal scrollBar.
+    abstractScrollArea.setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOn);
+    QCOMPARE(interface->childCount(), 2);
+    QWidget *horizontalScrollBar = abstractScrollArea.horizontalScrollBar();
+    QWidget *horizontalScrollBarContainer = horizontalScrollBar->parentWidget();
+    QVERIFY(verifyChild(horizontalScrollBarContainer, interface, 2, globalGeometry));
+
+    // Horizontal scrollBar widgets.
+    QLabel *secondLeftLabel = new QLabel(QLatin1String("L2"));
+    abstractScrollArea.addScrollBarWidget(secondLeftLabel, Qt::AlignLeft);
+    QCOMPARE(interface->childCount(), 2);
+
+    QLabel *firstLeftLabel = new QLabel(QLatin1String("L1"));
+    abstractScrollArea.addScrollBarWidget(firstLeftLabel, Qt::AlignLeft);
+    QCOMPARE(interface->childCount(), 2);
+
+    QLabel *secondRightLabel = new QLabel(QLatin1String("R2"));
+    abstractScrollArea.addScrollBarWidget(secondRightLabel, Qt::AlignRight);
+    QCOMPARE(interface->childCount(), 2);
+
+    QLabel *firstRightLabel = new QLabel(QLatin1String("R1"));
+    abstractScrollArea.addScrollBarWidget(firstRightLabel, Qt::AlignRight);
+    QCOMPARE(interface->childCount(), 2);
+
+    // Vertical scrollBar.
+    abstractScrollArea.setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOn);
+    QCOMPARE(interface->childCount(), 3);
+    QWidget *verticalScrollBar = abstractScrollArea.verticalScrollBar();
+    QWidget *verticalScrollBarContainer = verticalScrollBar->parentWidget();
+    QVERIFY(verifyChild(verticalScrollBarContainer, interface, 3, globalGeometry));
+
+    // Vertical scrollBar widgets.
+    QLabel *secondTopLabel = new QLabel(QLatin1String("T2"));
+    abstractScrollArea.addScrollBarWidget(secondTopLabel, Qt::AlignTop);
+    QCOMPARE(interface->childCount(), 3);
+
+    QLabel *firstTopLabel = new QLabel(QLatin1String("T1"));
+    abstractScrollArea.addScrollBarWidget(firstTopLabel, Qt::AlignTop);
+    QCOMPARE(interface->childCount(), 3);
+
+    QLabel *secondBottomLabel = new QLabel(QLatin1String("B2"));
+    abstractScrollArea.addScrollBarWidget(secondBottomLabel, Qt::AlignBottom);
+    QCOMPARE(interface->childCount(), 3);
+
+    QLabel *firstBottomLabel = new QLabel(QLatin1String("B1"));
+    abstractScrollArea.addScrollBarWidget(firstBottomLabel, Qt::AlignBottom);
+    QCOMPARE(interface->childCount(), 3);
+
+    // CornerWidget.
+    abstractScrollArea.setCornerWidget(new QLabel(QLatin1String("C")));
+    QCOMPARE(interface->childCount(), 4);
+    QWidget *cornerWidget = abstractScrollArea.cornerWidget();
+    QVERIFY(verifyChild(cornerWidget, interface, 4, globalGeometry));
+
+    // Test navigate.
+    QAccessibleInterface *target = 0;
+
+    // viewport -> Up -> NOTHING
+    const int viewportIndex = indexOfChild(interface, viewport);
+    QVERIFY(viewportIndex != -1);
+    QCOMPARE(interface->navigate(QAccessible::Up, viewportIndex, &target), -1);
+    QVERIFY(!target);
+
+    // viewport -> Left -> NOTHING
+    QCOMPARE(interface->navigate(QAccessible::Left, viewportIndex, &target), -1);
+    QVERIFY(!target);
+
+    // viewport -> Down -> horizontalScrollBarContainer
+    const int horizontalScrollBarContainerIndex = indexOfChild(interface, horizontalScrollBarContainer);
+    QVERIFY(horizontalScrollBarContainerIndex != -1);
+    QCOMPARE(interface->navigate(QAccessible::Down, viewportIndex, &target), 0);
+    QVERIFY(target);
+    QCOMPARE(target->object(), static_cast<QObject *>(horizontalScrollBarContainer));
+    delete target;
+    target = 0;
+
+    // horizontalScrollBarContainer -> Left -> NOTHING
+    QCOMPARE(interface->navigate(QAccessible::Left, horizontalScrollBarContainerIndex, &target), -1);
+    QVERIFY(!target);
+
+    // horizontalScrollBarContainer -> Down -> NOTHING
+    QVERIFY(horizontalScrollBarContainerIndex != -1);
+    QCOMPARE(interface->navigate(QAccessible::Down, horizontalScrollBarContainerIndex, &target), -1);
+    QVERIFY(!target);
+
+    // horizontalScrollBarContainer -> Right -> cornerWidget
+    const int cornerWidgetIndex = indexOfChild(interface, cornerWidget);
+    QVERIFY(cornerWidgetIndex != -1);
+    QCOMPARE(interface->navigate(QAccessible::Right, horizontalScrollBarContainerIndex, &target), 0);
+    QVERIFY(target);
+    QCOMPARE(target->object(), static_cast<QObject *>(cornerWidget));
+    delete target;
+    target = 0;
+
+    // cornerWidget -> Down -> NOTHING
+    QCOMPARE(interface->navigate(QAccessible::Down, cornerWidgetIndex, &target), -1);
+    QVERIFY(!target);
+
+    // cornerWidget -> Right -> NOTHING
+    QVERIFY(cornerWidgetIndex != -1);
+    QCOMPARE(interface->navigate(QAccessible::Right, cornerWidgetIndex, &target), -1);
+    QVERIFY(!target);
+
+    // cornerWidget -> Up ->  verticalScrollBarContainer
+    const int verticalScrollBarContainerIndex = indexOfChild(interface, verticalScrollBarContainer);
+    QVERIFY(verticalScrollBarContainerIndex != -1);
+    QCOMPARE(interface->navigate(QAccessible::Up, cornerWidgetIndex, &target), 0);
+    QVERIFY(target);
+    QCOMPARE(target->object(), static_cast<QObject *>(verticalScrollBarContainer));
+    delete target;
+    target = 0;
+
+    // verticalScrollBarContainer -> Right -> NOTHING
+    QCOMPARE(interface->navigate(QAccessible::Right, verticalScrollBarContainerIndex, &target), -1);
+    QVERIFY(!target);
+
+    // verticalScrollBarContainer -> Up -> NOTHING
+    QCOMPARE(interface->navigate(QAccessible::Up, verticalScrollBarContainerIndex, &target), -1);
+    QVERIFY(!target);
+
+    // verticalScrollBarContainer -> Left -> viewport
+    QCOMPARE(interface->navigate(QAccessible::Left, verticalScrollBarContainerIndex, &target), 0);
+    QVERIFY(target);
+    QCOMPARE(target->object(), static_cast<QObject *>(viewport));
+    delete target;
+    target = 0;
+
+    QCOMPARE(verifyHierarchy(interface), 0);
+
+    delete interface;
+    }
+
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::scrollAreaTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    {
+    QScrollArea scrollArea;
+    scrollArea.show();
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(&scrollArea);
+    QTest::qWait(100);
+#endif
+    QAccessibleInterface *interface = QAccessible::queryAccessibleInterface(&scrollArea);
+    QVERIFY(interface);
+    QCOMPARE(interface->childCount(), 1); // The viewport.
+    delete interface;
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::tableWidgetTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    {
+    QTableWidget *w = new QTableWidget(8,4);
+    for (int r = 0; r < 8; ++r) {
+        for (int c = 0; c < 4; ++c) {
+            w->setItem(r, c, new QTableWidgetItem(tr("%1,%2").arg(c).arg(r)));
+        }
+    }
+    w->resize(100, 100);
+    w->show();
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(w);
+    QTest::qWait(100);
+#endif
+    QAccessibleInterface *client = QAccessible::queryAccessibleInterface(w);
+    QCOMPARE(client->role(0), QAccessible::Client);
+    QCOMPARE(client->childCount(), 3);
+    QAccessibleInterface *view = 0;
+    client->navigate(QAccessible::Child, 1, &view);
+    QCOMPARE(view->role(0), QAccessible::Table);
+    QAccessibleInterface *ifRow;
+    view->navigate(QAccessible::Child, 2, &ifRow);
+    QCOMPARE(ifRow->role(0), QAccessible::Row);
+    QAccessibleInterface *item;
+    int entry = ifRow->navigate(QAccessible::Child, 1, &item);
+    QCOMPARE(entry, 1);
+    QCOMPARE(item , (QAccessibleInterface*)0);
+    QCOMPARE(ifRow->text(QAccessible::Name, 2), QLatin1String("0,0"));
+    QCOMPARE(ifRow->text(QAccessible::Name, 3), QLatin1String("1,0"));
+
+    QCOMPARE(verifyHierarchy(client),  0);
+
+    delete ifRow;
+    delete view;
+    delete client;
+    delete w;
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+
+}
+
+class QtTestTableModel: public QAbstractTableModel
+{
+    Q_OBJECT
+
+signals:
+    void invalidIndexEncountered() const;
+
+public:
+    QtTestTableModel(int rows = 0, int columns = 0, QObject *parent = 0)
+        : QAbstractTableModel(parent),
+          row_count(rows),
+          column_count(columns) {}
+
+    int rowCount(const QModelIndex& = QModelIndex()) const { return row_count; }
+    int columnCount(const QModelIndex& = QModelIndex()) const { return column_count; }
+
+    QVariant data(const QModelIndex &idx, int role) const
+    {
+        if (!idx.isValid() || idx.row() >= row_count || idx.column() >= column_count) {
+            qWarning() << "Invalid modelIndex [%d,%d,%p]" << idx;
+            emit invalidIndexEncountered();
+            return QVariant();
+        }
+
+        if (role == Qt::DisplayRole || role == Qt::EditRole)
+            return QString("[%1,%2,%3]").arg(idx.row()).arg(idx.column()).arg(0);
+
+        return QVariant();
+    }
+
+    void removeLastRow()
+    {
+        beginRemoveRows(QModelIndex(), row_count - 1, row_count - 1);
+        --row_count;
+        endRemoveRows();
+    }
+
+    void removeAllRows()
+    {
+        beginRemoveRows(QModelIndex(), 0, row_count - 1);
+        row_count = 0;
+        endRemoveRows();
+    }
+
+    void removeLastColumn()
+    {
+        beginRemoveColumns(QModelIndex(), column_count - 1, column_count - 1);
+        --column_count;
+        endRemoveColumns();
+    }
+
+    void removeAllColumns()
+    {
+        beginRemoveColumns(QModelIndex(), 0, column_count - 1);
+        column_count = 0;
+        endRemoveColumns();
+    }
+
+    void reset()
+    {
+        QAbstractTableModel::reset();
+    }
+
+    int row_count;
+    int column_count;
+};
+
+class QtTestDelegate : public QItemDelegate
+{
+public:
+    QtTestDelegate(QWidget *parent = 0) : QItemDelegate(parent) {}
+
+    virtual QSize sizeHint(const QStyleOptionViewItem &/*option*/, const QModelIndex &/*index*/) const
+    {
+        return QSize(100,50);
+    }
+};
+
+void tst_QAccessibility::tableViewTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    {
+    QtTestTableModel *model = new QtTestTableModel(3, 4);
+    QTableView *w = new QTableView();
+    w->setModel(model);
+    w->setItemDelegate(new QtTestDelegate(w));
+    w->resize(450,200);
+    w->resizeColumnsToContents();
+    w->resizeRowsToContents();
+    w->show();
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(w);
+    QTest::qWait(100);
+#endif
+    QAccessibleInterface *client = QAccessible::queryAccessibleInterface(w);
+    QAccessibleInterface *table2;
+    client->navigate(QAccessible::Child, 1, &table2);
+    QVERIFY(table2);
+    QCOMPARE(table2->role(1), QAccessible::Row);
+    QAccessibleInterface *toprow = 0;
+    table2->navigate(QAccessible::Child, 1, &toprow);
+    QVERIFY(toprow);
+    QCOMPARE(toprow->role(1), QAccessible::RowHeader);
+    QCOMPARE(toprow->role(2), QAccessible::ColumnHeader);
+    delete toprow;
+
+    // call childAt() for each child until we reach the bottom,
+    // and do it for each row in the table
+    for (int y = 1; y < 5; ++y) {  // this includes the special header
+        for (int x = 1; x < 6; ++x) {
+            QCOMPARE(client->role(0), QAccessible::Client);
+            QRect globalRect = client->rect(0);
+            QVERIFY(globalRect.isValid());
+            // make sure we don't hit the vertical header  #####
+            QPoint p = globalRect.topLeft() + QPoint(8, 8);
+            p.ry() += 50 * (y - 1);
+            p.rx() += 100 * (x - 1);
+            int index = client->childAt(p.x(), p.y());
+            QCOMPARE(index, 1);
+            QCOMPARE(client->role(index), QAccessible::Table);
+
+            // navigate to table/viewport
+            QAccessibleInterface *table;
+            client->navigate(QAccessible::Child, index, &table);
+            QVERIFY(table);
+            index = table->childAt(p.x(), p.y());
+            QCOMPARE(index, y);
+            QCOMPARE(table->role(index), QAccessible::Row);
+            QAccessibleInterface *row;
+            QCOMPARE(table->role(1), QAccessible::Row);
+
+            // navigate to the row
+            table->navigate(QAccessible::Child, index, &row);
+            QVERIFY(row);
+            QCOMPARE(row->role(1), QAccessible::RowHeader);
+            index = row->childAt(p.x(), p.y());
+            QVERIFY(index > 0);
+            if (x == 1 && y == 1) {
+                QCOMPARE(row->role(index), QAccessible::RowHeader);
+                QCOMPARE(row->text(QAccessible::Name, index), QLatin1String(""));
+            } else if (x > 1 && y > 1) {
+                QCOMPARE(row->role(index), QAccessible::Cell);
+                QCOMPARE(row->text(QAccessible::Name, index), QString::fromAscii("[%1,%2,0]").arg(y - 2).arg(x - 2));
+            } else if (x == 1) {
+                QCOMPARE(row->role(index), QAccessible::RowHeader);
+                QCOMPARE(row->text(QAccessible::Name, index), QString::fromAscii("%1").arg(y - 1));
+            } else if (y == 1) {
+                QCOMPARE(row->role(index), QAccessible::ColumnHeader);
+                QCOMPARE(row->text(QAccessible::Name, index), QString::fromAscii("%1").arg(x - 1));
+            }
+            delete table;
+            delete row;
+        }
+    }
+    delete table2;
+    delete client;
+    delete w;
+    delete model;
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::calendarWidgetTest()
+{
+#ifndef QT_NO_CALENDARWIDGET
+#ifdef QTEST_ACCESSIBILITY
+    {
+    QCalendarWidget calendarWidget;
+
+    QAccessibleInterface *interface = QAccessible::queryAccessibleInterface(&calendarWidget);
+    QVERIFY(interface);
+    QCOMPARE(interface->role(0), QAccessible::Table);
+    QVERIFY(!interface->rect(0).isValid());
+    QVERIFY(!interface->rect(1).isValid());
+    QCOMPARE(interface->childAt(200, 200), -1);
+
+    calendarWidget.resize(400, 300);
+    calendarWidget.show();
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(&calendarWidget);
+    QTest::qWait(100);
+#endif
+
+    // 1 = navigationBar, 2 = view.
+    QCOMPARE(interface->childCount(), 2);
+
+    const QRect globalGeometry = QRect(calendarWidget.mapToGlobal(QPoint(0, 0)),
+                                       calendarWidget.size());
+    QCOMPARE(interface->rect(0), globalGeometry);
+
+    QWidget *navigationBar = 0;
+    foreach (QObject *child, calendarWidget.children()) {
+        if (child->objectName() == QLatin1String("qt_calendar_navigationbar")) {
+            navigationBar = static_cast<QWidget *>(child);
+            break;
+        }
+    }
+    QVERIFY(navigationBar);
+    QVERIFY(verifyChild(navigationBar, interface, 1, globalGeometry));
+
+    QAbstractItemView *calendarView = 0;
+    foreach (QObject *child, calendarWidget.children()) {
+        if (child->objectName() == QLatin1String("qt_calendar_calendarview")) {
+            calendarView = static_cast<QAbstractItemView *>(child);
+            break;
+        }
+    }
+    QVERIFY(calendarView);
+    QVERIFY(verifyChild(calendarView, interface, 2, globalGeometry));
+
+    // Hide navigation bar.
+    calendarWidget.setNavigationBarVisible(false);
+    QCOMPARE(interface->childCount(), 1);
+    QVERIFY(!navigationBar->isVisible());
+
+    QVERIFY(verifyChild(calendarView, interface, 1, globalGeometry));
+
+    // Show navigation bar.
+    calendarWidget.setNavigationBarVisible(true);
+    QCOMPARE(interface->childCount(), 2);
+    QVERIFY(navigationBar->isVisible());
+
+    // Navigate to the navigation bar via Child.
+    QAccessibleInterface *navigationBarInterface = 0;
+    QCOMPARE(interface->navigate(QAccessible::Child, 1, &navigationBarInterface), 0);
+    QVERIFY(navigationBarInterface);
+    QCOMPARE(navigationBarInterface->object(), (QObject*)navigationBar);
+    delete navigationBarInterface;
+    navigationBarInterface = 0;
+
+    // Navigate to the view via Child.
+    QAccessibleInterface *calendarViewInterface = 0;
+    QCOMPARE(interface->navigate(QAccessible::Child, 2, &calendarViewInterface), 0);
+    QVERIFY(calendarViewInterface);
+    QCOMPARE(calendarViewInterface->object(), (QObject*)calendarView);
+    delete calendarViewInterface;
+    calendarViewInterface = 0;
+
+    QAccessibleInterface *doesNotExistsInterface = 0;
+    QCOMPARE(interface->navigate(QAccessible::Child, 3, &doesNotExistsInterface), -1);
+    QVERIFY(!doesNotExistsInterface);
+
+    // Navigate from navigation bar -> view (Down).
+    QCOMPARE(interface->navigate(QAccessible::Down, 1, &calendarViewInterface), 0);
+    QVERIFY(calendarViewInterface);
+    QCOMPARE(calendarViewInterface->object(), (QObject*)calendarView);
+    delete calendarViewInterface;
+    calendarViewInterface = 0;
+
+    // Navigate from view -> navigation bar (Up).
+    QCOMPARE(interface->navigate(QAccessible::Up, 2, &navigationBarInterface), 0);
+    QVERIFY(navigationBarInterface);
+    QCOMPARE(navigationBarInterface->object(), (QObject*)navigationBar);
+    delete navigationBarInterface;
+    navigationBarInterface = 0;
+
+    }
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+#endif // QT_NO_CALENDARWIDGET
+}
+
+void tst_QAccessibility::dockWidgetTest()
+{
+#ifndef QT_NO_DOCKWIDGET
+
+#ifdef QTEST_ACCESSIBILITY
+    // Set up a proper main window with two dock widgets
+    QMainWindow *mw = new QMainWindow();
+    QFrame *central = new QFrame(mw);
+    mw->setCentralWidget(central);
+    QMenuBar *mb = new QMenuBar(mw);
+    mb->addAction(tr("&File"));
+    mw->setMenuBar(mb);
+
+    QDockWidget *dock1 = new QDockWidget(mw);
+    mw->addDockWidget(Qt::LeftDockWidgetArea, dock1);
+    QPushButton *pb1 = new QPushButton(tr("Push me"), dock1);
+    dock1->setWidget(pb1);
+
+    QDockWidget *dock2 = new QDockWidget(mw);
+    mw->addDockWidget(Qt::BottomDockWidgetArea, dock2);
+    QPushButton *pb2 = new QPushButton(tr("Push me"), dock2);
+    dock2->setWidget(pb2);
+
+    mw->resize(600,400);
+    mw->show();
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(mw);
+    QTest::qWait(100);
+#endif
+
+    QAccessibleInterface *accMainWindow = QAccessible::queryAccessibleInterface(mw);
+    // 4 children: menu bar, dock1, dock2, and central widget
+    QCOMPARE(accMainWindow->childCount(), 4);
+    QAccessibleInterface *accDock1 = 0;
+    for (int i = 1; i <= 4; ++i) {
+        if (accMainWindow->role(i) == QAccessible::Window) {
+            accMainWindow->navigate(QAccessible::Child, i, &accDock1);
+            if (accDock1 && qobject_cast<QDockWidget*>(accDock1->object()) == dock1) {
+                break;
+            } else {
+                delete accDock1;
+            }
+        }
+    }
+    QVERIFY(accDock1);
+    QCOMPARE(accDock1->role(0), QAccessible::Window);
+    QCOMPARE(accDock1->role(1), QAccessible::TitleBar);
+    QVERIFY(accDock1->rect(0).contains(accDock1->rect(1)));
+
+    QPoint globalPos = dock1->mapToGlobal(QPoint(0,0));
+    globalPos.rx()+=5;  //### query style
+    globalPos.ry()+=5;
+    int entry = accDock1->childAt(globalPos.x(), globalPos.y());    //###
+    QAccessibleInterface *accTitleBar;
+    entry = accDock1->navigate(QAccessible::Child, entry, &accTitleBar);
+    QCOMPARE(accTitleBar->role(0), QAccessible::TitleBar);
+    QCOMPARE(accDock1->indexOfChild(accTitleBar), 1);
+    QAccessibleInterface *acc;
+    entry = accTitleBar->navigate(QAccessible::Ancestor, 1, &acc);
+    QVERIFY(acc);
+    QCOMPARE(entry, 0);
+    QCOMPARE(acc->role(0), QAccessible::Window);
+
+
+    delete accTitleBar;
+    delete accDock1;
+    delete pb1;
+    delete pb2;
+    delete dock1;
+    delete dock2;
+    delete mw;
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+#endif // QT_NO_DOCKWIDGET
+}
+
+void tst_QAccessibility::pushButtonTest()
+{
+#if !defined(QT3_SUPPORT)
+    qWarning( "Should never get here without Qt3Support");
+    return ;
+#else
+#ifdef QTEST_ACCESSIBILITY
+    // Set up a proper main window with two dock widgets
+    QWidget *toplevel = createGUI();
+    QObject *topRight = toplevel->findChild<QObject *>("topRight");
+
+    toplevel->show();
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(toplevel);
+    QTest::qWait(100);
+#endif
+    QPushButton *pb = qobject_cast<QPushButton*>(topRight->findChild<QPushButton *>("pbOk"));
+    QPoint pt = pb->mapToGlobal(pb->geometry().topLeft());
+    QRect rect(pt, pb->geometry().size());
+    pt = rect.center();
+
+    QAccessibleInterface *accToplevel = QAccessible::queryAccessibleInterface(toplevel);
+    QAccessibleInterface *acc;
+    QAccessibleInterface *acc2;
+    int entry = accToplevel->childAt(pt.x(), pt.y());
+    int child = accToplevel->navigate(QAccessible::Child, entry, &acc);
+    if (acc) {
+        entry = acc->childAt(pt.x(), pt.y());
+        child = acc->navigate(QAccessible::Child, entry, &acc2);
+        delete acc;
+        acc = acc2;
+    }
+    QCOMPARE(acc->role(0), QAccessible::PushButton);
+    QCOMPARE(acc->rect(0), rect);
+    QCOMPARE(acc->childAt(pt.x(), pt.y()), 0);
+
+    delete acc;
+    delete accToplevel;
+    delete toplevel;
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+#endif //QT3_SUPPORT
+}
+
+void tst_QAccessibility::comboBoxTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+#if defined(Q_OS_WINCE)
+    if (!IsValidCEPlatform()) {
+        QSKIP("Test skipped on Windows Mobile test hardware", SkipAll);
+    }
+#endif
+    QWidget *w = new QWidget();
+    QComboBox *cb = new QComboBox(w);
+    cb->addItems(QStringList() << "one" << "two" << "three");
+    w->show();
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(w);
+    QTest::qWait(100);
+#endif
+    QAccessibleInterface *acc = QAccessible::queryAccessibleInterface(w);
+    delete acc;
+
+    acc = QAccessible::queryAccessibleInterface(cb);
+
+    QRect accRect = acc->rect(0);
+    for (int i = 1; i < acc->childCount(); ++i) {
+        QVERIFY(accRect.contains(acc->rect(i)));
+    }
+    QCOMPARE(acc->doAction(QAccessible::Press, 2), true);
+    QTest::qWait(400);
+    QAccessibleInterface *accList = 0;
+    int entry = acc->navigate(QAccessible::Child, 3, &accList);
+    QCOMPARE(entry, 0);
+    QAccessibleInterface *acc2 = 0;
+    entry = accList->navigate(QAccessible::Ancestor, 1, &acc2);
+    QCOMPARE(entry, 0);
+    QCOMPARE(verifyHierarchy(acc), 0);
+    delete acc2;
+
+    delete accList;
+    delete acc;
+    delete w;
+
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+
+}
+
+void tst_QAccessibility::treeWidgetTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QWidget *w = new QWidget;
+    QTreeWidget *tree = new QTreeWidget(w);
+    QHBoxLayout *l = new QHBoxLayout(w);
+    l->addWidget(tree);
+    for (int i = 0; i < 10; ++i) {
+        QStringList strings = QStringList() << QString::fromAscii("row: %1").arg(i)
+                                            << QString("column 1") << QString("column 2");
+
+        tree->addTopLevelItem(new QTreeWidgetItem(strings));
+    }
+    w->show();
+//    QTest::qWait(1000);
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(w);
+    QTest::qWait(100);
+#endif
+
+    QAccessibleInterface *acc = QAccessible::queryAccessibleInterface(tree);
+    QAccessibleInterface *accViewport = 0;
+    int entry = acc->navigate(QAccessible::Child, 1, &accViewport);
+    QVERIFY(accViewport);
+    QCOMPARE(entry, 0);
+    QAccessibleInterface *accTreeItem = 0;
+    entry = accViewport->navigate(QAccessible::Child, 1, &accTreeItem);
+    QCOMPARE(entry, 0);
+
+    QAccessibleInterface *accTreeItem2 = 0;
+    entry = accTreeItem->navigate(QAccessible::Sibling, 3, &accTreeItem2);
+    QCOMPARE(entry, 0);
+    QCOMPARE(accTreeItem2->text(QAccessible::Name, 0), QLatin1String("row: 1"));
+
+    // test selected/focused state
+    QItemSelectionModel *selModel = tree->selectionModel();
+    QVERIFY(selModel);
+    selModel->select(QItemSelection(tree->model()->index(0, 0), tree->model()->index(3, 0)), QItemSelectionModel::Select);
+    selModel->setCurrentIndex(tree->model()->index(1, 0), QItemSelectionModel::Current);
+
+    for (int i = 1; i < 10 ; ++i) {
+        QAccessible::State expected;
+        if (i <= 5 && i >= 2)
+            expected = QAccessible::Selected;
+        if (i == 3)
+            expected |= QAccessible::Focused;
+
+        QCOMPARE(accViewport->state(i) & (QAccessible::Focused | QAccessible::Selected), expected);
+    }
+
+    // Test sanity of its navigation functions
+    QCOMPARE(verifyHierarchy(acc), 0);
+
+    delete accTreeItem2;
+    delete accTreeItem;
+    delete accViewport;
+    delete acc;
+    delete w;
+
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::labelTest()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QString text = "Hello World";
+    QLabel *label = new QLabel(text);
+    label->show();
+
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(label);
+#endif
+    QTest::qWait(100);
+
+    QAccessibleInterface *acc_label = QAccessible::queryAccessibleInterface(label);
+    QVERIFY(acc_label);
+
+    QCOMPARE(acc_label->text(QAccessible::Name, 0), text);
+
+    delete acc_label;
+    delete label;
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs accessibility support.", SkipAll);
+#endif
+}
+
+void tst_QAccessibility::accelerators()
+{
+#ifdef QTEST_ACCESSIBILITY
+    QWidget *window = new QWidget;
+    QHBoxLayout *lay = new QHBoxLayout(window);
+    QLabel *label = new QLabel(tr("&Line edit"), window);
+    QLineEdit *le = new QLineEdit(window);
+    lay->addWidget(label);
+    lay->addWidget(le);
+    label->setBuddy(le);
+
+    window->show();
+
+    QAccessibleInterface *accLineEdit = QAccessible::queryAccessibleInterface(le);
+    QCOMPARE(accLineEdit->text(QAccessible::Accelerator, 0), QKeySequence(Qt::ALT).toString() + QLatin1String("L"));
+    label->setText(tr("Q &"));
+    QCOMPARE(accLineEdit->text(QAccessible::Accelerator, 0), QString());
+    label->setText(tr("Q &&"));
+    QCOMPARE(accLineEdit->text(QAccessible::Accelerator, 0), QString());
+    label->setText(tr("Q && A"));
+    QCOMPARE(accLineEdit->text(QAccessible::Accelerator, 0), QString());
+    label->setText(tr("Q &&&A"));
+    QCOMPARE(accLineEdit->text(QAccessible::Accelerator, 0), QKeySequence(Qt::ALT).toString() + QLatin1String("A"));
+    label->setText(tr("Q &&A"));
+    QCOMPARE(accLineEdit->text(QAccessible::Accelerator, 0), QString());
+    label->setText(tr("Q &A&B"));
+    QCOMPARE(accLineEdit->text(QAccessible::Accelerator, 0), QKeySequence(Qt::ALT).toString() + QLatin1String("A"));
+
+#if defined(Q_WS_X11)
+    qt_x11_wait_for_window_manager(window);
+#endif
+    QTest::qWait(100);
+    delete window;
+    QTestAccessibility::clearEvents();
+#else
+    QSKIP("Test needs Qt >= 0x040000 and accessibility support.", SkipAll);
+#endif
+}
+
+
+QTEST_MAIN(tst_QAccessibility)
+
+#else // Q_OS_WINCE
+
+QTEST_NOOP_MAIN
+
+#endif  // Q_OS_WINCE
+
+#include "tst_qaccessibility.moc"