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 "content/browser/accessibility/browser_accessibility_manager_mac.h" 6 7 #import "base/logging.h" 8 #import "content/browser/accessibility/browser_accessibility_cocoa.h" 9 #import "content/browser/accessibility/browser_accessibility_mac.h" 10 #include "content/common/accessibility_messages.h" 11 12 namespace content { 13 14 // static 15 BrowserAccessibilityManager* BrowserAccessibilityManager::Create( 16 const ui::AXTreeUpdate& initial_tree, 17 BrowserAccessibilityDelegate* delegate, 18 BrowserAccessibilityFactory* factory) { 19 return new BrowserAccessibilityManagerMac( 20 NULL, initial_tree, delegate, factory); 21 } 22 23 BrowserAccessibilityManagerMac::BrowserAccessibilityManagerMac( 24 NSView* parent_view, 25 const ui::AXTreeUpdate& initial_tree, 26 BrowserAccessibilityDelegate* delegate, 27 BrowserAccessibilityFactory* factory) 28 : BrowserAccessibilityManager(initial_tree, delegate, factory), 29 parent_view_(parent_view), 30 created_live_region_(false) { 31 } 32 33 // static 34 ui::AXTreeUpdate BrowserAccessibilityManagerMac::GetEmptyDocument() { 35 ui::AXNodeData empty_document; 36 empty_document.id = 0; 37 empty_document.role = ui::AX_ROLE_ROOT_WEB_AREA; 38 empty_document.state = 39 1 << ui::AX_STATE_READ_ONLY; 40 ui::AXTreeUpdate update; 41 update.nodes.push_back(empty_document); 42 return update; 43 } 44 45 BrowserAccessibility* BrowserAccessibilityManagerMac::GetFocus( 46 BrowserAccessibility* root) { 47 BrowserAccessibility* node = GetActiveDescendantFocus(root); 48 return node; 49 } 50 51 void BrowserAccessibilityManagerMac::NotifyAccessibilityEvent( 52 ui::AXEvent event_type, 53 BrowserAccessibility* node) { 54 if (!node->IsNative()) 55 return; 56 57 // Refer to AXObjectCache.mm (webkit). 58 NSString* event_id = @""; 59 switch (event_type) { 60 case ui::AX_EVENT_ACTIVEDESCENDANTCHANGED: 61 if (node->GetRole() == ui::AX_ROLE_TREE) { 62 event_id = NSAccessibilitySelectedRowsChangedNotification; 63 } else { 64 event_id = NSAccessibilityFocusedUIElementChangedNotification; 65 BrowserAccessibility* active_descendant_focus = 66 GetActiveDescendantFocus(GetRoot()); 67 if (active_descendant_focus) 68 node = active_descendant_focus; 69 } 70 71 break; 72 case ui::AX_EVENT_ALERT: 73 // Not used on Mac. 74 return; 75 case ui::AX_EVENT_BLUR: 76 // A no-op on Mac. 77 return; 78 case ui::AX_EVENT_CHECKED_STATE_CHANGED: 79 // Not used on Mac. 80 return; 81 case ui::AX_EVENT_CHILDREN_CHANGED: 82 // TODO(dtseng): no clear equivalent on Mac. 83 return; 84 case ui::AX_EVENT_FOCUS: 85 event_id = NSAccessibilityFocusedUIElementChangedNotification; 86 break; 87 case ui::AX_EVENT_LAYOUT_COMPLETE: 88 event_id = @"AXLayoutComplete"; 89 break; 90 case ui::AX_EVENT_LIVE_REGION_CHANGED: 91 event_id = @"AXLiveRegionChanged"; 92 break; 93 case ui::AX_EVENT_LOAD_COMPLETE: 94 event_id = @"AXLoadComplete"; 95 break; 96 case ui::AX_EVENT_MENU_LIST_VALUE_CHANGED: 97 // Not used on Mac. 98 return; 99 case ui::AX_EVENT_ROW_COUNT_CHANGED: 100 event_id = NSAccessibilityRowCountChangedNotification; 101 break; 102 case ui::AX_EVENT_ROW_COLLAPSED: 103 event_id = @"AXRowCollapsed"; 104 break; 105 case ui::AX_EVENT_ROW_EXPANDED: 106 event_id = @"AXRowExpanded"; 107 break; 108 case ui::AX_EVENT_SCROLLED_TO_ANCHOR: 109 // Not used on Mac. 110 return; 111 case ui::AX_EVENT_SELECTED_CHILDREN_CHANGED: 112 event_id = NSAccessibilitySelectedChildrenChangedNotification; 113 break; 114 case ui::AX_EVENT_TEXT_SELECTION_CHANGED: 115 event_id = NSAccessibilitySelectedTextChangedNotification; 116 break; 117 case ui::AX_EVENT_VALUE_CHANGED: 118 event_id = NSAccessibilityValueChangedNotification; 119 break; 120 case ui::AX_EVENT_ARIA_ATTRIBUTE_CHANGED: 121 // Not used on Mac. 122 return; 123 case ui::AX_EVENT_AUTOCORRECTION_OCCURED: 124 // Not used on Mac. 125 return; 126 case ui::AX_EVENT_INVALID_STATUS_CHANGED: 127 // Not used on Mac. 128 return; 129 case ui::AX_EVENT_LOCATION_CHANGED: 130 // Not used on Mac. 131 return; 132 case ui::AX_EVENT_MENU_LIST_ITEM_SELECTED: 133 // Not used on Mac. 134 return; 135 case ui::AX_EVENT_TEXT_CHANGED: 136 // Not used on Mac. 137 return; 138 default: 139 LOG(WARNING) << "Unknown accessibility event: " << event_type; 140 return; 141 } 142 BrowserAccessibilityCocoa* native_node = node->ToBrowserAccessibilityCocoa(); 143 DCHECK(native_node); 144 NSAccessibilityPostNotification(native_node, event_id); 145 } 146 147 void BrowserAccessibilityManagerMac::OnNodeCreationFinished(ui::AXNode* node) { 148 BrowserAccessibility* obj = GetFromAXNode(node); 149 if (obj && obj->HasStringAttribute(ui::AX_ATTR_LIVE_STATUS)) 150 created_live_region_ = true; 151 } 152 153 void BrowserAccessibilityManagerMac::OnTreeUpdateFinished() { 154 if (!created_live_region_) 155 return; 156 157 // This code is to work around a bug in VoiceOver, where a new live 158 // region that gets added is ignored. VoiceOver seems to only scan the 159 // page for live regions once. By recreating the NSAccessibility 160 // object for the root of the tree, we force VoiceOver to clear out its 161 // internal state and find newly-added live regions this time. 162 BrowserAccessibilityMac* root = 163 static_cast<BrowserAccessibilityMac*>(GetRoot()); 164 root->RecreateNativeObject(); 165 NotifyAccessibilityEvent(ui::AX_EVENT_CHILDREN_CHANGED, root); 166 167 created_live_region_ = false; 168 } 169 170 } // namespace content 171