|  | // 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_util.h" | 
|  | #include "base/strings/stringprintf.h" | 
|  | #include "base/strings/utf_string_conversions.h" | 
|  | #include "content/browser/accessibility/browser_accessibility_manager_android.h" | 
|  | #include "content/common/accessibility_messages.h" | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | // 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(ui::AX_EVENT_LOCATION_CHANGED, this); | 
|  | } | 
|  |  | 
|  | bool BrowserAccessibilityAndroid::PlatformIsLeaf() const { | 
|  | if (InternalChildCount() == 0) | 
|  | 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. | 
|  | if (GetRole() == ui::AX_ROLE_DATE || GetRole() == ui::AX_ROLE_INPUT_TIME) | 
|  | return true; | 
|  |  | 
|  | 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 (HasOnlyStaticTextChildren()) | 
|  | return true; | 
|  | } | 
|  |  | 
|  | return BrowserAccessibility::PlatformIsLeaf(); | 
|  | } | 
|  |  | 
|  | 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 (!PlatformIsLeaf()) | 
|  | return false; | 
|  |  | 
|  | return IsFocusable() || !GetText().empty(); | 
|  | } | 
|  |  | 
|  | 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_ENABLED); | 
|  | } | 
|  |  | 
|  | bool BrowserAccessibilityAndroid::IsFocusable() const { | 
|  | bool focusable = HasState(ui::AX_STATE_FOCUSABLE); | 
|  | if (IsIframe() || | 
|  | GetRole() == ui::AX_ROLE_WEB_AREA) { | 
|  | focusable = false; | 
|  | } | 
|  | return focusable; | 
|  | } | 
|  |  | 
|  | bool BrowserAccessibilityAndroid::IsFocused() const { | 
|  | return manager()->GetFocus(manager()->GetRoot()) == this; | 
|  | } | 
|  |  | 
|  | bool BrowserAccessibilityAndroid::IsHeading() const { | 
|  | BrowserAccessibilityAndroid* parent = | 
|  | static_cast<BrowserAccessibilityAndroid*>(GetParent()); | 
|  | 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::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 = "android.widget.EditText"; | 
|  | break; | 
|  | case ui::AX_ROLE_SLIDER: | 
|  | class_name = "android.widget.SeekBar"; | 
|  | 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 = "android.widget.Spinner"; | 
|  | break; | 
|  | case ui::AX_ROLE_BUTTON: | 
|  | case ui::AX_ROLE_MENU_BUTTON: | 
|  | class_name = "android.widget.Button"; | 
|  | break; | 
|  | case ui::AX_ROLE_CHECK_BOX: | 
|  | case ui::AX_ROLE_SWITCH: | 
|  | class_name = "android.widget.CheckBox"; | 
|  | break; | 
|  | case ui::AX_ROLE_RADIO_BUTTON: | 
|  | class_name = "android.widget.RadioButton"; | 
|  | break; | 
|  | case ui::AX_ROLE_TOGGLE_BUTTON: | 
|  | class_name = "android.widget.ToggleButton"; | 
|  | break; | 
|  | case ui::AX_ROLE_CANVAS: | 
|  | case ui::AX_ROLE_IMAGE: | 
|  | case ui::AX_ROLE_SVG_ROOT: | 
|  | class_name = "android.widget.Image"; | 
|  | break; | 
|  | case ui::AX_ROLE_METER: | 
|  | case ui::AX_ROLE_PROGRESS_INDICATOR: | 
|  | class_name = "android.widget.ProgressBar"; | 
|  | break; | 
|  | case ui::AX_ROLE_TAB_LIST: | 
|  | class_name = "android.widget.TabWidget"; | 
|  | break; | 
|  | case ui::AX_ROLE_GRID: | 
|  | case ui::AX_ROLE_TABLE: | 
|  | class_name = "android.widget.GridView"; | 
|  | break; | 
|  | case ui::AX_ROLE_LIST: | 
|  | case ui::AX_ROLE_LIST_BOX: | 
|  | case ui::AX_ROLE_DESCRIPTION_LIST: | 
|  | class_name = "android.widget.ListView"; | 
|  | break; | 
|  | case ui::AX_ROLE_DIALOG: | 
|  | class_name = "android.app.Dialog"; | 
|  | break; | 
|  | case ui::AX_ROLE_ROOT_WEB_AREA: | 
|  | class_name = "android.webkit.WebView"; | 
|  | break; | 
|  | case ui::AX_ROLE_MENU_ITEM: | 
|  | case ui::AX_ROLE_MENU_ITEM_CHECK_BOX: | 
|  | case ui::AX_ROLE_MENU_ITEM_RADIO: | 
|  | class_name = "android.view.MenuItem"; | 
|  | break; | 
|  | default: | 
|  | class_name = "android.view.View"; | 
|  | break; | 
|  | } | 
|  |  | 
|  | return class_name; | 
|  | } | 
|  |  | 
|  | base::string16 BrowserAccessibilityAndroid::GetText() const { | 
|  | if (IsIframe() || | 
|  | GetRole() == ui::AX_ROLE_WEB_AREA) { | 
|  | return base::string16(); | 
|  | } | 
|  |  | 
|  | // See comment in browser_accessibility_win.cc for details. | 
|  | // The difference here is that 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 = GetString16Attribute(ui::AX_ATTR_VALUE); | 
|  | 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; | 
|  | } | 
|  | } | 
|  |  | 
|  | // 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) { | 
|  | int color = GetIntAttribute(ui::AX_ATTR_COLOR_VALUE); | 
|  | int red = (color >> 16) & 0xFF; | 
|  | int green = (color >> 8) & 0xFF; | 
|  | int blue = color & 0xFF; | 
|  | return base::UTF8ToUTF16( | 
|  | base::StringPrintf("#%02X%02X%02X", red, green, blue)); | 
|  | } | 
|  |  | 
|  | // Always prefer visible text if this is a link. Sites sometimes add | 
|  | // a "title" attribute to a link with more information, but we can't | 
|  | // lose the link text. | 
|  | base::string16 name = GetString16Attribute(ui::AX_ATTR_NAME); | 
|  | if (!name.empty() && GetRole() == ui::AX_ROLE_LINK) | 
|  | return name; | 
|  |  | 
|  | // If there's no text value, the basic rule is: prefer description | 
|  | // (aria-labelledby or aria-label), then help (title), then name | 
|  | // (inner text), then value (control value).  However, if | 
|  | // title_elem_id is set, that means there's a label element | 
|  | // supplying the name and then name takes precedence over help. | 
|  | // TODO(dmazzoni): clean this up by providing more granular labels in | 
|  | // Blink, making the platform-specific mapping to accessible text simpler. | 
|  | base::string16 description = GetString16Attribute(ui::AX_ATTR_DESCRIPTION); | 
|  | base::string16 help = GetString16Attribute(ui::AX_ATTR_HELP); | 
|  |  | 
|  | base::string16 placeholder; | 
|  | switch (GetRole()) { | 
|  | case ui::AX_ROLE_DATE: | 
|  | case ui::AX_ROLE_INPUT_TIME: | 
|  | case ui::AX_ROLE_TEXT_FIELD: | 
|  | GetHtmlAttribute("placeholder", &placeholder); | 
|  | } | 
|  |  | 
|  | int title_elem_id = GetIntAttribute( | 
|  | ui::AX_ATTR_TITLE_UI_ELEMENT); | 
|  | base::string16 text; | 
|  | if (!description.empty()) | 
|  | text = description; | 
|  | else if (!name.empty()) { | 
|  | text = name; | 
|  | if (!help.empty()) { | 
|  | // TODO(vkuzkokov): This is not the best way to pass 2 texts but this is | 
|  | // how Blink seems to be doing it. | 
|  | text += base::ASCIIToUTF16(" "); | 
|  | text += help; | 
|  | } | 
|  | } else if (!help.empty()) | 
|  | text = help; | 
|  | else if (!placeholder.empty()) | 
|  | text = placeholder; | 
|  | else if (!value.empty()) | 
|  | text = value; | 
|  | else if (title_elem_id) { | 
|  | BrowserAccessibility* title_elem = | 
|  | manager()->GetFromID(title_elem_id); | 
|  | if (title_elem) | 
|  | text = static_cast<BrowserAccessibilityAndroid*>(title_elem)->GetText(); | 
|  | } | 
|  |  | 
|  | // This is called from PlatformIsLeaf, so don't call PlatformChildCount | 
|  | // from within this! | 
|  | if (text.empty() && | 
|  | (HasOnlyStaticTextChildren() || | 
|  | (IsFocusable() && HasOnlyTextAndImageChildren()))) { | 
|  | for (uint32 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 name, 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; | 
|  | } | 
|  |  | 
|  | 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; | 
|  | } | 
|  | } | 
|  | 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; | 
|  | } | 
|  | 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) { | 
|  | // 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_WEB_AREA) { | 
|  | // If this is a web area inside of an iframe, try to use the bounds of | 
|  | // the containing element. | 
|  | BrowserAccessibility* parent = GetParent(); | 
|  | while (parent && (parent->GetLocation().width() == 0 || | 
|  | parent->GetLocation().height() == 0)) { | 
|  | parent = parent->GetParent(); | 
|  | } | 
|  | if (parent) | 
|  | bounds = parent->GetLocation(); | 
|  | else | 
|  | bounds = GetLocation(); | 
|  | } else { | 
|  | // Otherwise this is something like a scrollable div, just use the | 
|  | // bounds of this object itself. | 
|  | bounds = GetLocation(); | 
|  | } | 
|  |  | 
|  | // 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 = GetString16Attribute(ui::AX_ATTR_VALUE); | 
|  | 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>* starts, | 
|  | std::vector<int32>* 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>* line_starts, | 
|  | std::vector<int32>* 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 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->GetLocation().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 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>* word_starts, | 
|  | std::vector<int32>* word_ends, | 
|  | int offset) { | 
|  | if (GetRole() == ui::AX_ROLE_INLINE_TEXT_BOX) { | 
|  | const std::vector<int32>& starts = GetIntListAttribute( | 
|  | ui::AX_ATTR_WORD_STARTS); | 
|  | const std::vector<int32>& 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 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 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 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::HasOnlyStaticTextChildren() const { | 
|  | // This is called from PlatformIsLeaf, so don't call PlatformChildCount | 
|  | // from within this! | 
|  | for (uint32 i = 0; i < InternalChildCount(); i++) { | 
|  | BrowserAccessibility* child = InternalGetChild(i); | 
|  | if (child->GetRole() != ui::AX_ROLE_STATIC_TEXT) | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool BrowserAccessibilityAndroid::HasOnlyTextAndImageChildren() const { | 
|  | // This is called from PlatformIsLeaf, so don't call PlatformChildCount | 
|  | // from within this! | 
|  | for (uint32 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 { | 
|  | base::string16 html_tag = GetString16Attribute( | 
|  | ui::AX_ATTR_HTML_TAG); | 
|  | return html_tag == base::ASCIIToUTF16("iframe"); | 
|  | } | 
|  |  | 
|  | void BrowserAccessibilityAndroid::OnDataChanged() { | 
|  | BrowserAccessibility::OnDataChanged(); | 
|  |  | 
|  | if (IsEditableText()) { | 
|  | base::string16 value = GetString16Attribute(ui::AX_ATTR_VALUE); | 
|  | if (value != new_value_) { | 
|  | old_value_ = new_value_; | 
|  | new_value_ = value; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (GetRole() == ui::AX_ROLE_ALERT && first_time_) | 
|  | manager()->NotifyAccessibilityEvent(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(ui::AX_EVENT_SHOW, | 
|  | this); | 
|  | } | 
|  | cached_text_ = text; | 
|  | } | 
|  | } | 
|  |  | 
|  | int BrowserAccessibilityAndroid::CountChildrenWithRole(ui::AXRole role) const { | 
|  | int count = 0; | 
|  | for (uint32 i = 0; i < PlatformChildCount(); i++) { | 
|  | if (PlatformGetChild(i)->GetRole() == role) | 
|  | count++; | 
|  | } | 
|  | return count; | 
|  | } | 
|  |  | 
|  | }  // namespace content |