Flutter Engine Uber Docs
Docs for the entire Flutter Engine repo.
 
Loading...
Searching...
No Matches
accessibility_bridge_windows_unittests.cc
Go to the documentation of this file.
1// Copyright 2013 The Flutter Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
6
7#include <comdef.h>
8#include <comutil.h>
9#include <oleacc.h>
10
11#include <vector>
12
13#include "flutter/fml/macros.h"
22#include "gmock/gmock.h"
23#include "gtest/gtest.h"
24
25namespace flutter {
26namespace testing {
27
28namespace {
29using ::testing::NiceMock;
30
31// A structure representing a Win32 MSAA event targeting a specified node.
32struct MsaaEvent {
33 std::shared_ptr<FlutterPlatformNodeDelegateWindows> node_delegate;
35};
36
37// Accessibility bridge delegate that captures events dispatched to the OS.
38class AccessibilityBridgeWindowsSpy : public AccessibilityBridgeWindows {
39 public:
41
42 explicit AccessibilityBridgeWindowsSpy(FlutterWindowsEngine* engine,
43 FlutterWindowsView* view)
44 : AccessibilityBridgeWindows(view) {}
45
46 void DispatchWinAccessibilityEvent(
47 std::shared_ptr<FlutterPlatformNodeDelegateWindows> node_delegate,
49 dispatched_events_.push_back({node_delegate, event_type});
50 }
51
52 void SetFocus(std::shared_ptr<FlutterPlatformNodeDelegateWindows>
53 node_delegate) override {
54 focused_nodes_.push_back(std::move(node_delegate));
55 }
56
57 void ResetRecords() {
58 dispatched_events_.clear();
59 focused_nodes_.clear();
60 }
61
62 const std::vector<MsaaEvent>& dispatched_events() const {
63 return dispatched_events_;
64 }
65
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();
72 });
73 return ids;
74 }
75
76 protected:
77 std::weak_ptr<FlutterPlatformNodeDelegate> GetFocusedNode() override {
78 return focused_nodes_.back();
79 }
80
81 private:
82 std::vector<MsaaEvent> dispatched_events_;
83 std::vector<std::shared_ptr<FlutterPlatformNodeDelegate>> focused_nodes_;
84
85 FML_DISALLOW_COPY_AND_ASSIGN(AccessibilityBridgeWindowsSpy);
86};
87
88// A FlutterWindowsView whose accessibility bridge is an
89// AccessibilityBridgeWindowsSpy.
90class FlutterWindowsViewSpy : public FlutterWindowsView {
91 public:
92 FlutterWindowsViewSpy(FlutterWindowsEngine* engine,
93 std::unique_ptr<WindowBindingHandler> handler)
94 : FlutterWindowsView(kImplicitViewId,
95 engine,
96 std::move(handler),
97 false,
98 BoxConstraints()) {}
99
100 protected:
101 virtual std::shared_ptr<AccessibilityBridgeWindows>
102 CreateAccessibilityBridge() override {
103 return std::make_shared<AccessibilityBridgeWindowsSpy>(GetEngine(), this);
104 }
105
106 private:
107 FML_DISALLOW_COPY_AND_ASSIGN(FlutterWindowsViewSpy);
108};
109
110// Returns an engine instance configured with dummy project path values, and
111// overridden methods for sending platform messages, so that the engine can
112// respond as if the framework were connected.
113std::unique_ptr<FlutterWindowsEngine> GetTestEngine() {
114 FlutterDesktopEngineProperties properties = {};
115 properties.assets_path = L"C:\\foo\\flutter_assets";
116 properties.icu_data_path = L"C:\\foo\\icudtl.dat";
117 properties.aot_library_path = L"C:\\foo\\aot.so";
118 FlutterProjectBundle project(properties);
119 auto engine = std::make_unique<FlutterWindowsEngine>(project);
120
121 EngineModifier modifier(engine.get());
122 modifier.embedder_api().UpdateSemanticsEnabled =
123 [](FLUTTER_API_SYMBOL(FlutterEngine) engine, bool enabled) {
124 return kSuccess;
125 };
126
128 std::make_shared<MockKeyResponseController>());
129
130 engine->Run();
131 return engine;
132}
133
134// Populates the AXTree associated with the specified bridge with test data.
135//
136// node0
137// / \
138// node1 node2
139// / \
140// node3 node4
141//
142// node0 and node2 are grouping nodes. node1 and node2 are static text nodes.
143// node4 is a static text node with no text, and hence has the "ignored" state.
144void PopulateAXTree(std::shared_ptr<AccessibilityBridge> bridge) {
145 // Add node 0: root.
147 auto empty_flags = FlutterSemanticsFlags{};
148 std::vector<int32_t> node0_children{1, 2};
149 node0.child_count = node0_children.size();
150 node0.children_in_traversal_order = node0_children.data();
151 node0.children_in_hit_test_order = node0_children.data();
152 node0.flags2 = &empty_flags;
153
154 // Add node 1: text child of node 0.
156 node1.label = "prefecture";
157 node1.value = "Kyoto";
158 node1.flags2 = &empty_flags;
159
160 // Add node 2: subtree child of node 0.
162 std::vector<int32_t> node2_children{3, 4};
163 node2.child_count = node2_children.size();
164 node2.children_in_traversal_order = node2_children.data();
165 node2.children_in_hit_test_order = node2_children.data();
166 node2.flags2 = &empty_flags;
167
168 // Add node 3: text child of node 2.
170 node3.label = "city";
171 node3.value = "Uji";
172 node3.flags2 = &empty_flags;
173
174 // Add node 4: text child (with no text) of node 2.
176 node4.flags2 = &empty_flags;
177
178 bridge->AddFlutterSemanticsNodeUpdate(node0);
179 bridge->AddFlutterSemanticsNodeUpdate(node1);
180 bridge->AddFlutterSemanticsNodeUpdate(node2);
181 bridge->AddFlutterSemanticsNodeUpdate(node3);
182 bridge->AddFlutterSemanticsNodeUpdate(node4);
183 bridge->CommitUpdates();
184}
185
186ui::AXNode* AXNodeFromID(std::shared_ptr<AccessibilityBridge> bridge,
187 int32_t id) {
188 auto node_delegate = bridge->GetFlutterPlatformNodeDelegateFromID(id).lock();
189 return node_delegate ? node_delegate->GetAXNode() : nullptr;
190}
191
192std::shared_ptr<AccessibilityBridgeWindowsSpy> GetAccessibilityBridgeSpy(
193 FlutterWindowsView& view) {
194 return std::static_pointer_cast<AccessibilityBridgeWindowsSpy>(
195 view.accessibility_bridge().lock());
196}
197
198void ExpectWinEventFromAXEvent(int32_t node_id,
200 ax::mojom::Event expected_event) {
201 auto engine = GetTestEngine();
202 FlutterWindowsViewSpy view{
203 engine.get(), std::make_unique<NiceMock<MockWindowBindingHandler>>()};
204 EngineModifier modifier{engine.get()};
205 modifier.SetImplicitView(&view);
206 view.OnUpdateSemanticsEnabled(true);
207
208 auto bridge = GetAccessibilityBridgeSpy(view);
209 PopulateAXTree(bridge);
210
211 bridge->ResetRecords();
212 bridge->OnAccessibilityEvent({AXNodeFromID(bridge, node_id),
213 {ax_event, ax::mojom::EventFrom::kNone, {}}});
214 ASSERT_EQ(bridge->dispatched_events().size(), 1);
215 EXPECT_EQ(bridge->dispatched_events()[0].event_type, expected_event);
216}
217
218void ExpectWinEventFromAXEventOnFocusNode(int32_t node_id,
220 ax::mojom::Event expected_event,
221 int32_t focus_id) {
222 auto engine = GetTestEngine();
223 FlutterWindowsViewSpy view{
224 engine.get(), std::make_unique<NiceMock<MockWindowBindingHandler>>()};
225 EngineModifier modifier{engine.get()};
226 modifier.SetImplicitView(&view);
227 view.OnUpdateSemanticsEnabled(true);
228
229 auto bridge = GetAccessibilityBridgeSpy(view);
230 PopulateAXTree(bridge);
231
232 bridge->ResetRecords();
233 auto focus_delegate =
234 bridge->GetFlutterPlatformNodeDelegateFromID(focus_id).lock();
235 bridge->SetFocus(std::static_pointer_cast<FlutterPlatformNodeDelegateWindows>(
236 focus_delegate));
237 bridge->OnAccessibilityEvent({AXNodeFromID(bridge, node_id),
238 {ax_event, ax::mojom::EventFrom::kNone, {}}});
239 ASSERT_EQ(bridge->dispatched_events().size(), 1);
240 EXPECT_EQ(bridge->dispatched_events()[0].event_type, expected_event);
241 EXPECT_EQ(bridge->dispatched_events()[0].node_delegate->GetAXNode()->id(),
242 focus_id);
243}
244
245} // namespace
246
248 auto engine = GetTestEngine();
249 FlutterWindowsViewSpy view{
250 engine.get(), std::make_unique<NiceMock<MockWindowBindingHandler>>()};
251 EngineModifier modifier{engine.get()};
252 modifier.SetImplicitView(&view);
253 view.OnUpdateSemanticsEnabled(true);
254
255 auto bridge = view.accessibility_bridge().lock();
256 PopulateAXTree(bridge);
257
258 auto node0_delegate = bridge->GetFlutterPlatformNodeDelegateFromID(0).lock();
259 auto node1_delegate = bridge->GetFlutterPlatformNodeDelegateFromID(1).lock();
260 EXPECT_EQ(node0_delegate->GetNativeViewAccessible(),
261 node1_delegate->GetParent());
262}
263
264TEST(AccessibilityBridgeWindows, GetParentOnRootRetunsNullptr) {
265 auto engine = GetTestEngine();
266 FlutterWindowsViewSpy view{
267 engine.get(), std::make_unique<NiceMock<MockWindowBindingHandler>>()};
268 EngineModifier modifier{engine.get()};
269 modifier.SetImplicitView(&view);
270 view.OnUpdateSemanticsEnabled(true);
271
272 auto bridge = view.accessibility_bridge().lock();
273 PopulateAXTree(bridge);
274
275 auto node0_delegate = bridge->GetFlutterPlatformNodeDelegateFromID(0).lock();
276 ASSERT_TRUE(node0_delegate->GetParent() == nullptr);
277}
278
279TEST(AccessibilityBridgeWindows, DispatchAccessibilityAction) {
280 auto engine = GetTestEngine();
281 FlutterWindowsViewSpy view{
282 engine.get(), std::make_unique<NiceMock<MockWindowBindingHandler>>()};
283 EngineModifier modifier{engine.get()};
284 modifier.SetImplicitView(&view);
285 view.OnUpdateSemanticsEnabled(true);
286
287 auto bridge = view.accessibility_bridge().lock();
288 PopulateAXTree(bridge);
289
291 modifier.embedder_api().SendSemanticsAction = MOCK_ENGINE_PROC(
292 SendSemanticsAction,
293 ([&actual_action](FLUTTER_API_SYMBOL(FlutterEngine) engine,
294 const FlutterSendSemanticsActionInfo* info) {
295 actual_action = info->action;
296 return kSuccess;
297 }));
298
301 EXPECT_EQ(actual_action, kFlutterSemanticsActionCopy);
302}
303
304TEST(AccessibilityBridgeWindows, OnAccessibilityEventAlert) {
305 ExpectWinEventFromAXEvent(0, ui::AXEventGenerator::Event::ALERT,
307}
308
309TEST(AccessibilityBridgeWindows, OnAccessibilityEventChildrenChanged) {
310 ExpectWinEventFromAXEvent(0, ui::AXEventGenerator::Event::CHILDREN_CHANGED,
312}
313
314TEST(AccessibilityBridgeWindows, OnAccessibilityEventFocusChanged) {
315 auto engine = GetTestEngine();
316 FlutterWindowsViewSpy view{
317 engine.get(), std::make_unique<NiceMock<MockWindowBindingHandler>>()};
318 EngineModifier modifier{engine.get()};
319 modifier.SetImplicitView(&view);
320 view.OnUpdateSemanticsEnabled(true);
321
322 auto bridge = GetAccessibilityBridgeSpy(view);
323 PopulateAXTree(bridge);
324
325 bridge->ResetRecords();
326 bridge->OnAccessibilityEvent({AXNodeFromID(bridge, 1),
329 {}}});
330 ASSERT_EQ(bridge->dispatched_events().size(), 1);
331 EXPECT_EQ(bridge->dispatched_events()[0].event_type,
333
334 ASSERT_EQ(bridge->focused_nodes().size(), 1);
335 EXPECT_EQ(bridge->focused_nodes()[0], 1);
336}
337
338TEST(AccessibilityBridgeWindows, OnAccessibilityEventIgnoredChanged) {
339 // Static test nodes with no text, hint, or scrollability are ignored.
340 ExpectWinEventFromAXEvent(4, ui::AXEventGenerator::Event::IGNORED_CHANGED,
342}
343
344TEST(AccessibilityBridgeWindows, OnAccessibilityImageAnnotationChanged) {
345 ExpectWinEventFromAXEvent(
348}
349
350TEST(AccessibilityBridgeWindows, OnAccessibilityLiveRegionChanged) {
351 ExpectWinEventFromAXEvent(1, ui::AXEventGenerator::Event::LIVE_REGION_CHANGED,
353}
354
355TEST(AccessibilityBridgeWindows, OnAccessibilityNameChanged) {
356 ExpectWinEventFromAXEvent(1, ui::AXEventGenerator::Event::NAME_CHANGED,
358}
359
365
366TEST(AccessibilityBridgeWindows, OnAccessibilityVScrollPosChanged) {
367 ExpectWinEventFromAXEvent(
370}
371
372TEST(AccessibilityBridgeWindows, OnAccessibilitySelectedChanged) {
373 ExpectWinEventFromAXEvent(1, ui::AXEventGenerator::Event::SELECTED_CHANGED,
375}
376
377TEST(AccessibilityBridgeWindows, OnAccessibilitySelectedChildrenChanged) {
378 ExpectWinEventFromAXEvent(
381}
382
383TEST(AccessibilityBridgeWindows, OnAccessibilitySubtreeCreated) {
384 ExpectWinEventFromAXEvent(0, ui::AXEventGenerator::Event::SUBTREE_CREATED,
386}
387
388TEST(AccessibilityBridgeWindows, OnAccessibilityValueChanged) {
389 ExpectWinEventFromAXEvent(1, ui::AXEventGenerator::Event::VALUE_CHANGED,
391}
392
393TEST(AccessibilityBridgeWindows, OnAccessibilityStateChanged) {
394 ExpectWinEventFromAXEvent(
397}
398
399TEST(AccessibilityBridgeWindows, OnDocumentSelectionChanged) {
400 ExpectWinEventFromAXEventOnFocusNode(
403}
404
405} // namespace testing
406} // namespace flutter
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...
#define FLUTTER_API_SYMBOL(symbol)
Definition embedder.h:67
FlutterSemanticsAction
Definition embedder.h:122
@ kFlutterSemanticsActionCopy
Copy the current selection to the clipboard.
Definition embedder.h:155
@ kFlutterSemanticsActionTap
Definition embedder.h:125
FlutterEngine engine
Definition main.cc:84
FlView * view
const gchar FlBinaryMessengerMessageHandler handler
#define FML_DISALLOW_COPY_AND_ASSIGN(TypeName)
Definition macros.h:27
void MockEmbedderApiForKeyboard(EngineModifier &modifier, std::shared_ptr< MockKeyResponseController > response_controller)
TEST(NativeAssetsManagerTest, NoAvailableAssets)
constexpr int64_t kImplicitViewId
Definition ref_ptr.h:261
#define MOCK_ENGINE_PROC(proc, mock_impl)
size_t child_count
The number of children this node has.
Definition embedder.h:1693
const char * label
A textual description of the node.
Definition embedder.h:1673
FlutterSemanticsFlags * flags2
Definition embedder.h:1735
FlutterSemanticsAction action
The semantics action.
Definition embedder.h:2810