| /* |
| * Copyright (C) 2012, 2019 Igalia S.L. |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2,1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Library General Public License for more details. |
| * |
| * You should have received a copy of the GNU Library General Public License |
| * along with this library; see the file COPYING.LIB. If not, write to |
| * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, |
| * Boston, MA 02110-1301, USA. |
| */ |
| |
| #include "config.h" |
| |
| #include "WebViewTest.h" |
| #include <wtf/MonotonicTime.h> |
| |
| // The libatspi headers don't use G_BEGIN_DECLS |
| extern "C" { |
| #include <atspi/atspi.h> |
| } |
| |
| struct AtspiEventDeleter { |
| void operator()(AtspiEvent* event) const |
| { |
| g_boxed_free(ATSPI_TYPE_EVENT, event); |
| } |
| }; |
| |
| using UniqueAtspiEvent = std::unique_ptr<AtspiEvent, AtspiEventDeleter>; |
| |
| class AccessibilityTest : public WebViewTest { |
| public: |
| MAKE_GLIB_TEST_FIXTURE(AccessibilityTest); |
| |
| GRefPtr<AtspiAccessible> findTestApplication() |
| { |
| // Only one desktop is supported by ATSPI at the moment. |
| GRefPtr<AtspiAccessible> desktop = adoptGRef(atspi_get_desktop(0)); |
| |
| // We can get warnings from atspi when trying to connect to applications. |
| Test::removeLogFatalFlag(G_LOG_LEVEL_WARNING); |
| int childCount = atspi_accessible_get_child_count(desktop.get(), nullptr); |
| Test::addLogFatalFlag(G_LOG_LEVEL_WARNING); |
| for (int i = 0; i < childCount; ++i) { |
| GRefPtr<AtspiAccessible> current = adoptGRef(atspi_accessible_get_child_at_index(desktop.get(), i, nullptr)); |
| if (!g_strcmp0(atspi_accessible_get_name(current.get(), nullptr), "TestWebKitAccessibility")) |
| return current; |
| } |
| |
| return 0; |
| } |
| |
| GRefPtr<AtspiAccessible> findDocumentWeb(AtspiAccessible* accessible) |
| { |
| int childCount = atspi_accessible_get_child_count(accessible, nullptr); |
| for (int i = 0; i < childCount; ++i) { |
| GRefPtr<AtspiAccessible> child = adoptGRef(atspi_accessible_get_child_at_index(accessible, i, nullptr)); |
| if (atspi_accessible_get_role(child.get(), nullptr) == ATSPI_ROLE_DOCUMENT_WEB) |
| return child; |
| |
| if (auto documentWeb = findDocumentWeb(child.get())) |
| return documentWeb; |
| } |
| return nullptr; |
| } |
| |
| GRefPtr<AtspiAccessible> findRootObject(AtspiAccessible* application) |
| { |
| // Find the document web, its parent is the scroll view (WebCore root object) and its parent is |
| // the GtkPlug (WebProcess root element). |
| auto documentWeb = findDocumentWeb(application); |
| if (!documentWeb) |
| return nullptr; |
| |
| auto parent = adoptGRef(atspi_accessible_get_parent(documentWeb.get(), nullptr)); |
| return parent ? adoptGRef(atspi_accessible_get_parent(parent.get(), nullptr)) : nullptr; |
| } |
| |
| void waitUntilChildrenRemoved(AtspiAccessible* accessible) |
| { |
| m_eventSource = accessible; |
| GRefPtr<AtspiEventListener> listener = adoptGRef(atspi_event_listener_new( |
| [](AtspiEvent* event, gpointer userData) { |
| auto* test = static_cast<AccessibilityTest*>(userData); |
| if (event->source == test->m_eventSource) |
| g_main_loop_quit(test->m_mainLoop); |
| }, this, nullptr)); |
| atspi_event_listener_register(listener.get(), "object:children-changed:remove", nullptr); |
| g_main_loop_run(m_mainLoop); |
| m_eventSource = nullptr; |
| } |
| |
| void startEventMonitor(AtspiAccessible* source, const Vector<const char*>& events) |
| { |
| m_eventMonitor.source = source; |
| m_eventMonitor.listener = adoptGRef(atspi_event_listener_new([](AtspiEvent* event, gpointer userData) { |
| auto* test = static_cast<AccessibilityTest*>(userData); |
| if (event->source == test->m_eventMonitor.source) |
| test->m_eventMonitor.events.append(static_cast<AtspiEvent*>(g_boxed_copy(ATSPI_TYPE_EVENT, event))); |
| }, this, nullptr)); |
| |
| for (const auto* event : events) |
| atspi_event_listener_register(m_eventMonitor.listener.get(), event, nullptr); |
| } |
| |
| Vector<UniqueAtspiEvent> stopEventMonitor(unsigned expectedEvents, std::optional<Seconds> timeout = std::nullopt) |
| { |
| // If events is empty wait for the events. |
| auto startTime = MonotonicTime::now(); |
| while (m_eventMonitor.events.size() < expectedEvents && MonotonicTime::now() - startTime < timeout.value_or(Seconds::infinity())) |
| g_main_context_iteration(nullptr, timeout ? FALSE : TRUE); |
| |
| auto events = WTFMove(m_eventMonitor.events); |
| m_eventMonitor = { nullptr, { }, nullptr }; |
| return events; |
| } |
| |
| static unsigned stateSetSize(AtspiStateSet* stateSet) |
| { |
| GArray* states = atspi_state_set_get_states(stateSet); |
| unsigned length = states->len; |
| g_array_free(states, TRUE); |
| return length; |
| } |
| |
| private: |
| AtspiAccessible* m_eventSource { nullptr }; |
| |
| struct { |
| GRefPtr<AtspiEventListener> listener; |
| Vector<UniqueAtspiEvent> events; |
| AtspiAccessible* source { nullptr }; |
| } m_eventMonitor; |
| }; |
| |
| static void testAccessibleBasicHierarchy(AccessibilityTest* test, gconstpointer) |
| { |
| test->showInWindow(); |
| test->loadHtml( |
| "<html>" |
| " <body>" |
| " <h1>This is a test</h1>" |
| " <p>This is a paragraph with some plain text.</p>" |
| " <p>This paragraph contains <a href=\"http://www.webkitgtk.org\">a link</a> in the middle.</p>" |
| " </body>" |
| "</html>", |
| nullptr); |
| test->waitUntilLoadFinished(); |
| |
| auto testApp = test->findTestApplication(); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(testApp.get())); |
| GUniquePtr<char> name(atspi_accessible_get_name(testApp.get(), nullptr)); |
| g_assert_cmpstr(name.get(), ==, "TestWebKitAccessibility"); |
| g_assert_cmpint(atspi_accessible_get_role(testApp.get(), nullptr), ==, ATSPI_ROLE_APPLICATION); |
| |
| auto rootObject = test->findRootObject(testApp.get()); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(rootObject.get())); |
| g_assert_cmpint(atspi_accessible_get_role(rootObject.get(), nullptr), ==, ATSPI_ROLE_FILLER); |
| |
| auto scrollView = adoptGRef(atspi_accessible_get_child_at_index(rootObject.get(), 0, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(scrollView.get())); |
| g_assert_cmpint(atspi_accessible_get_role(scrollView.get(), nullptr), ==, ATSPI_ROLE_SCROLL_PANE); |
| |
| auto documentWeb = adoptGRef(atspi_accessible_get_child_at_index(scrollView.get(), 0, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(documentWeb.get())); |
| g_assert_cmpint(atspi_accessible_get_role(documentWeb.get(), nullptr), ==, ATSPI_ROLE_DOCUMENT_WEB); |
| |
| auto h1 = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 0, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(h1.get())); |
| name.reset(atspi_accessible_get_name(h1.get(), nullptr)); |
| g_assert_cmpstr(name.get(), ==, "This is a test"); |
| g_assert_cmpint(atspi_accessible_get_role(h1.get(), nullptr), ==, ATSPI_ROLE_HEADING); |
| |
| auto p1 = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 1, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(p1.get())); |
| g_assert_cmpint(atspi_accessible_get_role(p1.get(), nullptr), ==, ATSPI_ROLE_PARAGRAPH); |
| |
| auto p2 = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 2, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(p2.get())); |
| g_assert_cmpint(atspi_accessible_get_role(p2.get(), nullptr), ==, ATSPI_ROLE_PARAGRAPH); |
| |
| auto link = adoptGRef(atspi_accessible_get_child_at_index(p2.get(), 0, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(link.get())); |
| name.reset(atspi_accessible_get_name(link.get(), nullptr)); |
| g_assert_cmpstr(name.get(), ==, "a link"); |
| g_assert_cmpint(atspi_accessible_get_role(link.get(), nullptr), ==, ATSPI_ROLE_LINK); |
| |
| test->loadHtml( |
| "<html>" |
| " <body>" |
| " <h1>This is another test</h1>" |
| " <img src='data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAIAAAACCAIAAAD91JpzAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH3AYWDTMVwnSZnwAAAB1pVFh0Q29tbWVudAAAAAAAQ3JlYXRlZCB3aXRoIEdJTVBkLmUHAAAAFklEQVQI12P8z8DAwMDAxMDAwMDAAAANHQEDK+mmyAAAAABJRU5ErkJggg=='/>" |
| " </body>" |
| "</html>", |
| nullptr); |
| #if USE(ATSPI) |
| // In atspi implementation the root object doesn't emit children-changed because it |
| // always has ManagesDescendants in state. |
| test->waitUntilLoadFinished(); |
| #else |
| // Check that children-changed::remove is emitted on the root object on navigation, |
| // and the a11y hierarchy is updated. |
| test->waitUntilChildrenRemoved(rootObject.get()); |
| #endif |
| |
| documentWeb = test->findDocumentWeb(testApp.get()); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(documentWeb.get())); |
| g_assert_cmpint(atspi_accessible_get_role(documentWeb.get(), nullptr), ==, ATSPI_ROLE_DOCUMENT_WEB); |
| |
| h1 = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 0, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(h1.get())); |
| name.reset(atspi_accessible_get_name(h1.get(), nullptr)); |
| g_assert_cmpstr(name.get(), ==, "This is another test"); |
| g_assert_cmpint(atspi_accessible_get_role(h1.get(), nullptr), ==, ATSPI_ROLE_HEADING); |
| |
| auto section = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 1, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(section.get())); |
| g_assert_cmpint(atspi_accessible_get_role(section.get(), nullptr), ==, ATSPI_ROLE_SECTION); |
| |
| auto img = adoptGRef(atspi_accessible_get_child_at_index(section.get(), 0, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(img.get())); |
| g_assert_cmpint(atspi_accessible_get_role(img.get(), nullptr), ==, ATSPI_ROLE_IMAGE); |
| } |
| |
| static void testAccessibleIgnoredObjects(AccessibilityTest* test, gconstpointer) |
| { |
| test->showInWindow(); |
| test->loadHtml( |
| "<html>" |
| " <body>" |
| " <div>" |
| " <p>Static text nodes are ignored, so this paragraph shouldn't have a child</p>" |
| " </div>" |
| " </body>" |
| "</html>", |
| nullptr); |
| test->waitUntilLoadFinished(); |
| |
| auto testApp = test->findTestApplication(); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(testApp.get())); |
| |
| auto documentWeb = test->findDocumentWeb(testApp.get()); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(documentWeb.get())); |
| g_assert_cmpint(atspi_accessible_get_child_count(documentWeb.get(), nullptr), ==, 1); |
| |
| auto p = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 0, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(p.get())); |
| g_assert_cmpint(atspi_accessible_get_role(p.get(), nullptr), ==, ATSPI_ROLE_PARAGRAPH); |
| g_assert_cmpint(atspi_accessible_get_child_count(p.get(), nullptr), ==, 0); |
| } |
| |
| static void testAccessibleChildrenChanged(AccessibilityTest* test, gconstpointer) |
| { |
| #if !USE(ATSPI) |
| g_test_skip("This test doesn't work with ATK"); |
| #else |
| test->showInWindow(); |
| test->loadHtml( |
| "<html>" |
| " <body>" |
| " <div id='grandparent'>" |
| " <div id='parent'>" |
| " <p>Foo</p><p>Bar</p>" |
| " </div>" |
| " </div>" |
| " </body>" |
| "</html>", |
| nullptr); |
| test->waitUntilLoadFinished(); |
| |
| auto testApp = test->findTestApplication(); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(testApp.get())); |
| |
| auto documentWeb = test->findDocumentWeb(testApp.get()); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(documentWeb.get())); |
| // The divs are not exposed to ATs. |
| g_assert_cmpint(atspi_accessible_get_child_count(documentWeb.get(), nullptr), ==, 2); |
| |
| auto foo = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 0, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(foo.get())); |
| g_assert_cmpint(atspi_accessible_get_child_count(foo.get(), nullptr), ==, 0); |
| |
| auto bar = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 1, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(bar.get())); |
| g_assert_cmpint(atspi_accessible_get_child_count(bar.get(), nullptr), ==, 0); |
| |
| // Add a new paragraph. |
| test->startEventMonitor(documentWeb.get(), { "object:children-changed:add", "object:children-changed:remove" }); |
| test->runJavaScriptAndWaitUntilFinished("let p = document.createElement('p'); p.innerText = 'Baz'; document.getElementById('parent').appendChild(p);", nullptr); |
| auto events = test->stopEventMonitor(1); |
| g_assert_cmpuint(events.size(), ==, 1); |
| g_assert_cmpstr(events[0]->type, ==, "object:children-changed:add"); |
| g_assert_cmpuint(events[0]->detail1, ==, 2); // New node added to position 2. |
| g_assert_true(G_VALUE_HOLDS_OBJECT(&events[0]->any_data)); |
| auto baz = adoptGRef(static_cast<AtspiAccessible*>(g_value_dup_object(&events[0]->any_data))); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(baz.get())); |
| g_assert_cmpint(atspi_accessible_get_role(baz.get(), nullptr), ==, ATSPI_ROLE_PARAGRAPH); |
| g_assert_true(atspi_accessible_get_parent(baz.get(), nullptr) == documentWeb.get()); |
| g_assert_cmpint(atspi_accessible_get_child_count(documentWeb.get(), nullptr), ==, 3); |
| GRefPtr<AtspiAccessible> child = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 0, nullptr)); |
| g_assert_true(foo.get() == child.get()); |
| child = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 1, nullptr)); |
| g_assert_true(bar.get() == child.get()); |
| child = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 2, nullptr)); |
| g_assert_true(baz.get() == child.get()); |
| child = nullptr; |
| events = { }; |
| |
| // Remove one of the paragraphs. |
| test->startEventMonitor(documentWeb.get(), { "object:children-changed:add", "object:children-changed:remove" }); |
| test->runJavaScriptAndWaitUntilFinished("let div = document.getElementById('parent'); div.removeChild(div.children[0]);", nullptr); |
| events = test->stopEventMonitor(1); |
| g_assert_cmpuint(events.size(), ==, 1); |
| g_assert_cmpstr(events[0]->type, ==, "object:children-changed:remove"); |
| g_assert_cmpuint(events[0]->detail1, ==, 0); // Remove node was at position 0. |
| g_assert_true(G_VALUE_HOLDS_OBJECT(&events[0]->any_data)); |
| g_assert_true(g_value_get_object(&events[0]->any_data) == foo.get()); |
| GRefPtr<AtspiStateSet> set = adoptGRef(atspi_accessible_get_state_set(foo.get())); |
| g_assert_true(atspi_state_set_contains(set.get(), ATSPI_STATE_DEFUNCT)); |
| g_assert_cmpint(atspi_accessible_get_child_count(documentWeb.get(), nullptr), ==, 2); |
| child = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 0, nullptr)); |
| g_assert_true(bar.get() == child.get()); |
| child = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 1, nullptr)); |
| g_assert_true(baz.get() == child.get()); |
| child = nullptr; |
| events = { }; |
| |
| // Changing role causes an internal detach + attach that shouldn't be exposed to ATs. |
| test->startEventMonitor(documentWeb.get(), { "object:children-changed:add", "object:children-changed:remove" }); |
| test->runJavaScriptAndWaitUntilFinished("document.getElementById('parent').children[0].role='button'", nullptr); |
| events = test->stopEventMonitor(0, 0.5_s); |
| g_assert_cmpuint(events.size(), ==, 0); |
| g_assert_cmpint(atspi_accessible_get_role(bar.get(), nullptr), ==, ATSPI_ROLE_PUSH_BUTTON); |
| events = { }; |
| |
| // Remove the container. |
| test->startEventMonitor(documentWeb.get(), { "object:children-changed:add", "object:children-changed:remove" }); |
| test->runJavaScriptAndWaitUntilFinished("document.getElementById('grandparent').removeChild(document.getElementById('parent'));", nullptr); |
| events = test->stopEventMonitor(2); |
| g_assert_cmpuint(events.size(), ==, 2); |
| g_assert_cmpstr(events[0]->type, ==, "object:children-changed:remove"); |
| g_assert_cmpstr(events[1]->type, ==, "object:children-changed:remove"); |
| g_assert_true(G_VALUE_HOLDS_OBJECT(&events[0]->any_data)); |
| g_assert_true(G_VALUE_HOLDS_OBJECT(&events[1]->any_data)); |
| g_assert_true(g_value_get_object(&events[0]->any_data) == bar.get() || g_value_get_object(&events[0]->any_data) == baz.get()); |
| g_assert_true(g_value_get_object(&events[1]->any_data) == bar.get() || g_value_get_object(&events[1]->any_data) == baz.get()); |
| g_assert_true(g_value_get_object(&events[0]->any_data) != g_value_get_object(&events[1]->any_data)); |
| set = adoptGRef(atspi_accessible_get_state_set(bar.get())); |
| g_assert_true(atspi_state_set_contains(set.get(), ATSPI_STATE_DEFUNCT)); |
| set = adoptGRef(atspi_accessible_get_state_set(baz.get())); |
| g_assert_true(atspi_state_set_contains(set.get(), ATSPI_STATE_DEFUNCT)); |
| g_assert_cmpint(atspi_accessible_get_child_count(documentWeb.get(), nullptr), ==, 0); |
| #endif |
| } |
| |
| static void testAccessibleAttributes(AccessibilityTest* test, gconstpointer) |
| { |
| test->showInWindow(); |
| test->loadHtml( |
| "<html>" |
| " <body>" |
| " <h2>Heading level 2</h2>" |
| " <a id=\"webkitgtk\" href=\"http://www.webkitgtk.org\">A link</a>" |
| " </body>" |
| "</html>", |
| nullptr); |
| test->waitUntilLoadFinished(); |
| |
| #if USE(ATSPI) |
| static const char* toolkitName = "WebKitGTK"; |
| #else |
| static const char* toolkitName = "WebKitGtk"; |
| #endif |
| |
| auto testApp = test->findTestApplication(); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(testApp.get())); |
| |
| auto documentWeb = test->findDocumentWeb(testApp.get()); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(documentWeb.get())); |
| g_assert_cmpint(atspi_accessible_get_child_count(documentWeb.get(), nullptr), ==, 2); |
| auto attributes = adoptGRef(atspi_accessible_get_attributes(documentWeb.get(), nullptr)); |
| #if USE(ATSPI) |
| g_assert_cmpuint(g_hash_table_size(attributes.get()), ==, 1); |
| #else |
| // FIXME: ATK includes dragged state, but dragging is not supported. |
| g_assert_cmpuint(g_hash_table_size(attributes.get()), ==, 2); |
| #endif |
| g_assert_cmpstr(static_cast<const char*>(g_hash_table_lookup(attributes.get(), "toolkit")), ==, toolkitName); |
| |
| auto h2 = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 0, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(h2.get())); |
| attributes = adoptGRef(atspi_accessible_get_attributes(h2.get(), nullptr)); |
| g_assert_cmpuint(g_hash_table_size(attributes.get()), ==, 4); |
| g_assert_cmpstr(static_cast<const char*>(g_hash_table_lookup(attributes.get(), "toolkit")), ==, toolkitName); |
| g_assert_cmpstr(static_cast<const char*>(g_hash_table_lookup(attributes.get(), "computed-role")), ==, "heading"); |
| g_assert_cmpstr(static_cast<const char*>(g_hash_table_lookup(attributes.get(), "level")), ==, "2"); |
| g_assert_cmpstr(static_cast<const char*>(g_hash_table_lookup(attributes.get(), "tag")), ==, "h2"); |
| |
| auto section = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 1, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(section.get())); |
| g_assert_cmpint(atspi_accessible_get_child_count(section.get(), nullptr), ==, 1); |
| |
| auto a = adoptGRef(atspi_accessible_get_child_at_index(section.get(), 0, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(a.get())); |
| attributes = adoptGRef(atspi_accessible_get_attributes(a.get(), nullptr)); |
| g_assert_cmpuint(g_hash_table_size(attributes.get()), ==, 4); |
| g_assert_cmpstr(static_cast<const char*>(g_hash_table_lookup(attributes.get(), "toolkit")), ==, toolkitName); |
| g_assert_cmpstr(static_cast<const char*>(g_hash_table_lookup(attributes.get(), "computed-role")), ==, "link"); |
| #if USE(ATSPI) |
| g_assert_cmpstr(static_cast<const char*>(g_hash_table_lookup(attributes.get(), "id")), ==, "webkitgtk"); |
| #else |
| g_assert_cmpstr(static_cast<const char*>(g_hash_table_lookup(attributes.get(), "html-id")), ==, "webkitgtk"); |
| #endif |
| g_assert_cmpstr(static_cast<const char*>(g_hash_table_lookup(attributes.get(), "tag")), ==, "a"); |
| } |
| |
| static void testAccessibleState(AccessibilityTest* test, gconstpointer) |
| { |
| test->showInWindow(800, 600); |
| test->loadHtml( |
| "<html>" |
| " <body>" |
| " <h1>State</h1>" |
| " <button>Button</button>" |
| " <button disabled>Disabled</button>" |
| " <button aria-pressed='true'>Toggle button</button>" |
| " <input type='radio' name='radio'/>Radio" |
| " <input type='radio' name='radio' checked/>Radio Checked" |
| " <input type='radio' name='radio' disabled/>Radio disabled" |
| " <input type='checkbox' name='check-disabled' checked disabled/>Checked Disabled" |
| " <input value='Entry' aria-required='true'>" |
| " <input value='Disabled' disabled>" |
| " <input value='Read only' readonly>" |
| " <textarea rows=5 autofocus></textarea>" |
| " </body>" |
| "</html>", |
| nullptr); |
| test->waitUntilLoadFinished(); |
| |
| auto testApp = test->findTestApplication(); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(testApp.get())); |
| |
| auto documentWeb = test->findDocumentWeb(testApp.get()); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(documentWeb.get())); |
| g_assert_cmpint(atspi_accessible_get_child_count(documentWeb.get(), nullptr), ==, 2); |
| |
| auto h1 = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 0, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(h1.get())); |
| GRefPtr<AtspiStateSet> stateSet = adoptGRef(atspi_accessible_get_state_set(h1.get())); |
| #if USE(ATSPI) |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 4); |
| #else |
| // FIXME: ATK includes orientation state in every element. |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 5); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_HORIZONTAL)); |
| #endif |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_ENABLED)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SENSITIVE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_VISIBLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SHOWING)); |
| |
| auto section = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 1, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(section.get())); |
| g_assert_cmpint(atspi_accessible_get_child_count(section.get(), nullptr), ==, 11); |
| |
| unsigned nextChild = 0; |
| auto button = adoptGRef(atspi_accessible_get_child_at_index(section.get(), nextChild++, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(button.get())); |
| stateSet = adoptGRef(atspi_accessible_get_state_set(button.get())); |
| #if USE(ATSPI) |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 5); |
| #else |
| // FIXME: ATK includes orientation state in every element. |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 6); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_HORIZONTAL)); |
| #endif |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_ENABLED)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SENSITIVE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_VISIBLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SHOWING)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_FOCUSABLE)); |
| |
| auto buttonDisabled = adoptGRef(atspi_accessible_get_child_at_index(section.get(), nextChild++, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(buttonDisabled.get())); |
| stateSet = adoptGRef(atspi_accessible_get_state_set(buttonDisabled.get())); |
| #if USE(ATSPI) |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 2); |
| #else |
| // FIXME: ATK includes orientation state in every element. |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 3); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_HORIZONTAL)); |
| #endif |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_VISIBLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SHOWING)); |
| |
| auto toggleButton = adoptGRef(atspi_accessible_get_child_at_index(section.get(), nextChild++, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(toggleButton.get())); |
| stateSet = adoptGRef(atspi_accessible_get_state_set(toggleButton.get())); |
| #if USE(ATSPI) |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 6); |
| #else |
| // FIXME: ATK includes orientation state in every element. |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 7); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_HORIZONTAL)); |
| #endif |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_ENABLED)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SENSITIVE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_VISIBLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SHOWING)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_FOCUSABLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_PRESSED)); |
| |
| auto radio = adoptGRef(atspi_accessible_get_child_at_index(section.get(), nextChild++, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(radio.get())); |
| stateSet = adoptGRef(atspi_accessible_get_state_set(radio.get())); |
| #if USE(ATSPI) |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 6); |
| #else |
| // FIXME: ATK includes orientation state in every element. |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 7); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_VERTICAL)); |
| #endif |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_ENABLED)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SENSITIVE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_VISIBLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SHOWING)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_FOCUSABLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_CHECKABLE)); |
| |
| auto radioChecked = adoptGRef(atspi_accessible_get_child_at_index(section.get(), nextChild++, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(radioChecked.get())); |
| stateSet = adoptGRef(atspi_accessible_get_state_set(radioChecked.get())); |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 7); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_ENABLED)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SENSITIVE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_VISIBLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SHOWING)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_FOCUSABLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_CHECKABLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_CHECKED)); |
| |
| auto radioDisabled = adoptGRef(atspi_accessible_get_child_at_index(section.get(), nextChild++, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(radioDisabled.get())); |
| stateSet = adoptGRef(atspi_accessible_get_state_set(radioDisabled.get())); |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 3); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_VISIBLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SHOWING)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_CHECKABLE)); |
| |
| auto checkbox = adoptGRef(atspi_accessible_get_child_at_index(section.get(), nextChild++, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(checkbox.get())); |
| stateSet = adoptGRef(atspi_accessible_get_state_set(checkbox.get())); |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 4); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_VISIBLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SHOWING)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_CHECKABLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_CHECKED)); |
| |
| auto entry = adoptGRef(atspi_accessible_get_child_at_index(section.get(), nextChild++, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(entry.get())); |
| stateSet = adoptGRef(atspi_accessible_get_state_set(entry.get())); |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 9); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_ENABLED)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SENSITIVE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_VISIBLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SHOWING)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_FOCUSABLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_EDITABLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SINGLE_LINE)); |
| #if USE(ATSPI) |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SELECTABLE_TEXT)); |
| #else |
| // FIXME: ATK doesn't implement selectable text, but includes oprientation. |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_HORIZONTAL)); |
| #endif |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_REQUIRED)); |
| |
| auto entryDisabled = adoptGRef(atspi_accessible_get_child_at_index(section.get(), nextChild++, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(entryDisabled.get())); |
| stateSet = adoptGRef(atspi_accessible_get_state_set(entryDisabled.get())); |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 5); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_VISIBLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SHOWING)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_EDITABLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SINGLE_LINE)); |
| #if USE(ATSPI) |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SELECTABLE_TEXT)); |
| #else |
| // FIXME: ATK doesn't implement selectable text, but includes oprientation. |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_HORIZONTAL)); |
| #endif |
| |
| auto entryReadOnly = adoptGRef(atspi_accessible_get_child_at_index(section.get(), nextChild++, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(entryReadOnly.get())); |
| stateSet = adoptGRef(atspi_accessible_get_state_set(entryReadOnly.get())); |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 8); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_ENABLED)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SENSITIVE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_VISIBLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SHOWING)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_FOCUSABLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SINGLE_LINE)); |
| #if USE(ATSPI) |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SELECTABLE_TEXT)); |
| #else |
| // FIXME: ATK doesn't implement selectable text, but includes oprientation. |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_HORIZONTAL)); |
| #endif |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_READ_ONLY)); |
| |
| auto textArea = adoptGRef(atspi_accessible_get_child_at_index(section.get(), nextChild++, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(textArea.get())); |
| stateSet = adoptGRef(atspi_accessible_get_state_set(textArea.get())); |
| g_assert_cmpuint(AccessibilityTest::stateSetSize(stateSet.get()), ==, 9); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_ENABLED)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SENSITIVE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_VISIBLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SHOWING)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_FOCUSABLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_FOCUSED)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_EDITABLE)); |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_MULTI_LINE)); |
| #if USE(ATSPI) |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_SELECTABLE_TEXT)); |
| #else |
| // FIXME: ATK doesn't implement selectable text, but includes oprientation. |
| g_assert_true(atspi_state_set_contains(stateSet.get(), ATSPI_STATE_HORIZONTAL)); |
| #endif |
| } |
| |
| static void testAccessibleStateChanged(AccessibilityTest* test, gconstpointer) |
| { |
| test->showInWindow(); |
| test->loadHtml( |
| "<html>" |
| " <body>" |
| " <input id='check' type='checkbox'/>Checkbox" |
| " <button id='toggle' aria-pressed='true'>Toggle button</button>" |
| " <input id='entry' value='Entry'>" |
| " </body>" |
| "</html>", |
| nullptr); |
| test->waitUntilLoadFinished(); |
| |
| auto testApp = test->findTestApplication(); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(testApp.get())); |
| |
| auto documentWeb = test->findDocumentWeb(testApp.get()); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(documentWeb.get())); |
| g_assert_cmpint(atspi_accessible_get_child_count(documentWeb.get(), nullptr), ==, 1); |
| |
| auto section = adoptGRef(atspi_accessible_get_child_at_index(documentWeb.get(), 0, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(section.get())); |
| g_assert_cmpint(atspi_accessible_get_child_count(section.get(), nullptr), ==, 3); |
| |
| unsigned nextChild = 0; |
| auto checkbox = adoptGRef(atspi_accessible_get_child_at_index(section.get(), nextChild++, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(checkbox.get())); |
| test->startEventMonitor(checkbox.get(), { "object:state-changed" }); |
| test->runJavaScriptAndWaitUntilFinished("document.getElementById('check').checked = true;", nullptr); |
| auto events = test->stopEventMonitor(1); |
| g_assert_cmpuint(events.size(), ==, 1); |
| g_assert_cmpstr(events[0]->type, ==, "object:state-changed:checked"); |
| g_assert_cmpuint(events[0]->detail1, ==, 1); |
| events = { }; |
| |
| auto toggleButton = adoptGRef(atspi_accessible_get_child_at_index(section.get(), nextChild++, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(toggleButton.get())); |
| test->startEventMonitor(toggleButton.get(), { "object:state-changed" }); |
| test->runJavaScriptAndWaitUntilFinished("document.getElementById('toggle').ariaPressed = false;", nullptr); |
| events = test->stopEventMonitor(1); |
| g_assert_cmpuint(events.size(), ==, 1); |
| g_assert_cmpstr(events[0]->type, ==, "object:state-changed:pressed"); |
| g_assert_cmpuint(events[0]->detail1, ==, 0); |
| events = { }; |
| |
| auto entry = adoptGRef(atspi_accessible_get_child_at_index(section.get(), nextChild++, nullptr)); |
| g_assert_true(ATSPI_IS_ACCESSIBLE(entry.get())); |
| test->startEventMonitor(entry.get(), { "object:state-changed" }); |
| test->runJavaScriptAndWaitUntilFinished("let e = document.getElementById('entry'); e.ariaRequired = true; e.focus();", nullptr); |
| events = test->stopEventMonitor(2); |
| #if !USE(ATSPI) |
| // FIXME: With ATK focused event is sometimes emitted twice for some reason. |
| if (events.size() == 3) { |
| events.removeFirstMatching([](auto& item) { |
| if (!strcmp(item->type, "object:state-changed:focused")) |
| return true; |
| return false; |
| }); |
| } |
| #endif |
| g_assert_cmpuint(events.size(), ==, 2); |
| if (!g_strcmp0(events[0]->type, "object:state-changed:focused")) { |
| g_assert_cmpuint(events[0]->detail1, ==, 1); |
| |
| g_assert_cmpstr(events[1]->type, ==, "object:state-changed:required"); |
| g_assert_cmpuint(events[1]->detail1, ==, 1); |
| } else if (!g_strcmp0(events[0]->type, "object:state-changed:required")) { |
| g_assert_cmpuint(events[0]->detail1, ==, 1); |
| |
| g_assert_cmpstr(events[1]->type, ==, "object:state-changed:focused"); |
| g_assert_cmpuint(events[1]->detail1, ==, 1); |
| } else |
| g_assert_not_reached(); |
| events = { }; |
| } |
| |
| void beforeAll() |
| { |
| AccessibilityTest::add("WebKitAccessibility", "accessible/basic-hierarchy", testAccessibleBasicHierarchy); |
| AccessibilityTest::add("WebKitAccessibility", "accessible/ignored-objects", testAccessibleIgnoredObjects); |
| AccessibilityTest::add("WebKitAccessibility", "accessible/children-changed", testAccessibleChildrenChanged); |
| AccessibilityTest::add("WebKitAccessibility", "accessible/attributes", testAccessibleAttributes); |
| AccessibilityTest::add("WebKitAccessibility", "accessible/state", testAccessibleState); |
| AccessibilityTest::add("WebKitAccessibility", "accessible/state-changed", testAccessibleStateChanged); |
| } |
| |
| void afterAll() |
| { |
| } |