// Copyright (c) 2010 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/scoped_comptr_win.h" #include "chrome/browser/browser.h" #include "chrome/browser/browser_window.h" #include "chrome/browser/renderer_host/render_widget_host_view_win.h" #include "chrome/browser/tab_contents/tab_contents.h" #include "chrome/common/chrome_switches.h" #include "chrome/common/notification_type.h" #include "chrome/test/in_process_browser_test.h" #include "chrome/test/ui_test_utils.h" #include "ia2_api_all.h" // Generated using std::auto_ptr; using std::vector; using std::wstring; namespace { class AccessibilityWinBrowserTest : public InProcessBrowserTest { public: AccessibilityWinBrowserTest() : screenreader_running_(FALSE) {} // InProcessBrowserTest void SetUpInProcessBrowserTestFixture(); void TearDownInProcessBrowserTestFixture(); protected: IAccessible* GetRenderWidgetHostViewClientAccessible(); private: BOOL screenreader_running_; }; void AccessibilityWinBrowserTest::SetUpInProcessBrowserTestFixture() { // This test assumes the windows system-wide SPI_SETSCREENREADER flag is // cleared. if (SystemParametersInfo(SPI_GETSCREENREADER, 0, &screenreader_running_, 0) && screenreader_running_) { // Clear the SPI_SETSCREENREADER flag and notify active applications about // the setting change. ::SystemParametersInfo(SPI_SETSCREENREADER, FALSE, NULL, 0); ::SendNotifyMessage( HWND_BROADCAST, WM_SETTINGCHANGE, SPI_GETSCREENREADER, 0); } } void AccessibilityWinBrowserTest::TearDownInProcessBrowserTestFixture() { if (screenreader_running_) { // Restore the SPI_SETSCREENREADER flag and notify active applications about // the setting change. ::SystemParametersInfo(SPI_SETSCREENREADER, TRUE, NULL, 0); ::SendNotifyMessage( HWND_BROADCAST, WM_SETTINGCHANGE, SPI_GETSCREENREADER, 0); } } class AccessibleChecker { public: AccessibleChecker( wstring expected_name, int32 expected_role, wstring expected_value); AccessibleChecker( wstring expected_name, wstring expected_role, wstring expected_value); // Append an AccessibleChecker that verifies accessibility information for // a child IAccessible. Order is important. void AppendExpectedChild(AccessibleChecker* expected_child); // Check that the name and role of the given IAccessible instance and its // descendants match the expected names and roles that this object was // initialized with. void CheckAccessible(IAccessible* accessible); // Set the expected value for this AccessibleChecker. void SetExpectedValue(wstring expected_value); // Set the expected state for this AccessibleChecker. void SetExpectedState(LONG expected_state); private: void CheckAccessibleName(IAccessible* accessible); void CheckAccessibleRole(IAccessible* accessible); void CheckAccessibleValue(IAccessible* accessible); void CheckAccessibleState(IAccessible* accessible); void CheckAccessibleChildren(IAccessible* accessible); private: typedef vector AccessibleCheckerVector; // Expected accessible name. Checked against IAccessible::get_accName. wstring name_; // Expected accessible role. Checked against IAccessible::get_accRole. CComVariant role_; // Expected accessible value. Checked against IAccessible::get_accValue. wstring value_; // Expected accessible state. Checked against IAccessible::get_accState. LONG state_; // Expected accessible children. Checked using IAccessible::get_accChildCount // and ::AccessibleChildren. AccessibleCheckerVector children_; }; VARIANT CreateI4Variant(LONG value) { VARIANT variant = {0}; V_VT(&variant) = VT_I4; V_I4(&variant) = value; return variant; } IAccessible* GetAccessibleFromResultVariant(IAccessible* parent, VARIANT *var) { switch (V_VT(var)) { case VT_DISPATCH: return CComQIPtr(V_DISPATCH(var)).Detach(); break; case VT_I4: { CComPtr dispatch; HRESULT hr = parent->get_accChild(CreateI4Variant(V_I4(var)), &dispatch); EXPECT_TRUE(SUCCEEDED(hr)); return CComQIPtr(dispatch).Detach(); break; } } return NULL; } HRESULT QueryIAccessible2(IAccessible* accessible, IAccessible2** accessible2) { // TODO(ctguil): For some reason querying the IAccessible2 interface from // IAccessible fails. ScopedComPtr service_provider; HRESULT hr = accessible->QueryInterface(service_provider.Receive()); if (FAILED(hr)) return hr; hr = service_provider->QueryService(IID_IAccessible2, accessible2); return hr; } // Sets result to true if the child is located in the parent's tree. An // exhustive search is perform here because we determine equality using // IAccessible2::get_uniqueID which is only supported by the child node. void AccessibleContainsAccessible( IAccessible* parent, IAccessible2* child, bool* result) { vector> accessible_list; accessible_list.push_back(ScopedComPtr(parent)); LONG unique_id; HRESULT hr = child->get_uniqueID(&unique_id); ASSERT_EQ(hr, S_OK); *result = false; while (accessible_list.size()) { ScopedComPtr accessible = accessible_list.back(); accessible_list.pop_back(); ScopedComPtr accessible2; hr = QueryIAccessible2(accessible, accessible2.Receive()); if (SUCCEEDED(hr)) { LONG child_id; accessible2->get_uniqueID(&child_id); if (child_id == unique_id) { *result = true; break; } } LONG child_count; hr = accessible->get_accChildCount(&child_count); ASSERT_EQ(hr, S_OK); if (child_count == 0) continue; auto_ptr child_array(new VARIANT[child_count]); LONG obtained_count = 0; hr = AccessibleChildren( accessible, 0, child_count, child_array.get(), &obtained_count); ASSERT_EQ(hr, S_OK); ASSERT_EQ(child_count, obtained_count); for (int index = 0; index < obtained_count; index++) { ScopedComPtr child_accessible( GetAccessibleFromResultVariant(accessible, &child_array.get()[index])); if (child_accessible.get()) accessible_list.push_back(ScopedComPtr(child_accessible)); } } } // Retrieve the MSAA client accessibility object for the Render Widget Host View // of the selected tab. IAccessible* AccessibilityWinBrowserTest::GetRenderWidgetHostViewClientAccessible() { HWND hwnd_render_widget_host_view = browser()->GetSelectedTabContents()->GetRenderWidgetHostView()-> GetNativeView(); IAccessible* accessible; HRESULT hr = AccessibleObjectFromWindow( hwnd_render_widget_host_view, OBJID_CLIENT, IID_IAccessible, reinterpret_cast(&accessible)); EXPECT_EQ(S_OK, hr); EXPECT_NE(accessible, reinterpret_cast(NULL)); return accessible; } AccessibleChecker::AccessibleChecker( wstring expected_name, int32 expected_role, wstring expected_value) : name_(expected_name), role_(expected_role), value_(expected_value), state_(-1) { } AccessibleChecker::AccessibleChecker( wstring expected_name, wstring expected_role, wstring expected_value) : name_(expected_name), role_(expected_role.c_str()), value_(expected_value), state_(-1) { } void AccessibleChecker::AppendExpectedChild( AccessibleChecker* expected_child) { children_.push_back(expected_child); } void AccessibleChecker::CheckAccessible(IAccessible* accessible) { CheckAccessibleName(accessible); CheckAccessibleRole(accessible); CheckAccessibleValue(accessible); CheckAccessibleState(accessible); CheckAccessibleChildren(accessible); } void AccessibleChecker::SetExpectedValue(wstring expected_value) { value_ = expected_value; } void AccessibleChecker::SetExpectedState(LONG expected_state) { state_ = expected_state; } void AccessibleChecker::CheckAccessibleName(IAccessible* accessible) { CComBSTR name; HRESULT hr = accessible->get_accName(CreateI4Variant(CHILDID_SELF), &name); if (name_.empty()) { // If the object doesn't have name S_FALSE should be returned. EXPECT_EQ(hr, S_FALSE); } else { // Test that the correct string was returned. EXPECT_EQ(hr, S_OK); EXPECT_STREQ(name_.c_str(), wstring(name.m_str, SysStringLen(name)).c_str()); } } void AccessibleChecker::CheckAccessibleRole(IAccessible* accessible) { VARIANT var_role = {0}; HRESULT hr = accessible->get_accRole(CreateI4Variant(CHILDID_SELF), &var_role); EXPECT_EQ(hr, S_OK); ASSERT_TRUE(role_ == var_role); } void AccessibleChecker::CheckAccessibleValue(IAccessible* accessible) { CComBSTR value; HRESULT hr = accessible->get_accValue(CreateI4Variant(CHILDID_SELF), &value); EXPECT_EQ(hr, S_OK); // Test that the correct string was returned. EXPECT_STREQ(value_.c_str(), wstring(value.m_str, SysStringLen(value)).c_str()); } void AccessibleChecker::CheckAccessibleState(IAccessible* accessible) { if (state_ < 0) return; VARIANT var_state = {0}; HRESULT hr = accessible->get_accState(CreateI4Variant(CHILDID_SELF), &var_state); EXPECT_EQ(hr, S_OK); EXPECT_EQ(VT_I4, V_VT(&var_state)); ASSERT_TRUE(state_ == V_I4(&var_state)); } void AccessibleChecker::CheckAccessibleChildren(IAccessible* parent) { LONG child_count = 0; HRESULT hr = parent->get_accChildCount(&child_count); EXPECT_EQ(hr, S_OK); ASSERT_EQ(child_count, children_.size()); auto_ptr child_array(new VARIANT[child_count]); LONG obtained_count = 0; hr = AccessibleChildren(parent, 0, child_count, child_array.get(), &obtained_count); ASSERT_EQ(hr, S_OK); ASSERT_EQ(child_count, obtained_count); VARIANT* child = child_array.get(); for (AccessibleCheckerVector::iterator child_checker = children_.begin(); child_checker != children_.end(); ++child_checker, ++child) { ScopedComPtr child_accessible; child_accessible.Attach(GetAccessibleFromResultVariant(parent, child)); ASSERT_TRUE(child_accessible.get()); (*child_checker)->CheckAccessible(child_accessible); } } IN_PROC_BROWSER_TEST_F(AccessibilityWinBrowserTest, TestRendererAccessibilityTree) { // By requesting an accessible chrome will believe a screen reader has been // detected. ScopedComPtr document_accessible( GetRenderWidgetHostViewClientAccessible()); // The initial accessible returned should have state STATE_SYSTEM_BUSY while // the accessibility tree is being requested from the renderer. VARIANT var_state; HRESULT hr = document_accessible-> get_accState(CreateI4Variant(CHILDID_SELF), &var_state); EXPECT_EQ(hr, S_OK); EXPECT_EQ(V_VT(&var_state), VT_I4); EXPECT_EQ(V_I4(&var_state), STATE_SYSTEM_BUSY); // Wait for the initial accessibility tree to load. ui_test_utils::WaitForNotification( NotificationType::RENDER_VIEW_HOST_ACCESSIBILITY_TREE_UPDATED); GURL tree_url( "data:text/html,Accessibility Win Test" "" ""); browser()->OpenURL(tree_url, GURL(), CURRENT_TAB, PageTransition::TYPED); ui_test_utils::WaitForNotification( NotificationType::RENDER_VIEW_HOST_ACCESSIBILITY_TREE_UPDATED); document_accessible = GetRenderWidgetHostViewClientAccessible(); ASSERT_NE(document_accessible.get(), reinterpret_cast(NULL)); AccessibleChecker button_checker(L"push", ROLE_SYSTEM_PUSHBUTTON, L"push"); AccessibleChecker checkbox_checker(L"", ROLE_SYSTEM_CHECKBUTTON, L""); AccessibleChecker body_checker(L"", L"BODY", L""); body_checker.AppendExpectedChild(&button_checker); body_checker.AppendExpectedChild(&checkbox_checker); AccessibleChecker document_checker(L"", ROLE_SYSTEM_DOCUMENT, L""); document_checker.AppendExpectedChild(&body_checker); // Check the accessible tree of the renderer. document_checker.CheckAccessible(document_accessible); // Check that document accessible has a parent accessible. ScopedComPtr parent_dispatch; hr = document_accessible->get_accParent(parent_dispatch.Receive()); EXPECT_EQ(hr, S_OK); EXPECT_NE(parent_dispatch, reinterpret_cast(NULL)); // Navigate to another page. GURL about_url("about:"); ui_test_utils::NavigateToURL(browser(), about_url); // Verify that the IAccessible reference still points to a valid object and // that calls to its methods fail since the tree is no longer valid after // the page navagation. CComBSTR name; hr = document_accessible->get_accName(CreateI4Variant(CHILDID_SELF), &name); ASSERT_EQ(E_FAIL, hr); } IN_PROC_BROWSER_TEST_F(AccessibilityWinBrowserTest, TestDynamicAccessibilityTree) { // By requesting an accessible chrome will believe a screen reader has been // detected. Request and wait for the accessibility tree to be updated. GURL tree_url( "data:text/html,
new text" "';\">old text
" ""); browser()->OpenURL(tree_url, GURL(), CURRENT_TAB, PageTransition::TYPED); ScopedComPtr document_accessible( GetRenderWidgetHostViewClientAccessible()); ui_test_utils::WaitForNotification( NotificationType::RENDER_VIEW_HOST_ACCESSIBILITY_TREE_UPDATED); AccessibleChecker text_checker(L"", ROLE_SYSTEM_TEXT, L"old text"); AccessibleChecker div1_checker(L"", L"DIV", L""); div1_checker.AppendExpectedChild(&text_checker); AccessibleChecker checkbox_checker(L"", ROLE_SYSTEM_CHECKBUTTON, L""); checkbox_checker.SetExpectedState( STATE_SYSTEM_FOCUSABLE | STATE_SYSTEM_READONLY); AccessibleChecker div2_checker(L"", L"DIV", L""); div2_checker.AppendExpectedChild(&checkbox_checker); AccessibleChecker document_checker(L"", ROLE_SYSTEM_DOCUMENT, L""); document_checker.AppendExpectedChild(&div1_checker); document_checker.AppendExpectedChild(&div2_checker); // TODO(ctguil): Fix: We should not be expecting busy state here. document_checker.SetExpectedState(STATE_SYSTEM_BUSY); // Check the accessible tree of the browser. document_accessible = GetRenderWidgetHostViewClientAccessible(); ASSERT_NE(document_accessible.get(), reinterpret_cast(NULL)); document_checker.CheckAccessible(document_accessible); // Perform the default action on the div1 which executes the script that // updates text node within the div. CComPtr div1_dispatch; HRESULT hr = document_accessible->get_accChild(CreateI4Variant(1), &div1_dispatch); EXPECT_EQ(hr, S_OK); CComQIPtr div1_accessible(div1_dispatch); hr = div1_accessible->accDoDefaultAction(CreateI4Variant(CHILDID_SELF)); EXPECT_EQ(hr, S_OK); ui_test_utils::WaitForNotification( NotificationType::RENDER_VIEW_HOST_ACCESSIBILITY_TREE_UPDATED); // Check that the accessibility tree of the browser has been updated. text_checker.SetExpectedValue(L"new text"); document_checker.CheckAccessible(document_accessible); // Perform the default action on the checkbox which marks it as checked. CComPtr div2_dispatch; hr = document_accessible->get_accChild(CreateI4Variant(2), &div2_dispatch); EXPECT_EQ(hr, S_OK); CComQIPtr div2_accessible(div2_dispatch); CComPtr checkbox_dispatch; hr = div2_accessible->get_accChild(CreateI4Variant(1), &checkbox_dispatch); EXPECT_EQ(hr, S_OK); CComQIPtr checkbox_accessible(checkbox_dispatch); hr = checkbox_accessible->accDoDefaultAction(CreateI4Variant(CHILDID_SELF)); EXPECT_EQ(hr, S_OK); ui_test_utils::WaitForNotification( NotificationType::RENDER_VIEW_HOST_ACCESSIBILITY_TREE_UPDATED); // Check that the accessibility tree of the browser has been updated. checkbox_checker.SetExpectedState( STATE_SYSTEM_CHECKED | STATE_SYSTEM_FOCUSABLE | STATE_SYSTEM_FOCUSED | STATE_SYSTEM_READONLY); document_checker.CheckAccessible(document_accessible); } // FAILS crbug.com/54220 // This test verifies that browser-side cache of the renderer accessibility // tree is reachable from the browser's tree. Tools that analyze windows // accessibility trees like AccExplorer32 should be able to drill into the // cached renderer accessibility tree. IN_PROC_BROWSER_TEST_F(AccessibilityWinBrowserTest, FAILS_ContainsRendererAccessibilityTree) { // By requesting an accessible chrome will believe a screen reader has been // detected. Request and wait for the accessibility tree to be updated. GURL tree_url("data:text/html,"); browser()->OpenURL(tree_url, GURL(), CURRENT_TAB, PageTransition::TYPED); ScopedComPtr document_accessible( GetRenderWidgetHostViewClientAccessible()); ui_test_utils::WaitForNotification( NotificationType::RENDER_VIEW_HOST_ACCESSIBILITY_TREE_UPDATED); // Get the accessibility object for the browser window. HWND browser_hwnd = browser()->window()->GetNativeHandle(); ScopedComPtr browser_accessible; HRESULT hr = AccessibleObjectFromWindow( browser_hwnd, OBJID_WINDOW, IID_IAccessible, reinterpret_cast(browser_accessible.Receive())); ASSERT_EQ(S_OK, hr); // Get the accessibility object for the renderer client document. document_accessible = GetRenderWidgetHostViewClientAccessible(); ScopedComPtr document_accessible2; hr = QueryIAccessible2(document_accessible, document_accessible2.Receive()); ASSERT_EQ(S_OK, hr); // TODO(ctguil): Pointer comparison of retrieved IAccessible pointers dosen't // seem to work for here. Perhaps make IAccessible2 available in views to make // unique id comparison available. bool found = false; ScopedComPtr parent = document_accessible; while (parent.get()) { ScopedComPtr parent_dispatch; hr = parent->get_accParent(parent_dispatch.Receive()); ASSERT_TRUE(SUCCEEDED(hr)); if (!parent_dispatch.get()) { ASSERT_EQ(hr, S_FALSE); break; } parent.Release(); hr = parent_dispatch.QueryInterface(parent.Receive()); ASSERT_EQ(S_OK, hr); if (parent.get() == browser_accessible.get()) { found = true; break; } } // If pointer comparison fails resort to the exhuasive search that can use // IAccessible2::get_uniqueID for equality comparison. if (!found) { AccessibleContainsAccessible( browser_accessible, document_accessible2, &found); } ASSERT_EQ(found, true); } } // namespace.