13 #include "flutter/fml/macros.h"
14 #include "flutter/shell/platform/embedder/embedder.h"
15 #include "flutter/shell/platform/embedder/test_utils/proc_table_replacement.h"
19 #include "flutter/shell/platform/windows/testing/engine_modifier.h"
20 #include "flutter/shell/platform/windows/testing/mock_window_binding_handler.h"
21 #include "flutter/shell/platform/windows/testing/test_keyboard.h"
22 #include "gmock/gmock.h"
23 #include "gtest/gtest.h"
29 using ::testing::NiceMock;
38 class AccessibilityBridgeWindowsSpy :
public AccessibilityBridgeWindows {
42 explicit AccessibilityBridgeWindowsSpy(FlutterWindowsEngine* engine,
43 FlutterWindowsView* view)
44 : AccessibilityBridgeWindows(view) {}
46 void DispatchWinAccessibilityEvent(
47 std::shared_ptr<FlutterPlatformNodeDelegateWindows>
node_delegate,
52 void SetFocus(std::shared_ptr<FlutterPlatformNodeDelegateWindows>
58 dispatched_events_.clear();
59 focused_nodes_.clear();
62 const std::vector<MsaaEvent>& dispatched_events()
const {
63 return dispatched_events_;
66 const std::vector<int32_t> focused_nodes()
const {
67 std::vector<int32_t> ids;
68 std::transform(focused_nodes_.begin(), focused_nodes_.end(),
69 std::back_inserter(ids),
70 [](std::shared_ptr<FlutterPlatformNodeDelegate> node) {
71 return node->GetAXNode()->id();
77 std::weak_ptr<FlutterPlatformNodeDelegate> GetFocusedNode()
override {
78 return focused_nodes_.back();
82 std::vector<MsaaEvent> dispatched_events_;
83 std::vector<std::shared_ptr<FlutterPlatformNodeDelegate>> focused_nodes_;
85 FML_DISALLOW_COPY_AND_ASSIGN(AccessibilityBridgeWindowsSpy);
90 class FlutterWindowsViewSpy :
public FlutterWindowsView {
92 FlutterWindowsViewSpy(FlutterWindowsEngine* engine,
93 std::unique_ptr<WindowBindingHandler> handler)
97 virtual std::shared_ptr<AccessibilityBridgeWindows>
98 CreateAccessibilityBridge()
override {
99 return std::make_shared<AccessibilityBridgeWindowsSpy>(GetEngine(),
this);
103 FML_DISALLOW_COPY_AND_ASSIGN(FlutterWindowsViewSpy);
109 std::unique_ptr<FlutterWindowsEngine> GetTestEngine() {
111 properties.
assets_path = L
"C:\\foo\\flutter_assets";
114 FlutterProjectBundle project(properties);
115 auto engine = std::make_unique<FlutterWindowsEngine>(project);
117 EngineModifier modifier(engine.get());
118 modifier.embedder_api().UpdateSemanticsEnabled =
119 [](FLUTTER_API_SYMBOL(FlutterEngine) engine,
bool enabled) {
123 MockEmbedderApiForKeyboard(modifier,
124 std::make_shared<MockKeyResponseController>());
140 void PopulateAXTree(std::shared_ptr<AccessibilityBridge> bridge) {
142 FlutterSemanticsNode2 node0{
sizeof(FlutterSemanticsNode2), 0};
143 auto empty_flags = FlutterSemanticsFlags{};
144 std::vector<int32_t> node0_children{1, 2};
145 node0.child_count = node0_children.size();
146 node0.children_in_traversal_order = node0_children.data();
147 node0.children_in_hit_test_order = node0_children.data();
148 node0.flags2 = &empty_flags;
151 FlutterSemanticsNode2 node1{
sizeof(FlutterSemanticsNode2), 1};
152 node1.label =
"prefecture";
153 node1.value =
"Kyoto";
154 node1.flags2 = &empty_flags;
157 FlutterSemanticsNode2 node2{
sizeof(FlutterSemanticsNode2), 2};
158 std::vector<int32_t> node2_children{3, 4};
159 node2.child_count = node2_children.size();
160 node2.children_in_traversal_order = node2_children.data();
161 node2.children_in_hit_test_order = node2_children.data();
162 node2.flags2 = &empty_flags;
165 FlutterSemanticsNode2 node3{
sizeof(FlutterSemanticsNode2), 3};
166 node3.label =
"city";
168 node3.flags2 = &empty_flags;
171 FlutterSemanticsNode2 node4{
sizeof(FlutterSemanticsNode2), 4};
172 node4.flags2 = &empty_flags;
174 bridge->AddFlutterSemanticsNodeUpdate(node0);
175 bridge->AddFlutterSemanticsNodeUpdate(node1);
176 bridge->AddFlutterSemanticsNodeUpdate(node2);
177 bridge->AddFlutterSemanticsNodeUpdate(node3);
178 bridge->AddFlutterSemanticsNodeUpdate(node4);
179 bridge->CommitUpdates();
182 ui::AXNode* AXNodeFromID(std::shared_ptr<AccessibilityBridge> bridge,
184 auto node_delegate = bridge->GetFlutterPlatformNodeDelegateFromID(
id).lock();
188 std::shared_ptr<AccessibilityBridgeWindowsSpy> GetAccessibilityBridgeSpy(
189 FlutterWindowsView& view) {
190 return std::static_pointer_cast<AccessibilityBridgeWindowsSpy>(
191 view.accessibility_bridge().lock());
194 void ExpectWinEventFromAXEvent(int32_t node_id,
195 ui::AXEventGenerator::Event ax_event,
196 ax::mojom::Event expected_event) {
197 auto engine = GetTestEngine();
198 FlutterWindowsViewSpy view{
199 engine.get(), std::make_unique<NiceMock<MockWindowBindingHandler>>()};
200 EngineModifier modifier{engine.get()};
201 modifier.SetImplicitView(&view);
202 view.OnUpdateSemanticsEnabled(
true);
204 auto bridge = GetAccessibilityBridgeSpy(view);
205 PopulateAXTree(bridge);
207 bridge->ResetRecords();
208 bridge->OnAccessibilityEvent({AXNodeFromID(bridge, node_id),
209 {ax_event, ax::mojom::EventFrom::kNone, {}}});
210 ASSERT_EQ(bridge->dispatched_events().size(), 1);
211 EXPECT_EQ(bridge->dispatched_events()[0].event_type, expected_event);
214 void ExpectWinEventFromAXEventOnFocusNode(int32_t node_id,
215 ui::AXEventGenerator::Event ax_event,
216 ax::mojom::Event expected_event,
218 auto engine = GetTestEngine();
219 FlutterWindowsViewSpy view{
220 engine.get(), std::make_unique<NiceMock<MockWindowBindingHandler>>()};
221 EngineModifier modifier{engine.get()};
222 modifier.SetImplicitView(&view);
223 view.OnUpdateSemanticsEnabled(
true);
225 auto bridge = GetAccessibilityBridgeSpy(view);
226 PopulateAXTree(bridge);
228 bridge->ResetRecords();
229 auto focus_delegate =
230 bridge->GetFlutterPlatformNodeDelegateFromID(focus_id).lock();
231 bridge->SetFocus(std::static_pointer_cast<FlutterPlatformNodeDelegateWindows>(
233 bridge->OnAccessibilityEvent({AXNodeFromID(bridge, node_id),
234 {ax_event, ax::mojom::EventFrom::kNone, {}}});
235 ASSERT_EQ(bridge->dispatched_events().size(), 1);
236 EXPECT_EQ(bridge->dispatched_events()[0].event_type, expected_event);
237 EXPECT_EQ(bridge->dispatched_events()[0].node_delegate->GetAXNode()->id(),
244 auto engine = GetTestEngine();
245 FlutterWindowsViewSpy view{
246 engine.get(), std::make_unique<NiceMock<MockWindowBindingHandler>>()};
247 EngineModifier modifier{engine.get()};
248 modifier.SetImplicitView(&view);
249 view.OnUpdateSemanticsEnabled(
true);
251 auto bridge = view.accessibility_bridge().lock();
252 PopulateAXTree(bridge);
254 auto node0_delegate = bridge->GetFlutterPlatformNodeDelegateFromID(0).lock();
255 auto node1_delegate = bridge->GetFlutterPlatformNodeDelegateFromID(1).lock();
256 EXPECT_EQ(node0_delegate->GetNativeViewAccessible(),
257 node1_delegate->GetParent());
261 auto engine = GetTestEngine();
262 FlutterWindowsViewSpy view{
263 engine.get(), std::make_unique<NiceMock<MockWindowBindingHandler>>()};
264 EngineModifier modifier{engine.get()};
265 modifier.SetImplicitView(&view);
266 view.OnUpdateSemanticsEnabled(
true);
268 auto bridge = view.accessibility_bridge().lock();
269 PopulateAXTree(bridge);
271 auto node0_delegate = bridge->GetFlutterPlatformNodeDelegateFromID(0).lock();
272 ASSERT_TRUE(node0_delegate->GetParent() ==
nullptr);
276 auto engine = GetTestEngine();
277 FlutterWindowsViewSpy view{
278 engine.get(), std::make_unique<NiceMock<MockWindowBindingHandler>>()};
279 EngineModifier modifier{engine.get()};
280 modifier.SetImplicitView(&view);
281 view.OnUpdateSemanticsEnabled(
true);
283 auto bridge = view.accessibility_bridge().lock();
284 PopulateAXTree(bridge);
286 FlutterSemanticsAction actual_action = kFlutterSemanticsActionTap;
287 modifier.embedder_api().SendSemanticsAction = MOCK_ENGINE_PROC(
290 const FlutterSendSemanticsActionInfo* info) {
291 actual_action = info->action;
297 EXPECT_EQ(actual_action, kFlutterSemanticsActionCopy);
301 ExpectWinEventFromAXEvent(0, ui::AXEventGenerator::Event::ALERT,
302 ax::mojom::Event::kAlert);
306 ExpectWinEventFromAXEvent(0, ui::AXEventGenerator::Event::CHILDREN_CHANGED,
307 ax::mojom::Event::kChildrenChanged);
311 auto engine = GetTestEngine();
312 FlutterWindowsViewSpy view{
313 engine.get(), std::make_unique<NiceMock<MockWindowBindingHandler>>()};
314 EngineModifier modifier{engine.get()};
315 modifier.SetImplicitView(&view);
316 view.OnUpdateSemanticsEnabled(
true);
318 auto bridge = GetAccessibilityBridgeSpy(view);
319 PopulateAXTree(bridge);
321 bridge->ResetRecords();
322 bridge->OnAccessibilityEvent({AXNodeFromID(bridge, 1),
323 {ui::AXEventGenerator::Event::FOCUS_CHANGED,
324 ax::mojom::EventFrom::kNone,
326 ASSERT_EQ(bridge->dispatched_events().size(), 1);
327 EXPECT_EQ(bridge->dispatched_events()[0].event_type,
328 ax::mojom::Event::kFocus);
330 ASSERT_EQ(bridge->focused_nodes().size(), 1);
331 EXPECT_EQ(bridge->focused_nodes()[0], 1);
336 ExpectWinEventFromAXEvent(4, ui::AXEventGenerator::Event::IGNORED_CHANGED,
337 ax::mojom::Event::kHide);
341 ExpectWinEventFromAXEvent(
342 1, ui::AXEventGenerator::Event::IMAGE_ANNOTATION_CHANGED,
343 ax::mojom::Event::kTextChanged);
347 ExpectWinEventFromAXEvent(1, ui::AXEventGenerator::Event::LIVE_REGION_CHANGED,
348 ax::mojom::Event::kLiveRegionChanged);
352 ExpectWinEventFromAXEvent(1, ui::AXEventGenerator::Event::NAME_CHANGED,
353 ax::mojom::Event::kTextChanged);
357 ExpectWinEventFromAXEvent(
358 1, ui::AXEventGenerator::Event::SCROLL_HORIZONTAL_POSITION_CHANGED,
359 ax::mojom::Event::kScrollPositionChanged);
363 ExpectWinEventFromAXEvent(
364 1, ui::AXEventGenerator::Event::SCROLL_VERTICAL_POSITION_CHANGED,
365 ax::mojom::Event::kScrollPositionChanged);
369 ExpectWinEventFromAXEvent(1, ui::AXEventGenerator::Event::SELECTED_CHANGED,
370 ax::mojom::Event::kValueChanged);
374 ExpectWinEventFromAXEvent(
375 2, ui::AXEventGenerator::Event::SELECTED_CHILDREN_CHANGED,
376 ax::mojom::Event::kSelectedChildrenChanged);
380 ExpectWinEventFromAXEvent(0, ui::AXEventGenerator::Event::SUBTREE_CREATED,
381 ax::mojom::Event::kShow);
385 ExpectWinEventFromAXEvent(1, ui::AXEventGenerator::Event::VALUE_CHANGED,
386 ax::mojom::Event::kValueChanged);
390 ExpectWinEventFromAXEvent(
391 1, ui::AXEventGenerator::Event::WIN_IACCESSIBLE_STATE_CHANGED,
392 ax::mojom::Event::kStateChanged);
396 ExpectWinEventFromAXEventOnFocusNode(
397 1, ui::AXEventGenerator::Event::DOCUMENT_SELECTION_CHANGED,
398 ax::mojom::Event::kDocumentSelectionChanged, 2);
std::shared_ptr< FlutterPlatformNodeDelegateWindows > node_delegate
ax::mojom::Event event_type
void DispatchAccessibilityAction(AccessibilityNodeId target, FlutterSemanticsAction action, fml::MallocMapping data) override
Dispatch accessibility action back to the Flutter framework. These actions are generated in the nativ...
void OnAccessibilityEvent(ui::AXEventGenerator::TargetedEvent targeted_event) override
Handle accessibility events generated due to accessibility tree changes. These events are needed to b...
TEST(AccessibilityBridgeWindows, GetParent)
constexpr FlutterViewId kImplicitViewId
const wchar_t * icu_data_path
const wchar_t * assets_path
const wchar_t * aot_library_path