// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include #include #include "base/strings/utf_string_conversions.h" #include "content/browser/renderer_host/render_view_host_impl.h" #include "content/public/browser/notification_service.h" #include "content/public/browser/notification_types.h" #include "content/public/browser/render_widget_host_view.h" #include "content/shell/browser/shell.h" #include "content/test/accessibility_browser_test_utils.h" #include "content/test/content_browser_test.h" #include "content/test/content_browser_test_utils.h" #if defined(OS_WIN) #include #include #include "base/win/scoped_com_initializer.h" #include "ui/base/win/atl_module.h" #endif // TODO(dmazzoni): Disabled accessibility tests on Win64. crbug.com/179717 #if defined(OS_WIN) && defined(ARCH_CPU_X86_64) #define MAYBE_TableSpan DISABLED_TableSpan #else #define MAYBE_TableSpan TableSpan #endif namespace content { class CrossPlatformAccessibilityBrowserTest : public ContentBrowserTest { public: CrossPlatformAccessibilityBrowserTest() {} // Tell the renderer to send an accessibility tree, then wait for the // notification that it's been received. const AccessibilityNodeDataTreeNode& GetAccessibilityNodeDataTree( AccessibilityMode accessibility_mode = AccessibilityModeComplete) { AccessibilityNotificationWaiter waiter( shell(), accessibility_mode, blink::WebAXEventLayoutComplete); waiter.WaitForNotification(); return waiter.GetAccessibilityNodeDataTree(); } // Make sure each node in the tree has an unique id. void RecursiveAssertUniqueIds( const AccessibilityNodeDataTreeNode& node, base::hash_set* ids) { ASSERT_TRUE(ids->find(node.id) == ids->end()); ids->insert(node.id); for (size_t i = 0; i < node.children.size(); i++) RecursiveAssertUniqueIds(node.children[i], ids); } // ContentBrowserTest virtual void SetUpInProcessBrowserTestFixture() OVERRIDE; virtual void TearDownInProcessBrowserTestFixture() OVERRIDE; protected: std::string GetAttr(const AccessibilityNodeData& node, const AccessibilityNodeData::StringAttribute attr); int GetIntAttr(const AccessibilityNodeData& node, const AccessibilityNodeData::IntAttribute attr); bool GetBoolAttr(const AccessibilityNodeData& node, const AccessibilityNodeData::BoolAttribute attr); private: #if defined(OS_WIN) scoped_ptr com_initializer_; #endif DISALLOW_COPY_AND_ASSIGN(CrossPlatformAccessibilityBrowserTest); }; void CrossPlatformAccessibilityBrowserTest::SetUpInProcessBrowserTestFixture() { #if defined(OS_WIN) ui::win::CreateATLModuleIfNeeded(); com_initializer_.reset(new base::win::ScopedCOMInitializer()); #endif } void CrossPlatformAccessibilityBrowserTest::TearDownInProcessBrowserTestFixture() { #if defined(OS_WIN) com_initializer_.reset(); #endif } // Convenience method to get the value of a particular AccessibilityNodeData // node attribute as a UTF-8 string. std::string CrossPlatformAccessibilityBrowserTest::GetAttr( const AccessibilityNodeData& node, const AccessibilityNodeData::StringAttribute attr) { for (size_t i = 0; i < node.string_attributes.size(); ++i) { if (node.string_attributes[i].first == attr) return node.string_attributes[i].second; } return std::string(); } // Convenience method to get the value of a particular AccessibilityNodeData // node integer attribute. int CrossPlatformAccessibilityBrowserTest::GetIntAttr( const AccessibilityNodeData& node, const AccessibilityNodeData::IntAttribute attr) { for (size_t i = 0; i < node.int_attributes.size(); ++i) { if (node.int_attributes[i].first == attr) return node.int_attributes[i].second; } return -1; } // Convenience method to get the value of a particular AccessibilityNodeData // node boolean attribute. bool CrossPlatformAccessibilityBrowserTest::GetBoolAttr( const AccessibilityNodeData& node, const AccessibilityNodeData::BoolAttribute attr) { for (size_t i = 0; i < node.bool_attributes.size(); ++i) { if (node.bool_attributes[i].first == attr) return node.bool_attributes[i].second; } return false; } // Marked flaky per http://crbug.com/101984 IN_PROC_BROWSER_TEST_F(CrossPlatformAccessibilityBrowserTest, DISABLED_WebpageAccessibility) { // Create a data url and load it. const char url_str[] = "data:text/html," "" "Accessibility Test" "" ""; GURL url(url_str); NavigateToURL(shell(), url); const AccessibilityNodeDataTreeNode& tree = GetAccessibilityNodeDataTree(); // Check properties of the root element of the tree. EXPECT_STREQ(url_str, GetAttr(tree, AccessibilityNodeData::ATTR_DOC_URL).c_str()); EXPECT_STREQ( "Accessibility Test", GetAttr(tree, AccessibilityNodeData::ATTR_DOC_TITLE).c_str()); EXPECT_STREQ( "html", GetAttr(tree, AccessibilityNodeData::ATTR_DOC_DOCTYPE).c_str()); EXPECT_STREQ( "text/html", GetAttr(tree, AccessibilityNodeData::ATTR_DOC_MIMETYPE).c_str()); EXPECT_STREQ( "Accessibility Test", GetAttr(tree, AccessibilityNodeData::ATTR_NAME).c_str()); EXPECT_EQ(blink::WebAXRoleRootWebArea, tree.role); // Check properites of the BODY element. ASSERT_EQ(1U, tree.children.size()); const AccessibilityNodeDataTreeNode& body = tree.children[0]; EXPECT_EQ(blink::WebAXRoleGroup, body.role); EXPECT_STREQ("body", GetAttr(body, AccessibilityNodeData::ATTR_HTML_TAG).c_str()); EXPECT_STREQ("block", GetAttr(body, AccessibilityNodeData::ATTR_DISPLAY).c_str()); // Check properties of the two children of the BODY element. ASSERT_EQ(2U, body.children.size()); const AccessibilityNodeDataTreeNode& button = body.children[0]; EXPECT_EQ(blink::WebAXRoleButton, button.role); EXPECT_STREQ( "input", GetAttr(button, AccessibilityNodeData::ATTR_HTML_TAG).c_str()); EXPECT_STREQ( "push", GetAttr(button, AccessibilityNodeData::ATTR_NAME).c_str()); EXPECT_STREQ( "inline-block", GetAttr(button, AccessibilityNodeData::ATTR_DISPLAY).c_str()); ASSERT_EQ(2U, button.html_attributes.size()); EXPECT_STREQ("type", button.html_attributes[0].first.c_str()); EXPECT_STREQ("button", button.html_attributes[0].second.c_str()); EXPECT_STREQ("value", button.html_attributes[1].first.c_str()); EXPECT_STREQ("push", button.html_attributes[1].second.c_str()); const AccessibilityNodeDataTreeNode& checkbox = body.children[1]; EXPECT_EQ(blink::WebAXRoleCheckBox, checkbox.role); EXPECT_STREQ( "input", GetAttr(checkbox, AccessibilityNodeData::ATTR_HTML_TAG).c_str()); EXPECT_STREQ( "inline-block", GetAttr(checkbox, AccessibilityNodeData::ATTR_DISPLAY).c_str()); ASSERT_EQ(1U, checkbox.html_attributes.size()); EXPECT_STREQ("type", checkbox.html_attributes[0].first.c_str()); EXPECT_STREQ("checkbox", checkbox.html_attributes[0].second.c_str()); } IN_PROC_BROWSER_TEST_F(CrossPlatformAccessibilityBrowserTest, UnselectedEditableTextAccessibility) { // Create a data url and load it. const char url_str[] = "data:text/html," "" "" "" ""; GURL url(url_str); NavigateToURL(shell(), url); const AccessibilityNodeDataTreeNode& tree = GetAccessibilityNodeDataTree(); ASSERT_EQ(1U, tree.children.size()); const AccessibilityNodeDataTreeNode& body = tree.children[0]; ASSERT_EQ(1U, body.children.size()); const AccessibilityNodeDataTreeNode& text = body.children[0]; EXPECT_EQ(blink::WebAXRoleTextField, text.role); EXPECT_STREQ( "input", GetAttr(text, AccessibilityNodeData::ATTR_HTML_TAG).c_str()); EXPECT_EQ(0, GetIntAttr(text, AccessibilityNodeData::ATTR_TEXT_SEL_START)); EXPECT_EQ(0, GetIntAttr(text, AccessibilityNodeData::ATTR_TEXT_SEL_END)); EXPECT_STREQ( "Hello, world.", GetAttr(text, AccessibilityNodeData::ATTR_VALUE).c_str()); // TODO(dmazzoni): as soon as more accessibility code is cross-platform, // this code should test that the accessible info is dynamically updated // if the selection or value changes. } IN_PROC_BROWSER_TEST_F(CrossPlatformAccessibilityBrowserTest, SelectedEditableTextAccessibility) { // Create a data url and load it. const char url_str[] = "data:text/html," "" "" "" ""; GURL url(url_str); NavigateToURL(shell(), url); const AccessibilityNodeDataTreeNode& tree = GetAccessibilityNodeDataTree(); ASSERT_EQ(1U, tree.children.size()); const AccessibilityNodeDataTreeNode& body = tree.children[0]; ASSERT_EQ(1U, body.children.size()); const AccessibilityNodeDataTreeNode& text = body.children[0]; EXPECT_EQ(blink::WebAXRoleTextField, text.role); EXPECT_STREQ( "input", GetAttr(text, AccessibilityNodeData::ATTR_HTML_TAG).c_str()); EXPECT_EQ(0, GetIntAttr(text, AccessibilityNodeData::ATTR_TEXT_SEL_START)); EXPECT_EQ(13, GetIntAttr(text, AccessibilityNodeData::ATTR_TEXT_SEL_END)); EXPECT_STREQ( "Hello, world.", GetAttr(text, AccessibilityNodeData::ATTR_VALUE).c_str()); } IN_PROC_BROWSER_TEST_F(CrossPlatformAccessibilityBrowserTest, MultipleInheritanceAccessibility) { // In a WebKit accessibility render tree for a table, each cell is a // child of both a row and a column, so it appears to use multiple // inheritance. Make sure that the AccessibilityNodeDataObject tree only // keeps one copy of each cell, and uses an indirect child id for the // additional reference to it. const char url_str[] = "data:text/html," "" "
12
"; GURL url(url_str); NavigateToURL(shell(), url); const AccessibilityNodeDataTreeNode& tree = GetAccessibilityNodeDataTree(); ASSERT_EQ(1U, tree.children.size()); const AccessibilityNodeDataTreeNode& table = tree.children[0]; EXPECT_EQ(blink::WebAXRoleTable, table.role); const AccessibilityNodeDataTreeNode& row = table.children[0]; EXPECT_EQ(blink::WebAXRoleRow, row.role); const AccessibilityNodeDataTreeNode& cell1 = row.children[0]; EXPECT_EQ(blink::WebAXRoleCell, cell1.role); const AccessibilityNodeDataTreeNode& cell2 = row.children[1]; EXPECT_EQ(blink::WebAXRoleCell, cell2.role); const AccessibilityNodeDataTreeNode& column1 = table.children[1]; EXPECT_EQ(blink::WebAXRoleColumn, column1.role); EXPECT_EQ(0U, column1.children.size()); EXPECT_EQ(1U, column1.intlist_attributes.size()); EXPECT_EQ(AccessibilityNodeData::ATTR_INDIRECT_CHILD_IDS, column1.intlist_attributes[0].first); const std::vector column1_indirect_child_ids = column1.intlist_attributes[0].second; EXPECT_EQ(1U, column1_indirect_child_ids.size()); EXPECT_EQ(cell1.id, column1_indirect_child_ids[0]); const AccessibilityNodeDataTreeNode& column2 = table.children[2]; EXPECT_EQ(blink::WebAXRoleColumn, column2.role); EXPECT_EQ(0U, column2.children.size()); EXPECT_EQ(AccessibilityNodeData::ATTR_INDIRECT_CHILD_IDS, column2.intlist_attributes[0].first); const std::vector column2_indirect_child_ids = column2.intlist_attributes[0].second; EXPECT_EQ(1U, column2_indirect_child_ids.size()); EXPECT_EQ(cell2.id, column2_indirect_child_ids[0]); } IN_PROC_BROWSER_TEST_F(CrossPlatformAccessibilityBrowserTest, MultipleInheritanceAccessibility2) { // Here's another html snippet where WebKit puts the same node as a child // of two different parents. Instead of checking the exact output, just // make sure that no id is reused in the resulting tree. const char url_str[] = "data:text/html," "" ""; GURL url(url_str); NavigateToURL(shell(), url); const AccessibilityNodeDataTreeNode& tree = GetAccessibilityNodeDataTree(); base::hash_set ids; RecursiveAssertUniqueIds(tree, &ids); } IN_PROC_BROWSER_TEST_F(CrossPlatformAccessibilityBrowserTest, IframeAccessibility) { // Create a data url and load it. const char url_str[] = "data:text/html," "" "" "" "" ""; GURL url(url_str); NavigateToURL(shell(), url); const AccessibilityNodeDataTreeNode& tree = GetAccessibilityNodeDataTree(); ASSERT_EQ(1U, tree.children.size()); const AccessibilityNodeDataTreeNode& body = tree.children[0]; ASSERT_EQ(3U, body.children.size()); const AccessibilityNodeDataTreeNode& button1 = body.children[0]; EXPECT_EQ(blink::WebAXRoleButton, button1.role); EXPECT_STREQ( "Button 1", GetAttr(button1, AccessibilityNodeData::ATTR_NAME).c_str()); const AccessibilityNodeDataTreeNode& iframe = body.children[1]; EXPECT_STREQ("iframe", GetAttr(iframe, AccessibilityNodeData::ATTR_HTML_TAG).c_str()); ASSERT_EQ(1U, iframe.children.size()); const AccessibilityNodeDataTreeNode& scroll_area = iframe.children[0]; EXPECT_EQ(blink::WebAXRoleScrollArea, scroll_area.role); ASSERT_EQ(1U, scroll_area.children.size()); const AccessibilityNodeDataTreeNode& sub_document = scroll_area.children[0]; EXPECT_EQ(blink::WebAXRoleWebArea, sub_document.role); ASSERT_EQ(1U, sub_document.children.size()); const AccessibilityNodeDataTreeNode& sub_body = sub_document.children[0]; ASSERT_EQ(1U, sub_body.children.size()); const AccessibilityNodeDataTreeNode& button2 = sub_body.children[0]; EXPECT_EQ(blink::WebAXRoleButton, button2.role); EXPECT_STREQ("Button 2", GetAttr(button2, AccessibilityNodeData::ATTR_NAME).c_str()); const AccessibilityNodeDataTreeNode& button3 = body.children[2]; EXPECT_EQ(blink::WebAXRoleButton, button3.role); EXPECT_STREQ("Button 3", GetAttr(button3, AccessibilityNodeData::ATTR_NAME).c_str()); } IN_PROC_BROWSER_TEST_F(CrossPlatformAccessibilityBrowserTest, DuplicateChildrenAccessibility) { // Here's another html snippet where WebKit has a parent node containing // two duplicate child nodes. Instead of checking the exact output, just // make sure that no id is reused in the resulting tree. const char url_str[] = "data:text/html," "" "

