Index: content/renderer/accessibility_node_serializer.cc |
diff --git a/content/renderer/accessibility_node_serializer.cc b/content/renderer/accessibility_node_serializer.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..b838ab10c7f64f43600f877469aaab90b6a73788 |
--- /dev/null |
+++ b/content/renderer/accessibility_node_serializer.cc |
@@ -0,0 +1,594 @@ |
+// Copyright (c) 2012 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 "content/renderer/accessibility_node_serializer.h" |
+ |
+#include <set> |
+ |
+#include "base/string_number_conversions.h" |
+#include "base/string_util.h" |
+#include "base/utf_string_conversions.h" |
+#include "third_party/WebKit/Source/WebKit/chromium/public/WebAccessibilityObject.h" |
+#include "third_party/WebKit/Source/WebKit/chromium/public/WebAccessibilityRole.h" |
+#include "third_party/WebKit/Source/WebKit/chromium/public/WebDocument.h" |
+#include "third_party/WebKit/Source/WebKit/chromium/public/WebDocumentType.h" |
+#include "third_party/WebKit/Source/WebKit/chromium/public/WebElement.h" |
+#include "third_party/WebKit/Source/WebKit/chromium/public/WebFormControlElement.h" |
+#include "third_party/WebKit/Source/WebKit/chromium/public/WebFrame.h" |
+#include "third_party/WebKit/Source/WebKit/chromium/public/WebInputElement.h" |
+#include "third_party/WebKit/Source/WebKit/chromium/public/WebNode.h" |
+#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebRect.h" |
+#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebSize.h" |
+#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebString.h" |
+#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebVector.h" |
+ |
+using content::AccessibilityNodeData; |
+using WebKit::WebAccessibilityRole; |
+using WebKit::WebAccessibilityObject; |
+using WebKit::WebDocument; |
+using WebKit::WebDocumentType; |
+using WebKit::WebElement; |
+using WebKit::WebNode; |
+using WebKit::WebVector; |
+ |
+namespace { |
+ |
+// Returns true if |ancestor| is the first unignored parent of |child|, |
+// which means that when walking up the parent chain from |child|, |
+// |ancestor| is the *first* ancestor that isn't marked as |
+// accessibilityIsIgnored(). |
+bool IsParentUnignoredOf(const WebAccessibilityObject& ancestor, |
+ const WebAccessibilityObject& child) { |
+ WebAccessibilityObject parent = child.parentObject(); |
+ while (!parent.isNull() && parent.accessibilityIsIgnored()) |
+ parent = parent.parentObject(); |
+ return parent.equals(ancestor); |
+} |
+ |
+// Provides a conversion between the WebKit::WebAccessibilityRole and a role |
+// supported on the Browser side. Listed alphabetically by the |
+// WebKit::WebAccessibilityRole (except for default role). |
+AccessibilityNodeData::Role ConvertRole(WebKit::WebAccessibilityRole role) { |
+ switch (role) { |
+ case WebKit::WebAccessibilityRoleAnnotation: |
+ return AccessibilityNodeData::ROLE_ANNOTATION; |
+ case WebKit::WebAccessibilityRoleApplication: |
+ return AccessibilityNodeData::ROLE_APPLICATION; |
+ case WebKit::WebAccessibilityRoleApplicationAlert: |
+ return AccessibilityNodeData::ROLE_ALERT; |
+ case WebKit::WebAccessibilityRoleApplicationAlertDialog: |
+ return AccessibilityNodeData::ROLE_ALERT_DIALOG; |
+ case WebKit::WebAccessibilityRoleApplicationDialog: |
+ return AccessibilityNodeData::ROLE_DIALOG; |
+ case WebKit::WebAccessibilityRoleApplicationLog: |
+ return AccessibilityNodeData::ROLE_LOG; |
+ case WebKit::WebAccessibilityRoleApplicationMarquee: |
+ return AccessibilityNodeData::ROLE_MARQUEE; |
+ case WebKit::WebAccessibilityRoleApplicationStatus: |
+ return AccessibilityNodeData::ROLE_STATUS; |
+ case WebKit::WebAccessibilityRoleApplicationTimer: |
+ return AccessibilityNodeData::ROLE_TIMER; |
+ case WebKit::WebAccessibilityRoleBrowser: |
+ return AccessibilityNodeData::ROLE_BROWSER; |
+ case WebKit::WebAccessibilityRoleBusyIndicator: |
+ return AccessibilityNodeData::ROLE_BUSY_INDICATOR; |
+ case WebKit::WebAccessibilityRoleButton: |
+ return AccessibilityNodeData::ROLE_BUTTON; |
+ case WebKit::WebAccessibilityRoleCell: |
+ return AccessibilityNodeData::ROLE_CELL; |
+ case WebKit::WebAccessibilityRoleCheckBox: |
+ return AccessibilityNodeData::ROLE_CHECKBOX; |
+ case WebKit::WebAccessibilityRoleColorWell: |
+ return AccessibilityNodeData::ROLE_COLOR_WELL; |
+ case WebKit::WebAccessibilityRoleColumn: |
+ return AccessibilityNodeData::ROLE_COLUMN; |
+ case WebKit::WebAccessibilityRoleColumnHeader: |
+ return AccessibilityNodeData::ROLE_COLUMN_HEADER; |
+ case WebKit::WebAccessibilityRoleComboBox: |
+ return AccessibilityNodeData::ROLE_COMBO_BOX; |
+ case WebKit::WebAccessibilityRoleDefinitionListDefinition: |
+ return AccessibilityNodeData::ROLE_DEFINITION_LIST_DEFINITION; |
+ case WebKit::WebAccessibilityRoleDefinitionListTerm: |
+ return AccessibilityNodeData::ROLE_DEFINITION_LIST_TERM; |
+ case WebKit::WebAccessibilityRoleDirectory: |
+ return AccessibilityNodeData::ROLE_DIRECTORY; |
+ case WebKit::WebAccessibilityRoleDisclosureTriangle: |
+ return AccessibilityNodeData::ROLE_DISCLOSURE_TRIANGLE; |
+ case WebKit::WebAccessibilityRoleDocument: |
+ return AccessibilityNodeData::ROLE_DOCUMENT; |
+ case WebKit::WebAccessibilityRoleDocumentArticle: |
+ return AccessibilityNodeData::ROLE_ARTICLE; |
+ case WebKit::WebAccessibilityRoleDocumentMath: |
+ return AccessibilityNodeData::ROLE_MATH; |
+ case WebKit::WebAccessibilityRoleDocumentNote: |
+ return AccessibilityNodeData::ROLE_NOTE; |
+ case WebKit::WebAccessibilityRoleDocumentRegion: |
+ return AccessibilityNodeData::ROLE_REGION; |
+ case WebKit::WebAccessibilityRoleDrawer: |
+ return AccessibilityNodeData::ROLE_DRAWER; |
+ case WebKit::WebAccessibilityRoleEditableText: |
+ return AccessibilityNodeData::ROLE_EDITABLE_TEXT; |
+ case WebKit::WebAccessibilityRoleFooter: |
+ return AccessibilityNodeData::ROLE_FOOTER; |
+ case WebKit::WebAccessibilityRoleGrid: |
+ return AccessibilityNodeData::ROLE_GRID; |
+ case WebKit::WebAccessibilityRoleGroup: |
+ return AccessibilityNodeData::ROLE_GROUP; |
+ case WebKit::WebAccessibilityRoleGrowArea: |
+ return AccessibilityNodeData::ROLE_GROW_AREA; |
+ case WebKit::WebAccessibilityRoleHeading: |
+ return AccessibilityNodeData::ROLE_HEADING; |
+ case WebKit::WebAccessibilityRoleHelpTag: |
+ return AccessibilityNodeData::ROLE_HELP_TAG; |
+ case WebKit::WebAccessibilityRoleIgnored: |
+ return AccessibilityNodeData::ROLE_IGNORED; |
+ case WebKit::WebAccessibilityRoleImage: |
+ return AccessibilityNodeData::ROLE_IMAGE; |
+ case WebKit::WebAccessibilityRoleImageMap: |
+ return AccessibilityNodeData::ROLE_IMAGE_MAP; |
+ case WebKit::WebAccessibilityRoleImageMapLink: |
+ return AccessibilityNodeData::ROLE_IMAGE_MAP_LINK; |
+ case WebKit::WebAccessibilityRoleIncrementor: |
+ return AccessibilityNodeData::ROLE_INCREMENTOR; |
+ case WebKit::WebAccessibilityRoleLandmarkApplication: |
+ return AccessibilityNodeData::ROLE_LANDMARK_APPLICATION; |
+ case WebKit::WebAccessibilityRoleLandmarkBanner: |
+ return AccessibilityNodeData::ROLE_LANDMARK_BANNER; |
+ case WebKit::WebAccessibilityRoleLandmarkComplementary: |
+ return AccessibilityNodeData::ROLE_LANDMARK_COMPLEMENTARY; |
+ case WebKit::WebAccessibilityRoleLandmarkContentInfo: |
+ return AccessibilityNodeData::ROLE_LANDMARK_CONTENTINFO; |
+ case WebKit::WebAccessibilityRoleLandmarkMain: |
+ return AccessibilityNodeData::ROLE_LANDMARK_MAIN; |
+ case WebKit::WebAccessibilityRoleLandmarkNavigation: |
+ return AccessibilityNodeData::ROLE_LANDMARK_NAVIGATION; |
+ case WebKit::WebAccessibilityRoleLandmarkSearch: |
+ return AccessibilityNodeData::ROLE_LANDMARK_SEARCH; |
+ case WebKit::WebAccessibilityRoleLink: |
+ return AccessibilityNodeData::ROLE_LINK; |
+ case WebKit::WebAccessibilityRoleList: |
+ return AccessibilityNodeData::ROLE_LIST; |
+ case WebKit::WebAccessibilityRoleListBox: |
+ return AccessibilityNodeData::ROLE_LISTBOX; |
+ case WebKit::WebAccessibilityRoleListBoxOption: |
+ return AccessibilityNodeData::ROLE_LISTBOX_OPTION; |
+ case WebKit::WebAccessibilityRoleListItem: |
+ return AccessibilityNodeData::ROLE_LIST_ITEM; |
+ case WebKit::WebAccessibilityRoleListMarker: |
+ return AccessibilityNodeData::ROLE_LIST_MARKER; |
+ case WebKit::WebAccessibilityRoleMatte: |
+ return AccessibilityNodeData::ROLE_MATTE; |
+ case WebKit::WebAccessibilityRoleMenu: |
+ return AccessibilityNodeData::ROLE_MENU; |
+ case WebKit::WebAccessibilityRoleMenuBar: |
+ return AccessibilityNodeData::ROLE_MENU_BAR; |
+ case WebKit::WebAccessibilityRoleMenuButton: |
+ return AccessibilityNodeData::ROLE_MENU_BUTTON; |
+ case WebKit::WebAccessibilityRoleMenuItem: |
+ return AccessibilityNodeData::ROLE_MENU_ITEM; |
+ case WebKit::WebAccessibilityRoleMenuListOption: |
+ return AccessibilityNodeData::ROLE_MENU_LIST_OPTION; |
+ case WebKit::WebAccessibilityRoleMenuListPopup: |
+ return AccessibilityNodeData::ROLE_MENU_LIST_POPUP; |
+ case WebKit::WebAccessibilityRoleOutline: |
+ return AccessibilityNodeData::ROLE_OUTLINE; |
+ case WebKit::WebAccessibilityRolePopUpButton: |
+ return AccessibilityNodeData::ROLE_POPUP_BUTTON; |
+ case WebKit::WebAccessibilityRoleProgressIndicator: |
+ return AccessibilityNodeData::ROLE_PROGRESS_INDICATOR; |
+ case WebKit::WebAccessibilityRoleRadioButton: |
+ return AccessibilityNodeData::ROLE_RADIO_BUTTON; |
+ case WebKit::WebAccessibilityRoleRadioGroup: |
+ return AccessibilityNodeData::ROLE_RADIO_GROUP; |
+ case WebKit::WebAccessibilityRoleRow: |
+ return AccessibilityNodeData::ROLE_ROW; |
+ case WebKit::WebAccessibilityRoleRowHeader: |
+ return AccessibilityNodeData::ROLE_ROW_HEADER; |
+ case WebKit::WebAccessibilityRoleRuler: |
+ return AccessibilityNodeData::ROLE_RULER; |
+ case WebKit::WebAccessibilityRoleRulerMarker: |
+ return AccessibilityNodeData::ROLE_RULER_MARKER; |
+ case WebKit::WebAccessibilityRoleScrollArea: |
+ return AccessibilityNodeData::ROLE_SCROLLAREA; |
+ case WebKit::WebAccessibilityRoleScrollBar: |
+ return AccessibilityNodeData::ROLE_SCROLLBAR; |
+ case WebKit::WebAccessibilityRoleSheet: |
+ return AccessibilityNodeData::ROLE_SHEET; |
+ case WebKit::WebAccessibilityRoleSlider: |
+ return AccessibilityNodeData::ROLE_SLIDER; |
+ case WebKit::WebAccessibilityRoleSliderThumb: |
+ return AccessibilityNodeData::ROLE_SLIDER_THUMB; |
+ case WebKit::WebAccessibilityRoleSplitGroup: |
+ return AccessibilityNodeData::ROLE_SPLIT_GROUP; |
+ case WebKit::WebAccessibilityRoleSplitter: |
+ return AccessibilityNodeData::ROLE_SPLITTER; |
+ case WebKit::WebAccessibilityRoleStaticText: |
+ return AccessibilityNodeData::ROLE_STATIC_TEXT; |
+ case WebKit::WebAccessibilityRoleSystemWide: |
+ return AccessibilityNodeData::ROLE_SYSTEM_WIDE; |
+ case WebKit::WebAccessibilityRoleTab: |
+ return AccessibilityNodeData::ROLE_TAB; |
+ case WebKit::WebAccessibilityRoleTabGroup: |
+ return AccessibilityNodeData::ROLE_TAB_GROUP_UNUSED; |
+ case WebKit::WebAccessibilityRoleTabList: |
+ return AccessibilityNodeData::ROLE_TAB_LIST; |
+ case WebKit::WebAccessibilityRoleTabPanel: |
+ return AccessibilityNodeData::ROLE_TAB_PANEL; |
+ case WebKit::WebAccessibilityRoleTable: |
+ return AccessibilityNodeData::ROLE_TABLE; |
+ case WebKit::WebAccessibilityRoleTableHeaderContainer: |
+ return AccessibilityNodeData::ROLE_TABLE_HEADER_CONTAINER; |
+ case WebKit::WebAccessibilityRoleTextArea: |
+ return AccessibilityNodeData::ROLE_TEXTAREA; |
+ case WebKit::WebAccessibilityRoleTextField: |
+ return AccessibilityNodeData::ROLE_TEXT_FIELD; |
+ case WebKit::WebAccessibilityRoleToolbar: |
+ return AccessibilityNodeData::ROLE_TOOLBAR; |
+ case WebKit::WebAccessibilityRoleTreeGrid: |
+ return AccessibilityNodeData::ROLE_TREE_GRID; |
+ case WebKit::WebAccessibilityRoleTreeItemRole: |
+ return AccessibilityNodeData::ROLE_TREE_ITEM; |
+ case WebKit::WebAccessibilityRoleTreeRole: |
+ return AccessibilityNodeData::ROLE_TREE; |
+ case WebKit::WebAccessibilityRoleUserInterfaceTooltip: |
+ return AccessibilityNodeData::ROLE_TOOLTIP; |
+ case WebKit::WebAccessibilityRoleValueIndicator: |
+ return AccessibilityNodeData::ROLE_VALUE_INDICATOR; |
+ case WebKit::WebAccessibilityRoleWebArea: |
+ return AccessibilityNodeData::ROLE_WEB_AREA; |
+ case WebKit::WebAccessibilityRoleWebCoreLink: |
+ return AccessibilityNodeData::ROLE_WEBCORE_LINK; |
+ case WebKit::WebAccessibilityRoleWindow: |
+ return AccessibilityNodeData::ROLE_WINDOW; |
+ |
+ default: |
+ return AccessibilityNodeData::ROLE_UNKNOWN; |
+ } |
+} |
+ |
+// Provides a conversion between the WebAccessibilityObject state |
+// accessors and a state bitmask that can be serialized and sent to the |
+// Browser process. Rare state are sent as boolean attributes instead. |
+uint32 ConvertState(const WebAccessibilityObject& o) { |
+ uint32 state = 0; |
+ if (o.isChecked()) |
+ state |= (1 << AccessibilityNodeData::STATE_CHECKED); |
+ |
+ if (o.isCollapsed()) |
+ state |= (1 << AccessibilityNodeData::STATE_COLLAPSED); |
+ |
+ if (o.canSetFocusAttribute()) |
+ state |= (1 << AccessibilityNodeData::STATE_FOCUSABLE); |
+ |
+ if (o.isFocused()) |
+ state |= (1 << AccessibilityNodeData::STATE_FOCUSED); |
+ |
+ if (o.roleValue() == WebKit::WebAccessibilityRolePopUpButton || |
+ o.ariaHasPopup()) { |
+ state |= (1 << AccessibilityNodeData::STATE_HASPOPUP); |
+ if (!o.isCollapsed()) |
+ state |= (1 << AccessibilityNodeData::STATE_EXPANDED); |
+ } |
+ |
+ if (o.isHovered()) |
+ state |= (1 << AccessibilityNodeData::STATE_HOTTRACKED); |
+ |
+ if (o.isIndeterminate()) |
+ state |= (1 << AccessibilityNodeData::STATE_INDETERMINATE); |
+ |
+ if (!o.isVisible()) |
+ state |= (1 << AccessibilityNodeData::STATE_INVISIBLE); |
+ |
+ if (o.isLinked()) |
+ state |= (1 << AccessibilityNodeData::STATE_LINKED); |
+ |
+ if (o.isMultiSelectable()) |
+ state |= (1 << AccessibilityNodeData::STATE_MULTISELECTABLE); |
+ |
+ if (o.isOffScreen()) |
+ state |= (1 << AccessibilityNodeData::STATE_OFFSCREEN); |
+ |
+ if (o.isPressed()) |
+ state |= (1 << AccessibilityNodeData::STATE_PRESSED); |
+ |
+ if (o.isPasswordField()) |
+ state |= (1 << AccessibilityNodeData::STATE_PROTECTED); |
+ |
+ if (o.isReadOnly()) |
+ state |= (1 << AccessibilityNodeData::STATE_READONLY); |
+ |
+ if (o.isRequired()) |
+ state |= (1 << AccessibilityNodeData::STATE_REQUIRED); |
+ |
+ if (o.canSetSelectedAttribute()) |
+ state |= (1 << AccessibilityNodeData::STATE_SELECTABLE); |
+ |
+ if (o.isSelected()) |
+ state |= (1 << AccessibilityNodeData::STATE_SELECTED); |
+ |
+ if (o.isVisited()) |
+ state |= (1 << AccessibilityNodeData::STATE_TRAVERSED); |
+ |
+ if (!o.isEnabled()) |
+ state |= (1 << AccessibilityNodeData::STATE_UNAVAILABLE); |
+ |
+ if (o.isVertical()) |
+ state |= (1 << AccessibilityNodeData::STATE_VERTICAL); |
+ |
+ if (o.isVisited()) |
+ state |= (1 << AccessibilityNodeData::STATE_VISITED); |
+ |
+ return state; |
+} |
+ |
+} // Anonymous namespace |
+ |
+namespace content { |
+ |
+void SerializeAccessibilityNode( |
+ const WebAccessibilityObject& src, |
+ AccessibilityNodeData* dst, |
+ bool include_children) { |
+ dst->name = src.title(); |
+ dst->role = ConvertRole(src.roleValue()); |
+ dst->state = ConvertState(src); |
+ dst->location = src.boundingBoxRect(); |
+ dst->id = src.axID(); |
+ |
+ if (src.valueDescription().length()) |
+ dst->value = src.valueDescription(); |
+ else |
+ dst->value = src.stringValue(); |
+ |
+ if (src.accessKey().length()) |
+ dst->string_attributes[dst->ATTR_ACCESS_KEY] = src.accessKey(); |
+ if (src.actionVerb().length()) |
+ dst->string_attributes[dst->ATTR_ACTION] = src.actionVerb(); |
+ if (src.isAriaReadOnly()) |
+ dst->bool_attributes[dst->ATTR_ARIA_READONLY] = true; |
+ if (src.isButtonStateMixed()) |
+ dst->bool_attributes[dst->ATTR_BUTTON_MIXED] = true; |
+ if (src.canSetValueAttribute()) |
+ dst->bool_attributes[dst->ATTR_CAN_SET_VALUE] = true; |
+ if (src.accessibilityDescription().length()) |
+ dst->string_attributes[dst->ATTR_DESCRIPTION] = |
+ src.accessibilityDescription(); |
+ if (src.hasComputedStyle()) |
+ dst->string_attributes[dst->ATTR_DISPLAY] = src.computedStyleDisplay(); |
+ if (src.helpText().length()) |
+ dst->string_attributes[dst->ATTR_HELP] = src.helpText(); |
+ if (src.keyboardShortcut().length()) |
+ dst->string_attributes[dst->ATTR_SHORTCUT] = src.keyboardShortcut(); |
+ if (src.titleUIElement().isValid()) { |
+ dst->int_attributes[dst->ATTR_TITLE_UI_ELEMENT] = |
+ src.titleUIElement().axID(); |
+ } |
+ if (!src.url().isEmpty()) |
+ dst->string_attributes[dst->ATTR_URL] = src.url().spec().utf16(); |
+ |
+ if (dst->role == dst->ROLE_TREE_ITEM) |
+ dst->int_attributes[dst->ATTR_HIERARCHICAL_LEVEL] = src.hierarchicalLevel(); |
+ |
+ if (dst->role == dst->ROLE_SLIDER) |
+ include_children = false; |
+ |
+ // Treat the active list box item as focused. |
+ if (dst->role == dst->ROLE_LISTBOX_OPTION && src.isSelectedOptionActive()) |
+ dst->state |= (1 << AccessibilityNodeData::STATE_FOCUSED); |
+ |
+ WebNode node = src.node(); |
+ bool is_iframe = false; |
+ |
+ if (!node.isNull() && node.isElementNode()) { |
+ WebElement element = node.to<WebElement>(); |
+ is_iframe = (element.tagName() == ASCIIToUTF16("IFRAME")); |
+ |
+ // TODO(ctguil): The tagName in WebKit is lower cased but |
+ // HTMLElement::nodeName calls localNameUpper. Consider adding |
+ // a WebElement method that returns the original lower cased tagName. |
+ dst->string_attributes[dst->ATTR_HTML_TAG] = |
+ StringToLowerASCII(string16(element.tagName())); |
+ for (unsigned i = 0; i < element.attributeCount(); ++i) { |
+ string16 name = StringToLowerASCII(string16( |
+ element.attributeLocalName(i))); |
+ string16 value = element.attributeValue(i); |
+ dst->html_attributes.push_back( |
+ std::pair<string16, string16>(name, value)); |
+ } |
+ |
+ if (dst->role == dst->ROLE_EDITABLE_TEXT || |
+ dst->role == dst->ROLE_TEXTAREA || |
+ dst->role == dst->ROLE_TEXT_FIELD) { |
+ // Jaws gets confused by children of text fields, so we ignore them. |
+ include_children = false; |
+ |
+ dst->int_attributes[dst->ATTR_TEXT_SEL_START] = src.selectionStart(); |
+ dst->int_attributes[dst->ATTR_TEXT_SEL_END] = src.selectionEnd(); |
+ |
+ WebVector<int> src_line_breaks; |
+ src.lineBreaks(src_line_breaks); |
+ dst->line_breaks.reserve(src_line_breaks.size()); |
+ for (size_t i = 0; i < src_line_breaks.size(); ++i) |
+ dst->line_breaks.push_back(src_line_breaks[i]); |
+ } |
+ |
+ // ARIA role. |
+ if (element.hasAttribute("role")) { |
+ dst->string_attributes[dst->ATTR_ROLE] = element.getAttribute("role"); |
+ } |
+ |
+ // Live region attributes |
+ if (element.hasAttribute("aria-atomic")) { |
+ dst->bool_attributes[dst->ATTR_LIVE_ATOMIC] = |
+ LowerCaseEqualsASCII(element.getAttribute("aria-atomic"), "true"); |
+ } |
+ if (element.hasAttribute("aria-busy")) { |
+ dst->bool_attributes[dst->ATTR_LIVE_BUSY] = |
+ LowerCaseEqualsASCII(element.getAttribute("aria-busy"), "true"); |
+ } |
+ if (element.hasAttribute("aria-live")) { |
+ dst->string_attributes[dst->ATTR_LIVE_STATUS] = |
+ element.getAttribute("aria-live"); |
+ } |
+ if (element.hasAttribute("aria-relevant")) { |
+ dst->string_attributes[dst->ATTR_LIVE_RELEVANT] = |
+ element.getAttribute("aria-relevant"); |
+ } |
+ } |
+ |
+ // Walk up the parent chain to set live region attributes of containers |
+ |
+ WebAccessibilityObject container_accessible = src; |
+ while (!container_accessible.isNull()) { |
+ WebNode container_node = container_accessible.node(); |
+ if (!container_node.isNull() && container_node.isElementNode()) { |
+ WebElement container_elem = |
+ container_node.to<WebElement>(); |
+ if (container_elem.hasAttribute("aria-atomic") && |
+ dst->bool_attributes.find(dst->ATTR_CONTAINER_LIVE_ATOMIC) == |
+ dst->bool_attributes.end()) { |
+ dst->bool_attributes[dst->ATTR_CONTAINER_LIVE_ATOMIC] = |
+ LowerCaseEqualsASCII(container_elem.getAttribute("aria-atomic"), |
+ "true"); |
+ } |
+ if (container_elem.hasAttribute("aria-busy") && |
+ dst->bool_attributes.find(dst->ATTR_CONTAINER_LIVE_BUSY) == |
+ dst->bool_attributes.end()) { |
+ dst->bool_attributes[dst->ATTR_CONTAINER_LIVE_BUSY] = |
+ LowerCaseEqualsASCII(container_elem.getAttribute("aria-busy"), |
+ "true"); |
+ } |
+ if (container_elem.hasAttribute("aria-live") && |
+ dst->string_attributes.find(dst->ATTR_CONTAINER_LIVE_STATUS) == |
+ dst->string_attributes.end()) { |
+ dst->string_attributes[dst->ATTR_CONTAINER_LIVE_STATUS] = |
+ container_elem.getAttribute("aria-live"); |
+ } |
+ if (container_elem.hasAttribute("aria-relevant") && |
+ dst->string_attributes.find(dst->ATTR_CONTAINER_LIVE_RELEVANT) == |
+ dst->string_attributes.end()) { |
+ dst->string_attributes[dst->ATTR_CONTAINER_LIVE_RELEVANT] = |
+ container_elem.getAttribute("aria-relevant"); |
+ } |
+ } |
+ container_accessible = container_accessible.parentObject(); |
+ } |
+ |
+ if (dst->role == dst->ROLE_PROGRESS_INDICATOR || |
+ dst->role == dst->ROLE_SCROLLBAR || |
+ dst->role == dst->ROLE_SLIDER) { |
+ dst->float_attributes[dst->ATTR_VALUE_FOR_RANGE] = src.valueForRange(); |
+ dst->float_attributes[dst->ATTR_MAX_VALUE_FOR_RANGE] = |
+ src.minValueForRange(); |
+ dst->float_attributes[dst->ATTR_MIN_VALUE_FOR_RANGE] = |
+ src.maxValueForRange(); |
+ } |
+ |
+ if (dst->role == dst->ROLE_DOCUMENT || |
+ dst->role == dst->ROLE_WEB_AREA) { |
+ const WebDocument& document = src.document(); |
+ if (dst->name.empty()) |
+ dst->name = document.title(); |
+ dst->string_attributes[dst->ATTR_DOC_TITLE] = document.title(); |
+ dst->string_attributes[dst->ATTR_DOC_URL] = document.url().spec().utf16(); |
+ dst->string_attributes[dst->ATTR_DOC_MIMETYPE] = |
+ ASCIIToUTF16(document.isXHTMLDocument() ? "text/xhtml" : "text/html"); |
+ dst->bool_attributes[dst->ATTR_DOC_LOADED] = src.isLoaded(); |
+ dst->float_attributes[dst->ATTR_DOC_LOADING_PROGRESS] = |
+ src.estimatedLoadingProgress(); |
+ |
+ const WebDocumentType& doctype = document.doctype(); |
+ if (!doctype.isNull()) |
+ dst->string_attributes[dst->ATTR_DOC_DOCTYPE] = doctype.name(); |
+ |
+ const gfx::Size& scroll_offset = document.frame()->scrollOffset(); |
+ dst->int_attributes[dst->ATTR_SCROLL_X] = scroll_offset.width(); |
+ dst->int_attributes[dst->ATTR_SCROLL_Y] = scroll_offset.height(); |
+ |
+ const gfx::Size& min_offset = document.frame()->minimumScrollOffset(); |
+ dst->int_attributes[dst->ATTR_SCROLL_X_MIN] = min_offset.width(); |
+ dst->int_attributes[dst->ATTR_SCROLL_Y_MIN] = min_offset.height(); |
+ |
+ const gfx::Size& max_offset = document.frame()->maximumScrollOffset(); |
+ dst->int_attributes[dst->ATTR_SCROLL_X_MAX] = max_offset.width(); |
+ dst->int_attributes[dst->ATTR_SCROLL_Y_MAX] = max_offset.height(); |
+ } |
+ |
+ if (dst->role == dst->ROLE_TABLE) { |
+ int column_count = src.columnCount(); |
+ int row_count = src.rowCount(); |
+ if (column_count > 0 && row_count > 0) { |
+ std::set<int> unique_cell_id_set; |
+ dst->int_attributes[dst->ATTR_TABLE_COLUMN_COUNT] = column_count; |
+ dst->int_attributes[dst->ATTR_TABLE_ROW_COUNT] = row_count; |
+ for (int i = 0; i < column_count * row_count; ++i) { |
+ WebAccessibilityObject cell = src.cellForColumnAndRow( |
+ i % column_count, i / column_count); |
+ int cell_id = -1; |
+ if (!cell.isNull()) { |
+ cell_id = cell.axID(); |
+ if (unique_cell_id_set.find(cell_id) == unique_cell_id_set.end()) { |
+ unique_cell_id_set.insert(cell_id); |
+ dst->unique_cell_ids.push_back(cell_id); |
+ } |
+ } |
+ dst->cell_ids.push_back(cell_id); |
+ } |
+ } |
+ } |
+ |
+ if (dst->role == dst->ROLE_CELL || |
+ dst->role == dst->ROLE_ROW_HEADER || |
+ dst->role == dst->ROLE_COLUMN_HEADER) { |
+ dst->int_attributes[dst->ATTR_TABLE_CELL_COLUMN_INDEX] = |
+ src.cellColumnIndex(); |
+ dst->int_attributes[dst->ATTR_TABLE_CELL_COLUMN_SPAN] = |
+ src.cellColumnSpan(); |
+ dst->int_attributes[dst->ATTR_TABLE_CELL_ROW_INDEX] = src.cellRowIndex(); |
+ dst->int_attributes[dst->ATTR_TABLE_CELL_ROW_SPAN] = src.cellRowSpan(); |
+ } |
+ |
+ if (include_children) { |
+ // Recursively create children. |
+ int child_count = src.childCount(); |
+ std::set<int32> child_ids; |
+ for (int i = 0; i < child_count; ++i) { |
+ WebAccessibilityObject child = src.childAt(i); |
+ int32 child_id = child.axID(); |
+ |
+ // The child may be invalid due to issues in webkit accessibility code. |
+ // Don't add children that are invalid thus preventing a crash. |
+ // https://bugs.webkit.org/show_bug.cgi?id=44149 |
+ // TODO(ctguil): We may want to remove this check as webkit stabilizes. |
+ if (!child.isValid()) |
+ continue; |
+ |
+ // Children may duplicated in the webkit accessibility tree. Only add a |
+ // child once for the web accessibility tree. |
+ // https://bugs.webkit.org/show_bug.cgi?id=58930 |
+ if (child_ids.find(child_id) != child_ids.end()) |
+ continue; |
+ child_ids.insert(child_id); |
+ |
+ // Some nodes appear in the tree in more than one place: for example, |
+ // a cell in a table appears as a child of both a row and a column. |
+ // Only recursively add child nodes that have this node as its |
+ // unignored parent. For child nodes that are actually parented to |
+ // somethinng else, store only the ID. |
+ // |
+ // As an exception, also add children of an iframe element. |
+ // https://bugs.webkit.org/show_bug.cgi?id=57066 |
+ if (is_iframe || IsParentUnignoredOf(src, child)) { |
+ dst->children.push_back(AccessibilityNodeData()); |
+ SerializeAccessibilityNode(child, |
+ &dst->children.back(), |
+ include_children); |
+ } else { |
+ dst->indirect_child_ids.push_back(child_id); |
+ } |
+ } |
+ } |
+} |
+ |
+} // namespace content |