1 // Copyright (c) 2012 The Chromium 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
5 #include "base/strings/string16.h"
6 #include "base/strings/utf_string_conversions.h"
7 #include "content/browser/accessibility/browser_accessibility.h"
8 #include "content/browser/accessibility/browser_accessibility_manager.h"
9 #if defined(OS_WIN)
10 #include "content/browser/accessibility/browser_accessibility_win.h"
11 #endif
12 #include "content/common/accessibility_messages.h"
13 #include "testing/gtest/include/gtest/gtest.h"
14
15 namespace content {
16 namespace {
17
18 // Subclass of BrowserAccessibility that counts the number of instances.
19 class CountedBrowserAccessibility : public BrowserAccessibility {
20 public:
CountedBrowserAccessibility()21 CountedBrowserAccessibility() {
22 global_obj_count_++;
23 native_ref_count_ = 1;
24 }
~CountedBrowserAccessibility()25 virtual ~CountedBrowserAccessibility() {
26 global_obj_count_--;
27 }
28
NativeAddReference()29 virtual void NativeAddReference() OVERRIDE {
30 native_ref_count_++;
31 }
32
NativeReleaseReference()33 virtual void NativeReleaseReference() OVERRIDE {
34 native_ref_count_--;
35 if (native_ref_count_ == 0)
36 delete this;
37 }
38
39 int native_ref_count_;
40 static int global_obj_count_;
41
42 #if defined(OS_WIN)
43 // Adds some padding to prevent a heap-buffer-overflow when an instance of
44 // this class is casted into a BrowserAccessibilityWin pointer.
45 // http://crbug.com/235508
46 // TODO(dmazzoni): Fix this properly.
47 static const size_t kDataSize = sizeof(int) + sizeof(BrowserAccessibility);
48 uint8 padding_[sizeof(BrowserAccessibilityWin) - kDataSize];
49 #endif
50 };
51
52 int CountedBrowserAccessibility::global_obj_count_ = 0;
53
54 // Factory that creates a CountedBrowserAccessibility.
55 class CountedBrowserAccessibilityFactory
56 : public BrowserAccessibilityFactory {
57 public:
~CountedBrowserAccessibilityFactory()58 virtual ~CountedBrowserAccessibilityFactory() {}
Create()59 virtual BrowserAccessibility* Create() OVERRIDE {
60 return new CountedBrowserAccessibility();
61 }
62 };
63
64 class TestBrowserAccessibilityDelegate
65 : public BrowserAccessibilityDelegate {
66 public:
TestBrowserAccessibilityDelegate()67 TestBrowserAccessibilityDelegate()
68 : got_fatal_error_(false) {}
69
AccessibilitySetFocus(int acc_obj_id)70 virtual void AccessibilitySetFocus(int acc_obj_id) OVERRIDE {}
AccessibilityDoDefaultAction(int acc_obj_id)71 virtual void AccessibilityDoDefaultAction(int acc_obj_id) OVERRIDE {}
AccessibilityShowMenu(const gfx::Point & point)72 virtual void AccessibilityShowMenu(const gfx::Point& point) OVERRIDE {}
AccessibilityScrollToMakeVisible(int acc_obj_id,const gfx::Rect & subfocus)73 virtual void AccessibilityScrollToMakeVisible(
74 int acc_obj_id, const gfx::Rect& subfocus) OVERRIDE {}
AccessibilityScrollToPoint(int acc_obj_id,const gfx::Point & point)75 virtual void AccessibilityScrollToPoint(
76 int acc_obj_id, const gfx::Point& point) OVERRIDE {}
AccessibilitySetTextSelection(int acc_obj_id,int start_offset,int end_offset)77 virtual void AccessibilitySetTextSelection(
78 int acc_obj_id, int start_offset, int end_offset) OVERRIDE {}
AccessibilityViewHasFocus() const79 virtual bool AccessibilityViewHasFocus() const OVERRIDE {
80 return false;
81 }
AccessibilityGetViewBounds() const82 virtual gfx::Rect AccessibilityGetViewBounds() const OVERRIDE {
83 return gfx::Rect();
84 }
AccessibilityOriginInScreen(const gfx::Rect & bounds) const85 virtual gfx::Point AccessibilityOriginInScreen(
86 const gfx::Rect& bounds) const OVERRIDE {
87 return gfx::Point();
88 }
AccessibilityHitTest(const gfx::Point & point)89 virtual void AccessibilityHitTest(const gfx::Point& point) OVERRIDE {}
AccessibilityFatalError()90 virtual void AccessibilityFatalError() OVERRIDE {
91 got_fatal_error_ = true;
92 }
AccessibilityGetAcceleratedWidget()93 virtual gfx::AcceleratedWidget AccessibilityGetAcceleratedWidget() OVERRIDE {
94 return gfx::kNullAcceleratedWidget;
95 }
AccessibilityGetNativeViewAccessible()96 virtual gfx::NativeViewAccessible AccessibilityGetNativeViewAccessible()
97 OVERRIDE {
98 return NULL;
99 }
AccessibilityGetChildFrame(int accessibility_node_id)100 virtual BrowserAccessibilityManager* AccessibilityGetChildFrame(
101 int accessibility_node_id) OVERRIDE {
102 return NULL;
103 }
AccessibilityGetParentFrame()104 virtual BrowserAccessibility* AccessibilityGetParentFrame() OVERRIDE {
105 return NULL;
106 }
107
got_fatal_error() const108 bool got_fatal_error() const { return got_fatal_error_; }
reset_got_fatal_error()109 void reset_got_fatal_error() { got_fatal_error_ = false; }
110
111 private:
112 bool got_fatal_error_;
113 };
114
115 } // anonymous namespace
116
TEST(BrowserAccessibilityManagerTest,TestNoLeaks)117 TEST(BrowserAccessibilityManagerTest, TestNoLeaks) {
118 // Create ui::AXNodeData objects for a simple document tree,
119 // representing the accessibility information used to initialize
120 // BrowserAccessibilityManager.
121 ui::AXNodeData button;
122 button.id = 2;
123 button.SetName("Button");
124 button.role = ui::AX_ROLE_BUTTON;
125 button.state = 0;
126
127 ui::AXNodeData checkbox;
128 checkbox.id = 3;
129 checkbox.SetName("Checkbox");
130 checkbox.role = ui::AX_ROLE_CHECK_BOX;
131 checkbox.state = 0;
132
133 ui::AXNodeData root;
134 root.id = 1;
135 root.SetName("Document");
136 root.role = ui::AX_ROLE_ROOT_WEB_AREA;
137 root.state = 0;
138 root.child_ids.push_back(2);
139 root.child_ids.push_back(3);
140
141 // Construct a BrowserAccessibilityManager with this
142 // ui::AXNodeData tree and a factory for an instance-counting
143 // BrowserAccessibility, and ensure that exactly 3 instances were
144 // created. Note that the manager takes ownership of the factory.
145 CountedBrowserAccessibility::global_obj_count_ = 0;
146 BrowserAccessibilityManager* manager =
147 BrowserAccessibilityManager::Create(
148 MakeAXTreeUpdate(root, button, checkbox),
149 NULL,
150 new CountedBrowserAccessibilityFactory());
151
152 ASSERT_EQ(3, CountedBrowserAccessibility::global_obj_count_);
153
154 // Delete the manager and test that all 3 instances are deleted.
155 delete manager;
156 ASSERT_EQ(0, CountedBrowserAccessibility::global_obj_count_);
157
158 // Construct a manager again, and this time save references to two of
159 // the three nodes in the tree.
160 manager =
161 BrowserAccessibilityManager::Create(
162 MakeAXTreeUpdate(root, button, checkbox),
163 NULL,
164 new CountedBrowserAccessibilityFactory());
165 ASSERT_EQ(3, CountedBrowserAccessibility::global_obj_count_);
166
167 CountedBrowserAccessibility* root_accessible =
168 static_cast<CountedBrowserAccessibility*>(manager->GetRoot());
169 root_accessible->NativeAddReference();
170 CountedBrowserAccessibility* child1_accessible =
171 static_cast<CountedBrowserAccessibility*>(
172 root_accessible->PlatformGetChild(1));
173 child1_accessible->NativeAddReference();
174
175 // Now delete the manager, and only one of the three nodes in the tree
176 // should be released.
177 delete manager;
178 ASSERT_EQ(2, CountedBrowserAccessibility::global_obj_count_);
179
180 // Release each of our references and make sure that each one results in
181 // the instance being deleted as its reference count hits zero.
182 root_accessible->NativeReleaseReference();
183 ASSERT_EQ(1, CountedBrowserAccessibility::global_obj_count_);
184 child1_accessible->NativeReleaseReference();
185 ASSERT_EQ(0, CountedBrowserAccessibility::global_obj_count_);
186 }
187
TEST(BrowserAccessibilityManagerTest,TestReuseBrowserAccessibilityObjects)188 TEST(BrowserAccessibilityManagerTest, TestReuseBrowserAccessibilityObjects) {
189 // Make sure that changes to a subtree reuse as many objects as possible.
190
191 // Tree 1:
192 //
193 // root
194 // child1
195 // child2
196 // child3
197
198 ui::AXNodeData tree1_child1;
199 tree1_child1.id = 2;
200 tree1_child1.SetName("Child1");
201 tree1_child1.role = ui::AX_ROLE_BUTTON;
202 tree1_child1.state = 0;
203
204 ui::AXNodeData tree1_child2;
205 tree1_child2.id = 3;
206 tree1_child2.SetName("Child2");
207 tree1_child2.role = ui::AX_ROLE_BUTTON;
208 tree1_child2.state = 0;
209
210 ui::AXNodeData tree1_child3;
211 tree1_child3.id = 4;
212 tree1_child3.SetName("Child3");
213 tree1_child3.role = ui::AX_ROLE_BUTTON;
214 tree1_child3.state = 0;
215
216 ui::AXNodeData tree1_root;
217 tree1_root.id = 1;
218 tree1_root.SetName("Document");
219 tree1_root.role = ui::AX_ROLE_ROOT_WEB_AREA;
220 tree1_root.state = 0;
221 tree1_root.child_ids.push_back(2);
222 tree1_root.child_ids.push_back(3);
223 tree1_root.child_ids.push_back(4);
224
225 // Tree 2:
226 //
227 // root
228 // child0 <-- inserted
229 // child1
230 // child2
231 // <-- child3 deleted
232
233 ui::AXNodeData tree2_child0;
234 tree2_child0.id = 5;
235 tree2_child0.SetName("Child0");
236 tree2_child0.role = ui::AX_ROLE_BUTTON;
237 tree2_child0.state = 0;
238
239 ui::AXNodeData tree2_root;
240 tree2_root.id = 1;
241 tree2_root.SetName("DocumentChanged");
242 tree2_root.role = ui::AX_ROLE_ROOT_WEB_AREA;
243 tree2_root.state = 0;
244 tree2_root.child_ids.push_back(5);
245 tree2_root.child_ids.push_back(2);
246 tree2_root.child_ids.push_back(3);
247
248 // Construct a BrowserAccessibilityManager with tree1.
249 CountedBrowserAccessibility::global_obj_count_ = 0;
250 BrowserAccessibilityManager* manager =
251 BrowserAccessibilityManager::Create(
252 MakeAXTreeUpdate(tree1_root,
253 tree1_child1, tree1_child2, tree1_child3),
254 NULL,
255 new CountedBrowserAccessibilityFactory());
256 ASSERT_EQ(4, CountedBrowserAccessibility::global_obj_count_);
257
258 // Save references to all of the objects.
259 CountedBrowserAccessibility* root_accessible =
260 static_cast<CountedBrowserAccessibility*>(manager->GetRoot());
261 root_accessible->NativeAddReference();
262 CountedBrowserAccessibility* child1_accessible =
263 static_cast<CountedBrowserAccessibility*>(
264 root_accessible->PlatformGetChild(0));
265 child1_accessible->NativeAddReference();
266 CountedBrowserAccessibility* child2_accessible =
267 static_cast<CountedBrowserAccessibility*>(
268 root_accessible->PlatformGetChild(1));
269 child2_accessible->NativeAddReference();
270 CountedBrowserAccessibility* child3_accessible =
271 static_cast<CountedBrowserAccessibility*>(
272 root_accessible->PlatformGetChild(2));
273 child3_accessible->NativeAddReference();
274
275 // Check the index in parent.
276 EXPECT_EQ(0, child1_accessible->GetIndexInParent());
277 EXPECT_EQ(1, child2_accessible->GetIndexInParent());
278 EXPECT_EQ(2, child3_accessible->GetIndexInParent());
279
280 // Process a notification containing the changed subtree.
281 std::vector<AccessibilityHostMsg_EventParams> params;
282 params.push_back(AccessibilityHostMsg_EventParams());
283 AccessibilityHostMsg_EventParams* msg = ¶ms[0];
284 msg->event_type = ui::AX_EVENT_CHILDREN_CHANGED;
285 msg->update.nodes.push_back(tree2_root);
286 msg->update.nodes.push_back(tree2_child0);
287 msg->id = tree2_root.id;
288 manager->OnAccessibilityEvents(params);
289
290 // There should be 5 objects now: the 4 from the new tree, plus the
291 // reference to child3 we kept.
292 EXPECT_EQ(5, CountedBrowserAccessibility::global_obj_count_);
293
294 // Check that our references to the root, child1, and child2 are still valid,
295 // but that the reference to child3 is now invalid.
296 EXPECT_TRUE(root_accessible->instance_active());
297 EXPECT_TRUE(child1_accessible->instance_active());
298 EXPECT_TRUE(child2_accessible->instance_active());
299 EXPECT_FALSE(child3_accessible->instance_active());
300
301 // Check that the index in parent has been updated.
302 EXPECT_EQ(1, child1_accessible->GetIndexInParent());
303 EXPECT_EQ(2, child2_accessible->GetIndexInParent());
304
305 // Release our references. The object count should only decrease by 1
306 // for child3.
307 root_accessible->NativeReleaseReference();
308 child1_accessible->NativeReleaseReference();
309 child2_accessible->NativeReleaseReference();
310 child3_accessible->NativeReleaseReference();
311
312 EXPECT_EQ(4, CountedBrowserAccessibility::global_obj_count_);
313
314 // Delete the manager and make sure all memory is cleaned up.
315 delete manager;
316 ASSERT_EQ(0, CountedBrowserAccessibility::global_obj_count_);
317 }
318
TEST(BrowserAccessibilityManagerTest,TestReuseBrowserAccessibilityObjects2)319 TEST(BrowserAccessibilityManagerTest, TestReuseBrowserAccessibilityObjects2) {
320 // Similar to the test above, but with a more complicated tree.
321
322 // Tree 1:
323 //
324 // root
325 // container
326 // child1
327 // grandchild1
328 // child2
329 // grandchild2
330 // child3
331 // grandchild3
332
333 ui::AXNodeData tree1_grandchild1;
334 tree1_grandchild1.id = 4;
335 tree1_grandchild1.SetName("GrandChild1");
336 tree1_grandchild1.role = ui::AX_ROLE_BUTTON;
337 tree1_grandchild1.state = 0;
338
339 ui::AXNodeData tree1_child1;
340 tree1_child1.id = 3;
341 tree1_child1.SetName("Child1");
342 tree1_child1.role = ui::AX_ROLE_BUTTON;
343 tree1_child1.state = 0;
344 tree1_child1.child_ids.push_back(4);
345
346 ui::AXNodeData tree1_grandchild2;
347 tree1_grandchild2.id = 6;
348 tree1_grandchild2.SetName("GrandChild1");
349 tree1_grandchild2.role = ui::AX_ROLE_BUTTON;
350 tree1_grandchild2.state = 0;
351
352 ui::AXNodeData tree1_child2;
353 tree1_child2.id = 5;
354 tree1_child2.SetName("Child2");
355 tree1_child2.role = ui::AX_ROLE_BUTTON;
356 tree1_child2.state = 0;
357 tree1_child2.child_ids.push_back(6);
358
359 ui::AXNodeData tree1_grandchild3;
360 tree1_grandchild3.id = 8;
361 tree1_grandchild3.SetName("GrandChild3");
362 tree1_grandchild3.role = ui::AX_ROLE_BUTTON;
363 tree1_grandchild3.state = 0;
364
365 ui::AXNodeData tree1_child3;
366 tree1_child3.id = 7;
367 tree1_child3.SetName("Child3");
368 tree1_child3.role = ui::AX_ROLE_BUTTON;
369 tree1_child3.state = 0;
370 tree1_child3.child_ids.push_back(8);
371
372 ui::AXNodeData tree1_container;
373 tree1_container.id = 2;
374 tree1_container.SetName("Container");
375 tree1_container.role = ui::AX_ROLE_GROUP;
376 tree1_container.state = 0;
377 tree1_container.child_ids.push_back(3);
378 tree1_container.child_ids.push_back(5);
379 tree1_container.child_ids.push_back(7);
380
381 ui::AXNodeData tree1_root;
382 tree1_root.id = 1;
383 tree1_root.SetName("Document");
384 tree1_root.role = ui::AX_ROLE_ROOT_WEB_AREA;
385 tree1_root.state = 0;
386 tree1_root.child_ids.push_back(2);
387
388 // Tree 2:
389 //
390 // root
391 // container
392 // child0 <-- inserted
393 // grandchild0 <--
394 // child1
395 // grandchild1
396 // child2
397 // grandchild2
398 // <-- child3 (and grandchild3) deleted
399
400 ui::AXNodeData tree2_grandchild0;
401 tree2_grandchild0.id = 9;
402 tree2_grandchild0.SetName("GrandChild0");
403 tree2_grandchild0.role = ui::AX_ROLE_BUTTON;
404 tree2_grandchild0.state = 0;
405
406 ui::AXNodeData tree2_child0;
407 tree2_child0.id = 10;
408 tree2_child0.SetName("Child0");
409 tree2_child0.role = ui::AX_ROLE_BUTTON;
410 tree2_child0.state = 0;
411 tree2_child0.child_ids.push_back(9);
412
413 ui::AXNodeData tree2_container;
414 tree2_container.id = 2;
415 tree2_container.SetName("Container");
416 tree2_container.role = ui::AX_ROLE_GROUP;
417 tree2_container.state = 0;
418 tree2_container.child_ids.push_back(10);
419 tree2_container.child_ids.push_back(3);
420 tree2_container.child_ids.push_back(5);
421
422 // Construct a BrowserAccessibilityManager with tree1.
423 CountedBrowserAccessibility::global_obj_count_ = 0;
424 BrowserAccessibilityManager* manager =
425 BrowserAccessibilityManager::Create(
426 MakeAXTreeUpdate(tree1_root, tree1_container,
427 tree1_child1, tree1_grandchild1,
428 tree1_child2, tree1_grandchild2,
429 tree1_child3, tree1_grandchild3),
430 NULL,
431 new CountedBrowserAccessibilityFactory());
432 ASSERT_EQ(8, CountedBrowserAccessibility::global_obj_count_);
433
434 // Save references to some objects.
435 CountedBrowserAccessibility* root_accessible =
436 static_cast<CountedBrowserAccessibility*>(manager->GetRoot());
437 root_accessible->NativeAddReference();
438 CountedBrowserAccessibility* container_accessible =
439 static_cast<CountedBrowserAccessibility*>(
440 root_accessible->PlatformGetChild(0));
441 container_accessible->NativeAddReference();
442 CountedBrowserAccessibility* child2_accessible =
443 static_cast<CountedBrowserAccessibility*>(
444 container_accessible->PlatformGetChild(1));
445 child2_accessible->NativeAddReference();
446 CountedBrowserAccessibility* child3_accessible =
447 static_cast<CountedBrowserAccessibility*>(
448 container_accessible->PlatformGetChild(2));
449 child3_accessible->NativeAddReference();
450
451 // Check the index in parent.
452 EXPECT_EQ(1, child2_accessible->GetIndexInParent());
453 EXPECT_EQ(2, child3_accessible->GetIndexInParent());
454
455 // Process a notification containing the changed subtree rooted at
456 // the container.
457 std::vector<AccessibilityHostMsg_EventParams> params;
458 params.push_back(AccessibilityHostMsg_EventParams());
459 AccessibilityHostMsg_EventParams* msg = ¶ms[0];
460 msg->event_type = ui::AX_EVENT_CHILDREN_CHANGED;
461 msg->update.nodes.push_back(tree2_container);
462 msg->update.nodes.push_back(tree2_child0);
463 msg->update.nodes.push_back(tree2_grandchild0);
464 msg->id = tree2_container.id;
465 manager->OnAccessibilityEvents(params);
466
467 // There should be 9 objects now: the 8 from the new tree, plus the
468 // reference to child3 we kept.
469 EXPECT_EQ(9, CountedBrowserAccessibility::global_obj_count_);
470
471 // Check that our references to the root and container and child2 are
472 // still valid, but that the reference to child3 is now invalid.
473 EXPECT_TRUE(root_accessible->instance_active());
474 EXPECT_TRUE(container_accessible->instance_active());
475 EXPECT_TRUE(child2_accessible->instance_active());
476 EXPECT_FALSE(child3_accessible->instance_active());
477
478 // Ensure that we retain the parent of the detached subtree.
479 EXPECT_EQ(root_accessible, container_accessible->GetParent());
480 EXPECT_EQ(0, container_accessible->GetIndexInParent());
481
482 // Check that the index in parent has been updated.
483 EXPECT_EQ(2, child2_accessible->GetIndexInParent());
484
485 // Release our references. The object count should only decrease by 1
486 // for child3.
487 root_accessible->NativeReleaseReference();
488 container_accessible->NativeReleaseReference();
489 child2_accessible->NativeReleaseReference();
490 child3_accessible->NativeReleaseReference();
491
492 EXPECT_EQ(8, CountedBrowserAccessibility::global_obj_count_);
493
494 // Delete the manager and make sure all memory is cleaned up.
495 delete manager;
496 ASSERT_EQ(0, CountedBrowserAccessibility::global_obj_count_);
497 }
498
TEST(BrowserAccessibilityManagerTest,TestMoveChildUp)499 TEST(BrowserAccessibilityManagerTest, TestMoveChildUp) {
500 // Tree 1:
501 //
502 // 1
503 // 2
504 // 3
505 // 4
506
507 ui::AXNodeData tree1_4;
508 tree1_4.id = 4;
509 tree1_4.state = 0;
510
511 ui::AXNodeData tree1_3;
512 tree1_3.id = 3;
513 tree1_3.state = 0;
514 tree1_3.child_ids.push_back(4);
515
516 ui::AXNodeData tree1_2;
517 tree1_2.id = 2;
518 tree1_2.state = 0;
519
520 ui::AXNodeData tree1_1;
521 tree1_1.id = 1;
522 tree1_1.role = ui::AX_ROLE_ROOT_WEB_AREA;
523 tree1_1.state = 0;
524 tree1_1.child_ids.push_back(2);
525 tree1_1.child_ids.push_back(3);
526
527 // Tree 2:
528 //
529 // 1
530 // 4 <-- moves up a level and gains child
531 // 6 <-- new
532 // 5 <-- new
533
534 ui::AXNodeData tree2_6;
535 tree2_6.id = 6;
536 tree2_6.state = 0;
537
538 ui::AXNodeData tree2_5;
539 tree2_5.id = 5;
540 tree2_5.state = 0;
541
542 ui::AXNodeData tree2_4;
543 tree2_4.id = 4;
544 tree2_4.state = 0;
545 tree2_4.child_ids.push_back(6);
546
547 ui::AXNodeData tree2_1;
548 tree2_1.id = 1;
549 tree2_1.state = 0;
550 tree2_1.child_ids.push_back(4);
551 tree2_1.child_ids.push_back(5);
552
553 // Construct a BrowserAccessibilityManager with tree1.
554 CountedBrowserAccessibility::global_obj_count_ = 0;
555 BrowserAccessibilityManager* manager =
556 BrowserAccessibilityManager::Create(
557 MakeAXTreeUpdate(tree1_1, tree1_2, tree1_3, tree1_4),
558 NULL,
559 new CountedBrowserAccessibilityFactory());
560 ASSERT_EQ(4, CountedBrowserAccessibility::global_obj_count_);
561
562 // Process a notification containing the changed subtree.
563 std::vector<AccessibilityHostMsg_EventParams> params;
564 params.push_back(AccessibilityHostMsg_EventParams());
565 AccessibilityHostMsg_EventParams* msg = ¶ms[0];
566 msg->event_type = ui::AX_EVENT_CHILDREN_CHANGED;
567 msg->update.nodes.push_back(tree2_1);
568 msg->update.nodes.push_back(tree2_4);
569 msg->update.nodes.push_back(tree2_5);
570 msg->update.nodes.push_back(tree2_6);
571 msg->id = tree2_1.id;
572 manager->OnAccessibilityEvents(params);
573
574 // There should be 4 objects now.
575 EXPECT_EQ(4, CountedBrowserAccessibility::global_obj_count_);
576
577 // Delete the manager and make sure all memory is cleaned up.
578 delete manager;
579 ASSERT_EQ(0, CountedBrowserAccessibility::global_obj_count_);
580 }
581
TEST(BrowserAccessibilityManagerTest,TestFatalError)582 TEST(BrowserAccessibilityManagerTest, TestFatalError) {
583 // Test that BrowserAccessibilityManager raises a fatal error
584 // (which will crash the renderer) if the same id is used in
585 // two places in the tree.
586
587 ui::AXNodeData root;
588 root.id = 1;
589 root.role = ui::AX_ROLE_ROOT_WEB_AREA;
590 root.child_ids.push_back(2);
591 root.child_ids.push_back(2);
592
593 CountedBrowserAccessibilityFactory* factory =
594 new CountedBrowserAccessibilityFactory();
595 scoped_ptr<TestBrowserAccessibilityDelegate> delegate(
596 new TestBrowserAccessibilityDelegate());
597 scoped_ptr<BrowserAccessibilityManager> manager;
598 ASSERT_FALSE(delegate->got_fatal_error());
599 manager.reset(BrowserAccessibilityManager::Create(
600 MakeAXTreeUpdate(root),
601 delegate.get(),
602 factory));
603 ASSERT_TRUE(delegate->got_fatal_error());
604
605 ui::AXNodeData root2;
606 root2.id = 1;
607 root2.role = ui::AX_ROLE_ROOT_WEB_AREA;
608 root2.child_ids.push_back(2);
609 root2.child_ids.push_back(3);
610
611 ui::AXNodeData child1;
612 child1.id = 2;
613 child1.child_ids.push_back(4);
614 child1.child_ids.push_back(5);
615
616 ui::AXNodeData child2;
617 child2.id = 3;
618 child2.child_ids.push_back(6);
619 child2.child_ids.push_back(5); // Duplicate
620
621 ui::AXNodeData grandchild4;
622 grandchild4.id = 4;
623
624 ui::AXNodeData grandchild5;
625 grandchild5.id = 5;
626
627 ui::AXNodeData grandchild6;
628 grandchild6.id = 6;
629
630 delegate->reset_got_fatal_error();
631 factory = new CountedBrowserAccessibilityFactory();
632 manager.reset(BrowserAccessibilityManager::Create(
633 MakeAXTreeUpdate(root2, child1, child2,
634 grandchild4, grandchild5, grandchild6),
635 delegate.get(),
636 factory));
637 ASSERT_TRUE(delegate->got_fatal_error());
638 }
639
TEST(BrowserAccessibilityManagerTest,BoundsForRange)640 TEST(BrowserAccessibilityManagerTest, BoundsForRange) {
641 ui::AXNodeData root;
642 root.id = 1;
643 root.role = ui::AX_ROLE_ROOT_WEB_AREA;
644
645 ui::AXNodeData static_text;
646 static_text.id = 2;
647 static_text.SetValue("Hello, world.");
648 static_text.role = ui::AX_ROLE_STATIC_TEXT;
649 static_text.location = gfx::Rect(100, 100, 29, 18);
650 root.child_ids.push_back(2);
651
652 ui::AXNodeData inline_text1;
653 inline_text1.id = 3;
654 inline_text1.SetValue("Hello, ");
655 inline_text1.role = ui::AX_ROLE_INLINE_TEXT_BOX;
656 inline_text1.location = gfx::Rect(100, 100, 29, 9);
657 inline_text1.AddIntAttribute(ui::AX_ATTR_TEXT_DIRECTION,
658 ui::AX_TEXT_DIRECTION_LR);
659 std::vector<int32> character_offsets1;
660 character_offsets1.push_back(6); // 0
661 character_offsets1.push_back(11); // 1
662 character_offsets1.push_back(16); // 2
663 character_offsets1.push_back(21); // 3
664 character_offsets1.push_back(26); // 4
665 character_offsets1.push_back(29); // 5
666 character_offsets1.push_back(29); // 6 (note that the space has no width)
667 inline_text1.AddIntListAttribute(
668 ui::AX_ATTR_CHARACTER_OFFSETS, character_offsets1);
669 static_text.child_ids.push_back(3);
670
671 ui::AXNodeData inline_text2;
672 inline_text2.id = 4;
673 inline_text2.SetValue("world.");
674 inline_text2.role = ui::AX_ROLE_INLINE_TEXT_BOX;
675 inline_text2.location = gfx::Rect(100, 109, 28, 9);
676 inline_text2.AddIntAttribute(ui::AX_ATTR_TEXT_DIRECTION,
677 ui::AX_TEXT_DIRECTION_LR);
678 std::vector<int32> character_offsets2;
679 character_offsets2.push_back(5);
680 character_offsets2.push_back(10);
681 character_offsets2.push_back(15);
682 character_offsets2.push_back(20);
683 character_offsets2.push_back(25);
684 character_offsets2.push_back(28);
685 inline_text2.AddIntListAttribute(
686 ui::AX_ATTR_CHARACTER_OFFSETS, character_offsets2);
687 static_text.child_ids.push_back(4);
688
689 scoped_ptr<BrowserAccessibilityManager> manager(
690 BrowserAccessibilityManager::Create(
691 MakeAXTreeUpdate(root, static_text, inline_text1, inline_text2),
692 NULL,
693 new CountedBrowserAccessibilityFactory()));
694
695 BrowserAccessibility* root_accessible = manager->GetRoot();
696 BrowserAccessibility* static_text_accessible =
697 root_accessible->PlatformGetChild(0);
698
699 EXPECT_EQ(gfx::Rect(100, 100, 6, 9).ToString(),
700 static_text_accessible->GetLocalBoundsForRange(0, 1).ToString());
701
702 EXPECT_EQ(gfx::Rect(100, 100, 26, 9).ToString(),
703 static_text_accessible->GetLocalBoundsForRange(0, 5).ToString());
704
705 EXPECT_EQ(gfx::Rect(100, 109, 5, 9).ToString(),
706 static_text_accessible->GetLocalBoundsForRange(7, 1).ToString());
707
708 EXPECT_EQ(gfx::Rect(100, 109, 25, 9).ToString(),
709 static_text_accessible->GetLocalBoundsForRange(7, 5).ToString());
710
711 EXPECT_EQ(gfx::Rect(100, 100, 29, 18).ToString(),
712 static_text_accessible->GetLocalBoundsForRange(5, 3).ToString());
713
714 EXPECT_EQ(gfx::Rect(100, 100, 29, 18).ToString(),
715 static_text_accessible->GetLocalBoundsForRange(0, 13).ToString());
716
717 // Test range that's beyond the text.
718 EXPECT_EQ(gfx::Rect(100, 100, 29, 18).ToString(),
719 static_text_accessible->GetLocalBoundsForRange(-1, 999).ToString());
720
721 // Test that we can call bounds for range on the parent element, too,
722 // and it still works.
723 EXPECT_EQ(gfx::Rect(100, 100, 29, 18).ToString(),
724 root_accessible->GetLocalBoundsForRange(0, 13).ToString());
725 }
726
TEST(BrowserAccessibilityManagerTest,BoundsForRangeBiDi)727 TEST(BrowserAccessibilityManagerTest, BoundsForRangeBiDi) {
728 // In this example, we assume that the string "123abc" is rendered with
729 // "123" going left-to-right and "abc" going right-to-left. In other
730 // words, on-screen it would look like "123cba". This is possible to
731 // acheive if the source string had unicode control characters
732 // to switch directions. This test doesn't worry about how, though - it just
733 // tests that if something like that were to occur, GetLocalBoundsForRange
734 // returns the correct bounds for different ranges.
735
736 ui::AXNodeData root;
737 root.id = 1;
738 root.role = ui::AX_ROLE_ROOT_WEB_AREA;
739
740 ui::AXNodeData static_text;
741 static_text.id = 2;
742 static_text.SetValue("123abc");
743 static_text.role = ui::AX_ROLE_STATIC_TEXT;
744 static_text.location = gfx::Rect(100, 100, 60, 20);
745 root.child_ids.push_back(2);
746
747 ui::AXNodeData inline_text1;
748 inline_text1.id = 3;
749 inline_text1.SetValue("123");
750 inline_text1.role = ui::AX_ROLE_INLINE_TEXT_BOX;
751 inline_text1.location = gfx::Rect(100, 100, 30, 20);
752 inline_text1.AddIntAttribute(ui::AX_ATTR_TEXT_DIRECTION,
753 ui::AX_TEXT_DIRECTION_LR);
754 std::vector<int32> character_offsets1;
755 character_offsets1.push_back(10); // 0
756 character_offsets1.push_back(20); // 1
757 character_offsets1.push_back(30); // 2
758 inline_text1.AddIntListAttribute(
759 ui::AX_ATTR_CHARACTER_OFFSETS, character_offsets1);
760 static_text.child_ids.push_back(3);
761
762 ui::AXNodeData inline_text2;
763 inline_text2.id = 4;
764 inline_text2.SetValue("abc");
765 inline_text2.role = ui::AX_ROLE_INLINE_TEXT_BOX;
766 inline_text2.location = gfx::Rect(130, 100, 30, 20);
767 inline_text2.AddIntAttribute(ui::AX_ATTR_TEXT_DIRECTION,
768 ui::AX_TEXT_DIRECTION_RL);
769 std::vector<int32> character_offsets2;
770 character_offsets2.push_back(10);
771 character_offsets2.push_back(20);
772 character_offsets2.push_back(30);
773 inline_text2.AddIntListAttribute(
774 ui::AX_ATTR_CHARACTER_OFFSETS, character_offsets2);
775 static_text.child_ids.push_back(4);
776
777 scoped_ptr<BrowserAccessibilityManager> manager(
778 BrowserAccessibilityManager::Create(
779 MakeAXTreeUpdate(root, static_text, inline_text1, inline_text2),
780 NULL,
781 new CountedBrowserAccessibilityFactory()));
782
783 BrowserAccessibility* root_accessible = manager->GetRoot();
784 BrowserAccessibility* static_text_accessible =
785 root_accessible->PlatformGetChild(0);
786
787 EXPECT_EQ(gfx::Rect(100, 100, 60, 20).ToString(),
788 static_text_accessible->GetLocalBoundsForRange(0, 6).ToString());
789
790 EXPECT_EQ(gfx::Rect(100, 100, 10, 20).ToString(),
791 static_text_accessible->GetLocalBoundsForRange(0, 1).ToString());
792
793 EXPECT_EQ(gfx::Rect(100, 100, 30, 20).ToString(),
794 static_text_accessible->GetLocalBoundsForRange(0, 3).ToString());
795
796 EXPECT_EQ(gfx::Rect(150, 100, 10, 20).ToString(),
797 static_text_accessible->GetLocalBoundsForRange(3, 1).ToString());
798
799 EXPECT_EQ(gfx::Rect(130, 100, 30, 20).ToString(),
800 static_text_accessible->GetLocalBoundsForRange(3, 3).ToString());
801
802 // This range is only two characters, but because of the direction switch
803 // the bounds are as wide as four characters.
804 EXPECT_EQ(gfx::Rect(120, 100, 40, 20).ToString(),
805 static_text_accessible->GetLocalBoundsForRange(2, 2).ToString());
806 }
807
808 #if defined(OS_WIN)
809 #define MAYBE_BoundsForRangeOnParentElement \
810 DISABLED_BoundsForRangeOnParentElement
811 #else
812 #define MAYBE_BoundsForRangeOnParentElement BoundsForRangeOnParentElement
813 #endif
TEST(BrowserAccessibilityManagerTest,MAYBE_BoundsForRangeOnParentElement)814 TEST(BrowserAccessibilityManagerTest, MAYBE_BoundsForRangeOnParentElement) {
815 ui::AXNodeData root;
816 root.id = 1;
817 root.role = ui::AX_ROLE_ROOT_WEB_AREA;
818 root.child_ids.push_back(2);
819
820 ui::AXNodeData div;
821 div.id = 2;
822 div.role = ui::AX_ROLE_DIV;
823 div.location = gfx::Rect(100, 100, 100, 20);
824 div.child_ids.push_back(3);
825 div.child_ids.push_back(4);
826 div.child_ids.push_back(5);
827
828 ui::AXNodeData static_text1;
829 static_text1.id = 3;
830 static_text1.SetValue("AB");
831 static_text1.role = ui::AX_ROLE_STATIC_TEXT;
832 static_text1.location = gfx::Rect(100, 100, 40, 20);
833 static_text1.child_ids.push_back(6);
834
835 ui::AXNodeData img;
836 img.id = 4;
837 img.role = ui::AX_ROLE_IMAGE;
838 img.location = gfx::Rect(140, 100, 20, 20);
839
840 ui::AXNodeData static_text2;
841 static_text2.id = 5;
842 static_text2.SetValue("CD");
843 static_text2.role = ui::AX_ROLE_STATIC_TEXT;
844 static_text2.location = gfx::Rect(160, 100, 40, 20);
845 static_text2.child_ids.push_back(7);
846
847 ui::AXNodeData inline_text1;
848 inline_text1.id = 6;
849 inline_text1.SetValue("AB");
850 inline_text1.role = ui::AX_ROLE_INLINE_TEXT_BOX;
851 inline_text1.location = gfx::Rect(100, 100, 40, 20);
852 inline_text1.AddIntAttribute(ui::AX_ATTR_TEXT_DIRECTION,
853 ui::AX_TEXT_DIRECTION_LR);
854 std::vector<int32> character_offsets1;
855 character_offsets1.push_back(20); // 0
856 character_offsets1.push_back(40); // 1
857 inline_text1.AddIntListAttribute(
858 ui::AX_ATTR_CHARACTER_OFFSETS, character_offsets1);
859
860 ui::AXNodeData inline_text2;
861 inline_text2.id = 7;
862 inline_text2.SetValue("CD");
863 inline_text2.role = ui::AX_ROLE_INLINE_TEXT_BOX;
864 inline_text2.location = gfx::Rect(160, 100, 40, 20);
865 inline_text2.AddIntAttribute(ui::AX_ATTR_TEXT_DIRECTION,
866 ui::AX_TEXT_DIRECTION_LR);
867 std::vector<int32> character_offsets2;
868 character_offsets2.push_back(20); // 0
869 character_offsets2.push_back(40); // 1
870 inline_text2.AddIntListAttribute(
871 ui::AX_ATTR_CHARACTER_OFFSETS, character_offsets2);
872
873 scoped_ptr<BrowserAccessibilityManager> manager(
874 BrowserAccessibilityManager::Create(
875 MakeAXTreeUpdate(
876 root, div, static_text1, img,
877 static_text2, inline_text1, inline_text2),
878 NULL,
879 new CountedBrowserAccessibilityFactory()));
880 BrowserAccessibility* root_accessible = manager->GetRoot();
881
882 EXPECT_EQ(gfx::Rect(100, 100, 20, 20).ToString(),
883 root_accessible->GetLocalBoundsForRange(0, 1).ToString());
884
885 EXPECT_EQ(gfx::Rect(100, 100, 40, 20).ToString(),
886 root_accessible->GetLocalBoundsForRange(0, 2).ToString());
887
888 EXPECT_EQ(gfx::Rect(100, 100, 80, 20).ToString(),
889 root_accessible->GetLocalBoundsForRange(0, 3).ToString());
890
891 EXPECT_EQ(gfx::Rect(120, 100, 60, 20).ToString(),
892 root_accessible->GetLocalBoundsForRange(1, 2).ToString());
893
894 EXPECT_EQ(gfx::Rect(120, 100, 80, 20).ToString(),
895 root_accessible->GetLocalBoundsForRange(1, 3).ToString());
896
897 EXPECT_EQ(gfx::Rect(100, 100, 100, 20).ToString(),
898 root_accessible->GetLocalBoundsForRange(0, 4).ToString());
899 }
900
TEST(BrowserAccessibilityManagerTest,NextPreviousInTreeOrder)901 TEST(BrowserAccessibilityManagerTest, NextPreviousInTreeOrder) {
902 ui::AXNodeData root;
903 root.id = 1;
904 root.role = ui::AX_ROLE_ROOT_WEB_AREA;
905
906 ui::AXNodeData node2;
907 node2.id = 2;
908 root.child_ids.push_back(2);
909
910 ui::AXNodeData node3;
911 node3.id = 3;
912 root.child_ids.push_back(3);
913
914 ui::AXNodeData node4;
915 node4.id = 4;
916 node3.child_ids.push_back(4);
917
918 ui::AXNodeData node5;
919 node5.id = 5;
920 root.child_ids.push_back(5);
921
922 scoped_ptr<BrowserAccessibilityManager> manager(
923 BrowserAccessibilityManager::Create(
924 MakeAXTreeUpdate(root, node2, node3, node4, node5),
925 NULL,
926 new CountedBrowserAccessibilityFactory()));
927
928 BrowserAccessibility* root_accessible = manager->GetRoot();
929 BrowserAccessibility* node2_accessible = root_accessible->PlatformGetChild(0);
930 BrowserAccessibility* node3_accessible = root_accessible->PlatformGetChild(1);
931 BrowserAccessibility* node4_accessible =
932 node3_accessible->PlatformGetChild(0);
933 BrowserAccessibility* node5_accessible = root_accessible->PlatformGetChild(2);
934
935 ASSERT_EQ(NULL, manager->NextInTreeOrder(NULL));
936 ASSERT_EQ(node2_accessible, manager->NextInTreeOrder(root_accessible));
937 ASSERT_EQ(node3_accessible, manager->NextInTreeOrder(node2_accessible));
938 ASSERT_EQ(node4_accessible, manager->NextInTreeOrder(node3_accessible));
939 ASSERT_EQ(node5_accessible, manager->NextInTreeOrder(node4_accessible));
940 ASSERT_EQ(NULL, manager->NextInTreeOrder(node5_accessible));
941
942 ASSERT_EQ(NULL, manager->PreviousInTreeOrder(NULL));
943 ASSERT_EQ(node4_accessible, manager->PreviousInTreeOrder(node5_accessible));
944 ASSERT_EQ(node3_accessible, manager->PreviousInTreeOrder(node4_accessible));
945 ASSERT_EQ(node2_accessible, manager->PreviousInTreeOrder(node3_accessible));
946 ASSERT_EQ(root_accessible, manager->PreviousInTreeOrder(node2_accessible));
947 }
948
949 } // namespace content
950