"; GURL url(url_str); NavigateToURL(shell(), url); const AccessibilityNodeDataTreeNode& tree = GetAccessibilityNodeDataTree(); base::hash_set ids; RecursiveAssertUniqueIds(tree, &ids); } IN_PROC_BROWSER_TEST_F(CrossPlatformAccessibilityBrowserTest, MAYBE_TableSpan) { // +---+---+---+ // | 1 | 2 | // +---+---+---+ // | 3 | 4 | // +---+---+---+ const char url_str[] = "data:text/html," "" "" " " " " " " " " " " " " "
12
34
"; GURL url(url_str); NavigateToURL(shell(), url); const AccessibilityNodeDataTreeNode& tree = GetAccessibilityNodeDataTree(); const AccessibilityNodeDataTreeNode& table = tree.children[0]; EXPECT_EQ(blink::WebAXRoleTable, table.role); ASSERT_GE(table.children.size(), 5U); EXPECT_EQ(blink::WebAXRoleRow, table.children[0].role); EXPECT_EQ(blink::WebAXRoleRow, table.children[1].role); EXPECT_EQ(blink::WebAXRoleColumn, table.children[2].role); EXPECT_EQ(blink::WebAXRoleColumn, table.children[3].role); EXPECT_EQ(blink::WebAXRoleColumn, table.children[4].role); EXPECT_EQ(3, GetIntAttr(table, AccessibilityNodeData::ATTR_TABLE_COLUMN_COUNT)); EXPECT_EQ(2, GetIntAttr(table, AccessibilityNodeData::ATTR_TABLE_ROW_COUNT)); const AccessibilityNodeDataTreeNode& cell1 = table.children[0].children[0]; const AccessibilityNodeDataTreeNode& cell2 = table.children[0].children[1]; const AccessibilityNodeDataTreeNode& cell3 = table.children[1].children[0]; const AccessibilityNodeDataTreeNode& cell4 = table.children[1].children[1]; ASSERT_EQ(AccessibilityNodeData::ATTR_CELL_IDS, table.intlist_attributes[0].first); const std::vector& table_cell_ids = table.intlist_attributes[0].second; ASSERT_EQ(6U, table_cell_ids.size()); EXPECT_EQ(cell1.id, table_cell_ids[0]); EXPECT_EQ(cell1.id, table_cell_ids[1]); EXPECT_EQ(cell2.id, table_cell_ids[2]); EXPECT_EQ(cell3.id, table_cell_ids[3]); EXPECT_EQ(cell4.id, table_cell_ids[4]); EXPECT_EQ(cell4.id, table_cell_ids[5]); EXPECT_EQ(0, GetIntAttr(cell1, AccessibilityNodeData::ATTR_TABLE_CELL_COLUMN_INDEX)); EXPECT_EQ(0, GetIntAttr(cell1, AccessibilityNodeData::ATTR_TABLE_CELL_ROW_INDEX)); EXPECT_EQ(2, GetIntAttr(cell1, AccessibilityNodeData::ATTR_TABLE_CELL_COLUMN_SPAN)); EXPECT_EQ(1, GetIntAttr(cell1, AccessibilityNodeData::ATTR_TABLE_CELL_ROW_SPAN)); EXPECT_EQ(2, GetIntAttr(cell2, AccessibilityNodeData::ATTR_TABLE_CELL_COLUMN_INDEX)); EXPECT_EQ(1, GetIntAttr(cell2, AccessibilityNodeData::ATTR_TABLE_CELL_COLUMN_SPAN)); EXPECT_EQ(0, GetIntAttr(cell3, AccessibilityNodeData::ATTR_TABLE_CELL_COLUMN_INDEX)); EXPECT_EQ(1, GetIntAttr(cell3, AccessibilityNodeData::ATTR_TABLE_CELL_COLUMN_SPAN)); EXPECT_EQ(1, GetIntAttr(cell4, AccessibilityNodeData::ATTR_TABLE_CELL_COLUMN_INDEX)); EXPECT_EQ(2, GetIntAttr(cell4, AccessibilityNodeData::ATTR_TABLE_CELL_COLUMN_SPAN)); } IN_PROC_BROWSER_TEST_F(CrossPlatformAccessibilityBrowserTest, WritableElement) { const char url_str[] = "data:text/html," "" "
" " Some text" "
"; GURL url(url_str); NavigateToURL(shell(), url); const AccessibilityNodeDataTreeNode& tree = GetAccessibilityNodeDataTree(); ASSERT_EQ(1U, tree.children.size()); const AccessibilityNodeDataTreeNode& textbox = tree.children[0]; EXPECT_EQ( true, GetBoolAttr(textbox, AccessibilityNodeData::ATTR_CAN_SET_VALUE)); } } // namespace content