blob: 925da6605c7b22bf5538c0a00bc66a465c8445d5 [file] [log] [blame]
// Copyright 2016 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 "base/logging.h"
#include "build/build_config.h"
#include "content/browser/accessibility/browser_accessibility.h"
#include "content/browser/accessibility/browser_accessibility_manager.h"
#include "content/browser/web_contents/web_contents_impl.h"
#include "content/public/test/browser_test_utils.h"
#include "content/public/test/content_browser_test.h"
#include "content/public/test/content_browser_test_utils.h"
#include "content/public/test/test_utils.h"
#include "content/shell/browser/shell.h"
#include "content/test/accessibility_browser_test_utils.h"
#include "net/base/data_url.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/gfx/codec/png_codec.h"
#include "url/gurl.h"
namespace content {
namespace {
class AccessibilityActionBrowserTest : public ContentBrowserTest {
public:
AccessibilityActionBrowserTest() {}
~AccessibilityActionBrowserTest() override {}
protected:
BrowserAccessibility* FindNode(ax::mojom::Role role,
const std::string& name_or_value) {
BrowserAccessibility* root = GetManager()->GetRoot();
CHECK(root);
return FindNodeInSubtree(*root, role, name_or_value);
}
BrowserAccessibilityManager* GetManager() {
WebContentsImpl* web_contents =
static_cast<WebContentsImpl*>(shell()->web_contents());
return web_contents->GetRootBrowserAccessibilityManager();
}
void GetBitmapFromImageDataURL(BrowserAccessibility* target,
SkBitmap* bitmap) {
std::string image_data_url =
target->GetStringAttribute(ax::mojom::StringAttribute::kImageDataUrl);
std::string mimetype;
std::string charset;
std::string png_data;
ASSERT_TRUE(net::DataURL::Parse(GURL(image_data_url), &mimetype, &charset,
&png_data));
ASSERT_EQ("image/png", mimetype);
ASSERT_TRUE(gfx::PNGCodec::Decode(
reinterpret_cast<const unsigned char*>(png_data.data()),
png_data.size(), bitmap));
}
private:
BrowserAccessibility* FindNodeInSubtree(BrowserAccessibility& node,
ax::mojom::Role role,
const std::string& name_or_value) {
const auto& name =
node.GetStringAttribute(ax::mojom::StringAttribute::kName);
const auto& value =
node.GetStringAttribute(ax::mojom::StringAttribute::kValue);
if (node.GetRole() == role &&
(name == name_or_value || value == name_or_value)) {
return &node;
}
for (unsigned int i = 0; i < node.PlatformChildCount(); ++i) {
BrowserAccessibility* result =
FindNodeInSubtree(*node.PlatformGetChild(i), role, name_or_value);
if (result)
return result;
}
return nullptr;
}
};
} // namespace
// Canvas tests rely on the harness producing pixel output in order to read back
// pixels from a canvas element. So we have to override the setup function.
class AccessibilityCanvasActionBrowserTest
: public AccessibilityActionBrowserTest {
public:
void SetUp() override {
EnablePixelOutput();
ContentBrowserTest::SetUp();
}
};
IN_PROC_BROWSER_TEST_F(AccessibilityActionBrowserTest, FocusAction) {
NavigateToURL(shell(), GURL(url::kAboutBlankURL));
AccessibilityNotificationWaiter waiter(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kLoadComplete);
GURL url(
"data:text/html,"
"<button>One</button>"
"<button>Two</button>"
"<button>Three</button>");
NavigateToURL(shell(), url);
waiter.WaitForNotification();
BrowserAccessibility* target = FindNode(ax::mojom::Role::kButton, "One");
ASSERT_NE(nullptr, target);
AccessibilityNotificationWaiter waiter2(
shell()->web_contents(), ui::kAXModeComplete, ax::mojom::Event::kFocus);
GetManager()->SetFocus(*target);
waiter2.WaitForNotification();
BrowserAccessibility* focus = GetManager()->GetFocus();
EXPECT_EQ(focus->GetId(), target->GetId());
}
IN_PROC_BROWSER_TEST_F(AccessibilityActionBrowserTest,
IncrementDecrementActions) {
NavigateToURL(shell(), GURL(url::kAboutBlankURL));
AccessibilityNotificationWaiter waiter(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kLoadComplete);
GURL url(
"data:text/html,"
"<input type=range min=2 value=8 max=10 step=2>");
NavigateToURL(shell(), url);
waiter.WaitForNotification();
BrowserAccessibility* target = FindNode(ax::mojom::Role::kSlider, "");
ASSERT_NE(nullptr, target);
EXPECT_EQ(8.0, target->GetFloatAttribute(
ax::mojom::FloatAttribute::kValueForRange));
// Increment, should result in value changing from 8 to 10.
{
AccessibilityNotificationWaiter waiter2(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kValueChanged);
GetManager()->Increment(*target);
waiter2.WaitForNotification();
}
EXPECT_EQ(10.0, target->GetFloatAttribute(
ax::mojom::FloatAttribute::kValueForRange));
// Increment, should result in value staying the same (max).
{
AccessibilityNotificationWaiter waiter2(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kValueChanged);
GetManager()->Increment(*target);
waiter2.WaitForNotification();
}
EXPECT_EQ(10.0, target->GetFloatAttribute(
ax::mojom::FloatAttribute::kValueForRange));
// Decrement, should result in value changing from 10 to 8.
{
AccessibilityNotificationWaiter waiter2(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kValueChanged);
GetManager()->Decrement(*target);
waiter2.WaitForNotification();
}
EXPECT_EQ(8.0, target->GetFloatAttribute(
ax::mojom::FloatAttribute::kValueForRange));
}
IN_PROC_BROWSER_TEST_F(AccessibilityActionBrowserTest, Scroll) {
NavigateToURL(shell(), GURL(url::kAboutBlankURL));
AccessibilityNotificationWaiter waiter(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kLoadComplete);
GURL url(
"data:text/html,"
"<div style='width:100; height:50; overflow:scroll' "
"aria-label='shakespeare'>"
"To be or not to be, that is the question."
"</div>");
NavigateToURL(shell(), url);
waiter.WaitForNotification();
BrowserAccessibility* target =
FindNode(ax::mojom::Role::kGenericContainer, "shakespeare");
EXPECT_NE(target, nullptr);
int y_before = target->GetIntAttribute(ax::mojom::IntAttribute::kScrollY);
AccessibilityNotificationWaiter waiter2(
shell()->web_contents(), ui::kAXModeComplete,
ax::mojom::Event::kScrollPositionChanged);
ui::AXActionData data;
data.action = ax::mojom::Action::kScrollDown;
data.target_node_id = target->GetId();
target->manager()->delegate()->AccessibilityPerformAction(data);
waiter2.WaitForNotification();
int y_after = target->GetIntAttribute(ax::mojom::IntAttribute::kScrollY);
EXPECT_GT(y_after, y_before);
}
IN_PROC_BROWSER_TEST_F(AccessibilityCanvasActionBrowserTest, CanvasGetImage) {
NavigateToURL(shell(), GURL(url::kAboutBlankURL));
AccessibilityNotificationWaiter waiter(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kLoadComplete);
GURL url(
"data:text/html,"
"<body>"
"<canvas aria-label='canvas' id='c' width='4' height='2'></canvas>"
"<script>\n"
" var c = document.getElementById('c').getContext('2d');\n"
" c.beginPath();\n"
" c.moveTo(0, 0.5);\n"
" c.lineTo(4, 0.5);\n"
" c.strokeStyle = '%23ff0000';\n"
" c.stroke();\n"
" c.beginPath();\n"
" c.moveTo(0, 1.5);\n"
" c.lineTo(4, 1.5);\n"
" c.strokeStyle = '%230000ff';\n"
" c.stroke();\n"
"</script>"
"</body>");
NavigateToURL(shell(), url);
waiter.WaitForNotification();
BrowserAccessibility* target = FindNode(ax::mojom::Role::kCanvas, "canvas");
ASSERT_NE(nullptr, target);
AccessibilityNotificationWaiter waiter2(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kImageFrameUpdated);
GetManager()->GetImageData(*target, gfx::Size());
waiter2.WaitForNotification();
SkBitmap bitmap;
GetBitmapFromImageDataURL(target, &bitmap);
ASSERT_EQ(4, bitmap.width());
ASSERT_EQ(2, bitmap.height());
EXPECT_EQ(SK_ColorRED, bitmap.getColor(0, 0));
EXPECT_EQ(SK_ColorRED, bitmap.getColor(1, 0));
EXPECT_EQ(SK_ColorRED, bitmap.getColor(2, 0));
EXPECT_EQ(SK_ColorRED, bitmap.getColor(3, 0));
EXPECT_EQ(SK_ColorBLUE, bitmap.getColor(0, 1));
EXPECT_EQ(SK_ColorBLUE, bitmap.getColor(1, 1));
EXPECT_EQ(SK_ColorBLUE, bitmap.getColor(2, 1));
EXPECT_EQ(SK_ColorBLUE, bitmap.getColor(3, 1));
}
IN_PROC_BROWSER_TEST_F(AccessibilityCanvasActionBrowserTest,
CanvasGetImageScale) {
NavigateToURL(shell(), GURL(url::kAboutBlankURL));
AccessibilityNotificationWaiter waiter(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kLoadComplete);
GURL url(
"data:text/html,"
"<body>"
"<canvas aria-label='canvas' id='c' width='40' height='20'></canvas>"
"<script>\n"
" var c = document.getElementById('c').getContext('2d');\n"
" c.fillStyle = '%2300ff00';\n"
" c.fillRect(0, 0, 40, 10);\n"
" c.fillStyle = '%23ff00ff';\n"
" c.fillRect(0, 10, 40, 10);\n"
"</script>"
"</body>");
NavigateToURL(shell(), url);
waiter.WaitForNotification();
BrowserAccessibility* target = FindNode(ax::mojom::Role::kCanvas, "canvas");
ASSERT_NE(nullptr, target);
AccessibilityNotificationWaiter waiter2(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kImageFrameUpdated);
GetManager()->GetImageData(*target, gfx::Size(4, 4));
waiter2.WaitForNotification();
SkBitmap bitmap;
GetBitmapFromImageDataURL(target, &bitmap);
ASSERT_EQ(4, bitmap.width());
ASSERT_EQ(2, bitmap.height());
EXPECT_EQ(SK_ColorGREEN, bitmap.getColor(0, 0));
EXPECT_EQ(SK_ColorGREEN, bitmap.getColor(1, 0));
EXPECT_EQ(SK_ColorGREEN, bitmap.getColor(2, 0));
EXPECT_EQ(SK_ColorGREEN, bitmap.getColor(3, 0));
EXPECT_EQ(SK_ColorMAGENTA, bitmap.getColor(0, 1));
EXPECT_EQ(SK_ColorMAGENTA, bitmap.getColor(1, 1));
EXPECT_EQ(SK_ColorMAGENTA, bitmap.getColor(2, 1));
EXPECT_EQ(SK_ColorMAGENTA, bitmap.getColor(3, 1));
}
IN_PROC_BROWSER_TEST_F(AccessibilityActionBrowserTest, ImgElementGetImage) {
NavigateToURL(shell(), GURL(url::kAboutBlankURL));
AccessibilityNotificationWaiter waiter(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kLoadComplete);
GURL url(
"data:text/html,"
"<body>"
"<img src='data:image/gif;base64,R0lGODdhAgADAKEDAAAA//"
"8AAAD/AP///ywAAAAAAgADAAACBEwkAAUAOw=='>"
"</body>");
NavigateToURL(shell(), url);
waiter.WaitForNotification();
BrowserAccessibility* target = FindNode(ax::mojom::Role::kImage, "");
ASSERT_NE(nullptr, target);
AccessibilityNotificationWaiter waiter2(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kImageFrameUpdated);
GetManager()->GetImageData(*target, gfx::Size());
waiter2.WaitForNotification();
SkBitmap bitmap;
GetBitmapFromImageDataURL(target, &bitmap);
ASSERT_EQ(2, bitmap.width());
ASSERT_EQ(3, bitmap.height());
EXPECT_EQ(SK_ColorRED, bitmap.getColor(0, 0));
EXPECT_EQ(SK_ColorRED, bitmap.getColor(1, 0));
EXPECT_EQ(SK_ColorGREEN, bitmap.getColor(0, 1));
EXPECT_EQ(SK_ColorGREEN, bitmap.getColor(1, 1));
EXPECT_EQ(SK_ColorBLUE, bitmap.getColor(0, 2));
EXPECT_EQ(SK_ColorBLUE, bitmap.getColor(1, 2));
}
IN_PROC_BROWSER_TEST_F(AccessibilityActionBrowserTest,
DoDefaultActionFocusesContentEditable) {
NavigateToURL(shell(), GURL(url::kAboutBlankURL));
AccessibilityNotificationWaiter waiter(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kLoadComplete);
GURL url(
"data:text/html,"
"<div><button>Before</button></div>"
"<div contenteditable>Editable text</div>"
"<div><button>After</button></div>");
NavigateToURL(shell(), url);
waiter.WaitForNotification();
BrowserAccessibility* target =
FindNode(ax::mojom::Role::kGenericContainer, "Editable text");
ASSERT_NE(nullptr, target);
AccessibilityNotificationWaiter waiter2(
shell()->web_contents(), ui::kAXModeComplete, ax::mojom::Event::kFocus);
GetManager()->DoDefaultAction(*target);
waiter2.WaitForNotification();
BrowserAccessibility* focus = GetManager()->GetFocus();
EXPECT_EQ(focus->GetId(), target->GetId());
}
IN_PROC_BROWSER_TEST_F(AccessibilityActionBrowserTest, InputSetValue) {
NavigateToURL(shell(), GURL(url::kAboutBlankURL));
AccessibilityNotificationWaiter waiter(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kLoadComplete);
GURL url(
"data:text/html,"
"<input aria-label='Answer' value='Before'>");
NavigateToURL(shell(), url);
waiter.WaitForNotification();
BrowserAccessibility* target =
FindNode(ax::mojom::Role::kTextField, "Answer");
ASSERT_NE(nullptr, target);
EXPECT_EQ("Before",
target->GetStringAttribute(ax::mojom::StringAttribute::kValue));
AccessibilityNotificationWaiter waiter2(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kValueChanged);
GetManager()->SetValue(*target, "After");
waiter2.WaitForNotification();
EXPECT_EQ("After",
target->GetStringAttribute(ax::mojom::StringAttribute::kValue));
}
IN_PROC_BROWSER_TEST_F(AccessibilityActionBrowserTest, TextareaSetValue) {
NavigateToURL(shell(), GURL(url::kAboutBlankURL));
AccessibilityNotificationWaiter waiter(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kLoadComplete);
GURL url(
"data:text/html,"
"<textarea aria-label='Answer'>Before</textarea>");
NavigateToURL(shell(), url);
waiter.WaitForNotification();
BrowserAccessibility* target =
FindNode(ax::mojom::Role::kTextField, "Answer");
ASSERT_NE(nullptr, target);
EXPECT_EQ("Before",
target->GetStringAttribute(ax::mojom::StringAttribute::kValue));
AccessibilityNotificationWaiter waiter2(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kValueChanged);
GetManager()->SetValue(*target, "Line1\nLine2");
waiter2.WaitForNotification();
EXPECT_EQ("Line1\nLine2",
target->GetStringAttribute(ax::mojom::StringAttribute::kValue));
// TODO(dmazzoni): On Android we use an ifdef to disable inline text boxes,
// which contain all of the line break information.
//
// We should do it with accessibility flags instead. http://crbug.com/672205
#if !defined(OS_ANDROID)
// Check that it really does contain two lines.
auto start_pos = target->CreatePositionAt(0);
auto end_of_line_1 = start_pos->CreateNextLineEndPosition(
ui::AXBoundaryBehavior::CrossBoundary);
EXPECT_EQ(5, end_of_line_1->text_offset());
#endif
}
IN_PROC_BROWSER_TEST_F(AccessibilityActionBrowserTest,
ContenteditableSetValue) {
NavigateToURL(shell(), GURL(url::kAboutBlankURL));
AccessibilityNotificationWaiter waiter(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kLoadComplete);
GURL url(
"data:text/html,"
"<div contenteditable aria-label='Answer'>Before</div>");
NavigateToURL(shell(), url);
waiter.WaitForNotification();
BrowserAccessibility* target =
FindNode(ax::mojom::Role::kGenericContainer, "Answer");
ASSERT_NE(nullptr, target);
EXPECT_EQ("Before",
target->GetStringAttribute(ax::mojom::StringAttribute::kValue));
AccessibilityNotificationWaiter waiter2(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kValueChanged);
GetManager()->SetValue(*target, "Line1\nLine2");
waiter2.WaitForNotification();
EXPECT_EQ("Line1\nLine2",
target->GetStringAttribute(ax::mojom::StringAttribute::kValue));
// TODO(dmazzoni): On Android we use an ifdef to disable inline text boxes,
// which contain all of the line break information.
//
// We should do it with accessibility flags instead. http://crbug.com/672205
#if !defined(OS_ANDROID)
// Check that it really does contain two lines.
auto start_pos = target->CreatePositionAt(0);
auto end_of_line_1 = start_pos->CreateNextLineEndPosition(
ui::AXBoundaryBehavior::CrossBoundary);
EXPECT_EQ(5, end_of_line_1->text_offset());
#endif
}
IN_PROC_BROWSER_TEST_F(AccessibilityActionBrowserTest, ShowContextMenu) {
NavigateToURL(shell(), GURL(url::kAboutBlankURL));
AccessibilityNotificationWaiter waiter(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kLoadComplete);
GURL url(
"data:text/html,"
"<a href='about:blank'>1</a>"
"<a href='about:blank'>2</a>");
NavigateToURL(shell(), url);
waiter.WaitForNotification();
BrowserAccessibility* target_node = FindNode(ax::mojom::Role::kLink, "2");
EXPECT_NE(target_node, nullptr);
// Register a ContextMenuFilter in the render process to wait for the
// ShowContextMenu event to be raised.
content::RenderProcessHost* render_process_host =
shell()->web_contents()->GetMainFrame()->GetProcess();
auto context_menu_filter = base::MakeRefCounted<ContextMenuFilter>();
render_process_host->AddFilter(context_menu_filter.get());
// Raise the ShowContextMenu event from the second link.
ui::AXActionData context_menu_action;
context_menu_action.action = ax::mojom::Action::kShowContextMenu;
target_node->AccessibilityPerformAction(context_menu_action);
context_menu_filter->Wait();
ContextMenuParams context_menu_params = context_menu_filter->get_params();
EXPECT_EQ(base::ASCIIToUTF16("2"), context_menu_params.link_text);
EXPECT_EQ(ui::MenuSourceType::MENU_SOURCE_NONE,
context_menu_params.source_type);
}
IN_PROC_BROWSER_TEST_F(AccessibilityActionBrowserTest,
AriaGridSelectedChangedEvent) {
NavigateToURL(shell(), GURL(url::kAboutBlankURL));
AccessibilityNotificationWaiter waiter(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kLoadComplete);
GURL url(
"data:text/html,"
"<body>"
"<script>"
"function tdclick(ele, event) {"
"var selected = ele.getAttribute('aria-selected');"
"ele.setAttribute('aria-selected', selected != 'true');"
"event.stopPropagation();"
"}"
"</script>"
"<table role='grid' multi aria-multiselectable='true'><tbody>"
"<tr>"
"<td role='gridcell' aria-selected='true' tabindex='0' "
"onclick='tdclick(this, event)'>A</td>"
"<td role='gridcell' aria-selected='false' tabindex='-1' "
"onclick='tdclick(this, event)'>B</td>"
"</tr>"
"</tbody></table>"
"</body>");
NavigateToURL(shell(), url);
waiter.WaitForNotification();
BrowserAccessibility* cell1 = FindNode(ax::mojom::Role::kCell, "A");
ASSERT_NE(nullptr, cell1);
BrowserAccessibility* cell2 = FindNode(ax::mojom::Role::kCell, "B");
ASSERT_NE(nullptr, cell2);
// Initial state
EXPECT_TRUE(cell1->GetBoolAttribute(ax::mojom::BoolAttribute::kSelected));
EXPECT_FALSE(cell2->GetBoolAttribute(ax::mojom::BoolAttribute::kSelected));
{
AccessibilityNotificationWaiter selection_waiter(
shell()->web_contents(), ui::kAXModeComplete,
ui::AXEventGenerator::Event::SELECTED_CHANGED);
GetManager()->DoDefaultAction(*cell2);
selection_waiter.WaitForNotification();
EXPECT_EQ(cell2->GetId(), selection_waiter.event_target_id());
EXPECT_TRUE(cell1->GetBoolAttribute(ax::mojom::BoolAttribute::kSelected));
EXPECT_TRUE(cell2->GetBoolAttribute(ax::mojom::BoolAttribute::kSelected));
}
{
AccessibilityNotificationWaiter selection_waiter(
shell()->web_contents(), ui::kAXModeComplete,
ui::AXEventGenerator::Event::SELECTED_CHANGED);
GetManager()->DoDefaultAction(*cell1);
selection_waiter.WaitForNotification();
EXPECT_EQ(cell1->GetId(), selection_waiter.event_target_id());
EXPECT_FALSE(cell1->GetBoolAttribute(ax::mojom::BoolAttribute::kSelected));
EXPECT_TRUE(cell2->GetBoolAttribute(ax::mojom::BoolAttribute::kSelected));
}
{
AccessibilityNotificationWaiter selection_waiter(
shell()->web_contents(), ui::kAXModeComplete,
ui::AXEventGenerator::Event::SELECTED_CHANGED);
GetManager()->DoDefaultAction(*cell2);
selection_waiter.WaitForNotification();
EXPECT_EQ(cell2->GetId(), selection_waiter.event_target_id());
EXPECT_FALSE(cell1->GetBoolAttribute(ax::mojom::BoolAttribute::kSelected));
EXPECT_FALSE(cell2->GetBoolAttribute(ax::mojom::BoolAttribute::kSelected));
}
}
IN_PROC_BROWSER_TEST_F(AccessibilityActionBrowserTest,
AriaControlsChangedEvent) {
NavigateToURL(shell(), GURL(url::kAboutBlankURL));
AccessibilityNotificationWaiter waiter(shell()->web_contents(),
ui::kAXModeComplete,
ax::mojom::Event::kLoadComplete);
GURL url(
"data:text/html,"
"<body>"
"<script>"
"function setcontrols(ele, event) {"
" ele.setAttribute('aria-controls', 'radio1 radio2');"
"}"
"</script>"
"<div id='radiogroup' role='radiogroup' aria-label='group'"
" onclick='setcontrols(this, event)'>"
"<div id='radio1' role='radio'>radio1</div>"
"<div id='radio2' role='radio'>radio2</div>"
"</div>"
"</body>");
NavigateToURL(shell(), url);
waiter.WaitForNotification();
BrowserAccessibility* target =
FindNode(ax::mojom::Role::kRadioGroup, "group");
ASSERT_NE(nullptr, target);
BrowserAccessibility* radio1 =
FindNode(ax::mojom::Role::kRadioButton, "radio1");
ASSERT_NE(nullptr, radio1);
BrowserAccessibility* radio2 =
FindNode(ax::mojom::Role::kRadioButton, "radio2");
ASSERT_NE(nullptr, radio2);
AccessibilityNotificationWaiter waiter2(
shell()->web_contents(), ui::kAXModeComplete,
ui::AXEventGenerator::Event::CONTROLS_CHANGED);
GetManager()->DoDefaultAction(*target);
waiter2.WaitForNotification();
auto&& control_list = target->GetData().GetIntListAttribute(
ax::mojom::IntListAttribute::kControlsIds);
EXPECT_EQ(2u, control_list.size());
auto find_radio1 =
std::find(control_list.cbegin(), control_list.cend(), radio1->GetId());
auto find_radio2 =
std::find(control_list.cbegin(), control_list.cend(), radio2->GetId());
EXPECT_NE(find_radio1, control_list.cend());
EXPECT_NE(find_radio2, control_list.cend());
}
IN_PROC_BROWSER_TEST_F(AccessibilityActionBrowserTest, FocusLostOnDeletedNode) {
NavigateToURL(shell(), GURL(url::kAboutBlankURL));
GURL url(
"data:text/html,"
"<button id='1'>1</button>"
"<iframe id='iframe' srcdoc=\""
"<button id='2'>2</button>"
"\"></iframe>");
NavigateToURL(shell(), url);
EnableAccessibilityForWebContents(shell()->web_contents());
auto FocusNodeAndReload = [this, &url](const std::string& node_name,
const std::string& focus_node_script) {
WaitForAccessibilityTreeToContainNodeWithName(shell()->web_contents(),
node_name);
BrowserAccessibility* node = FindNode(ax::mojom::Role::kButton, node_name);
ASSERT_NE(nullptr, node);
EXPECT_TRUE(ExecuteScript(shell(), focus_node_script));
WaitForAccessibilityFocusChange();
EXPECT_EQ(node->GetId(),
GetFocusedAccessibilityNodeInfo(shell()->web_contents()).id);
// Reloading the frames will achieve two things:
// 1. Force the deletion of the node being tested.
// 2. Lose focus on the node by focusing a new frame.
AccessibilityNotificationWaiter load_waiter(
shell()->web_contents(), ui::kAXModeComplete,
ax::mojom::Event::kLoadComplete);
NavigateToURL(shell(), url);
load_waiter.WaitForNotification();
};
FocusNodeAndReload("1", "document.getElementById('1').focus();");
FocusNodeAndReload("2",
"var iframe = document.getElementById('iframe');"
"var inner_doc = iframe.contentWindow.document;"
"inner_doc.getElementById('2').focus();");
}
} // namespace content