blob: 32b9e4255fc81c92707cff9ac21af97218e21f74 [file] [log] [blame]
// Copyright 2013 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/browser/accessibility/browser_accessibility_android.h"
#include "base/i18n/break_iterator.h"
#include "base/strings/string_number_conversions.h"
#include "base/strings/string_util.h"
#include "base/strings/stringprintf.h"
#include "base/strings/utf_string_conversions.h"
#include "content/app/strings/grit/content_strings.h"
#include "content/browser/accessibility/browser_accessibility_manager_android.h"
#include "content/common/accessibility_messages.h"
#include "content/public/common/content_client.h"
#include "third_party/skia/include/core/SkColor.h"
#include "ui/accessibility/platform/ax_android_constants.h"
namespace {
const base::char16 kSecurePasswordBullet = 0x2022;
// These are enums from android.text.InputType in Java:
enum {
ANDROID_TEXT_INPUTTYPE_TYPE_NULL = 0,
ANDROID_TEXT_INPUTTYPE_TYPE_DATETIME = 0x4,
ANDROID_TEXT_INPUTTYPE_TYPE_DATETIME_DATE = 0x14,
ANDROID_TEXT_INPUTTYPE_TYPE_DATETIME_TIME = 0x24,
ANDROID_TEXT_INPUTTYPE_TYPE_NUMBER = 0x2,
ANDROID_TEXT_INPUTTYPE_TYPE_PHONE = 0x3,
ANDROID_TEXT_INPUTTYPE_TYPE_TEXT = 0x1,
ANDROID_TEXT_INPUTTYPE_TYPE_TEXT_URI = 0x11,
ANDROID_TEXT_INPUTTYPE_TYPE_TEXT_WEB_EDIT_TEXT = 0xa1,
ANDROID_TEXT_INPUTTYPE_TYPE_TEXT_WEB_EMAIL = 0xd1,
ANDROID_TEXT_INPUTTYPE_TYPE_TEXT_WEB_PASSWORD = 0xe1
};
// These are enums from android.view.View in Java:
enum {
ANDROID_VIEW_VIEW_ACCESSIBILITY_LIVE_REGION_NONE = 0,
ANDROID_VIEW_VIEW_ACCESSIBILITY_LIVE_REGION_POLITE = 1,
ANDROID_VIEW_VIEW_ACCESSIBILITY_LIVE_REGION_ASSERTIVE = 2
};
// These are enums from
// android.view.accessibility.AccessibilityNodeInfo.RangeInfo in Java:
enum {
ANDROID_VIEW_ACCESSIBILITY_RANGE_TYPE_FLOAT = 1
};
} // namespace
namespace content {
// static
BrowserAccessibility* BrowserAccessibility::Create() {
return new BrowserAccessibilityAndroid();
}
BrowserAccessibilityAndroid::BrowserAccessibilityAndroid() {
first_time_ = true;
}
bool BrowserAccessibilityAndroid::IsNative() const {
return true;
}
void BrowserAccessibilityAndroid::OnLocationChanged() {
manager()->NotifyAccessibilityEvent(
BrowserAccessibilityEvent::FromTreeChange,
ui::AX_EVENT_LOCATION_CHANGED,
this);
}
base::string16 BrowserAccessibilityAndroid::GetValue() const {
base::string16 value = BrowserAccessibility::GetValue();
// Optionally replace entered password text with bullet characters
// based on a user preference.
if (IsPassword()) {
bool should_expose = static_cast<BrowserAccessibilityManagerAndroid*>(
manager())->ShouldExposePasswordText();
if (!should_expose) {
value = base::string16(value.size(), kSecurePasswordBullet);
}
}
return value;
}
bool BrowserAccessibilityAndroid::PlatformIsLeaf() const {
if (BrowserAccessibility::PlatformIsLeaf())
return true;
// Iframes are always allowed to contain children.
if (IsIframe() ||
GetRole() == ui::AX_ROLE_ROOT_WEB_AREA ||
GetRole() == ui::AX_ROLE_WEB_AREA) {
return false;
}
// If it has a focusable child, we definitely can't leave out children.
if (HasFocusableChild())
return false;
// Date and time controls should drop their children.
switch (GetRole()) {
case ui::AX_ROLE_DATE:
case ui::AX_ROLE_DATE_TIME:
case ui::AX_ROLE_INPUT_TIME:
return true;
default:
break;
}
BrowserAccessibilityManagerAndroid* manager_android =
static_cast<BrowserAccessibilityManagerAndroid*>(manager());
if (manager_android->prune_tree_for_screen_reader()) {
// Headings with text can drop their children.
base::string16 name = GetText();
if (GetRole() == ui::AX_ROLE_HEADING && !name.empty())
return true;
// Focusable nodes with text can drop their children.
if (HasState(ui::AX_STATE_FOCUSABLE) && !name.empty())
return true;
// Nodes with only static text as children can drop their children.
if (HasOnlyTextChildren())
return true;
}
return false;
}
bool BrowserAccessibilityAndroid::IsCheckable() const {
bool checkable = false;
bool is_aria_pressed_defined;
bool is_mixed;
GetAriaTristate("aria-pressed", &is_aria_pressed_defined, &is_mixed);
if (GetRole() == ui::AX_ROLE_CHECK_BOX ||
GetRole() == ui::AX_ROLE_RADIO_BUTTON ||
GetRole() == ui::AX_ROLE_MENU_ITEM_CHECK_BOX ||
GetRole() == ui::AX_ROLE_MENU_ITEM_RADIO ||
is_aria_pressed_defined) {
checkable = true;
}
if (HasState(ui::AX_STATE_CHECKED))
checkable = true;
return checkable;
}
bool BrowserAccessibilityAndroid::IsChecked() const {
return (HasState(ui::AX_STATE_CHECKED) || HasState(ui::AX_STATE_PRESSED));
}
bool BrowserAccessibilityAndroid::IsClickable() const {
// If it has a default action, it's definitely clickable.
if (HasIntAttribute(ui::AX_ATTR_ACTION))
return true;
// Otherwise return true if it's focusable, but skip web areas and iframes.
if (IsIframe() || (GetRole() == ui::AX_ROLE_ROOT_WEB_AREA))
return false;
return IsFocusable();
}
bool BrowserAccessibilityAndroid::IsCollapsed() const {
return HasState(ui::AX_STATE_COLLAPSED);
}
bool BrowserAccessibilityAndroid::IsCollection() const {
return (GetRole() == ui::AX_ROLE_GRID ||
GetRole() == ui::AX_ROLE_LIST ||
GetRole() == ui::AX_ROLE_LIST_BOX ||
GetRole() == ui::AX_ROLE_DESCRIPTION_LIST ||
GetRole() == ui::AX_ROLE_TABLE ||
GetRole() == ui::AX_ROLE_TREE);
}
bool BrowserAccessibilityAndroid::IsCollectionItem() const {
return (GetRole() == ui::AX_ROLE_CELL ||
GetRole() == ui::AX_ROLE_COLUMN_HEADER ||
GetRole() == ui::AX_ROLE_DESCRIPTION_LIST_TERM ||
GetRole() == ui::AX_ROLE_LIST_BOX_OPTION ||
GetRole() == ui::AX_ROLE_LIST_ITEM ||
GetRole() == ui::AX_ROLE_ROW_HEADER ||
GetRole() == ui::AX_ROLE_TREE_ITEM);
}
bool BrowserAccessibilityAndroid::IsContentInvalid() const {
std::string invalid;
return GetHtmlAttribute("aria-invalid", &invalid);
}
bool BrowserAccessibilityAndroid::IsDismissable() const {
return false; // No concept of "dismissable" on the web currently.
}
bool BrowserAccessibilityAndroid::IsEditableText() const {
return GetRole() == ui::AX_ROLE_TEXT_FIELD;
}
bool BrowserAccessibilityAndroid::IsEnabled() const {
return !HasState(ui::AX_STATE_DISABLED);
}
bool BrowserAccessibilityAndroid::IsExpanded() const {
return HasState(ui::AX_STATE_EXPANDED);
}
bool BrowserAccessibilityAndroid::IsFocusable() const {
// If it's an iframe element, or the root element of a child frame,
// only mark it as focusable if the element has an explicit name.
// Otherwise mark it as not focusable to avoid the user landing on
// empty container elements in the tree.
if (IsIframe() ||
(GetRole() == ui::AX_ROLE_ROOT_WEB_AREA && PlatformGetParent()))
return HasStringAttribute(ui::AX_ATTR_NAME);
return HasState(ui::AX_STATE_FOCUSABLE);
}
bool BrowserAccessibilityAndroid::IsFocused() const {
return manager()->GetFocus() == this;
}
bool BrowserAccessibilityAndroid::IsHeading() const {
BrowserAccessibilityAndroid* parent =
static_cast<BrowserAccessibilityAndroid*>(PlatformGetParent());
if (parent && parent->IsHeading())
return true;
return (GetRole() == ui::AX_ROLE_COLUMN_HEADER ||
GetRole() == ui::AX_ROLE_HEADING ||
GetRole() == ui::AX_ROLE_ROW_HEADER);
}
bool BrowserAccessibilityAndroid::IsHierarchical() const {
return (GetRole() == ui::AX_ROLE_LIST ||
GetRole() == ui::AX_ROLE_DESCRIPTION_LIST ||
GetRole() == ui::AX_ROLE_TREE);
}
bool BrowserAccessibilityAndroid::IsLink() const {
return (GetRole() == ui::AX_ROLE_LINK ||
GetRole() == ui::AX_ROLE_IMAGE_MAP_LINK);
}
bool BrowserAccessibilityAndroid::IsMultiLine() const {
return HasState(ui::AX_STATE_MULTILINE);
}
bool BrowserAccessibilityAndroid::IsPassword() const {
return HasState(ui::AX_STATE_PROTECTED);
}
bool BrowserAccessibilityAndroid::IsRangeType() const {
return (GetRole() == ui::AX_ROLE_PROGRESS_INDICATOR ||
GetRole() == ui::AX_ROLE_METER ||
GetRole() == ui::AX_ROLE_SCROLL_BAR ||
GetRole() == ui::AX_ROLE_SLIDER);
}
bool BrowserAccessibilityAndroid::IsScrollable() const {
return (HasIntAttribute(ui::AX_ATTR_SCROLL_X_MAX) &&
GetRole() != ui::AX_ROLE_SCROLL_AREA);
}
bool BrowserAccessibilityAndroid::IsSelected() const {
return HasState(ui::AX_STATE_SELECTED);
}
bool BrowserAccessibilityAndroid::IsSlider() const {
return GetRole() == ui::AX_ROLE_SLIDER;
}
bool BrowserAccessibilityAndroid::IsVisibleToUser() const {
return !HasState(ui::AX_STATE_INVISIBLE);
}
bool BrowserAccessibilityAndroid::IsInterestingOnAndroid() const {
// The root is not interesting if it doesn't have a title, even
// though it's focusable.
if (GetRole() == ui::AX_ROLE_ROOT_WEB_AREA && GetText().empty())
return false;
// Focusable nodes are always interesting. Note that IsFocusable()
// already skips over things like iframes and child frames that are
// technically focusable but shouldn't be exposed as focusable on Android.
if (IsFocusable())
return true;
// If it's not focusable but has a control role, then it's interesting.
if (IsControl())
return true;
// Otherwise, the interesting nodes are leaf nodes with non-whitespace text.
return PlatformIsLeaf() &&
!base::ContainsOnlyChars(GetText(), base::kWhitespaceUTF16);
}
const BrowserAccessibilityAndroid*
BrowserAccessibilityAndroid::GetSoleInterestingNodeFromSubtree() const {
if (IsInterestingOnAndroid())
return this;
const BrowserAccessibilityAndroid* sole_interesting_node = nullptr;
for (uint32_t i = 0; i < PlatformChildCount(); ++i) {
const BrowserAccessibilityAndroid* interesting_node =
static_cast<const BrowserAccessibilityAndroid*>(PlatformGetChild(i))->
GetSoleInterestingNodeFromSubtree();
if (interesting_node && sole_interesting_node) {
// If there are two interesting nodes, return nullptr.
return nullptr;
} else if (interesting_node) {
sole_interesting_node = interesting_node;
}
}
return sole_interesting_node;
}
bool BrowserAccessibilityAndroid::CanOpenPopup() const {
return HasState(ui::AX_STATE_HASPOPUP);
}
const char* BrowserAccessibilityAndroid::GetClassName() const {
const char* class_name = NULL;
switch (GetRole()) {
case ui::AX_ROLE_SEARCH_BOX:
case ui::AX_ROLE_SPIN_BUTTON:
case ui::AX_ROLE_TEXT_FIELD:
class_name = ui::kAXEditTextClassname;
break;
case ui::AX_ROLE_SLIDER:
class_name = ui::kAXSeekBarClassname;
break;
case ui::AX_ROLE_COLOR_WELL:
case ui::AX_ROLE_COMBO_BOX:
case ui::AX_ROLE_DATE:
case ui::AX_ROLE_POP_UP_BUTTON:
case ui::AX_ROLE_INPUT_TIME:
class_name = ui::kAXSpinnerClassname;
break;
case ui::AX_ROLE_BUTTON:
case ui::AX_ROLE_MENU_BUTTON:
class_name = ui::kAXButtonClassname;
break;
case ui::AX_ROLE_CHECK_BOX:
case ui::AX_ROLE_SWITCH:
class_name = ui::kAXCheckBoxClassname;
break;
case ui::AX_ROLE_RADIO_BUTTON:
class_name = ui::kAXRadioButtonClassname;
break;
case ui::AX_ROLE_TOGGLE_BUTTON:
class_name = ui::kAXToggleButtonClassname;
break;
case ui::AX_ROLE_CANVAS:
case ui::AX_ROLE_IMAGE:
case ui::AX_ROLE_SVG_ROOT:
class_name = ui::kAXImageClassname;
break;
case ui::AX_ROLE_METER:
case ui::AX_ROLE_PROGRESS_INDICATOR:
class_name = ui::kAXProgressBarClassname;
break;
case ui::AX_ROLE_TAB_LIST:
class_name = ui::kAXTabWidgetClassname;
break;
case ui::AX_ROLE_GRID:
case ui::AX_ROLE_TABLE:
class_name = ui::kAXGridViewClassname;
break;
case ui::AX_ROLE_LIST:
case ui::AX_ROLE_LIST_BOX:
case ui::AX_ROLE_DESCRIPTION_LIST:
class_name = ui::kAXListViewClassname;
break;
case ui::AX_ROLE_DIALOG:
class_name = ui::kAXDialogClassname;
break;
case ui::AX_ROLE_ROOT_WEB_AREA:
if (PlatformGetParent() == nullptr)
class_name = ui::kAXWebViewClassname;
else
class_name = ui::kAXViewClassname;
break;
case ui::AX_ROLE_MENU_ITEM:
case ui::AX_ROLE_MENU_ITEM_CHECK_BOX:
case ui::AX_ROLE_MENU_ITEM_RADIO:
class_name = ui::kAXMenuItemClassname;
break;
default:
class_name = ui::kAXViewClassname;
break;
}
return class_name;
}
base::string16 BrowserAccessibilityAndroid::GetText() const {
if (IsIframe() ||
GetRole() == ui::AX_ROLE_WEB_AREA) {
return base::string16();
}
// In accordance with ARIA, some elements use their contents as an
// accessibility name, so some elements will have the same name as their
// child. While most platforms expect this, it causes Android to speak the
// name twice. So in this case we should remove the name from the parent.
if (GetRole() == ui::AX_ROLE_LIST_ITEM &&
GetIntAttribute(ui::AX_ATTR_NAME_FROM) == ui::AX_NAME_FROM_CONTENTS) {
// This is an approximation of "PlatformChildCount() > 0" because we can't
// call PlatformChildCount from here.
if (InternalChildCount() > 0 && !HasOnlyTextChildren())
return base::string16();
}
// We can only expose one accessible name on Android,
// not 2 or 3 like on Windows or Mac.
// First, always return the |value| attribute if this is an
// input field.
base::string16 value = GetValue();
if (!value.empty()) {
if (HasState(ui::AX_STATE_EDITABLE))
return value;
switch (GetRole()) {
case ui::AX_ROLE_COMBO_BOX:
case ui::AX_ROLE_POP_UP_BUTTON:
case ui::AX_ROLE_TEXT_FIELD:
return value;
default:
break;
}
}
// For color wells, the color is stored in separate attributes.
// Perhaps we could return color names in the future?
if (GetRole() == ui::AX_ROLE_COLOR_WELL) {
unsigned int color =
static_cast<unsigned int>(GetIntAttribute(ui::AX_ATTR_COLOR_VALUE));
unsigned int red = SkColorGetR(color);
unsigned int green = SkColorGetG(color);
unsigned int blue = SkColorGetB(color);
return base::UTF8ToUTF16(
base::StringPrintf("#%02X%02X%02X", red, green, blue));
}
base::string16 text = GetString16Attribute(ui::AX_ATTR_NAME);
base::string16 description = GetString16Attribute(ui::AX_ATTR_DESCRIPTION);
if (!description.empty()) {
if (!text.empty())
text += base::ASCIIToUTF16(" ");
text += description;
}
if (text.empty())
text = value;
// If this is the root element, give up now, allow it to have no
// accessible text. For almost all other focusable nodes we try to
// get text from contents, but for the root element that's redundant
// and often way too verbose.
if (GetRole() == ui::AX_ROLE_ROOT_WEB_AREA)
return text;
// This is called from PlatformIsLeaf, so don't call PlatformChildCount
// from within this!
if (text.empty() && (HasOnlyTextChildren() ||
(IsFocusable() && HasOnlyTextAndImageChildren()))) {
for (uint32_t i = 0; i < InternalChildCount(); i++) {
BrowserAccessibility* child = InternalGetChild(i);
text += static_cast<BrowserAccessibilityAndroid*>(child)->GetText();
}
}
if (text.empty() && (IsLink() || GetRole() == ui::AX_ROLE_IMAGE)) {
base::string16 url = GetString16Attribute(ui::AX_ATTR_URL);
// Given a url like http://foo.com/bar/baz.png, just return the
// base text, e.g., "baz".
int trailing_slashes = 0;
while (url.size() - trailing_slashes > 0 &&
url[url.size() - trailing_slashes - 1] == '/') {
trailing_slashes++;
}
if (trailing_slashes)
url = url.substr(0, url.size() - trailing_slashes);
size_t slash_index = url.rfind('/');
if (slash_index != std::string::npos)
url = url.substr(slash_index + 1);
size_t dot_index = url.rfind('.');
if (dot_index != std::string::npos)
url = url.substr(0, dot_index);
text = url;
}
return text;
}
base::string16 BrowserAccessibilityAndroid::GetRoleDescription() const {
content::ContentClient* content_client = content::GetContentClient();
// As a special case, if we have a heading level return a string like
// "heading level 1", etc.
if (GetRole() == ui::AX_ROLE_HEADING) {
int level = GetIntAttribute(ui::AX_ATTR_HIERARCHICAL_LEVEL);
if (level >= 1 && level <= 6) {
std::vector<base::string16> values;
values.push_back(base::IntToString16(level));
return base::ReplaceStringPlaceholders(
content_client->GetLocalizedString(IDS_AX_ROLE_HEADING_WITH_LEVEL),
values, nullptr);
}
}
int message_id = -1;
switch (GetRole()) {
case ui::AX_ROLE_ABBR:
// No role description.
break;
case ui::AX_ROLE_ALERT_DIALOG:
message_id = IDS_AX_ROLE_ALERT_DIALOG;
break;
case ui::AX_ROLE_ALERT:
message_id = IDS_AX_ROLE_ALERT;
break;
case ui::AX_ROLE_ANCHOR:
// No role description.
break;
case ui::AX_ROLE_ANNOTATION:
// No role description.
break;
case ui::AX_ROLE_APPLICATION:
message_id = IDS_AX_ROLE_APPLICATION;
break;
case ui::AX_ROLE_ARTICLE:
message_id = IDS_AX_ROLE_ARTICLE;
break;
case ui::AX_ROLE_AUDIO:
message_id = IDS_AX_MEDIA_AUDIO_ELEMENT;
break;
case ui::AX_ROLE_BANNER:
message_id = IDS_AX_ROLE_BANNER;
break;
case ui::AX_ROLE_BLOCKQUOTE:
message_id = IDS_AX_ROLE_BLOCKQUOTE;
break;
case ui::AX_ROLE_BUSY_INDICATOR:
message_id = IDS_AX_ROLE_BUSY_INDICATOR;
break;
case ui::AX_ROLE_BUTTON:
message_id = IDS_AX_ROLE_BUTTON;
break;
case ui::AX_ROLE_BUTTON_DROP_DOWN:
message_id = IDS_AX_ROLE_BUTTON_DROP_DOWN;
break;
case ui::AX_ROLE_CANVAS:
// No role description.
break;
case ui::AX_ROLE_CAPTION:
// No role description.
break;
case ui::AX_ROLE_CELL:
message_id = IDS_AX_ROLE_CELL;
break;
case ui::AX_ROLE_CHECK_BOX:
message_id = IDS_AX_ROLE_CHECK_BOX;
break;
case ui::AX_ROLE_CLIENT:
// No role description.
break;
case ui::AX_ROLE_COLOR_WELL:
message_id = IDS_AX_ROLE_COLOR_WELL;
break;
case ui::AX_ROLE_COLUMN_HEADER:
message_id = IDS_AX_ROLE_COLUMN_HEADER;
break;
case ui::AX_ROLE_COLUMN:
// No role description.
break;
case ui::AX_ROLE_COMBO_BOX:
message_id = IDS_AX_ROLE_COMBO_BOX;
break;
case ui::AX_ROLE_COMPLEMENTARY:
message_id = IDS_AX_ROLE_COMPLEMENTARY;
break;
case ui::AX_ROLE_CONTENT_INFO:
message_id = IDS_AX_ROLE_CONTENT_INFO;
break;
case ui::AX_ROLE_DATE:
message_id = IDS_AX_ROLE_DATE;
break;
case ui::AX_ROLE_DATE_TIME:
message_id = IDS_AX_ROLE_DATE_TIME;
break;
case ui::AX_ROLE_DEFINITION:
message_id = IDS_AX_ROLE_DEFINITION;
break;
case ui::AX_ROLE_DESCRIPTION_LIST_DETAIL:
message_id = IDS_AX_ROLE_DEFINITION;
break;
case ui::AX_ROLE_DESCRIPTION_LIST:
// No role description.
break;
case ui::AX_ROLE_DESCRIPTION_LIST_TERM:
// No role description.
break;
case ui::AX_ROLE_DESKTOP:
// No role description.
break;
case ui::AX_ROLE_DETAILS:
// No role description.
break;
case ui::AX_ROLE_DIALOG:
message_id = IDS_AX_ROLE_DIALOG;
break;
case ui::AX_ROLE_DIRECTORY:
message_id = IDS_AX_ROLE_DIRECTORY;
break;
case ui::AX_ROLE_DISCLOSURE_TRIANGLE:
message_id = IDS_AX_ROLE_DISCLOSURE_TRIANGLE;
break;
case ui::AX_ROLE_DIV:
// No role description.
break;
case ui::AX_ROLE_DOCUMENT:
message_id = IDS_AX_ROLE_DOCUMENT;
break;
case ui::AX_ROLE_EMBEDDED_OBJECT:
message_id = IDS_AX_ROLE_EMBEDDED_OBJECT;
break;
case ui::AX_ROLE_FEED:
message_id = IDS_AX_ROLE_FEED;
break;
case ui::AX_ROLE_FIGCAPTION:
// No role description.
break;
case ui::AX_ROLE_FIGURE:
message_id = IDS_AX_ROLE_GRAPHIC;
break;
case ui::AX_ROLE_FOOTER:
message_id = IDS_AX_ROLE_FOOTER;
break;
case ui::AX_ROLE_FORM:
// No role description.
break;
case ui::AX_ROLE_GRID:
message_id = IDS_AX_ROLE_TABLE;
break;
case ui::AX_ROLE_GROUP:
// No role description.
break;
case ui::AX_ROLE_HEADING:
// Note that code above this switch statement handles headings with
// a level, returning a string like "heading level 1", etc.
message_id = IDS_AX_ROLE_HEADING;
break;
case ui::AX_ROLE_IFRAME:
// No role description.
break;
case ui::AX_ROLE_IFRAME_PRESENTATIONAL:
// No role description.
break;
case ui::AX_ROLE_IGNORED:
// No role description.
break;
case ui::AX_ROLE_IMAGE_MAP_LINK:
message_id = IDS_AX_ROLE_LINK;
break;
case ui::AX_ROLE_IMAGE_MAP:
message_id = IDS_AX_ROLE_GRAPHIC;
break;
case ui::AX_ROLE_IMAGE:
message_id = IDS_AX_ROLE_GRAPHIC;
break;
case ui::AX_ROLE_INLINE_TEXT_BOX:
// No role description.
break;
case ui::AX_ROLE_INPUT_TIME:
message_id = IDS_AX_ROLE_INPUT_TIME;
break;
case ui::AX_ROLE_LABEL_TEXT:
// No role description.
break;
case ui::AX_ROLE_LEGEND:
// No role description.
break;
case ui::AX_ROLE_LINE_BREAK:
// No role description.
break;
case ui::AX_ROLE_LINK:
message_id = IDS_AX_ROLE_LINK;
break;
case ui::AX_ROLE_LIST_BOX_OPTION:
// No role description.
break;
case ui::AX_ROLE_LIST_BOX:
message_id = IDS_AX_ROLE_LIST_BOX;
break;
case ui::AX_ROLE_LIST_ITEM:
// No role description.
break;
case ui::AX_ROLE_LIST_MARKER:
// No role description.
break;
case ui::AX_ROLE_LIST:
// No role description.
break;
case ui::AX_ROLE_LOCATION_BAR:
// No role description.
break;
case ui::AX_ROLE_LOG:
message_id = IDS_AX_ROLE_LOG;
break;
case ui::AX_ROLE_MAIN:
message_id = IDS_AX_ROLE_MAIN_CONTENT;
break;
case ui::AX_ROLE_MARK:
message_id = IDS_AX_ROLE_MARK;
break;
case ui::AX_ROLE_MARQUEE:
message_id = IDS_AX_ROLE_MARQUEE;
break;
case ui::AX_ROLE_MATH:
message_id = IDS_AX_ROLE_MATH;
break;
case ui::AX_ROLE_MENU:
message_id = IDS_AX_ROLE_MENU;
break;
case ui::AX_ROLE_MENU_BAR:
message_id = IDS_AX_ROLE_MENU_BAR;
break;
case ui::AX_ROLE_MENU_BUTTON:
message_id = IDS_AX_ROLE_MENU_BUTTON;
break;
case ui::AX_ROLE_MENU_ITEM:
message_id = IDS_AX_ROLE_MENU_ITEM;
break;
case ui::AX_ROLE_MENU_ITEM_CHECK_BOX:
message_id = IDS_AX_ROLE_CHECK_BOX;
break;
case ui::AX_ROLE_MENU_ITEM_RADIO:
message_id = IDS_AX_ROLE_RADIO;
break;
case ui::AX_ROLE_MENU_LIST_OPTION:
// No role description.
break;
case ui::AX_ROLE_MENU_LIST_POPUP:
// No role description.
break;
case ui::AX_ROLE_METER:
message_id = IDS_AX_ROLE_METER;
break;
case ui::AX_ROLE_NAVIGATION:
message_id = IDS_AX_ROLE_NAVIGATIONAL_LINK;
break;
case ui::AX_ROLE_NOTE:
message_id = IDS_AX_ROLE_NOTE;
break;
case ui::AX_ROLE_OUTLINE:
message_id = IDS_AX_ROLE_OUTLINE;
break;
case ui::AX_ROLE_PANE:
// No role description.
break;
case ui::AX_ROLE_PARAGRAPH:
// No role description.
break;
case ui::AX_ROLE_POP_UP_BUTTON:
message_id = IDS_AX_ROLE_POP_UP_BUTTON;
break;
case ui::AX_ROLE_PRE:
// No role description.
break;
case ui::AX_ROLE_PRESENTATIONAL:
// No role description.
break;
case ui::AX_ROLE_PROGRESS_INDICATOR:
message_id = IDS_AX_ROLE_PROGRESS_INDICATOR;
break;
case ui::AX_ROLE_RADIO_BUTTON:
message_id = IDS_AX_ROLE_RADIO;
break;
case ui::AX_ROLE_RADIO_GROUP:
message_id = IDS_AX_ROLE_RADIO_GROUP;
break;
case ui::AX_ROLE_REGION:
message_id = IDS_AX_ROLE_REGION;
break;
case ui::AX_ROLE_ROOT_WEB_AREA:
// No role description.
break;
case ui::AX_ROLE_ROW_HEADER:
message_id = IDS_AX_ROLE_ROW_HEADER;
break;
case ui::AX_ROLE_ROW:
// No role description.
break;
case ui::AX_ROLE_RUBY:
// No role description.
break;
case ui::AX_ROLE_RULER:
message_id = IDS_AX_ROLE_RULER;
break;
case ui::AX_ROLE_SVG_ROOT:
message_id = IDS_AX_ROLE_GRAPHIC;
break;
case ui::AX_ROLE_SCROLL_AREA:
// No role description.
break;
case ui::AX_ROLE_SCROLL_BAR:
message_id = IDS_AX_ROLE_SCROLL_BAR;
break;
case ui::AX_ROLE_SEAMLESS_WEB_AREA:
// No role description.
break;
case ui::AX_ROLE_SEARCH:
message_id = IDS_AX_ROLE_SEARCH;
break;
case ui::AX_ROLE_SEARCH_BOX:
message_id = IDS_AX_ROLE_SEARCH_BOX;
break;
case ui::AX_ROLE_SLIDER:
message_id = IDS_AX_ROLE_SLIDER;
break;
case ui::AX_ROLE_SLIDER_THUMB:
// No role description.
break;
case ui::AX_ROLE_SPIN_BUTTON_PART:
// No role description.
break;
case ui::AX_ROLE_SPIN_BUTTON:
message_id = IDS_AX_ROLE_SPIN_BUTTON;
break;
case ui::AX_ROLE_SPLITTER:
message_id = IDS_AX_ROLE_SPLITTER;
break;
case ui::AX_ROLE_STATIC_TEXT:
// No role description.
break;
case ui::AX_ROLE_STATUS:
message_id = IDS_AX_ROLE_STATUS;
break;
case ui::AX_ROLE_SWITCH:
message_id = IDS_AX_ROLE_SWITCH;
break;
case ui::AX_ROLE_TAB_GROUP:
// No role description.
break;
case ui::AX_ROLE_TAB_LIST:
message_id = IDS_AX_ROLE_TAB_LIST;
break;
case ui::AX_ROLE_TAB_PANEL:
message_id = IDS_AX_ROLE_TAB_PANEL;
break;
case ui::AX_ROLE_TAB:
message_id = IDS_AX_ROLE_TAB;
break;
case ui::AX_ROLE_TABLE_HEADER_CONTAINER:
// No role description.
break;
case ui::AX_ROLE_TABLE:
message_id = IDS_AX_ROLE_TABLE;
break;
case ui::AX_ROLE_TERM:
message_id = IDS_AX_ROLE_DESCRIPTION_TERM;
break;
case ui::AX_ROLE_TEXT_FIELD:
// No role description.
break;
case ui::AX_ROLE_TIME:
message_id = IDS_AX_ROLE_TIME;
break;
case ui::AX_ROLE_TIMER:
message_id = IDS_AX_ROLE_TIMER;
break;
case ui::AX_ROLE_TITLE_BAR:
// No role description.
break;
case ui::AX_ROLE_TOGGLE_BUTTON:
message_id = IDS_AX_ROLE_TOGGLE_BUTTON;
break;
case ui::AX_ROLE_TOOLBAR:
message_id = IDS_AX_ROLE_TOOLBAR;
break;
case ui::AX_ROLE_TREE_GRID:
message_id = IDS_AX_ROLE_TREE_GRID;
break;
case ui::AX_ROLE_TREE_ITEM:
message_id = IDS_AX_ROLE_TREE_ITEM;
break;
case ui::AX_ROLE_TREE:
message_id = IDS_AX_ROLE_TREE;
break;
case ui::AX_ROLE_UNKNOWN:
// No role description.
break;
case ui::AX_ROLE_TOOLTIP:
message_id = IDS_AX_ROLE_TOOLTIP;
break;
case ui::AX_ROLE_VIDEO:
message_id = IDS_AX_MEDIA_VIDEO_ELEMENT;
break;
case ui::AX_ROLE_WEB_AREA:
// No role description.
break;
case ui::AX_ROLE_WEB_VIEW:
// No role description.
break;
case ui::AX_ROLE_WINDOW:
// No role description.
break;
case ui::AX_ROLE_NONE:
// No role description.
break;
}
if (message_id != -1)
return content_client->GetLocalizedString(message_id);
return base::string16();
}
int BrowserAccessibilityAndroid::GetItemIndex() const {
int index = 0;
switch (GetRole()) {
case ui::AX_ROLE_LIST_ITEM:
case ui::AX_ROLE_LIST_BOX_OPTION:
case ui::AX_ROLE_TREE_ITEM:
index = GetIntAttribute(ui::AX_ATTR_POS_IN_SET) - 1;
break;
case ui::AX_ROLE_SLIDER:
case ui::AX_ROLE_PROGRESS_INDICATOR: {
// Return a percentage here for live feedback in an AccessibilityEvent.
// The exact value is returned in RangeCurrentValue.
float min = GetFloatAttribute(ui::AX_ATTR_MIN_VALUE_FOR_RANGE);
float max = GetFloatAttribute(ui::AX_ATTR_MAX_VALUE_FOR_RANGE);
float value = GetFloatAttribute(ui::AX_ATTR_VALUE_FOR_RANGE);
if (max > min && value >= min && value <= max)
index = static_cast<int>(((value - min)) * 100 / (max - min));
break;
}
default:
break;
}
return index;
}
int BrowserAccessibilityAndroid::GetItemCount() const {
int count = 0;
switch (GetRole()) {
case ui::AX_ROLE_LIST:
case ui::AX_ROLE_LIST_BOX:
case ui::AX_ROLE_DESCRIPTION_LIST:
count = PlatformChildCount();
break;
case ui::AX_ROLE_SLIDER:
case ui::AX_ROLE_PROGRESS_INDICATOR:
// An AccessibilityEvent can only return integer information about a
// seek control, so we return a percentage. The real range is returned
// in RangeMin and RangeMax.
count = 100;
break;
default:
break;
}
return count;
}
bool BrowserAccessibilityAndroid::CanScrollForward() const {
if (IsSlider()) {
float value = GetFloatAttribute(ui::AX_ATTR_VALUE_FOR_RANGE);
float max = GetFloatAttribute(ui::AX_ATTR_MAX_VALUE_FOR_RANGE);
return value < max;
} else {
return GetScrollX() < GetMaxScrollX() ||
GetScrollY() < GetMaxScrollY();
}
}
bool BrowserAccessibilityAndroid::CanScrollBackward() const {
if (IsSlider()) {
float value = GetFloatAttribute(ui::AX_ATTR_VALUE_FOR_RANGE);
float min = GetFloatAttribute(ui::AX_ATTR_MIN_VALUE_FOR_RANGE);
return value > min;
} else {
return GetScrollX() > GetMinScrollX() ||
GetScrollY() > GetMinScrollY();
}
}
bool BrowserAccessibilityAndroid::CanScrollUp() const {
return GetScrollY() > GetMinScrollY();
}
bool BrowserAccessibilityAndroid::CanScrollDown() const {
return GetScrollY() < GetMaxScrollY();
}
bool BrowserAccessibilityAndroid::CanScrollLeft() const {
return GetScrollX() > GetMinScrollX();
}
bool BrowserAccessibilityAndroid::CanScrollRight() const {
return GetScrollX() < GetMaxScrollX();
}
int BrowserAccessibilityAndroid::GetScrollX() const {
int value = 0;
GetIntAttribute(ui::AX_ATTR_SCROLL_X, &value);
return value;
}
int BrowserAccessibilityAndroid::GetScrollY() const {
int value = 0;
GetIntAttribute(ui::AX_ATTR_SCROLL_Y, &value);
return value;
}
int BrowserAccessibilityAndroid::GetMinScrollX() const {
return GetIntAttribute(ui::AX_ATTR_SCROLL_X_MIN);
}
int BrowserAccessibilityAndroid::GetMinScrollY() const {
return GetIntAttribute(ui::AX_ATTR_SCROLL_Y_MIN);
}
int BrowserAccessibilityAndroid::GetMaxScrollX() const {
return GetIntAttribute(ui::AX_ATTR_SCROLL_X_MAX);
}
int BrowserAccessibilityAndroid::GetMaxScrollY() const {
return GetIntAttribute(ui::AX_ATTR_SCROLL_Y_MAX);
}
bool BrowserAccessibilityAndroid::Scroll(int direction) const {
int x = GetIntAttribute(ui::AX_ATTR_SCROLL_X);
int x_min = GetIntAttribute(ui::AX_ATTR_SCROLL_X_MIN);
int x_max = GetIntAttribute(ui::AX_ATTR_SCROLL_X_MAX);
int y = GetIntAttribute(ui::AX_ATTR_SCROLL_Y);
int y_min = GetIntAttribute(ui::AX_ATTR_SCROLL_Y_MIN);
int y_max = GetIntAttribute(ui::AX_ATTR_SCROLL_Y_MAX);
// Figure out the bounding box of the visible portion of this scrollable
// view so we know how much to scroll by.
gfx::Rect bounds;
if (GetRole() == ui::AX_ROLE_ROOT_WEB_AREA && !PlatformGetParent()) {
// If this is the root web area, use the bounds of the view to determine
// how big one page is.
if (!manager()->delegate())
return false;
bounds = manager()->delegate()->AccessibilityGetViewBounds();
} else if (GetRole() == ui::AX_ROLE_ROOT_WEB_AREA && PlatformGetParent()) {
// If this is a web area inside of an iframe, try to use the bounds of
// the containing element.
BrowserAccessibility* parent = PlatformGetParent();
while (parent && (parent->GetPageBoundsRect().width() == 0 ||
parent->GetPageBoundsRect().height() == 0)) {
parent = parent->PlatformGetParent();
}
if (parent)
bounds = parent->GetPageBoundsRect();
else
bounds = GetPageBoundsRect();
} else {
// Otherwise this is something like a scrollable div, just use the
// bounds of this object itself.
bounds = GetPageBoundsRect();
}
// Scroll by 80% of one page.
int page_x = std::max(bounds.width() * 4 / 5, 1);
int page_y = std::max(bounds.height() * 4 / 5, 1);
if (direction == FORWARD)
direction = y_max > y_min ? DOWN : RIGHT;
if (direction == BACKWARD)
direction = y_max > y_min ? UP : LEFT;
switch (direction) {
case UP:
y = std::min(std::max(y - page_y, y_min), y_max);
break;
case DOWN:
y = std::min(std::max(y + page_y, y_min), y_max);
break;
case LEFT:
x = std::min(std::max(x - page_x, x_min), x_max);
break;
case RIGHT:
x = std::min(std::max(x + page_x, x_min), x_max);
break;
default:
NOTREACHED();
}
manager()->SetScrollOffset(*this, gfx::Point(x, y));
return true;
}
// Given arbitrary old_value_ and new_value_, we must come up with reasonable
// edit metrics. Although edits like "apple" > "apples" are typical, anything
// is possible, such as "apple" > "applesauce", "apple" > "boot", or "" >
// "supercalifragilisticexpialidocious". So we consider old_value_ to be of the
// form AXB and new_value_ to be of the form AYB, where X and Y are the pieces
// that don't match. We take the X to be the "removed" characters and Y to be
// the "added" characters.
int BrowserAccessibilityAndroid::GetTextChangeFromIndex() const {
// This is len(A)
return CommonPrefixLength(old_value_, new_value_);
}
int BrowserAccessibilityAndroid::GetTextChangeAddedCount() const {
// This is len(AYB) - (len(A) + len(B)), or len(Y), the added characters.
return new_value_.length() - CommonEndLengths(old_value_, new_value_);
}
int BrowserAccessibilityAndroid::GetTextChangeRemovedCount() const {
// This is len(AXB) - (len(A) + len(B)), or len(X), the removed characters.
return old_value_.length() - CommonEndLengths(old_value_, new_value_);
}
// static
size_t BrowserAccessibilityAndroid::CommonPrefixLength(
const base::string16 a,
const base::string16 b) {
size_t a_len = a.length();
size_t b_len = b.length();
size_t i = 0;
while (i < a_len &&
i < b_len &&
a[i] == b[i]) {
i++;
}
return i;
}
// static
size_t BrowserAccessibilityAndroid::CommonSuffixLength(
const base::string16 a,
const base::string16 b) {
size_t a_len = a.length();
size_t b_len = b.length();
size_t i = 0;
while (i < a_len &&
i < b_len &&
a[a_len - i - 1] == b[b_len - i - 1]) {
i++;
}
return i;
}
// static
size_t BrowserAccessibilityAndroid::CommonEndLengths(
const base::string16 a,
const base::string16 b) {
size_t prefix_len = CommonPrefixLength(a, b);
// Remove the matching prefix before finding the suffix. Otherwise, if
// old_value_ is "a" and new_value_ is "aa", "a" will be double-counted as
// both a prefix and a suffix of "aa".
base::string16 a_body = a.substr(prefix_len, std::string::npos);
base::string16 b_body = b.substr(prefix_len, std::string::npos);
size_t suffix_len = CommonSuffixLength(a_body, b_body);
return prefix_len + suffix_len;
}
base::string16 BrowserAccessibilityAndroid::GetTextChangeBeforeText() const {
return old_value_;
}
int BrowserAccessibilityAndroid::GetSelectionStart() const {
int sel_start = 0;
GetIntAttribute(ui::AX_ATTR_TEXT_SEL_START, &sel_start);
return sel_start;
}
int BrowserAccessibilityAndroid::GetSelectionEnd() const {
int sel_end = 0;
GetIntAttribute(ui::AX_ATTR_TEXT_SEL_END, &sel_end);
return sel_end;
}
int BrowserAccessibilityAndroid::GetEditableTextLength() const {
base::string16 value = GetValue();
return value.length();
}
int BrowserAccessibilityAndroid::AndroidInputType() const {
std::string html_tag = GetStringAttribute(
ui::AX_ATTR_HTML_TAG);
if (html_tag != "input")
return ANDROID_TEXT_INPUTTYPE_TYPE_NULL;
std::string type;
if (!GetHtmlAttribute("type", &type))
return ANDROID_TEXT_INPUTTYPE_TYPE_TEXT;
if (type.empty() || type == "text" || type == "search")
return ANDROID_TEXT_INPUTTYPE_TYPE_TEXT;
else if (type == "date")
return ANDROID_TEXT_INPUTTYPE_TYPE_DATETIME_DATE;
else if (type == "datetime" || type == "datetime-local")
return ANDROID_TEXT_INPUTTYPE_TYPE_DATETIME;
else if (type == "email")
return ANDROID_TEXT_INPUTTYPE_TYPE_TEXT_WEB_EMAIL;
else if (type == "month")
return ANDROID_TEXT_INPUTTYPE_TYPE_DATETIME_DATE;
else if (type == "number")
return ANDROID_TEXT_INPUTTYPE_TYPE_NUMBER;
else if (type == "password")
return ANDROID_TEXT_INPUTTYPE_TYPE_TEXT_WEB_PASSWORD;
else if (type == "tel")
return ANDROID_TEXT_INPUTTYPE_TYPE_PHONE;
else if (type == "time")
return ANDROID_TEXT_INPUTTYPE_TYPE_DATETIME_TIME;
else if (type == "url")
return ANDROID_TEXT_INPUTTYPE_TYPE_TEXT_URI;
else if (type == "week")
return ANDROID_TEXT_INPUTTYPE_TYPE_DATETIME;
return ANDROID_TEXT_INPUTTYPE_TYPE_NULL;
}
int BrowserAccessibilityAndroid::AndroidLiveRegionType() const {
std::string live = GetStringAttribute(
ui::AX_ATTR_LIVE_STATUS);
if (live == "polite")
return ANDROID_VIEW_VIEW_ACCESSIBILITY_LIVE_REGION_POLITE;
else if (live == "assertive")
return ANDROID_VIEW_VIEW_ACCESSIBILITY_LIVE_REGION_ASSERTIVE;
return ANDROID_VIEW_VIEW_ACCESSIBILITY_LIVE_REGION_NONE;
}
int BrowserAccessibilityAndroid::AndroidRangeType() const {
return ANDROID_VIEW_ACCESSIBILITY_RANGE_TYPE_FLOAT;
}
int BrowserAccessibilityAndroid::RowCount() const {
if (GetRole() == ui::AX_ROLE_GRID ||
GetRole() == ui::AX_ROLE_TABLE) {
return CountChildrenWithRole(ui::AX_ROLE_ROW);
}
if (GetRole() == ui::AX_ROLE_LIST ||
GetRole() == ui::AX_ROLE_LIST_BOX ||
GetRole() == ui::AX_ROLE_DESCRIPTION_LIST ||
GetRole() == ui::AX_ROLE_TREE) {
return PlatformChildCount();
}
return 0;
}
int BrowserAccessibilityAndroid::ColumnCount() const {
if (GetRole() == ui::AX_ROLE_GRID ||
GetRole() == ui::AX_ROLE_TABLE) {
return CountChildrenWithRole(ui::AX_ROLE_COLUMN);
}
return 0;
}
int BrowserAccessibilityAndroid::RowIndex() const {
if (GetRole() == ui::AX_ROLE_LIST_ITEM ||
GetRole() == ui::AX_ROLE_LIST_BOX_OPTION ||
GetRole() == ui::AX_ROLE_TREE_ITEM) {
return GetIndexInParent();
}
return GetIntAttribute(ui::AX_ATTR_TABLE_CELL_ROW_INDEX);
}
int BrowserAccessibilityAndroid::RowSpan() const {
return GetIntAttribute(ui::AX_ATTR_TABLE_CELL_ROW_SPAN);
}
int BrowserAccessibilityAndroid::ColumnIndex() const {
return GetIntAttribute(ui::AX_ATTR_TABLE_CELL_COLUMN_INDEX);
}
int BrowserAccessibilityAndroid::ColumnSpan() const {
return GetIntAttribute(ui::AX_ATTR_TABLE_CELL_COLUMN_SPAN);
}
float BrowserAccessibilityAndroid::RangeMin() const {
return GetFloatAttribute(ui::AX_ATTR_MIN_VALUE_FOR_RANGE);
}
float BrowserAccessibilityAndroid::RangeMax() const {
return GetFloatAttribute(ui::AX_ATTR_MAX_VALUE_FOR_RANGE);
}
float BrowserAccessibilityAndroid::RangeCurrentValue() const {
return GetFloatAttribute(ui::AX_ATTR_VALUE_FOR_RANGE);
}
void BrowserAccessibilityAndroid::GetGranularityBoundaries(
int granularity,
std::vector<int32_t>* starts,
std::vector<int32_t>* ends,
int offset) {
switch (granularity) {
case ANDROID_ACCESSIBILITY_NODE_INFO_MOVEMENT_GRANULARITY_LINE:
GetLineBoundaries(starts, ends, offset);
break;
case ANDROID_ACCESSIBILITY_NODE_INFO_MOVEMENT_GRANULARITY_WORD:
GetWordBoundaries(starts, ends, offset);
break;
default:
NOTREACHED();
}
}
void BrowserAccessibilityAndroid::GetLineBoundaries(
std::vector<int32_t>* line_starts,
std::vector<int32_t>* line_ends,
int offset) {
// If this node has no children, treat it as all one line.
if (GetText().size() > 0 && !InternalChildCount()) {
line_starts->push_back(offset);
line_ends->push_back(offset + GetText().size());
}
// If this is a static text node, get the line boundaries from the
// inline text boxes if possible.
if (GetRole() == ui::AX_ROLE_STATIC_TEXT) {
int last_y = 0;
for (uint32_t i = 0; i < InternalChildCount(); i++) {
BrowserAccessibilityAndroid* child =
static_cast<BrowserAccessibilityAndroid*>(InternalGetChild(i));
CHECK_EQ(ui::AX_ROLE_INLINE_TEXT_BOX, child->GetRole());
// TODO(dmazzoni): replace this with a proper API to determine
// if two inline text boxes are on the same line. http://crbug.com/421771
int y = child->GetPageBoundsRect().y();
if (i == 0) {
line_starts->push_back(offset);
} else if (y != last_y) {
line_ends->push_back(offset);
line_starts->push_back(offset);
}
offset += child->GetText().size();
last_y = y;
}
line_ends->push_back(offset);
return;
}
// Otherwise, call GetLineBoundaries recursively on the children.
for (uint32_t i = 0; i < InternalChildCount(); i++) {
BrowserAccessibilityAndroid* child =
static_cast<BrowserAccessibilityAndroid*>(InternalGetChild(i));
child->GetLineBoundaries(line_starts, line_ends, offset);
offset += child->GetText().size();
}
}
void BrowserAccessibilityAndroid::GetWordBoundaries(
std::vector<int32_t>* word_starts,
std::vector<int32_t>* word_ends,
int offset) {
if (GetRole() == ui::AX_ROLE_INLINE_TEXT_BOX) {
const std::vector<int32_t>& starts =
GetIntListAttribute(ui::AX_ATTR_WORD_STARTS);
const std::vector<int32_t>& ends =
GetIntListAttribute(ui::AX_ATTR_WORD_ENDS);
for (size_t i = 0; i < starts.size(); ++i) {
word_starts->push_back(offset + starts[i]);
word_ends->push_back(offset + ends[i]);
}
return;
}
base::string16 concatenated_text;
for (uint32_t i = 0; i < InternalChildCount(); i++) {
BrowserAccessibilityAndroid* child =
static_cast<BrowserAccessibilityAndroid*>(InternalGetChild(i));
base::string16 child_text = child->GetText();
concatenated_text += child->GetText();
}
base::string16 text = GetText();
if (text.empty() || concatenated_text == text) {
// Great - this node is just the concatenation of its children, so
// we can get the word boundaries recursively.
for (uint32_t i = 0; i < InternalChildCount(); i++) {
BrowserAccessibilityAndroid* child =
static_cast<BrowserAccessibilityAndroid*>(InternalGetChild(i));
child->GetWordBoundaries(word_starts, word_ends, offset);
offset += child->GetText().size();
}
} else {
// This node has its own accessible text that doesn't match its
// visible text - like alt text for an image or something with an
// aria-label, so split the text into words locally.
base::i18n::BreakIterator iter(text, base::i18n::BreakIterator::BREAK_WORD);
if (!iter.Init())
return;
while (iter.Advance()) {
if (iter.IsWord()) {
word_starts->push_back(iter.prev());
word_ends->push_back(iter.pos());
}
}
}
}
bool BrowserAccessibilityAndroid::HasFocusableChild() const {
// This is called from PlatformIsLeaf, so don't call PlatformChildCount
// from within this!
for (uint32_t i = 0; i < InternalChildCount(); i++) {
BrowserAccessibility* child = InternalGetChild(i);
if (child->HasState(ui::AX_STATE_FOCUSABLE))
return true;
if (static_cast<BrowserAccessibilityAndroid*>(child)->HasFocusableChild())
return true;
}
return false;
}
bool BrowserAccessibilityAndroid::HasNonEmptyValue() const {
return IsEditableText() && !GetValue().empty();
}
bool BrowserAccessibilityAndroid::HasOnlyTextChildren() const {
// This is called from PlatformIsLeaf, so don't call PlatformChildCount
// from within this!
for (uint32_t i = 0; i < InternalChildCount(); i++) {
BrowserAccessibility* child = InternalGetChild(i);
if (!child->IsTextOnlyObject())
return false;
}
return true;
}
bool BrowserAccessibilityAndroid::HasOnlyTextAndImageChildren() const {
// This is called from PlatformIsLeaf, so don't call PlatformChildCount
// from within this!
for (uint32_t i = 0; i < InternalChildCount(); i++) {
BrowserAccessibility* child = InternalGetChild(i);
if (child->GetRole() != ui::AX_ROLE_STATIC_TEXT &&
child->GetRole() != ui::AX_ROLE_IMAGE) {
return false;
}
}
return true;
}
bool BrowserAccessibilityAndroid::IsIframe() const {
return (GetRole() == ui::AX_ROLE_IFRAME ||
GetRole() == ui::AX_ROLE_IFRAME_PRESENTATIONAL);
}
void BrowserAccessibilityAndroid::OnDataChanged() {
BrowserAccessibility::OnDataChanged();
if (IsEditableText()) {
base::string16 value = GetValue();
if (value != new_value_) {
old_value_ = new_value_;
new_value_ = value;
}
}
if (GetRole() == ui::AX_ROLE_ALERT && first_time_) {
manager()->NotifyAccessibilityEvent(
BrowserAccessibilityEvent::FromTreeChange,
ui::AX_EVENT_ALERT,
this);
}
base::string16 live;
if (GetString16Attribute(
ui::AX_ATTR_CONTAINER_LIVE_STATUS, &live)) {
NotifyLiveRegionUpdate(live);
}
first_time_ = false;
}
void BrowserAccessibilityAndroid::NotifyLiveRegionUpdate(
base::string16& aria_live) {
if (!base::EqualsASCII(aria_live, aria_strings::kAriaLivePolite) &&
!base::EqualsASCII(aria_live, aria_strings::kAriaLiveAssertive))
return;
base::string16 text = GetText();
if (cached_text_ != text) {
if (!text.empty()) {
manager()->NotifyAccessibilityEvent(
BrowserAccessibilityEvent::FromTreeChange,
ui::AX_EVENT_SHOW,
this);
}
cached_text_ = text;
}
}
int BrowserAccessibilityAndroid::CountChildrenWithRole(ui::AXRole role) const {
int count = 0;
for (uint32_t i = 0; i < PlatformChildCount(); i++) {
if (PlatformGetChild(i)->GetRole() == role)
count++;
}
return count;
}
} // namespace content