| /* |
| * Copyright (C) 2008, 2009, 2011 Apple Inc. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of |
| * its contributors may be used to endorse or promote products derived |
| * from this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY |
| * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED |
| * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
| * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY |
| * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES |
| * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
| * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND |
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF |
| * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #include "third_party/blink/renderer/modules/accessibility/ax_object.h" |
| |
| #include <algorithm> |
| |
| #include "third_party/blink/public/common/features.h" |
| #include "third_party/blink/public/mojom/input/focus_type.mojom-blink.h" |
| #include "third_party/blink/renderer/core/aom/accessible_node.h" |
| #include "third_party/blink/renderer/core/aom/accessible_node_list.h" |
| #include "third_party/blink/renderer/core/css/resolver/style_resolver.h" |
| #include "third_party/blink/renderer/core/display_lock/display_lock_utilities.h" |
| #include "third_party/blink/renderer/core/dom/dom_node_ids.h" |
| #include "third_party/blink/renderer/core/dom/focus_params.h" |
| #include "third_party/blink/renderer/core/dom/node_computed_style.h" |
| #include "third_party/blink/renderer/core/frame/local_frame.h" |
| #include "third_party/blink/renderer/core/frame/local_frame_view.h" |
| #include "third_party/blink/renderer/core/frame/settings.h" |
| #include "third_party/blink/renderer/core/html/canvas/html_canvas_element.h" |
| #include "third_party/blink/renderer/core/html/custom/element_internals.h" |
| #include "third_party/blink/renderer/core/html/forms/html_input_element.h" |
| #include "third_party/blink/renderer/core/html/forms/html_select_element.h" |
| #include "third_party/blink/renderer/core/html/html_dialog_element.h" |
| #include "third_party/blink/renderer/core/html/html_frame_owner_element.h" |
| #include "third_party/blink/renderer/core/html/parser/html_parser_idioms.h" |
| #include "third_party/blink/renderer/core/input/context_menu_allowed_scope.h" |
| #include "third_party/blink/renderer/core/input/event_handler.h" |
| #include "third_party/blink/renderer/core/input_type_names.h" |
| #include "third_party/blink/renderer/core/layout/layout_box.h" |
| #include "third_party/blink/renderer/core/layout/layout_box_model_object.h" |
| #include "third_party/blink/renderer/core/layout/layout_view.h" |
| #include "third_party/blink/renderer/core/page/chrome_client.h" |
| #include "third_party/blink/renderer/core/page/focus_controller.h" |
| #include "third_party/blink/renderer/core/page/page.h" |
| #include "third_party/blink/renderer/core/page/scrolling/top_document_root_scroller_controller.h" |
| #include "third_party/blink/renderer/modules/accessibility/ax_menu_list.h" |
| #include "third_party/blink/renderer/modules/accessibility/ax_menu_list_option.h" |
| #include "third_party/blink/renderer/modules/accessibility/ax_menu_list_popup.h" |
| #include "third_party/blink/renderer/modules/accessibility/ax_object_cache_impl.h" |
| #include "third_party/blink/renderer/modules/accessibility/ax_range.h" |
| #include "third_party/blink/renderer/modules/accessibility/ax_sparse_attribute_setter.h" |
| #include "third_party/blink/renderer/platform/instrumentation/use_counter.h" |
| #include "third_party/blink/renderer/platform/language.h" |
| #include "third_party/blink/renderer/platform/runtime_enabled_features.h" |
| #include "third_party/blink/renderer/platform/text/platform_locale.h" |
| #include "third_party/blink/renderer/platform/wtf/hash_set.h" |
| #include "third_party/blink/renderer/platform/wtf/std_lib_extras.h" |
| #include "third_party/blink/renderer/platform/wtf/text/wtf_string.h" |
| #include "third_party/blink/renderer/platform/wtf/wtf_size_t.h" |
| #include "third_party/skia/include/core/SkMatrix44.h" |
| #include "ui/accessibility/ax_node_data.h" |
| #include "ui/accessibility/ax_role_properties.h" |
| |
| namespace blink { |
| |
| namespace { |
| |
| struct RoleHashTraits : HashTraits<ax::mojom::blink::Role> { |
| static const bool kEmptyValueIsZero = true; |
| static ax::mojom::blink::Role EmptyValue() { |
| return ax::mojom::blink::Role::kUnknown; |
| } |
| }; |
| |
| using ARIARoleMap = HashMap<String, |
| ax::mojom::blink::Role, |
| CaseFoldingHash, |
| HashTraits<String>, |
| RoleHashTraits>; |
| |
| struct RoleEntry { |
| const char* aria_role; |
| ax::mojom::blink::Role webcore_role; |
| }; |
| |
| // Mapping of ARIA role name to internal role name. |
| const RoleEntry kRoles[] = { |
| {"alert", ax::mojom::blink::Role::kAlert}, |
| {"alertdialog", ax::mojom::blink::Role::kAlertDialog}, |
| {"application", ax::mojom::blink::Role::kApplication}, |
| {"article", ax::mojom::blink::Role::kArticle}, |
| {"banner", ax::mojom::blink::Role::kBanner}, |
| {"blockquote", ax::mojom::blink::Role::kBlockquote}, |
| {"button", ax::mojom::blink::Role::kButton}, |
| {"caption", ax::mojom::blink::Role::kCaption}, |
| {"cell", ax::mojom::blink::Role::kCell}, |
| {"code", ax::mojom::blink::Role::kCode}, |
| {"checkbox", ax::mojom::blink::Role::kCheckBox}, |
| {"columnheader", ax::mojom::blink::Role::kColumnHeader}, |
| {"combobox", ax::mojom::blink::Role::kComboBoxGrouping}, |
| {"comment", ax::mojom::blink::Role::kComment}, |
| {"complementary", ax::mojom::blink::Role::kComplementary}, |
| {"contentinfo", ax::mojom::blink::Role::kContentInfo}, |
| {"definition", ax::mojom::blink::Role::kDefinition}, |
| {"deletion", ax::mojom::blink::Role::kContentDeletion}, |
| {"dialog", ax::mojom::blink::Role::kDialog}, |
| {"directory", ax::mojom::blink::Role::kDirectory}, |
| // ------------------------------------------------- |
| // DPub Roles: |
| // www.w3.org/TR/dpub-aam-1.0/#mapping_role_table |
| {"doc-abstract", ax::mojom::blink::Role::kDocAbstract}, |
| {"doc-acknowledgments", ax::mojom::blink::Role::kDocAcknowledgments}, |
| {"doc-afterword", ax::mojom::blink::Role::kDocAfterword}, |
| {"doc-appendix", ax::mojom::blink::Role::kDocAppendix}, |
| {"doc-backlink", ax::mojom::blink::Role::kDocBackLink}, |
| {"doc-biblioentry", ax::mojom::blink::Role::kDocBiblioEntry}, |
| {"doc-bibliography", ax::mojom::blink::Role::kDocBibliography}, |
| {"doc-biblioref", ax::mojom::blink::Role::kDocBiblioRef}, |
| {"doc-chapter", ax::mojom::blink::Role::kDocChapter}, |
| {"doc-colophon", ax::mojom::blink::Role::kDocColophon}, |
| {"doc-conclusion", ax::mojom::blink::Role::kDocConclusion}, |
| {"doc-cover", ax::mojom::blink::Role::kDocCover}, |
| {"doc-credit", ax::mojom::blink::Role::kDocCredit}, |
| {"doc-credits", ax::mojom::blink::Role::kDocCredits}, |
| {"doc-dedication", ax::mojom::blink::Role::kDocDedication}, |
| {"doc-endnote", ax::mojom::blink::Role::kDocEndnote}, |
| {"doc-endnotes", ax::mojom::blink::Role::kDocEndnotes}, |
| {"doc-epigraph", ax::mojom::blink::Role::kDocEpigraph}, |
| {"doc-epilogue", ax::mojom::blink::Role::kDocEpilogue}, |
| {"doc-errata", ax::mojom::blink::Role::kDocErrata}, |
| {"doc-example", ax::mojom::blink::Role::kDocExample}, |
| {"doc-footnote", ax::mojom::blink::Role::kDocFootnote}, |
| {"doc-foreword", ax::mojom::blink::Role::kDocForeword}, |
| {"doc-glossary", ax::mojom::blink::Role::kDocGlossary}, |
| {"doc-glossref", ax::mojom::blink::Role::kDocGlossRef}, |
| {"doc-index", ax::mojom::blink::Role::kDocIndex}, |
| {"doc-introduction", ax::mojom::blink::Role::kDocIntroduction}, |
| {"doc-noteref", ax::mojom::blink::Role::kDocNoteRef}, |
| {"doc-notice", ax::mojom::blink::Role::kDocNotice}, |
| {"doc-pagebreak", ax::mojom::blink::Role::kDocPageBreak}, |
| {"doc-pagelist", ax::mojom::blink::Role::kDocPageList}, |
| {"doc-part", ax::mojom::blink::Role::kDocPart}, |
| {"doc-preface", ax::mojom::blink::Role::kDocPreface}, |
| {"doc-prologue", ax::mojom::blink::Role::kDocPrologue}, |
| {"doc-pullquote", ax::mojom::blink::Role::kDocPullquote}, |
| {"doc-qna", ax::mojom::blink::Role::kDocQna}, |
| {"doc-subtitle", ax::mojom::blink::Role::kDocSubtitle}, |
| {"doc-tip", ax::mojom::blink::Role::kDocTip}, |
| {"doc-toc", ax::mojom::blink::Role::kDocToc}, |
| // End DPub roles. |
| // ------------------------------------------------- |
| {"document", ax::mojom::blink::Role::kDocument}, |
| {"emphasis", ax::mojom::blink::Role::kEmphasis}, |
| {"feed", ax::mojom::blink::Role::kFeed}, |
| {"figure", ax::mojom::blink::Role::kFigure}, |
| {"form", ax::mojom::blink::Role::kForm}, |
| {"generic", ax::mojom::blink::Role::kGenericContainer}, |
| // ------------------------------------------------- |
| // ARIA Graphics module roles: |
| // https://rawgit.com/w3c/graphics-aam/master/ |
| {"graphics-document", ax::mojom::blink::Role::kGraphicsDocument}, |
| {"graphics-object", ax::mojom::blink::Role::kGraphicsObject}, |
| {"graphics-symbol", ax::mojom::blink::Role::kGraphicsSymbol}, |
| // End ARIA Graphics module roles. |
| // ------------------------------------------------- |
| {"grid", ax::mojom::blink::Role::kGrid}, |
| {"gridcell", ax::mojom::blink::Role::kCell}, |
| {"group", ax::mojom::blink::Role::kGroup}, |
| {"heading", ax::mojom::blink::Role::kHeading}, |
| {"img", ax::mojom::blink::Role::kImage}, |
| {"insertion", ax::mojom::blink::Role::kContentInsertion}, |
| {"link", ax::mojom::blink::Role::kLink}, |
| {"list", ax::mojom::blink::Role::kList}, |
| {"listbox", ax::mojom::blink::Role::kListBox}, |
| {"listitem", ax::mojom::blink::Role::kListItem}, |
| {"log", ax::mojom::blink::Role::kLog}, |
| {"main", ax::mojom::blink::Role::kMain}, |
| {"marquee", ax::mojom::blink::Role::kMarquee}, |
| {"math", ax::mojom::blink::Role::kMath}, |
| {"menu", ax::mojom::blink::Role::kMenu}, |
| {"menubar", ax::mojom::blink::Role::kMenuBar}, |
| {"menuitem", ax::mojom::blink::Role::kMenuItem}, |
| {"menuitemcheckbox", ax::mojom::blink::Role::kMenuItemCheckBox}, |
| {"menuitemradio", ax::mojom::blink::Role::kMenuItemRadio}, |
| {"mark", ax::mojom::blink::Role::kMark}, |
| {"meter", ax::mojom::blink::Role::kMeter}, |
| {"navigation", ax::mojom::blink::Role::kNavigation}, |
| {"none", ax::mojom::blink::Role::kNone}, |
| {"note", ax::mojom::blink::Role::kNote}, |
| {"option", ax::mojom::blink::Role::kListBoxOption}, |
| {"paragraph", ax::mojom::blink::Role::kParagraph}, |
| {"presentation", ax::mojom::blink::Role::kPresentational}, |
| {"progressbar", ax::mojom::blink::Role::kProgressIndicator}, |
| {"radio", ax::mojom::blink::Role::kRadioButton}, |
| {"radiogroup", ax::mojom::blink::Role::kRadioGroup}, |
| // TODO(accessibility) region should only be mapped |
| // if name present. See http://crbug.com/840819. |
| {"region", ax::mojom::blink::Role::kRegion}, |
| {"row", ax::mojom::blink::Role::kRow}, |
| {"rowgroup", ax::mojom::blink::Role::kRowGroup}, |
| {"rowheader", ax::mojom::blink::Role::kRowHeader}, |
| {"scrollbar", ax::mojom::blink::Role::kScrollBar}, |
| {"search", ax::mojom::blink::Role::kSearch}, |
| {"searchbox", ax::mojom::blink::Role::kSearchBox}, |
| {"separator", ax::mojom::blink::Role::kSplitter}, |
| {"slider", ax::mojom::blink::Role::kSlider}, |
| {"spinbutton", ax::mojom::blink::Role::kSpinButton}, |
| {"status", ax::mojom::blink::Role::kStatus}, |
| {"strong", ax::mojom::blink::Role::kStrong}, |
| {"suggestion", ax::mojom::blink::Role::kSuggestion}, |
| {"switch", ax::mojom::blink::Role::kSwitch}, |
| {"tab", ax::mojom::blink::Role::kTab}, |
| {"table", ax::mojom::blink::Role::kTable}, |
| {"tablist", ax::mojom::blink::Role::kTabList}, |
| {"tabpanel", ax::mojom::blink::Role::kTabPanel}, |
| {"term", ax::mojom::blink::Role::kTerm}, |
| {"text", ax::mojom::blink::Role::kStaticText}, |
| {"textbox", ax::mojom::blink::Role::kTextField}, |
| {"time", ax::mojom::blink::Role::kTime}, |
| {"timer", ax::mojom::blink::Role::kTimer}, |
| {"toolbar", ax::mojom::blink::Role::kToolbar}, |
| {"tooltip", ax::mojom::blink::Role::kTooltip}, |
| {"tree", ax::mojom::blink::Role::kTree}, |
| {"treegrid", ax::mojom::blink::Role::kTreeGrid}, |
| {"treeitem", ax::mojom::blink::Role::kTreeItem}}; |
| |
| struct InternalRoleEntry { |
| ax::mojom::blink::Role webcore_role; |
| const char* internal_role_name; |
| }; |
| |
| const InternalRoleEntry kInternalRoles[] = { |
| {ax::mojom::blink::Role::kNone, "None"}, |
| {ax::mojom::blink::Role::kAbbr, "Abbr"}, |
| {ax::mojom::blink::Role::kAlertDialog, "AlertDialog"}, |
| {ax::mojom::blink::Role::kAlert, "Alert"}, |
| {ax::mojom::blink::Role::kAnchor, "Anchor"}, |
| {ax::mojom::blink::Role::kComment, "Comment"}, |
| {ax::mojom::blink::Role::kApplication, "Application"}, |
| {ax::mojom::blink::Role::kArticle, "Article"}, |
| {ax::mojom::blink::Role::kAudio, "Audio"}, |
| {ax::mojom::blink::Role::kBanner, "Banner"}, |
| {ax::mojom::blink::Role::kBlockquote, "Blockquote"}, |
| {ax::mojom::blink::Role::kButton, "Button"}, |
| {ax::mojom::blink::Role::kCanvas, "Canvas"}, |
| {ax::mojom::blink::Role::kCaption, "Caption"}, |
| {ax::mojom::blink::Role::kCaret, "Caret"}, |
| {ax::mojom::blink::Role::kCell, "Cell"}, |
| {ax::mojom::blink::Role::kCheckBox, "CheckBox"}, |
| {ax::mojom::blink::Role::kClient, "Client"}, |
| {ax::mojom::blink::Role::kCode, "Code"}, |
| {ax::mojom::blink::Role::kColorWell, "ColorWell"}, |
| {ax::mojom::blink::Role::kColumnHeader, "ColumnHeader"}, |
| {ax::mojom::blink::Role::kColumn, "Column"}, |
| {ax::mojom::blink::Role::kComboBoxGrouping, "ComboBox"}, |
| {ax::mojom::blink::Role::kComboBoxMenuButton, "ComboBox"}, |
| {ax::mojom::blink::Role::kComplementary, "Complementary"}, |
| {ax::mojom::blink::Role::kContentDeletion, "ContentDeletion"}, |
| {ax::mojom::blink::Role::kContentInsertion, "ContentInsertion"}, |
| {ax::mojom::blink::Role::kContentInfo, "ContentInfo"}, |
| {ax::mojom::blink::Role::kDate, "Date"}, |
| {ax::mojom::blink::Role::kDateTime, "DateTime"}, |
| {ax::mojom::blink::Role::kDefinition, "Definition"}, |
| {ax::mojom::blink::Role::kDescriptionListDetail, "DescriptionListDetail"}, |
| {ax::mojom::blink::Role::kDescriptionList, "DescriptionList"}, |
| {ax::mojom::blink::Role::kDescriptionListTerm, "DescriptionListTerm"}, |
| {ax::mojom::blink::Role::kDesktop, "Desktop"}, |
| {ax::mojom::blink::Role::kDetails, "Details"}, |
| {ax::mojom::blink::Role::kDialog, "Dialog"}, |
| {ax::mojom::blink::Role::kDirectory, "Directory"}, |
| {ax::mojom::blink::Role::kDisclosureTriangle, "DisclosureTriangle"}, |
| // -------------------------------------------------------------- |
| // DPub Roles: |
| // https://www.w3.org/TR/dpub-aam-1.0/#mapping_role_table |
| {ax::mojom::blink::Role::kDocAbstract, "DocAbstract"}, |
| {ax::mojom::blink::Role::kDocAcknowledgments, "DocAcknowledgments"}, |
| {ax::mojom::blink::Role::kDocAfterword, "DocAfterword"}, |
| {ax::mojom::blink::Role::kDocAppendix, "DocAppendix"}, |
| {ax::mojom::blink::Role::kDocBackLink, "DocBackLink"}, |
| {ax::mojom::blink::Role::kDocBiblioEntry, "DocBiblioentry"}, |
| {ax::mojom::blink::Role::kDocBibliography, "DocBibliography"}, |
| {ax::mojom::blink::Role::kDocBiblioRef, "DocBiblioref"}, |
| {ax::mojom::blink::Role::kDocChapter, "DocChapter"}, |
| {ax::mojom::blink::Role::kDocColophon, "DocColophon"}, |
| {ax::mojom::blink::Role::kDocConclusion, "DocConclusion"}, |
| {ax::mojom::blink::Role::kDocCover, "DocCover"}, |
| {ax::mojom::blink::Role::kDocCredit, "DocCredit"}, |
| {ax::mojom::blink::Role::kDocCredits, "DocCredits"}, |
| {ax::mojom::blink::Role::kDocDedication, "DocDedication"}, |
| {ax::mojom::blink::Role::kDocEndnote, "DocEndnote"}, |
| {ax::mojom::blink::Role::kDocEndnotes, "DocEndnotes"}, |
| {ax::mojom::blink::Role::kDocEpigraph, "DocEpigraph"}, |
| {ax::mojom::blink::Role::kDocEpilogue, "DocEpilogue"}, |
| {ax::mojom::blink::Role::kDocErrata, "DocErrata"}, |
| {ax::mojom::blink::Role::kDocExample, "DocExample"}, |
| {ax::mojom::blink::Role::kDocFootnote, "DocFootnote"}, |
| {ax::mojom::blink::Role::kDocForeword, "DocForeword"}, |
| {ax::mojom::blink::Role::kDocGlossary, "DocGlossary"}, |
| {ax::mojom::blink::Role::kDocGlossRef, "DocGlossref"}, |
| {ax::mojom::blink::Role::kDocIndex, "DocIndex"}, |
| {ax::mojom::blink::Role::kDocIntroduction, "DocIntroduction"}, |
| {ax::mojom::blink::Role::kDocNoteRef, "DocNoteref"}, |
| {ax::mojom::blink::Role::kDocNotice, "DocNotice"}, |
| {ax::mojom::blink::Role::kDocPageBreak, "DocPagebreak"}, |
| {ax::mojom::blink::Role::kDocPageList, "DocPagelist"}, |
| {ax::mojom::blink::Role::kDocPart, "DocPart"}, |
| {ax::mojom::blink::Role::kDocPreface, "DocPreface"}, |
| {ax::mojom::blink::Role::kDocPrologue, "DocPrologue"}, |
| {ax::mojom::blink::Role::kDocPullquote, "DocPullquote"}, |
| {ax::mojom::blink::Role::kDocQna, "DocQna"}, |
| {ax::mojom::blink::Role::kDocSubtitle, "DocSubtitle"}, |
| {ax::mojom::blink::Role::kDocTip, "DocTip"}, |
| {ax::mojom::blink::Role::kDocToc, "DocToc"}, |
| // End DPub roles. |
| // -------------------------------------------------------------- |
| {ax::mojom::blink::Role::kDocument, "Document"}, |
| {ax::mojom::blink::Role::kEmbeddedObject, "EmbeddedObject"}, |
| {ax::mojom::blink::Role::kEmphasis, "Emphasis"}, |
| {ax::mojom::blink::Role::kFeed, "feed"}, |
| {ax::mojom::blink::Role::kFigcaption, "Figcaption"}, |
| {ax::mojom::blink::Role::kFigure, "Figure"}, |
| {ax::mojom::blink::Role::kFooter, "Footer"}, |
| {ax::mojom::blink::Role::kFooterAsNonLandmark, "FooterAsNonLandmark"}, |
| {ax::mojom::blink::Role::kForm, "Form"}, |
| {ax::mojom::blink::Role::kGenericContainer, "GenericContainer"}, |
| // -------------------------------------------------------------- |
| // ARIA Graphics module roles: |
| // https://rawgit.com/w3c/graphics-aam/master/#mapping_role_table |
| {ax::mojom::blink::Role::kGraphicsDocument, "GraphicsDocument"}, |
| {ax::mojom::blink::Role::kGraphicsObject, "GraphicsObject"}, |
| {ax::mojom::blink::Role::kGraphicsSymbol, "GraphicsSymbol"}, |
| // End ARIA Graphics module roles. |
| // -------------------------------------------------------------- |
| {ax::mojom::blink::Role::kGrid, "Grid"}, |
| {ax::mojom::blink::Role::kGroup, "Group"}, |
| {ax::mojom::blink::Role::kHeader, "Header"}, |
| {ax::mojom::blink::Role::kHeaderAsNonLandmark, "HeaderAsNonLandmark"}, |
| {ax::mojom::blink::Role::kHeading, "Heading"}, |
| {ax::mojom::blink::Role::kIframePresentational, "IframePresentational"}, |
| {ax::mojom::blink::Role::kIframe, "Iframe"}, |
| {ax::mojom::blink::Role::kIgnored, "Ignored"}, |
| {ax::mojom::blink::Role::kImageMap, "ImageMap"}, |
| {ax::mojom::blink::Role::kImage, "Image"}, |
| {ax::mojom::blink::Role::kImeCandidate, "ImeCandidate"}, |
| {ax::mojom::blink::Role::kInlineTextBox, "InlineTextBox"}, |
| {ax::mojom::blink::Role::kInputTime, "InputTime"}, |
| {ax::mojom::blink::Role::kKeyboard, "Keyboard"}, |
| {ax::mojom::blink::Role::kLabelText, "Label"}, |
| {ax::mojom::blink::Role::kLayoutTable, "LayoutTable"}, |
| {ax::mojom::blink::Role::kLayoutTableCell, "LayoutCellTable"}, |
| {ax::mojom::blink::Role::kLayoutTableRow, "LayoutRowTable"}, |
| {ax::mojom::blink::Role::kLegend, "Legend"}, |
| {ax::mojom::blink::Role::kLink, "Link"}, |
| {ax::mojom::blink::Role::kLineBreak, "LineBreak"}, |
| {ax::mojom::blink::Role::kListBox, "ListBox"}, |
| {ax::mojom::blink::Role::kListBoxOption, "ListBoxOption"}, |
| {ax::mojom::blink::Role::kListGrid, "ListGrid"}, |
| {ax::mojom::blink::Role::kListItem, "ListItem"}, |
| {ax::mojom::blink::Role::kListMarker, "ListMarker"}, |
| {ax::mojom::blink::Role::kList, "List"}, |
| {ax::mojom::blink::Role::kLog, "Log"}, |
| {ax::mojom::blink::Role::kMain, "Main"}, |
| {ax::mojom::blink::Role::kMark, "Mark"}, |
| {ax::mojom::blink::Role::kMarquee, "Marquee"}, |
| {ax::mojom::blink::Role::kMath, "Math"}, |
| {ax::mojom::blink::Role::kMenuBar, "MenuBar"}, |
| {ax::mojom::blink::Role::kMenuButton, "MenuButton"}, |
| {ax::mojom::blink::Role::kMenuItem, "MenuItem"}, |
| {ax::mojom::blink::Role::kMenuItemCheckBox, "MenuItemCheckBox"}, |
| {ax::mojom::blink::Role::kMenuItemRadio, "MenuItemRadio"}, |
| {ax::mojom::blink::Role::kMenuListOption, "MenuListOption"}, |
| {ax::mojom::blink::Role::kMenuListPopup, "MenuListPopup"}, |
| {ax::mojom::blink::Role::kMenu, "Menu"}, |
| {ax::mojom::blink::Role::kMeter, "Meter"}, |
| {ax::mojom::blink::Role::kNavigation, "Navigation"}, |
| {ax::mojom::blink::Role::kNote, "Note"}, |
| {ax::mojom::blink::Role::kPane, "Pane"}, |
| {ax::mojom::blink::Role::kParagraph, "Paragraph"}, |
| {ax::mojom::blink::Role::kPdfActionableHighlight, "PdfActionableHighlight"}, |
| {ax::mojom::blink::Role::kPluginObject, "PluginObject"}, |
| {ax::mojom::blink::Role::kPopUpButton, "PopUpButton"}, |
| {ax::mojom::blink::Role::kPortal, "Portal"}, |
| {ax::mojom::blink::Role::kPre, "Pre"}, |
| {ax::mojom::blink::Role::kPresentational, "Presentational"}, |
| {ax::mojom::blink::Role::kProgressIndicator, "ProgressIndicator"}, |
| {ax::mojom::blink::Role::kRadioButton, "RadioButton"}, |
| {ax::mojom::blink::Role::kRadioGroup, "RadioGroup"}, |
| {ax::mojom::blink::Role::kRegion, "Region"}, |
| {ax::mojom::blink::Role::kRootWebArea, "WebArea"}, |
| {ax::mojom::blink::Role::kRow, "Row"}, |
| {ax::mojom::blink::Role::kRowGroup, "RowGroup"}, |
| {ax::mojom::blink::Role::kRowHeader, "RowHeader"}, |
| {ax::mojom::blink::Role::kRuby, "Ruby"}, |
| {ax::mojom::blink::Role::kRubyAnnotation, "RubyAnnotation"}, |
| {ax::mojom::blink::Role::kSection, "Section"}, |
| {ax::mojom::blink::Role::kSvgRoot, "SVGRoot"}, |
| {ax::mojom::blink::Role::kScrollBar, "ScrollBar"}, |
| {ax::mojom::blink::Role::kScrollView, "ScrollView"}, |
| {ax::mojom::blink::Role::kSearch, "Search"}, |
| {ax::mojom::blink::Role::kSearchBox, "SearchBox"}, |
| {ax::mojom::blink::Role::kSlider, "Slider"}, |
| {ax::mojom::blink::Role::kSliderThumb, "SliderThumb"}, |
| {ax::mojom::blink::Role::kSpinButton, "SpinButton"}, |
| {ax::mojom::blink::Role::kSplitter, "Splitter"}, |
| {ax::mojom::blink::Role::kStaticText, "StaticText"}, |
| {ax::mojom::blink::Role::kStatus, "Status"}, |
| {ax::mojom::blink::Role::kStrong, "Strong"}, |
| {ax::mojom::blink::Role::kSuggestion, "Suggestion"}, |
| {ax::mojom::blink::Role::kSwitch, "Switch"}, |
| {ax::mojom::blink::Role::kTab, "Tab"}, |
| {ax::mojom::blink::Role::kTabList, "TabList"}, |
| {ax::mojom::blink::Role::kTabPanel, "TabPanel"}, |
| {ax::mojom::blink::Role::kTable, "Table"}, |
| {ax::mojom::blink::Role::kTableHeaderContainer, "TableHeaderContainer"}, |
| {ax::mojom::blink::Role::kTerm, "Term"}, |
| {ax::mojom::blink::Role::kTextField, "TextField"}, |
| {ax::mojom::blink::Role::kTextFieldWithComboBox, "ComboBox"}, |
| {ax::mojom::blink::Role::kTime, "Time"}, |
| {ax::mojom::blink::Role::kTimer, "Timer"}, |
| {ax::mojom::blink::Role::kTitleBar, "TitleBar"}, |
| {ax::mojom::blink::Role::kToggleButton, "ToggleButton"}, |
| {ax::mojom::blink::Role::kToolbar, "Toolbar"}, |
| {ax::mojom::blink::Role::kTreeGrid, "TreeGrid"}, |
| {ax::mojom::blink::Role::kTreeItem, "TreeItem"}, |
| {ax::mojom::blink::Role::kTree, "Tree"}, |
| {ax::mojom::blink::Role::kTooltip, "UserInterfaceTooltip"}, |
| {ax::mojom::blink::Role::kUnknown, "Unknown"}, |
| {ax::mojom::blink::Role::kVideo, "Video"}, |
| {ax::mojom::blink::Role::kWebArea, "WebArea"}, |
| {ax::mojom::blink::Role::kWebView, "WebView"}, |
| {ax::mojom::blink::Role::kWindow, "Window"}}; |
| |
| static_assert(base::size(kInternalRoles) == |
| static_cast<size_t>(ax::mojom::blink::Role::kMaxValue) + 1, |
| "Not all internal roles have an entry in internalRoles array"); |
| |
| // Roles which we need to map in the other direction |
| const RoleEntry kReverseRoles[] = { |
| {"banner", ax::mojom::blink::Role::kHeader}, |
| {"button", ax::mojom::blink::Role::kToggleButton}, |
| {"combobox", ax::mojom::blink::Role::kPopUpButton}, |
| {"contentinfo", ax::mojom::blink::Role::kFooter}, |
| {"menuitem", ax::mojom::blink::Role::kMenuButton}, |
| {"menuitem", ax::mojom::blink::Role::kMenuListOption}, |
| {"progressbar", ax::mojom::blink::Role::kMeter}, |
| {"region", ax::mojom::blink::Role::kSection}, |
| {"textbox", ax::mojom::blink::Role::kTextField}, |
| {"combobox", ax::mojom::blink::Role::kComboBoxMenuButton}, |
| {"combobox", ax::mojom::blink::Role::kTextFieldWithComboBox}}; |
| |
| static ARIARoleMap* CreateARIARoleMap() { |
| ARIARoleMap* role_map = new ARIARoleMap; |
| |
| for (size_t i = 0; i < base::size(kRoles); ++i) |
| role_map->Set(String(kRoles[i].aria_role), kRoles[i].webcore_role); |
| |
| return role_map; |
| } |
| |
| static Vector<AtomicString>* CreateRoleNameVector() { |
| Vector<AtomicString>* role_name_vector = |
| new Vector<AtomicString>(base::size(kInternalRoles)); |
| for (wtf_size_t i = 0; i < base::size(kInternalRoles); i++) |
| (*role_name_vector)[i] = g_null_atom; |
| |
| for (wtf_size_t i = 0; i < base::size(kRoles); ++i) { |
| (*role_name_vector)[static_cast<wtf_size_t>(kRoles[i].webcore_role)] = |
| AtomicString(kRoles[i].aria_role); |
| } |
| |
| for (wtf_size_t i = 0; i < base::size(kReverseRoles); ++i) { |
| (*role_name_vector)[static_cast<wtf_size_t>( |
| kReverseRoles[i].webcore_role)] = |
| AtomicString(kReverseRoles[i].aria_role); |
| } |
| |
| return role_name_vector; |
| } |
| |
| static Vector<AtomicString>* CreateInternalRoleNameVector() { |
| Vector<AtomicString>* internal_role_name_vector = |
| new Vector<AtomicString>(base::size(kInternalRoles)); |
| for (wtf_size_t i = 0; i < base::size(kInternalRoles); i++) { |
| (*internal_role_name_vector)[static_cast<wtf_size_t>( |
| kInternalRoles[i].webcore_role)] = |
| AtomicString(kInternalRoles[i].internal_role_name); |
| } |
| |
| return internal_role_name_vector; |
| } |
| |
| HTMLDialogElement* GetActiveDialogElement(Node* node) { |
| return node->GetDocument().ActiveModalDialog(); |
| } |
| |
| } // namespace |
| |
| unsigned AXObject::number_of_live_ax_objects_ = 0; |
| |
| AXObject::AXObject(AXObjectCacheImpl& ax_object_cache) |
| : id_(0), |
| have_children_(false), |
| role_(ax::mojom::blink::Role::kUnknown), |
| aria_role_(ax::mojom::blink::Role::kUnknown), |
| last_known_is_ignored_value_(kDefaultBehavior), |
| last_known_is_ignored_but_included_in_tree_value_(kDefaultBehavior), |
| explicit_container_id_(0), |
| parent_(nullptr), |
| last_modification_count_(-1), |
| cached_is_ignored_(false), |
| cached_is_ignored_but_included_in_tree_(false), |
| cached_is_inert_or_aria_hidden_(false), |
| cached_is_descendant_of_leaf_node_(false), |
| cached_is_descendant_of_disabled_node_(false), |
| cached_has_inherited_presentational_role_(false), |
| cached_is_editable_root_(false), |
| cached_live_region_root_(nullptr), |
| cached_aria_column_index_(0), |
| cached_aria_row_index_(0), |
| ax_object_cache_(&ax_object_cache) { |
| ++number_of_live_ax_objects_; |
| } |
| |
| AXObject::~AXObject() { |
| DCHECK(IsDetached()); |
| --number_of_live_ax_objects_; |
| } |
| |
| void AXObject::Init() { |
| role_ = DetermineAccessibilityRole(); |
| } |
| |
| void AXObject::Detach() { |
| // Clear any children and call detachFromParent on them so that |
| // no children are left with dangling pointers to their parent. |
| ClearChildren(); |
| |
| ax_object_cache_ = nullptr; |
| } |
| |
| bool AXObject::IsDetached() const { |
| return !ax_object_cache_; |
| } |
| |
| const AtomicString& AXObject::GetAOMPropertyOrARIAAttribute( |
| AOMStringProperty property) const { |
| Element* element = this->GetElement(); |
| if (!element) |
| return g_null_atom; |
| |
| return AccessibleNode::GetPropertyOrARIAAttribute(element, property); |
| } |
| |
| Element* AXObject::GetAOMPropertyOrARIAAttribute( |
| AOMRelationProperty property) const { |
| Element* element = this->GetElement(); |
| if (!element) |
| return nullptr; |
| |
| return AccessibleNode::GetPropertyOrARIAAttribute(element, property); |
| } |
| |
| bool AXObject::HasAOMProperty(AOMRelationListProperty property, |
| HeapVector<Member<Element>>& result) const { |
| Element* element = this->GetElement(); |
| if (!element) |
| return false; |
| |
| return AccessibleNode::GetProperty(element, property, result); |
| } |
| |
| bool AXObject::HasAOMPropertyOrARIAAttribute( |
| AOMRelationListProperty property, |
| HeapVector<Member<Element>>& result) const { |
| Element* element = this->GetElement(); |
| if (!element) |
| return false; |
| |
| return AccessibleNode::GetPropertyOrARIAAttribute(element, property, result); |
| } |
| |
| bool AXObject::HasAOMPropertyOrARIAAttribute(AOMBooleanProperty property, |
| bool& result) const { |
| Element* element = this->GetElement(); |
| if (!element) |
| return false; |
| |
| bool is_null = true; |
| result = |
| AccessibleNode::GetPropertyOrARIAAttribute(element, property, is_null); |
| return !is_null; |
| } |
| |
| bool AXObject::AOMPropertyOrARIAAttributeIsTrue( |
| AOMBooleanProperty property) const { |
| bool result; |
| if (HasAOMPropertyOrARIAAttribute(property, result)) |
| return result; |
| return false; |
| } |
| |
| bool AXObject::AOMPropertyOrARIAAttributeIsFalse( |
| AOMBooleanProperty property) const { |
| bool result; |
| if (HasAOMPropertyOrARIAAttribute(property, result)) |
| return !result; |
| return false; |
| } |
| |
| bool AXObject::HasAOMPropertyOrARIAAttribute(AOMUIntProperty property, |
| uint32_t& result) const { |
| Element* element = this->GetElement(); |
| if (!element) |
| return false; |
| |
| bool is_null = true; |
| result = |
| AccessibleNode::GetPropertyOrARIAAttribute(element, property, is_null); |
| return !is_null; |
| } |
| |
| bool AXObject::HasAOMPropertyOrARIAAttribute(AOMIntProperty property, |
| int32_t& result) const { |
| Element* element = this->GetElement(); |
| if (!element) |
| return false; |
| |
| bool is_null = true; |
| result = |
| AccessibleNode::GetPropertyOrARIAAttribute(element, property, is_null); |
| return !is_null; |
| } |
| |
| bool AXObject::HasAOMPropertyOrARIAAttribute(AOMFloatProperty property, |
| float& result) const { |
| Element* element = this->GetElement(); |
| if (!element) |
| return false; |
| |
| bool is_null = true; |
| result = |
| AccessibleNode::GetPropertyOrARIAAttribute(element, property, is_null); |
| return !is_null; |
| } |
| |
| bool AXObject::HasAOMPropertyOrARIAAttribute(AOMStringProperty property, |
| AtomicString& result) const { |
| Element* element = this->GetElement(); |
| if (!element) |
| return false; |
| |
| result = AccessibleNode::GetPropertyOrARIAAttribute(element, property); |
| return !result.IsNull(); |
| } |
| |
| AccessibleNode* AXObject::GetAccessibleNode() const { |
| Element* element = GetElement(); |
| if (!element) |
| return nullptr; |
| |
| return element->ExistingAccessibleNode(); |
| } |
| |
| void AXObject::GetSparseAXAttributes( |
| AXSparseAttributeClient& sparse_attribute_client) const { |
| AXSparseAttributeAOMPropertyClient property_client(*ax_object_cache_, |
| sparse_attribute_client); |
| HashSet<QualifiedName> shadowed_aria_attributes; |
| AccessibleNode* accessible_node = GetAccessibleNode(); |
| |
| // Virtual nodes for AOM are still tied to the AXTree. |
| if (accessible_node && IsVirtualObject()) |
| accessible_node->GetAllAOMProperties(&property_client, |
| shadowed_aria_attributes); |
| |
| Element* element = GetElement(); |
| if (!element) |
| return; |
| |
| AXSparseAttributeSetterMap& ax_sparse_attribute_setter_map = |
| GetSparseAttributeSetterMap(); |
| AttributeCollection attributes = element->AttributesWithoutUpdate(); |
| HashSet<QualifiedName> set_attributes; |
| for (const Attribute& attr : attributes) { |
| set_attributes.insert(attr.GetName()); |
| if (shadowed_aria_attributes.Contains(attr.GetName())) |
| continue; |
| |
| AXSparseAttributeSetter* setter = |
| ax_sparse_attribute_setter_map.at(attr.GetName()); |
| if (setter) |
| setter->Run(*this, sparse_attribute_client, attr.Value()); |
| } |
| if (!element->DidAttachInternals()) |
| return; |
| const auto& internals_attributes = |
| element->EnsureElementInternals().GetAttributes(); |
| for (const QualifiedName& attr : internals_attributes.Keys()) { |
| if (set_attributes.Contains(attr)) |
| continue; |
| AXSparseAttributeSetter* setter = ax_sparse_attribute_setter_map.at(attr); |
| if (setter) { |
| setter->Run(*this, sparse_attribute_client, |
| internals_attributes.at(attr)); |
| } |
| } |
| } |
| |
| void AXObject::Serialize(ui::AXNodeData* node_data) { |
| AccessibilityExpanded expanded = IsExpanded(); |
| if (expanded) { |
| if (expanded == kExpandedCollapsed) |
| node_data->AddState(ax::mojom::blink::State::kCollapsed); |
| else if (expanded == kExpandedExpanded) |
| node_data->AddState(ax::mojom::blink::State::kExpanded); |
| } |
| |
| if (CanSetFocusAttribute()) |
| node_data->AddState(ax::mojom::blink::State::kFocusable); |
| |
| if (HasPopup() != ax::mojom::blink::HasPopup::kFalse) |
| node_data->SetHasPopup(HasPopup()); |
| else if (RoleValue() == ax::mojom::blink::Role::kPopUpButton) |
| node_data->SetHasPopup(ax::mojom::blink::HasPopup::kMenu); |
| |
| if (IsAutofillAvailable()) |
| node_data->AddState(ax::mojom::blink::State::kAutofillAvailable); |
| |
| if (IsDefault()) |
| node_data->AddState(ax::mojom::blink::State::kDefault); |
| |
| // aria-grabbed is deprecated in WAI-ARIA 1.1. |
| if (IsGrabbed() != kGrabbedStateUndefined) { |
| node_data->AddBoolAttribute(ax::mojom::blink::BoolAttribute::kGrabbed, |
| IsGrabbed() == kGrabbedStateTrue); |
| } |
| |
| if (IsHovered()) |
| node_data->AddState(ax::mojom::blink::State::kHovered); |
| |
| if (!IsVisible()) |
| node_data->AddState(ax::mojom::blink::State::kInvisible); |
| |
| if (IsLinked()) |
| node_data->AddState(ax::mojom::blink::State::kLinked); |
| |
| if (IsMultiline()) |
| node_data->AddState(ax::mojom::blink::State::kMultiline); |
| |
| if (IsMultiSelectable()) |
| node_data->AddState(ax::mojom::blink::State::kMultiselectable); |
| |
| if (IsPasswordField()) |
| node_data->AddState(ax::mojom::blink::State::kProtected); |
| |
| if (IsRequired()) |
| node_data->AddState(ax::mojom::blink::State::kRequired); |
| |
| if (IsEditable()) |
| node_data->AddState(ax::mojom::blink::State::kEditable); |
| |
| if (IsSelected() != blink::kSelectedStateUndefined) { |
| node_data->AddBoolAttribute(ax::mojom::blink::BoolAttribute::kSelected, |
| IsSelected() == blink::kSelectedStateTrue); |
| node_data->AddBoolAttribute( |
| ax::mojom::blink::BoolAttribute::kSelectedFromFocus, |
| IsSelectedFromFocus()); |
| } |
| |
| if (IsRichlyEditable()) |
| node_data->AddState(ax::mojom::blink::State::kRichlyEditable); |
| |
| if (IsVisited()) |
| node_data->AddState(ax::mojom::blink::State::kVisited); |
| |
| if (Orientation() == kAccessibilityOrientationVertical) |
| node_data->AddState(ax::mojom::blink::State::kVertical); |
| else if (Orientation() == blink::kAccessibilityOrientationHorizontal) |
| node_data->AddState(ax::mojom::blink::State::kHorizontal); |
| |
| if (AccessibilityIsIgnored()) |
| node_data->AddState(ax::mojom::blink::State::kIgnored); |
| |
| if (GetTextAlign() != ax::mojom::blink::TextAlign::kNone) { |
| node_data->SetTextAlign(GetTextAlign()); |
| } |
| |
| // If this is an HTMLFrameOwnerElement (such as an iframe), we may need |
| // to embed the ID of the child frame. |
| if (auto* html_frame_owner_element = |
| DynamicTo<HTMLFrameOwnerElement>(GetElement())) { |
| if (Frame* child_frame = html_frame_owner_element->ContentFrame()) { |
| base::Optional<base::UnguessableToken> child_token = |
| child_frame->GetEmbeddingToken(); |
| if (child_token && !(IsDetached() || ChildCountIncludingIgnored())) { |
| node_data->AddStringAttribute( |
| ax::mojom::blink::StringAttribute::kChildTreeId, |
| child_token->ToString()); |
| } |
| } |
| } |
| } |
| |
| bool AXObject::IsAXNodeObject() const { |
| return false; |
| } |
| |
| bool AXObject::IsAXLayoutObject() const { |
| return false; |
| } |
| |
| bool AXObject::IsAXInlineTextBox() const { |
| return false; |
| } |
| |
| bool AXObject::IsList() const { |
| return ui::IsList(RoleValue()); |
| } |
| |
| bool AXObject::IsAXListBox() const { |
| return false; |
| } |
| |
| bool AXObject::IsAXListBoxOption() const { |
| return false; |
| } |
| |
| bool AXObject::IsMenuList() const { |
| return false; |
| } |
| |
| bool AXObject::IsMenuListOption() const { |
| return false; |
| } |
| |
| bool AXObject::IsMenuListPopup() const { |
| return false; |
| } |
| |
| bool AXObject::IsMockObject() const { |
| return false; |
| } |
| |
| bool AXObject::IsProgressIndicator() const { |
| return false; |
| } |
| |
| bool AXObject::IsAXRadioInput() const { |
| return false; |
| } |
| |
| bool AXObject::IsSlider() const { |
| return false; |
| } |
| |
| bool AXObject::IsAXSVGRoot() const { |
| return false; |
| } |
| |
| bool AXObject::IsValidationMessage() const { |
| return false; |
| } |
| |
| bool AXObject::IsVirtualObject() const { |
| return false; |
| } |
| |
| ax::mojom::blink::Role AXObject::RoleValue() const { |
| return role_; |
| } |
| |
| bool AXObject::IsARIATextControl() const { |
| return AriaRoleAttribute() == ax::mojom::blink::Role::kTextField || |
| AriaRoleAttribute() == ax::mojom::blink::Role::kSearchBox || |
| AriaRoleAttribute() == ax::mojom::blink::Role::kTextFieldWithComboBox; |
| } |
| |
| bool AXObject::IsAnchor() const { |
| return IsLink() && !IsNativeImage(); |
| } |
| |
| bool AXObject::IsButton() const { |
| return ui::IsButton(RoleValue()); |
| } |
| |
| bool AXObject::IsCanvas() const { |
| return RoleValue() == ax::mojom::blink::Role::kCanvas; |
| } |
| |
| bool AXObject::IsCheckbox() const { |
| return RoleValue() == ax::mojom::blink::Role::kCheckBox; |
| } |
| |
| bool AXObject::IsCheckboxOrRadio() const { |
| return IsCheckbox() || IsRadioButton(); |
| } |
| |
| bool AXObject::IsColorWell() const { |
| return RoleValue() == ax::mojom::blink::Role::kColorWell; |
| } |
| |
| bool AXObject::IsControl() const { |
| return ui::IsControl(RoleValue()); |
| } |
| |
| bool AXObject::IsDefault() const { |
| return false; |
| } |
| |
| bool AXObject::IsFieldset() const { |
| return false; |
| } |
| |
| bool AXObject::IsHeading() const { |
| return ui::IsHeading(RoleValue()); |
| } |
| |
| bool AXObject::IsImage() const { |
| // Canvas is not currently included so that it is not exposed unless there is |
| // a label, fallback content or something to make it accessible. This decision |
| // may be revisited at a later date. |
| return ui::IsImage(RoleValue()) && |
| RoleValue() != ax::mojom::blink::Role::kCanvas; |
| } |
| |
| bool AXObject::IsInputImage() const { |
| return false; |
| } |
| |
| bool AXObject::IsLink() const { |
| return ui::IsLink(RoleValue()); |
| } |
| |
| bool AXObject::IsInPageLinkTarget() const { |
| return false; |
| } |
| |
| bool AXObject::IsImageMapLink() const { |
| return false; |
| } |
| |
| bool AXObject::IsMenu() const { |
| return RoleValue() == ax::mojom::blink::Role::kMenu; |
| } |
| |
| bool AXObject::IsMenuButton() const { |
| return RoleValue() == ax::mojom::blink::Role::kMenuButton; |
| } |
| |
| bool AXObject::IsCheckable() const { |
| switch (RoleValue()) { |
| case ax::mojom::blink::Role::kCheckBox: |
| case ax::mojom::blink::Role::kMenuItemCheckBox: |
| case ax::mojom::blink::Role::kMenuItemRadio: |
| case ax::mojom::blink::Role::kRadioButton: |
| case ax::mojom::blink::Role::kSwitch: |
| case ax::mojom::blink::Role::kToggleButton: |
| return true; |
| case ax::mojom::blink::Role::kTreeItem: |
| case ax::mojom::blink::Role::kListBoxOption: |
| case ax::mojom::blink::Role::kMenuListOption: |
| return AriaCheckedIsPresent(); |
| default: |
| return false; |
| } |
| } |
| |
| // Why this is here instead of AXNodeObject: |
| // Because an AXMenuListOption (<option>) can |
| // have an ARIA role of menuitemcheckbox/menuitemradio |
| // yet does not inherit from AXNodeObject |
| ax::mojom::blink::CheckedState AXObject::CheckedState() const { |
| if (!IsCheckable()) |
| return ax::mojom::blink::CheckedState::kNone; |
| |
| // Try ARIA checked/pressed state |
| const ax::mojom::blink::Role role = RoleValue(); |
| const auto prop = role == ax::mojom::blink::Role::kToggleButton |
| ? AOMStringProperty::kPressed |
| : AOMStringProperty::kChecked; |
| const AtomicString& checked_attribute = GetAOMPropertyOrARIAAttribute(prop); |
| if (checked_attribute) { |
| if (EqualIgnoringASCIICase(checked_attribute, "mixed")) { |
| // Only checkable role that doesn't support mixed is the switch. |
| if (role != ax::mojom::blink::Role::kSwitch) |
| return ax::mojom::blink::CheckedState::kMixed; |
| } |
| |
| // Anything other than "false" should be treated as "true". |
| return EqualIgnoringASCIICase(checked_attribute, "false") |
| ? ax::mojom::blink::CheckedState::kFalse |
| : ax::mojom::blink::CheckedState::kTrue; |
| } |
| |
| // Native checked state |
| if (role != ax::mojom::blink::Role::kToggleButton) { |
| const Node* node = this->GetNode(); |
| if (!node) |
| return ax::mojom::blink::CheckedState::kNone; |
| |
| // Expose native checkbox mixed state as accessibility mixed state. However, |
| // do not expose native radio mixed state as accessibility mixed state. |
| // This would confuse the JAWS screen reader, which reports a mixed radio as |
| // both checked and partially checked, but a native mixed native radio |
| // button sinply means no radio buttons have been checked in the group yet. |
| if (IsNativeCheckboxInMixedState(node)) |
| return ax::mojom::blink::CheckedState::kMixed; |
| |
| auto* html_input_element = DynamicTo<HTMLInputElement>(node); |
| if (html_input_element && html_input_element->ShouldAppearChecked()) { |
| return ax::mojom::blink::CheckedState::kTrue; |
| } |
| } |
| |
| return ax::mojom::blink::CheckedState::kFalse; |
| } |
| |
| bool AXObject::IsNativeCheckboxInMixedState(const Node* node) { |
| const auto* input = DynamicTo<HTMLInputElement>(node); |
| if (!input) |
| return false; |
| |
| const auto inputType = input->type(); |
| if (inputType != input_type_names::kCheckbox) |
| return false; |
| return input->ShouldAppearIndeterminate(); |
| } |
| |
| bool AXObject::IsLandmarkRelated() const { |
| switch (RoleValue()) { |
| case ax::mojom::blink::Role::kApplication: |
| case ax::mojom::blink::Role::kArticle: |
| case ax::mojom::blink::Role::kBanner: |
| case ax::mojom::blink::Role::kComplementary: |
| case ax::mojom::blink::Role::kContentInfo: |
| case ax::mojom::blink::Role::kDocAcknowledgments: |
| case ax::mojom::blink::Role::kDocAfterword: |
| case ax::mojom::blink::Role::kDocAppendix: |
| case ax::mojom::blink::Role::kDocBibliography: |
| case ax::mojom::blink::Role::kDocChapter: |
| case ax::mojom::blink::Role::kDocConclusion: |
| case ax::mojom::blink::Role::kDocCredits: |
| case ax::mojom::blink::Role::kDocEndnotes: |
| case ax::mojom::blink::Role::kDocEpilogue: |
| case ax::mojom::blink::Role::kDocErrata: |
| case ax::mojom::blink::Role::kDocForeword: |
| case ax::mojom::blink::Role::kDocGlossary: |
| case ax::mojom::blink::Role::kDocIntroduction: |
| case ax::mojom::blink::Role::kDocPart: |
| case ax::mojom::blink::Role::kDocPreface: |
| case ax::mojom::blink::Role::kDocPrologue: |
| case ax::mojom::blink::Role::kDocToc: |
| case ax::mojom::blink::Role::kFooter: |
| case ax::mojom::blink::Role::kForm: |
| case ax::mojom::blink::Role::kHeader: |
| case ax::mojom::blink::Role::kMain: |
| case ax::mojom::blink::Role::kNavigation: |
| case ax::mojom::blink::Role::kRegion: |
| case ax::mojom::blink::Role::kSearch: |
| case ax::mojom::blink::Role::kSection: |
| return true; |
| default: |
| return false; |
| } |
| } |
| |
| bool AXObject::IsMenuRelated() const { |
| return ui::IsMenuRelated(RoleValue()); |
| } |
| |
| bool AXObject::IsMeter() const { |
| return RoleValue() == ax::mojom::blink::Role::kMeter; |
| } |
| |
| bool AXObject::IsNativeImage() const { |
| return false; |
| } |
| |
| bool AXObject::IsNativeSpinButton() const { |
| return false; |
| } |
| |
| bool AXObject::IsNativeTextControl() const { |
| return false; |
| } |
| |
| bool AXObject::IsNonNativeTextControl() const { |
| return false; |
| } |
| |
| bool AXObject::IsPasswordField() const { |
| return false; |
| } |
| |
| bool AXObject::IsPasswordFieldAndShouldHideValue() const { |
| Settings* settings = GetDocument()->GetSettings(); |
| if (!settings || settings->GetAccessibilityPasswordValuesEnabled()) |
| return false; |
| |
| return IsPasswordField(); |
| } |
| |
| bool AXObject::IsPresentational() const { |
| return ui::IsPresentational(RoleValue()); |
| } |
| |
| bool AXObject::IsTextObject() const { |
| // Objects with |ax::mojom::blink::Role::kLineBreak| are HTML <br> elements |
| // and are not backed by DOM text nodes. We can't mark them as text objects |
| // for that reason. |
| switch (RoleValue()) { |
| case ax::mojom::blink::Role::kInlineTextBox: |
| case ax::mojom::blink::Role::kStaticText: |
| return true; |
| default: |
| return false; |
| } |
| } |
| |
| bool AXObject::IsRangeValueSupported() const { |
| if (RoleValue() == ax::mojom::blink::Role::kSplitter) { |
| // According to the ARIA spec, role="separator" acts as a splitter only |
| // when focusable, and supports a range only in that case. |
| return CanSetFocusAttribute(); |
| } |
| return ui::IsRangeValueSupported(RoleValue()); |
| } |
| |
| bool AXObject::IsClickable() const { |
| return ui::IsClickable(RoleValue()); |
| } |
| |
| bool AXObject::AccessibilityIsIgnored() const { |
| UpdateDistributionForFlatTreeTraversal(); |
| UpdateCachedAttributeValuesIfNeeded(); |
| return cached_is_ignored_; |
| } |
| |
| bool AXObject::AccessibilityIsIgnoredButIncludedInTree() const { |
| UpdateDistributionForFlatTreeTraversal(); |
| UpdateCachedAttributeValuesIfNeeded(); |
| return cached_is_ignored_but_included_in_tree_; |
| } |
| |
| // AccessibilityIsIncludedInTree should be true for all nodes that should be |
| // included in the tree, even if they are ignored |
| bool AXObject::AccessibilityIsIncludedInTree() const { |
| return !AccessibilityIsIgnored() || AccessibilityIsIgnoredButIncludedInTree(); |
| } |
| |
| void AXObject::UpdateCachedAttributeValuesIfNeeded() const { |
| if (IsDetached()) |
| return; |
| |
| AXObjectCacheImpl& cache = AXObjectCache(); |
| |
| if (cache.ModificationCount() == last_modification_count_) |
| return; |
| |
| last_modification_count_ = cache.ModificationCount(); |
| cached_background_color_ = ComputeBackgroundColor(); |
| cached_is_inert_or_aria_hidden_ = ComputeIsInertOrAriaHidden(); |
| cached_is_descendant_of_leaf_node_ = !!LeafNodeAncestor(); |
| cached_is_descendant_of_disabled_node_ = !!DisabledAncestor(); |
| cached_has_inherited_presentational_role_ = |
| !!InheritsPresentationalRoleFrom(); |
| cached_is_ignored_ = ComputeAccessibilityIsIgnored(); |
| cached_is_ignored_but_included_in_tree_ = |
| cached_is_ignored_ && ComputeAccessibilityIsIgnoredButIncludedInTree(); |
| cached_is_editable_root_ = ComputeIsEditableRoot(); |
| // Compute live region root, which can be from any ARIA live value, including |
| // "off", or from an automatic ARIA live value, e.g. from role="status". |
| // TODO(dmazzoni): remove this const_cast. |
| AtomicString aria_live; |
| cached_live_region_root_ = |
| IsLiveRegionRoot() |
| ? const_cast<AXObject*>(this) |
| : (ParentObjectIfExists() ? ParentObjectIfExists()->LiveRegionRoot() |
| : nullptr); |
| cached_aria_column_index_ = ComputeAriaColumnIndex(); |
| cached_aria_row_index_ = ComputeAriaRowIndex(); |
| |
| bool ignored_states_changed = false; |
| if (cached_is_ignored_ != LastKnownIsIgnoredValue()) { |
| last_known_is_ignored_value_ = |
| cached_is_ignored_ ? kIgnoreObject : kIncludeObject; |
| ignored_states_changed = true; |
| } |
| |
| if (cached_is_ignored_but_included_in_tree_ != |
| LastKnownIsIgnoredButIncludedInTreeValue()) { |
| last_known_is_ignored_but_included_in_tree_value_ = |
| cached_is_ignored_but_included_in_tree_ ? kIncludeObject |
| : kIgnoreObject; |
| ignored_states_changed = true; |
| } |
| |
| if (ignored_states_changed) { |
| if (AXObject* parent = ParentObjectIfExists()) |
| parent->ChildrenChanged(); |
| } |
| |
| if (GetLayoutObject() && GetLayoutObject()->IsText()) { |
| cached_local_bounding_box_rect_for_accessibility_ = |
| GetLayoutObject()->LocalBoundingBoxRectForAccessibility(); |
| } |
| } |
| |
| bool AXObject::AccessibilityIsIgnoredByDefault( |
| IgnoredReasons* ignored_reasons) const { |
| return DefaultObjectInclusion(ignored_reasons) == kIgnoreObject; |
| } |
| |
| AXObjectInclusion AXObject::AccessibilityPlatformIncludesObject() const { |
| if (IsA<AXMenuListPopup>(this) || IsA<AXMenuListOption>(this)) |
| return kIncludeObject; |
| |
| return kDefaultBehavior; |
| } |
| |
| AXObjectInclusion AXObject::DefaultObjectInclusion( |
| IgnoredReasons* ignored_reasons) const { |
| if (IsInertOrAriaHidden()) { |
| // Keep focusable elements that are aria-hidden in tree, so that they can |
| // still fire events such as focus and value changes. Handle the case where |
| // a <select> element is aria-hidden. |
| if (RoleValue() != ax::mojom::blink::Role::kMenuListPopup && |
| !CanSetFocusAttribute()) { |
| if (ignored_reasons) |
| ComputeIsInertOrAriaHidden(ignored_reasons); |
| return kIgnoreObject; |
| } |
| } |
| |
| return AccessibilityPlatformIncludesObject(); |
| } |
| |
| bool AXObject::IsInertOrAriaHidden() const { |
| UpdateCachedAttributeValuesIfNeeded(); |
| return cached_is_inert_or_aria_hidden_; |
| } |
| |
| bool AXObject::ComputeIsInertOrAriaHidden( |
| IgnoredReasons* ignored_reasons) const { |
| if (GetNode()) { |
| if (GetNode()->IsInert()) { |
| if (ignored_reasons) { |
| HTMLDialogElement* dialog = GetActiveDialogElement(GetNode()); |
| if (dialog) { |
| AXObject* dialog_object = AXObjectCache().GetOrCreate(dialog); |
| if (dialog_object) { |
| ignored_reasons->push_back( |
| IgnoredReason(kAXActiveModalDialog, dialog_object)); |
| } else { |
| ignored_reasons->push_back(IgnoredReason(kAXInertElement)); |
| } |
| } else { |
| const AXObject* inert_root_el = InertRoot(); |
| if (inert_root_el == this) { |
| ignored_reasons->push_back(IgnoredReason(kAXInertElement)); |
| } else { |
| ignored_reasons->push_back( |
| IgnoredReason(kAXInertSubtree, inert_root_el)); |
| } |
| } |
| } |
| return true; |
| } else if (IsBlockedByAriaModalDialog(ignored_reasons)) { |
| return true; |
| } |
| } else { |
| AXObject* parent = ParentObject(); |
| if (parent && parent->IsInertOrAriaHidden()) { |
| if (ignored_reasons) |
| parent->ComputeIsInertOrAriaHidden(ignored_reasons); |
| return true; |
| } |
| } |
| |
| const AXObject* hidden_root = AriaHiddenRoot(); |
| if (hidden_root) { |
| if (ignored_reasons) { |
| if (hidden_root == this) { |
| ignored_reasons->push_back(IgnoredReason(kAXAriaHiddenElement)); |
| } else { |
| ignored_reasons->push_back( |
| IgnoredReason(kAXAriaHiddenSubtree, hidden_root)); |
| } |
| } |
| return true; |
| } |
| |
| return false; |
| } |
| |
| bool AXObject::IsBlockedByAriaModalDialog( |
| IgnoredReasons* ignored_reasons) const { |
| AXObject* active_aria_modal_dialog = |
| AXObjectCache().GetActiveAriaModalDialog(); |
| |
| // On platforms that don't require manual pruning of the accessibility tree, |
| // the active aria modal dialog should never be set, so has no effect. |
| if (!active_aria_modal_dialog) |
| return false; |
| |
| if (this == active_aria_modal_dialog || |
| IsDescendantOf(*active_aria_modal_dialog)) |
| return false; |
| |
| if (ignored_reasons) { |
| ignored_reasons->push_back( |
| IgnoredReason(kAXAriaModalDialog, active_aria_modal_dialog)); |
| } |
| return true; |
| } |
| |
| bool AXObject::IsVisible() const { |
| return !IsInertOrAriaHidden() && !IsHiddenViaStyle(); |
| } |
| |
| bool AXObject::IsDescendantOfLeafNode() const { |
| UpdateCachedAttributeValuesIfNeeded(); |
| return cached_is_descendant_of_leaf_node_; |
| } |
| |
| AXObject* AXObject::LeafNodeAncestor() const { |
| if (AXObject* parent = ParentObject()) { |
| if (!parent->CanHaveChildren()) |
| return parent; |
| |
| return parent->LeafNodeAncestor(); |
| } |
| |
| return nullptr; |
| } |
| |
| const AXObject* AXObject::AriaHiddenRoot() const { |
| for (const AXObject* object = this; object; object = object->ParentObject()) { |
| if (object->AOMPropertyOrARIAAttributeIsTrue(AOMBooleanProperty::kHidden)) |
| return object; |
| } |
| |
| return nullptr; |
| } |
| |
| const AXObject* AXObject::InertRoot() const { |
| const AXObject* object = this; |
| if (!RuntimeEnabledFeatures::InertAttributeEnabled()) |
| return nullptr; |
| |
| while (object && !object->IsAXNodeObject()) |
| object = object->ParentObject(); |
| Node* node = object->GetNode(); |
| auto* element = DynamicTo<Element>(node); |
| if (!element) |
| element = FlatTreeTraversal::ParentElement(*node); |
| |
| while (element) { |
| if (element->FastHasAttribute(html_names::kInertAttr)) |
| return AXObjectCache().GetOrCreate(element); |
| element = FlatTreeTraversal::ParentElement(*element); |
| } |
| |
| return nullptr; |
| } |
| |
| bool AXObject::DispatchEventToAOMEventListeners(Event& event) { |
| HeapVector<Member<AccessibleNode>> event_path; |
| for (AXObject* ancestor = this; ancestor; |
| ancestor = ancestor->ParentObject()) { |
| AccessibleNode* ancestor_accessible_node = ancestor->GetAccessibleNode(); |
| if (!ancestor_accessible_node) |
| continue; |
| |
| if (!ancestor_accessible_node->HasEventListeners(event.type())) |
| continue; |
| |
| event_path.push_back(ancestor_accessible_node); |
| } |
| |
| // Short-circuit: if there are no AccessibleNodes attached anywhere |
| // in the ancestry of this node, exit. |
| if (!event_path.size()) |
| return false; |
| |
| // Check if the user has granted permission for this domain to use |
| // AOM event listeners yet. This may trigger an infobar, but we shouldn't |
| // block, so whatever decision the user makes will apply to the next |
| // event received after that. |
| // |
| // Note that we only ask the user about this permission the first |
| // time an event is received that actually would have triggered an |
| // event listener. However, if the user grants this permission, it |
| // persists for this origin from then on. |
| if (!AXObjectCache().CanCallAOMEventListeners()) { |
| AXObjectCache().RequestAOMEventListenerPermission(); |
| return false; |
| } |
| |
| // Since we now know the AOM is being used in this document, get the |
| // AccessibleNode for the target element and create it if necessary - |
| // otherwise we wouldn't be able to set the event target. However note |
| // that if it didn't previously exist it won't be part of the event path. |
| AccessibleNode* target = GetAccessibleNode(); |
| if (!target) { |
| if (Element* element = GetElement()) |
| target = element->accessibleNode(); |
| } |
| if (!target) |
| return false; |
| event.SetTarget(target); |
| |
| // Capturing phase. |
| event.SetEventPhase(Event::kCapturingPhase); |
| for (int i = static_cast<int>(event_path.size()) - 1; i >= 0; i--) { |
| // Don't call capturing event listeners on the target. Note that |
| // the target may not necessarily be in the event path which is why |
| // we check here. |
| if (event_path[i] == target) |
| break; |
| |
| event.SetCurrentTarget(event_path[i]); |
| event_path[i]->FireEventListeners(event); |
| if (event.PropagationStopped()) |
| return true; |
| } |
| |
| // Targeting phase. |
| event.SetEventPhase(Event::kAtTarget); |
| event.SetCurrentTarget(event_path[0]); |
| event_path[0]->FireEventListeners(event); |
| if (event.PropagationStopped()) |
| return true; |
| |
| // Bubbling phase. |
| event.SetEventPhase(Event::kBubblingPhase); |
| for (wtf_size_t i = 1; i < event_path.size(); i++) { |
| event.SetCurrentTarget(event_path[i]); |
| event_path[i]->FireEventListeners(event); |
| if (event.PropagationStopped()) |
| return true; |
| } |
| |
| if (event.defaultPrevented()) |
| return true; |
| |
| return false; |
| } |
| |
| bool AXObject::IsDescendantOfDisabledNode() const { |
| UpdateCachedAttributeValuesIfNeeded(); |
| return cached_is_descendant_of_disabled_node_; |
| } |
| |
| const AXObject* AXObject::DisabledAncestor() const { |
| bool disabled = false; |
| if (HasAOMPropertyOrARIAAttribute(AOMBooleanProperty::kDisabled, disabled)) { |
| if (disabled) |
| return this; |
| return nullptr; |
| } |
| |
| if (AXObject* parent = ParentObject()) |
| return parent->DisabledAncestor(); |
| |
| return nullptr; |
| } |
| |
| bool AXObject::ComputeAccessibilityIsIgnoredButIncludedInTree() const { |
| if (!GetNode()) |
| return false; |
| |
| // Use a flag to control whether or not the <html> element is included |
| // in the accessibility tree. Either way it's always marked as "ignored", |
| // but eventually we want to always include it in the tree to simplify |
| // some logic. |
| if (GetNode() && IsA<HTMLHtmlElement>(GetNode())) |
| return RuntimeEnabledFeatures::AccessibilityExposeHTMLElementEnabled(); |
| |
| // If the node is part of the user agent shadow dom, or has the explicit |
| // internal Role::kIgnored, they aren't interesting for paragraph navigation |
| // or LabelledBy/DescribedBy relationships. |
| if (RoleValue() == ax::mojom::blink::Role::kIgnored || |
| GetNode()->IsInUserAgentShadowRoot()) { |
| return false; |
| } |
| |
| // Keep the internal accessibility tree consistent for videos which lack |
| // a player and also inner text. |
| if (RoleValue() == ax::mojom::blink::Role::kVideo || |
| RoleValue() == ax::mojom::blink::Role::kAudio) { |
| return true; |
| } |
| |
| // Always pass through Line Breaking objects, this is necessary to |
| // detect paragraph edges, which are defined as hard-line breaks. |
| if (IsLineBreakingObject()) |
| return true; |
| |
| // Allow the browser side ax tree to access "visibility: [hidden|collapse]" |
| // and "display: none" nodes. This is useful for APIs that return the node |
| // referenced by aria-labeledby and aria-describedby. |
| // An element must have an id attribute or it cannot be referenced by |
| // aria-labelledby or aria-describedby. |
| if (RuntimeEnabledFeatures::AccessibilityExposeDisplayNoneEnabled()) { |
| if (Element* element = GetElement()) { |
| if (element->FastHasAttribute(html_names::kIdAttr) && |
| IsHiddenViaStyle()) { |
| return true; |
| } |
| } |
| } else if (GetLayoutObject()) { |
| if (GetLayoutObject()->Style()->Visibility() != EVisibility::kVisible) |
| return true; |
| } |
| |
| // Allow the browser side ax tree to access "aria-hidden" nodes. |
| // This is useful for APIs that return the node referenced by |
| // aria-labeledby and aria-describedby. |
| if (GetLayoutObject() && AriaHiddenRoot()) |
| return true; |
| |
| return false; |
| } |
| |
| const AXObject* AXObject::DatetimeAncestor(int max_levels_to_check) const { |
| switch (RoleValue()) { |
| case ax::mojom::blink::Role::kDateTime: |
| case ax::mojom::blink::Role::kDate: |
| case ax::mojom::blink::Role::kInputTime: |
| case ax::mojom::blink::Role::kTime: |
| return this; |
| default: |
| break; |
| } |
| |
| if (max_levels_to_check == 0) |
| return nullptr; |
| |
| if (AXObject* parent = ParentObject()) |
| return parent->DatetimeAncestor(max_levels_to_check - 1); |
| |
| return nullptr; |
| } |
| |
| bool AXObject::LastKnownIsIgnoredValue() const { |
| if (last_known_is_ignored_value_ == kDefaultBehavior) { |
| last_known_is_ignored_value_ = |
| AccessibilityIsIgnored() ? kIgnoreObject : kIncludeObject; |
| } |
| |
| return last_known_is_ignored_value_ == kIgnoreObject; |
| } |
| |
| void AXObject::SetLastKnownIsIgnoredValue(bool is_ignored) { |
| last_known_is_ignored_value_ = is_ignored ? kIgnoreObject : kIncludeObject; |
| } |
| |
| bool AXObject::LastKnownIsIgnoredButIncludedInTreeValue() const { |
| if (last_known_is_ignored_but_included_in_tree_value_ == kDefaultBehavior) { |
| last_known_is_ignored_but_included_in_tree_value_ = |
| AccessibilityIsIgnoredButIncludedInTree() ? kIncludeObject |
| : kIgnoreObject; |
| } |
| |
| return last_known_is_ignored_but_included_in_tree_value_ == kIncludeObject; |
| } |
| |
| void AXObject::SetLastKnownIsIgnoredButIncludedInTreeValue( |
| bool is_ignored_but_included_in_tree) { |
| last_known_is_ignored_but_included_in_tree_value_ = |
| is_ignored_but_included_in_tree ? kIncludeObject : kIgnoreObject; |
| } |
| |
| bool AXObject::HasInheritedPresentationalRole() const { |
| UpdateCachedAttributeValuesIfNeeded(); |
| return cached_has_inherited_presentational_role_; |
| } |
| |
| bool AXObject::CanSetValueAttribute() const { |
| switch (RoleValue()) { |
| case ax::mojom::blink::Role::kColorWell: |
| case ax::mojom::blink::Role::kDate: |
| case ax::mojom::blink::Role::kDateTime: |
| case ax::mojom::blink::Role::kInputTime: |
| case ax::mojom::blink::Role::kScrollBar: |
| case ax::mojom::blink::Role::kSearchBox: |
| case ax::mojom::blink::Role::kSlider: |
| case ax::mojom::blink::Role::kSpinButton: |
| case ax::mojom::blink::Role::kSplitter: |
| case ax::mojom::blink::Role::kTextField: |
| case ax::mojom::blink::Role::kTextFieldWithComboBox: |
| return Restriction() == kRestrictionNone; |
| default: |
| return false; |
| } |
| } |
| |
| bool AXObject::IsFocusableStyleUsingBestAvailableState() const { |
| auto* element = GetElement(); |
| DCHECK(element); |
| |
| // If this element's layout tree does not need an update, it means that we can |
| // rely on Element's IsFocusableStyle directly, which is the best available |
| // source of information. |
| // Note that we also allow this to be used if we're in a style recalc, since |
| // we might get here through layout object attachment. In that case, the dirty |
| // bits may not have been cleared yet, but all relevant style and layout tree |
| // should be up to date. Note that this quirk can be fixed by deferring AX |
| // tree updates to happen after the layout tree attachment has finished. |
| if (GetDocument()->InStyleRecalc() || |
| !GetDocument()->NeedsLayoutTreeUpdateForNodeIncludingDisplayLocked( |
| *element)) { |
| return element->IsFocusableStyle(); |
| } |
| |
| // The best available source of information is now the AX tree, so use that to |
| // figure out whether we have focusable style. |
| |
| // If we're in a canvas subtree, then use the canvas visibility instead of |
| // self visibility. The elements in a canvas subtree are fallback elements, |
| // which are not necessarily rendered but are allowed to be focusable. |
| if (element->IsInCanvasSubtree()) { |
| const HTMLCanvasElement* canvas = |
| Traversal<HTMLCanvasElement>::FirstAncestorOrSelf(*element); |
| DCHECK(canvas); |
| return canvas->GetLayoutObject() && |
| canvas->GetLayoutObject()->Style()->Visibility() == |
| EVisibility::kVisible; |
| } |
| |
| return GetLayoutObject() && |
| GetLayoutObject()->Style()->Visibility() == EVisibility::kVisible; |
| } |
| |
| // This does not use Element::IsFocusable(), as that can sometimes recalculate |
| // styles because of IsFocusableStyle() check, resetting the document lifecycle. |
| bool AXObject::CanSetFocusAttribute() const { |
| if (IsDetached()) |
| return false; |
| |
| // Objects within a portal are not focusable. |
| // Note that they are ignored but can be included in the tree. |
| bool inside_portal = GetDocument() && GetDocument()->GetPage() && |
| GetDocument()->GetPage()->InsidePortal(); |
| if (inside_portal) |
| return false; |
| |
| // Display-locked nodes that have content-visibility: hidden are not exposed |
| // to accessibility in any way, so they are not focusable. Note that for |
| // content-visibility: auto cases, `ShouldIgnoreNodeDueToDisplayLock()` would |
| // return false, since we're not ignoring the element in that case. |
| if (GetNode() && |
| DisplayLockUtilities::ShouldIgnoreNodeDueToDisplayLock( |
| *GetNode(), DisplayLockActivationReason::kAccessibility)) { |
| return false; |
| } |
| |
| // Focusable: web area -- this is the only focusable non-element. Web areas |
| // inside portals are not focusable though (portal contents cannot get focus). |
| if (IsWebArea()) |
| return true; |
| |
| // NOT focusable: objects with no DOM node, e.g. extra layout blocks inserted |
| // as filler, or objects where the node is not an element, such as a text |
| // node or an HTML comment. |
| Element* elem = GetElement(); |
| if (!elem) |
| return false; |
| |
| // NOT focusable: inert elements. |
| if (elem->IsInert()) |
| return false; |
| |
| // NOT focusable: disabled form controls. |
| if (IsDisabledFormControl(elem)) |
| return false; |
| |
| // Focusable: options in a combobox or listbox. |
| // Even though they are not treated as supporting focus by Blink (the parent |
| // widget is), they are considered focusable in the accessibility sense, |
| // behaving like potential active descendants, and handling focus actions. |
| // Menu list options are handled before visibility check, because they |
| // are considered focusable even when part of collapsed drop down. |
| if (RoleValue() == ax::mojom::blink::Role::kMenuListOption) |
| return true; |
| |
| // NOT focusable: hidden elements. |
| // TODO(aleventhal) Consider caching visibility when it's safe to compute. |
| if (!IsA<HTMLAreaElement>(elem) && !IsFocusableStyleUsingBestAvailableState()) |
| return false; |
| |
| // Focusable: options in a combobox or listbox. |
| // Similar to menu list option treatment above, but not focusable if hidden. |
| if (RoleValue() == ax::mojom::blink::Role::kListBoxOption) |
| return true; |
| |
| // Focusable: element supports focus. |
| if (elem->SupportsFocus()) |
| return true; |
| |
| // TODO(accessibility) Focusable: scrollable with the keyboard. |
| // Keyboard-focusable scroll containers feature: |
| // https://www.chromestatus.com/feature/5231964663578624 |
| // When adding here, remove similar check from ::NameFromContents(). |
| // if (RuntimeEnabledFeatures::KeyboardFocusableScrollersEnabled() && |
| // IsUserScrollable()) { |
| // return true; |
| // } |
| |
| // Focusable: can be an active descendant. |
| if (CanBeActiveDescendant()) |
| return true; |
| |
| // NOT focusable: everything else. |
| return false; |
| } |
| |
| // From ARIA 1.1. |
| // 1. The value of aria-activedescendant refers to an element that is either a |
| // descendant of the element with DOM focus or is a logical descendant as |
| // indicated by the aria-owns attribute. 2. The element with DOM focus is a |
| // textbox with aria-controls referring to an element that supports |
| // aria-activedescendant, and the value of aria-activedescendant specified for |
| // the textbox refers to either a descendant of the element controlled by the |
| // textbox or is a logical descendant of that controlled element as indicated by |
| // the aria-owns attribute. |
| bool AXObject::CanBeActiveDescendant() const { |
| // Require an element with an id attribute. |
| // TODO(accessibility): this code currently requires both an id and role |
| // attribute, as well as an ancestor or controlling aria-activedescendant. |
| // However, with element reflection it may be possible to set an active |
| // descendant without an id, so at some point we may need to remove the |
| // requirement for an id attribute. |
| if (!GetElement() || !GetElement()->FastHasAttribute(html_names::kIdAttr)) |
| return false; |
| |
| // Does not make sense to use aria-activedescendant to point to a |
| // presentational object. |
| if (IsPresentational()) |
| return false; |
| |
| // Does not make sense to use aria-activedescendant to point to an HTML |
| // element that requires real focus, therefore an ARIA role is necessary. |
| if (AriaRoleAttribute() == ax::mojom::blink::Role::kUnknown) |
| return false; |
| |
| return IsARIAControlledByTextboxWithActiveDescendant() || |
| AncestorExposesActiveDescendant(); |
| } |
| |
| void AXObject::UpdateDistributionForFlatTreeTraversal() const { |
| Node* node = GetNode(); |
| if (!node) { |
| AXObject* parent = this->ParentObject(); |
| while (!node && parent) { |
| node = parent->GetNode(); |
| parent = parent->ParentObject(); |
| } |
| } |
| |
| if (node) |
| node->UpdateDistributionForFlatTreeTraversal(); |
| |
| // TODO(aboxhall): Instead of this, propagate inert down through frames |
| Document* document = GetDocument(); |
| while (document && document->LocalOwner()) { |
| document->LocalOwner()->UpdateDistributionForFlatTreeTraversal(); |
| document = document->LocalOwner()->ownerDocument(); |
| } |
| } |
| |
| bool AXObject::IsARIAControlledByTextboxWithActiveDescendant() const { |
| if (IsDetached()) |
| return false; |
| |
| // This situation should mostly arise when using an active descendant on a |
| // textbox inside an ARIA 1.1 combo box widget, which points to the selected |
| // option in a list. In such situations, the active descendant is useful only |
| // when the textbox is focused. Therefore, we don't currently need to keep |
| // track of all aria-controls relationships. |
| const AXObject* focused_object = AXObjectCache().FocusedObject(); |
| if (!focused_object || !focused_object->IsTextControl()) |
| return false; |
| |
| if (!focused_object->GetAOMPropertyOrARIAAttribute( |
| AOMRelationProperty::kActiveDescendant)) { |
| return false; |
| } |
| |
| HeapVector<Member<Element>> controlled_by_elements; |
| if (!focused_object->HasAOMPropertyOrARIAAttribute( |
| AOMRelationListProperty::kControls, controlled_by_elements)) { |
| return false; |
| } |
| |
| for (const auto& controlled_by_element : controlled_by_elements) { |
| const AXObject* controlled_by_object = |
| AXObjectCache().GetOrCreate(controlled_by_element); |
| if (!controlled_by_object) |
| continue; |
| |
| const AXObject* object = this; |
| while (object && object != controlled_by_object) |
| object = object->ParentObjectUnignored(); |
| if (object) |
| return true; |
| } |
| |
| return false; |
| } |
| |
| bool AXObject::AncestorExposesActiveDescendant() const { |
| const AXObject* parent = ParentObjectUnignored(); |
| if (!parent) |
| return false; |
| |
| if (parent->GetAOMPropertyOrARIAAttribute( |
| AOMRelationProperty::kActiveDescendant)) { |
| return true; |
| } |
| |
| return parent->AncestorExposesActiveDescendant(); |
| } |
| |
| bool AXObject::HasIndirectChildren() const { |
| return RoleValue() == ax::mojom::blink::Role::kTableHeaderContainer; |
| } |
| |
| bool AXObject::CanSetSelectedAttribute() const { |
| // Sub-widget elements can be selected if not disabled (native or ARIA) |
| return IsSubWidget() && Restriction() != kRestrictionDisabled; |
| } |
| |
| bool AXObject::IsSubWidget() const { |
| switch (RoleValue()) { |
| case ax::mojom::blink::Role::kCell: |
| case ax::mojom::blink::Role::kColumnHeader: |
| case ax::mojom::blink::Role::kRowHeader: |
| case ax::mojom::blink::Role::kColumn: |
| case ax::mojom::blink::Role::kRow: |
| // If it has an explicit ARIA role, it's a subwidget. |
| // |
| // Reasoning: |
| // Static table cells are not selectable, but ARIA grid cells |
| // and rows definitely are according to the spec. To support |
| // ARIA 1.0, it's sufficient to just check if there's any |
| // ARIA role at all, because if so then it must be a grid-related |
| // role so it must be selectable. |
| // |
| // TODO: an ARIA 1.1+ role of "cell", or a role of "row" inside |
| // an ARIA 1.1 role of "table", should not be selectable. We may |
| // need to create separate role enums for grid cells vs table cells |
| // to implement this. |
| if (AriaRoleAttribute() != ax::mojom::blink::Role::kUnknown) |
| return true; |
| |
| // Otherwise it's only a subwidget if it's in a grid or treegrid, |
| // not in a table. |
| return std::any_of( |
| UnignoredAncestorsBegin(), UnignoredAncestorsEnd(), |
| [](const AXObject& ancestor) { |
| return ancestor.RoleValue() == ax::mojom::blink::Role::kGrid || |
| ancestor.RoleValue() == ax::mojom::blink::Role::kTreeGrid; |
| }); |
| |
| case ax::mojom::blink::Role::kListBoxOption: |
| case ax::mojom::blink::Role::kMenuListOption: |
| case ax::mojom::blink::Role::kTab: |
| case ax::mojom::blink::Role::kTreeItem: |
| return true; |
| default: |
| return false; |
| } |
| } |
| |
| bool AXObject::SupportsARIASetSizeAndPosInSet() const { |
| return ui::IsSetLike(RoleValue()) || ui::IsItemLike(RoleValue()); |
| } |
| |
| // Simplify whitespace, but preserve a single leading and trailing whitespace |
| // character if it's present. |
| // static |
| String AXObject::CollapseWhitespace(const String& str) { |
| StringBuilder result; |
| if (!str.IsEmpty() && IsHTMLSpace<UChar>(str[0])) |
| result.Append(' '); |
| result.Append(str.SimplifyWhiteSpace(IsHTMLSpace<UChar>)); |
| if (!str.IsEmpty() && IsHTMLSpace<UChar>(str[str.length() - 1])) |
| result.Append(' '); |
| return result.ToString(); |
| } |
| |
| String AXObject::ComputedName() const { |
| ax::mojom::blink::NameFrom name_from; |
| AXObjectVector name_objects; |
| return GetName(name_from, &name_objects); |
| } |
| |
| String AXObject::GetName(ax::mojom::blink::NameFrom& name_from, |
| AXObject::AXObjectVector* name_objects) const { |
| HeapHashSet<Member<const AXObject>> visited; |
| AXRelatedObjectVector related_objects; |
| // For purposes of computing a text alternative, if an ignored node is |
| // included in the tree, assume that it is the target of aria-labelledby or |
| // aria-describedby, since we can't tell yet whether that's the case. If it |
| // isn't exposed, the AT will never see the name anyways. |
| bool hidden_and_ignored_but_included_in_tree = |
| IsHiddenForTextAlternativeCalculation() && |
| AccessibilityIsIgnoredButIncludedInTree(); |
| // Initialize |name_from|, as TextAlternative() might never set it in some |
| // cases. |
| name_from = ax::mojom::blink::NameFrom::kNone; |
| String text = TextAlternative(false, hidden_and_ignored_but_included_in_tree, |
| visited, name_from, &related_objects, nullptr); |
| |
| ax::mojom::blink::Role role = RoleValue(); |
| if (!GetNode() || (!IsA<HTMLBRElement>(GetNode()) && |
| role != ax::mojom::blink::Role::kStaticText && |
| role != ax::mojom::blink::Role::kInlineTextBox)) |
| text = CollapseWhitespace(text); |
| |
| if (name_objects) { |
| name_objects->clear(); |
| for (NameSourceRelatedObject* related_object : related_objects) |
| name_objects->push_back(related_object->object); |
| } |
| |
| return text; |
| } |
| |
| String AXObject::GetName(NameSources* name_sources) const { |
| AXObjectSet visited; |
| ax::mojom::blink::NameFrom tmp_name_from; |
| AXRelatedObjectVector tmp_related_objects; |
| // For purposes of computing a text alternative, if an ignored node is |
| // included in the tree, assume that it is the target of aria-labelledby or |
| // aria-describedby, since we can't tell yet whether that's the case. If it |
| // isn't exposed, the AT will never see the name anyways. |
| bool hidden_and_ignored_but_included_in_tree = |
| IsHiddenForTextAlternativeCalculation() && |
| AccessibilityIsIgnoredButIncludedInTree(); |
| String text = |
| TextAlternative(false, hidden_and_ignored_but_included_in_tree, visited, |
| tmp_name_from, &tmp_related_objects, name_sources); |
| text = text.SimplifyWhiteSpace(IsHTMLSpace<UChar>); |
| return text; |
| } |
| |
| String AXObject::RecursiveTextAlternative(const AXObject& ax_obj, |
| bool in_aria_labelled_by_traversal, |
| AXObjectSet& visited) { |
| ax::mojom::blink::NameFrom tmp_name_from; |
| return RecursiveTextAlternative(ax_obj, in_aria_labelled_by_traversal, |
| visited, tmp_name_from); |
| } |
| |
| String AXObject::RecursiveTextAlternative( |
| const AXObject& ax_obj, |
| bool in_aria_labelled_by_traversal, |
| AXObjectSet& visited, |
| ax::mojom::blink::NameFrom& name_from) { |
| if (visited.Contains(&ax_obj) && !in_aria_labelled_by_traversal) |
| return String(); |
| |
| return ax_obj.TextAlternative(true, in_aria_labelled_by_traversal, visited, |
| name_from, nullptr, nullptr); |
| } |
| |
| bool AXObject::IsHiddenViaStyle() const { |
| Node* node = GetNode(); |
| if (!node) |
| return false; |
| |
| // Display-locked nodes are always hidden. |
| if (DisplayLockUtilities::NearestLockedExclusiveAncestor(*node)) |
| return true; |
| |
| if (GetLayoutObject()) |
| return GetLayoutObject()->Style()->Visibility() != EVisibility::kVisible; |
| |
| if (Element* element = DynamicTo<Element>(node)) { |
| const ComputedStyle* style = element->GetComputedStyle(); |
| return !style || style->IsEnsuredInDisplayNone() || |
| style->Visibility() != EVisibility::kVisible; |
| } |
| return false; |
| } |
| |
| bool AXObject::IsHiddenForTextAlternativeCalculation() const { |
| if (AOMPropertyOrARIAAttributeIsFalse(AOMBooleanProperty::kHidden)) |
| return false; |
| |
| auto* node = GetNode(); |
| if (!node) |
| return false; |
| |
| // Display-locked elements are available for text/name resolution. |
| if (DisplayLockUtilities::NearestLockedExclusiveAncestor(*node)) |
| return false; |
| |
| if (GetLayoutObject()) |
| return GetLayoutObject()->Style()->Visibility() != EVisibility::kVisible; |
| |
| if (IsA<HTMLNoScriptElement>(node)) |
| return true; |
| |
| // This is an obscure corner case: if a node has no LayoutObject, that means |
| // it's not rendered, but we still may be exploring it as part of a text |
| // alternative calculation, for example if it was explicitly referenced by |
| // aria-labelledby. So we need to explicitly call the style resolver to check |
| // whether it's invisible or display:none, rather than relying on the style |
| // cached in the LayoutObject. |
| Document* document = GetDocument(); |
| if (!document || !document->GetFrame()) |
| return false; |
| auto* element = DynamicTo<Element>(node); |
| if (element && node->isConnected()) { |
| const ComputedStyle* style = element->EnsureComputedStyle(); |
| if (!style) |
| return false; |
| return style->Display() == EDisplay::kNone || |
| style->Visibility() != EVisibility::kVisible; |
| } |
| return false; |
| } |
| |
| String AXObject::AriaTextAlternative(bool recursive, |
| bool in_aria_labelled_by_traversal, |
| AXObjectSet& visited, |
| ax::mojom::blink::NameFrom& name_from, |
| AXRelatedObjectVector* related_objects, |
| NameSources* name_sources, |
| bool* found_text_alternative) const { |
| String text_alternative; |
| bool already_visited = visited.Contains(this); |
| visited.insert(this); |
| |
| // Step 2A from: http://www.w3.org/TR/accname-aam-1.1 |
| // If you change this logic, update AXNodeObject::nameFromLabelElement, too. |
| if (!in_aria_labelled_by_traversal && |
| IsHiddenForTextAlternativeCalculation()) { |
| *found_text_alternative = true; |
| return String(); |
| } |
| |
| // Step 2B from: http://www.w3.org/TR/accname-aam-1.1 |
| // If you change this logic, update AXNodeObject::nameFromLabelElement, too. |
| if (!in_aria_labelled_by_traversal && !already_visited) { |
| name_from = ax::mojom::blink::NameFrom::kRelatedElement; |
| |
| // Check ARIA attributes. |
| const QualifiedName& attr = |
| HasAttribute(html_names::kAriaLabeledbyAttr) && |
| !HasAttribute(html_names::kAriaLabelledbyAttr) |
| ? html_names::kAriaLabeledbyAttr |
| : html_names::kAriaLabelledbyAttr; |
| |
| if (name_sources) { |
| name_sources->push_back(NameSource(*found_text_alternative, attr)); |
| name_sources->back().type = name_from; |
| } |
| |
| Element* element = GetElement(); |
| if (element) { |
| HeapVector<Member<Element>> elements_from_attribute; |
| Vector<String> ids; |
| ElementsFromAttribute(elements_from_attribute, attr, ids); |
| |
| const AtomicString& aria_labelledby = GetAttribute(attr); |
| |
| if (!aria_labelledby.IsNull()) { |
| if (name_sources) |
| name_sources->back().attribute_value = aria_labelledby; |
| |
| // Operate on a copy of |visited| so that if |name_sources| is not |
| // null, the set of visited objects is preserved unmodified for future |
| // calculations. |
| AXObjectSet visited_copy = visited; |
| text_alternative = TextFromElements( |
| true, visited, elements_from_attribute, related_objects); |
| if (!ids.IsEmpty()) |
| AXObjectCache().UpdateReverseRelations(this, ids); |
| if (!text_alternative.IsNull()) { |
| if (name_sources) { |
| NameSource& source = name_sources->back(); |
| source.type = name_from; |
| source.related_objects = *related_objects; |
| source.text = text_alternative; |
| *found_text_alternative = true; |
| } else { |
| *found_text_alternative = true; |
| return text_alternative; |
| } |
| } else if (name_sources) { |
| name_sources->back().invalid = true; |
| } |
| } |
| } |
| } |
| |
| // Step 2C from: http://www.w3.org/TR/accname-aam-1.1 |
| // If you change this logic, update AXNodeObject::nameFromLabelElement, too. |
| name_from = ax::mojom::blink::NameFrom::kAttribute; |
| if (name_sources) { |
| name_sources->push_back( |
| NameSource(*found_text_alternative, html_names::kAriaLabelAttr)); |
| name_sources->back().type = name_from; |
| } |
| const AtomicString& aria_label = |
| GetAOMPropertyOrARIAAttribute(AOMStringProperty::kLabel); |
| if (!aria_label.IsEmpty()) { |
| text_alternative = aria_label; |
| |
| if (name_sources) { |
| NameSource& source = name_sources->back(); |
| source.text = text_alternative; |
| source.attribute_value = aria_label; |
| *found_text_alternative = true; |
| } else { |
| *found_text_alternative = true; |
| return text_alternative; |
| } |
| } |
| |
| return text_alternative; |
| } |
| |
| String AXObject::TextFromElements( |
| bool in_aria_labelledby_traversal, |
| AXObjectSet& visited, |
| HeapVector<Member<Element>>& elements, |
| AXRelatedObjectVector* related_objects) const { |
| StringBuilder accumulated_text; |
| bool found_valid_element = false; |
| AXRelatedObjectVector local_related_objects; |
| |
| for (const auto& element : elements) { |
| AXObject* ax_element = AXObjectCache().GetOrCreate(element); |
| if (ax_element) { |
| found_valid_element = true; |
| |
| String result = RecursiveTextAlternative( |
| *ax_element, in_aria_labelledby_traversal, visited); |
| visited.insert(ax_element); |
| local_related_objects.push_back( |
| MakeGarbageCollected<NameSourceRelatedObject>(ax_element, result)); |
| if (!result.IsEmpty()) { |
| if (!accumulated_text.IsEmpty()) |
| accumulated_text.Append(' '); |
| accumulated_text.Append(result); |
| } |
| } |
| } |
| if (!found_valid_element) |
| return String(); |
| if (related_objects) |
| *related_objects = local_related_objects; |
| return accumulated_text.ToString(); |
| } |
| |
| void AXObject::TokenVectorFromAttribute(Vector<String>& tokens, |
| const QualifiedName& attribute) const { |
| Node* node = this->GetNode(); |
| if (!node || !node->IsElementNode()) |
| return; |
| |
| String attribute_value = GetAttribute(attribute).GetString(); |
| if (attribute_value.IsEmpty()) |
| return; |
| |
| attribute_value = attribute_value.SimplifyWhiteSpace(); |
| attribute_value.Split(' ', tokens); |
| } |
| |
| void AXObject::ElementsFromAttribute(HeapVector<Member<Element>>& elements, |
| const QualifiedName& attribute, |
| Vector<String>& ids) const { |
| // We compute the attr-associated elements, which are either explicitly set |
| // element references set via the IDL, or computed from the content attribute. |
| TokenVectorFromAttribute(ids, attribute); |
| Element* element = GetElement(); |
| if (!element) |
| return; |
| |
| base::Optional<HeapVector<Member<Element>>> attr_associated_elements = |
| element->GetElementArrayAttribute(attribute); |
| if (!attr_associated_elements) |
| return; |
| |
| for (const auto& element : attr_associated_elements.value()) |
| elements.push_back(element); |
| } |
| |
| void AXObject::AriaLabelledbyElementVector( |
| HeapVector<Member<Element>>& elements, |
| Vector<String>& ids) const { |
| // Try both spellings, but prefer aria-labelledby, which is the official spec. |
| ElementsFromAttribute(elements, html_names::kAriaLabelledbyAttr, ids); |
| if (!ids.size()) |
| ElementsFromAttribute(elements, html_names::kAriaLabeledbyAttr, ids); |
| } |
| |
| String AXObject::TextFromAriaLabelledby(AXObjectSet& visited, |
| AXRelatedObjectVector* related_objects, |
| Vector<String>& ids) const { |
| HeapVector<Member<Element>> elements; |
| AriaLabelledbyElementVector(elements, ids); |
| return TextFromElements(true, visited, elements, related_objects); |
| } |
| |
| String AXObject::TextFromAriaDescribedby(AXRelatedObjectVector* related_objects, |
| Vector<String>& ids) const { |
| AXObjectSet visited; |
| HeapVector<Member<Element>> elements; |
| ElementsFromAttribute(elements, html_names::kAriaDescribedbyAttr, ids); |
| return TextFromElements(true, visited, elements, related_objects); |
| } |
| |
| RGBA32 AXObject::BackgroundColor() const { |
| UpdateCachedAttributeValuesIfNeeded(); |
| return cached_background_color_; |
| } |
| |
| AccessibilityOrientation AXObject::Orientation() const { |
| // In ARIA 1.1, the default value for aria-orientation changed from |
| // horizontal to undefined. |
| return kAccessibilityOrientationUndefined; |
| } |
| |
| void AXObject::Markers(Vector<DocumentMarker::MarkerType>&, |
| Vector<AXRange>&) const {} |
| |
| void AXObject::TextCharacterOffsets(Vector<int>&) const {} |
| |
| void AXObject::GetWordBoundaries(Vector<int>& word_starts, |
| Vector<int>& word_ends) const {} |
| |
| ax::mojom::blink::DefaultActionVerb AXObject::Action() const { |
| Element* action_element = ActionElement(); |
| if (!action_element) |
| return ax::mojom::blink::DefaultActionVerb::kNone; |
| |
| // TODO(dmazzoni): Ensure that combo box text field is handled here. |
| if (IsTextControl()) |
| return ax::mojom::blink::DefaultActionVerb::kActivate; |
| |
| if (IsCheckable()) { |
| return CheckedState() != ax::mojom::blink::CheckedState::kTrue |
| ? ax::mojom::blink::DefaultActionVerb::kCheck |
| : ax::mojom::blink::DefaultActionVerb::kUncheck; |
| } |
| |
| // If this object cannot receive focus and has a button role, use click as |
| // the default action. On the AuraLinux platform, the press action is a |
| // signal to users that they can trigger the action using the keyboard, while |
| // a click action means the user should trigger the action via a simulated |
| // click. If this object cannot receive focus, it's impossible to trigger it |
| // with a key press. |
| if (RoleValue() == ax::mojom::blink::Role::kButton && !CanSetFocusAttribute()) |
| return ax::mojom::blink::DefaultActionVerb::kClick; |
| |
| switch (RoleValue()) { |
| case ax::mojom::blink::Role::kButton: |
| case ax::mojom::blink::Role::kDisclosureTriangle: |
| case ax::mojom::blink::Role::kToggleButton: |
| return ax::mojom::blink::DefaultActionVerb::kPress; |
| case ax::mojom::blink::Role::kListBoxOption: |
| case ax::mojom::blink::Role::kMenuItemRadio: |
| case ax::mojom::blink::Role::kMenuItem: |
| case ax::mojom::blink::Role::kMenuListOption: |
| return ax::mojom::blink::DefaultActionVerb::kSelect; |
| case ax::mojom::blink::Role::kLink: |
| return ax::mojom::blink::DefaultActionVerb::kJump; |
| case ax::mojom::blink::Role::kComboBoxMenuButton: |
| case ax::mojom::blink::Role::kPopUpButton: |
| return ax::mojom::blink::DefaultActionVerb::kOpen; |
| default: |
| if (action_element == GetNode()) |
| return ax::mojom::blink::DefaultActionVerb::kClick; |
| return ax::mojom::blink::DefaultActionVerb::kClickAncestor; |
| } |
| } |
| |
| bool AXObject::AriaPressedIsPresent() const { |
| AtomicString result; |
| return HasAOMPropertyOrARIAAttribute(AOMStringProperty::kPressed, result); |
| } |
| |
| bool AXObject::AriaCheckedIsPresent() const { |
| AtomicString result; |
| return HasAOMPropertyOrARIAAttribute(AOMStringProperty::kChecked, result); |
| } |
| |
| bool AXObject::SupportsARIAExpanded() const { |
| switch (RoleValue()) { |
| case ax::mojom::blink::Role::kApplication: |
| case ax::mojom::blink::Role::kButton: |
| case ax::mojom::blink::Role::kCheckBox: |
| case ax::mojom::blink::Role::kColumnHeader: |
| case ax::mojom::blink::Role::kComboBoxGrouping: |
| case ax::mojom::blink::Role::kComboBoxMenuButton: |
| case ax::mojom::blink::Role::kDisclosureTriangle: |
| case ax::mojom::blink::Role::kListBox: |
| case ax::mojom::blink::Role::kLink: |
| case ax::mojom::blink::Role::kPopUpButton: |
| case ax::mojom::blink::Role::kMenuButton: |
| case ax::mojom::blink::Role::kMenuItem: |
| case ax::mojom::blink::Role::kMenuItemCheckBox: |
| case ax::mojom::blink::Role::kMenuItemRadio: |
| case ax::mojom::blink::Role::kRow: |
| case ax::mojom::blink::Role::kRowHeader: |
| case ax::mojom::blink::Role::kSwitch: |
| case ax::mojom::blink::Role::kTab: |
| case ax::mojom::blink::Role::kTextFieldWithComboBox: |
| case ax::mojom::blink::Role::kTreeItem: |
| return true; |
| case ax::mojom::blink::Role::kCell: |
| // TODO(Accessibility): aria-expanded is supported on grid cells but not |
| // on cells inside a static table. Consider creating separate internal |
| // roles so that we can easily distinguish these two types. See also |
| // IsSubWidget(). |
| return true; |
| default: |
| return false; |
| } |
| } |
| |
| bool IsGlobalARIAAttribute(const AtomicString& name) { |
| if (!name.StartsWith("ARIA")) |
| return false; |
| if (name.StartsWith("ARIA-ATOMIC")) |
| return true; |
| if (name.StartsWith("ARIA-BUSY")) |
| return true; |
| if (name.StartsWith("ARIA-CONTROLS")) |
| return true; |
| if (name.StartsWith("ARIA-CURRENT")) |
| return true; |
| if (name.StartsWith("ARIA-DESCRIBEDBY")) |
| return true; |
| if (name.StartsWith("ARIA-DETAILS")) |
| return true; |
| if (name.StartsWith("ARIA-DISABLED")) |
| return true; |
| if (name.StartsWith("ARIA-DROPEFFECT")) |
| return true; |
| if (name.StartsWith("ARIA-ERRORMESSAGE")) |
| return true; |
| if (name.StartsWith("ARIA-FLOWTO")) |
| return true; |
| if (name.StartsWith("ARIA-GRABBED")) |
| return true; |
| if (name.StartsWith("ARIA-HASPOPUP")) |
| return true; |
| if (name.StartsWith("ARIA-HIDDEN")) |
| return true; |
| if (name.StartsWith("ARIA-INVALID")) |
| return true; |
| if (name.StartsWith("ARIA-KEYSHORTCUTS")) |
| return true; |
| if (name.StartsWith("ARIA-LABEL")) |
| return true; |
| if (name.StartsWith("ARIA-LABELEDBY")) |
| return true; |
| if (name.StartsWith("ARIA-LABELLEDBY")) |
| return true; |
| if (name.StartsWith("ARIA-LIVE")) |
| return true; |
| if (name.StartsWith("ARIA-OWNS")) |
| return true; |
| if (name.StartsWith("ARIA-RELEVANT")) |
| return true; |
| if (name.StartsWith("ARIA-ROLEDESCRIPTION")) |
| return true; |
| return false; |
| } |
| |
| bool AXObject::HasGlobalARIAAttribute() const { |
| auto* element = GetElement(); |
| if (!element) |
| return false; |
| |
| AttributeCollection attributes = element->AttributesWithoutUpdate(); |
| for (const Attribute& attr : attributes) { |
| // Attributes cache their uppercase names. |
| auto name = attr.GetName().LocalNameUpper(); |
| if (IsGlobalARIAAttribute(name)) |
| return true; |
| } |
| if (!element->DidAttachInternals()) |
| return false; |
| const auto& internals_attributes = |
| element->EnsureElementInternals().GetAttributes(); |
| for (const QualifiedName& attr : internals_attributes.Keys()) { |
| if (IsGlobalARIAAttribute(attr.LocalNameUpper())) |
| return true; |
| } |
| return false; |
| } |
| |
| int AXObject::IndexInParent() const { |
| DCHECK(AccessibilityIsIncludedInTree()) |
| << "IndexInParent is only valid when a node is included in the tree"; |
| if (!ParentObjectIncludedInTree()) |
| return 0; |
| |
| const AXObjectVector& siblings = |
| ParentObjectIncludedInTree()->ChildrenIncludingIgnored(); |
| wtf_size_t index = siblings.Find(this); |
| DCHECK(index != kNotFound); |
| return (index == kNotFound) ? 0 : static_cast<int>(index); |
| } |
| |
| bool AXObject::IsLiveRegionRoot() const { |
| const AtomicString& live_region = LiveRegionStatus(); |
| return !live_region.IsEmpty(); |
| } |
| |
| bool AXObject::IsActiveLiveRegionRoot() const { |
| const AtomicString& live_region = LiveRegionStatus(); |
| return !live_region.IsEmpty() && !EqualIgnoringASCIICase(live_region, "off"); |
| } |
| |
| AXRestriction AXObject::Restriction() const { |
| // According to ARIA, all elements of the base markup can be disabled. |
| // According to CORE-AAM, any focusable descendant of aria-disabled |
| // ancestor is also disabled. |
| bool is_disabled; |
| if (HasAOMPropertyOrARIAAttribute(AOMBooleanProperty::kDisabled, |
| is_disabled)) { |
| // Has aria-disabled, overrides native markup determining disabled. |
| if (is_disabled) |
| return kRestrictionDisabled; |
| } else if (CanSetFocusAttribute() && IsDescendantOfDisabledNode()) { |
| // aria-disabled on an ancestor propagates to focusable descendants. |
| return kRestrictionDisabled; |
| } |
| |
| // Check aria-readonly if supported by current role. |
| bool is_read_only; |
| if (SupportsARIAReadOnly() && |
| HasAOMPropertyOrARIAAttribute(AOMBooleanProperty::kReadOnly, |
| is_read_only)) { |
| // ARIA overrides other readonly state markup. |
| return is_read_only ? kRestrictionReadOnly : kRestrictionNone; |
| } |
| |
| // This is a node that is not readonly and not disabled. |
| return kRestrictionNone; |
| } |
| |
| ax::mojom::blink::Role AXObject::DetermineAccessibilityRole() { |
| aria_role_ = DetermineAriaRoleAttribute(); |
| return aria_role_; |
| } |
| |
| ax::mojom::blink::Role AXObject::AriaRoleAttribute() const { |
| return aria_role_; |
| } |
| |
| ax::mojom::blink::Role AXObject::DetermineAriaRoleAttribute() const { |
| const AtomicString& aria_role = |
| GetAOMPropertyOrARIAAttribute(AOMStringProperty::kRole); |
| if (aria_role.IsNull() || aria_role.IsEmpty()) |
| return ax::mojom::blink::Role::kUnknown; |
| |
| ax::mojom::blink::Role role = AriaRoleToWebCoreRole(aria_role); |
| |
| switch (role) { |
| case ax::mojom::blink::Role::kComment: |
| case ax::mojom::blink::Role::kMark: |
| case ax::mojom::blink::Role::kSuggestion: |
| UseCounter::Count(GetDocument(), WebFeature::kARIAAnnotations); |
| if (GetElement() && |
| !RuntimeEnabledFeatures::AccessibilityExposeARIAAnnotationsEnabled( |
| GetElement()->GetExecutionContext())) { |
| role = ax::mojom::blink::Role::kGenericContainer; |
| } |
| break; |
| default: |
| break; |
| } |
| |
| // ARIA states if an item can get focus, it should not be presentational. |
| // It also states user agents should ignore the presentational role if |
| // the element has global ARIA states and properties. |
| if ((role == ax::mojom::blink::Role::kNone || |
| role == ax::mojom::blink::Role::kPresentational) && |
| (CanSetFocusAttribute() || HasGlobalARIAAttribute())) |
| return ax::mojom::blink::Role::kUnknown; |
| |
| if (role == ax::mojom::blink::Role::kButton) |
| role = ButtonRoleType(); |
| |
| role = RemapAriaRoleDueToParent(role); |
| |
| // Distinguish between different uses of the "combobox" role: |
| // |
| // ax::mojom::blink::Role::kComboBoxGrouping: |
| // <div role="combobox"><input></div> |
| // ax::mojom::blink::Role::kTextFieldWithComboBox: |
| // <input role="combobox"> |
| // ax::mojom::blink::Role::kComboBoxMenuButton: |
| // <div tabindex=0 role="combobox">Select</div> |
| if (role == ax::mojom::blink::Role::kComboBoxGrouping) { |
| if (IsNativeTextControl()) |
| role = ax::mojom::blink::Role::kTextFieldWithComboBox; |
| else if (GetElement() && GetElement()->SupportsFocus()) |
| role = ax::mojom::blink::Role::kComboBoxMenuButton; |
| } |
| |
| if (role != ax::mojom::blink::Role::kUnknown) |
| return role; |
| |
| return ax::mojom::blink::Role::kUnknown; |
| } |
| |
| ax::mojom::blink::Role AXObject::RemapAriaRoleDueToParent( |
| ax::mojom::blink::Role role) const { |
| // Some objects change their role based on their parent. |
| // However, asking for the unignoredParent calls accessibilityIsIgnored(), |
| // which can trigger a loop. While inside the call stack of creating an |
| // element, we need to avoid accessibilityIsIgnored(). |
| // https://bugs.webkit.org/show_bug.cgi?id=65174 |
| |
| // Don't return table roles unless inside a table-like container. |
| switch (role) { |
| case ax::mojom::blink::Role::kRow: |
| case ax::mojom::blink::Role::kRowGroup: |
| case ax::mojom::blink::Role::kCell: |
| case ax::mojom::blink::Role::kRowHeader: |
| case ax::mojom::blink::Role::kColumnHeader: |
| for (AXObject* ancestor = ParentObjectUnignored(); ancestor; |
| ancestor = ancestor->ParentObjectUnignored()) { |
| ax::mojom::blink::Role ancestor_aria_role = |
| ancestor->AriaRoleAttribute(); |
| if (ancestor_aria_role == ax::mojom::blink::Role::kCell) |
| return ax::mojom::blink::Role::kGenericContainer; // In another cell, |
| // illegal. |
| if (ancestor->IsTableLikeRole()) |
| return role; // Inside a table: ARIA role is legal. |
| } |
| return ax::mojom::blink::Role::kGenericContainer; // Not in a table. |
| default: |
| break; |
| } |
| |
| if (role != ax::mojom::blink::Role::kListBoxOption && |
| role != ax::mojom::blink::Role::kMenuItem) |
| return role; |
| |
| for (AXObject* parent = ParentObject(); |
| parent && !parent->AccessibilityIsIgnored(); |
| parent = parent->ParentObject()) { |
| ax::mojom::blink::Role parent_aria_role = parent->AriaRoleAttribute(); |
| |
| // Selects and listboxes both have options as child roles, but they map to |
| // different roles within WebCore. |
| if (role == ax::mojom::blink::Role::kListBoxOption && |
| parent_aria_role == ax::mojom::blink::Role::kMenu) |
| return ax::mojom::blink::Role::kMenuItem; |
| // An aria "menuitem" may map to MenuButton or MenuItem depending on its |
| // parent. |
| if (role == ax::mojom::blink::Role::kMenuItem && |
| parent_aria_role == ax::mojom::blink::Role::kGroup) |
| return ax::mojom::blink::Role::kMenuButton; |
| |
| // If the parent had a different role, then we don't need to continue |
| // searching up the chain. |
| if (parent_aria_role != ax::mojom::blink::Role::kUnknown) |
| break; |
| } |
| |
| return role; |
| } |
| |
| bool AXObject::IsEditableRoot() const { |
| UpdateCachedAttributeValuesIfNeeded(); |
| return cached_is_editable_root_; |
| } |
| |
| AXObject* AXObject::LiveRegionRoot() const { |
| UpdateCachedAttributeValuesIfNeeded(); |
| return cached_live_region_root_; |
| } |
| |
| bool AXObject::LiveRegionAtomic() const { |
| bool atomic = false; |
| if (HasAOMPropertyOrARIAAttribute(AOMBooleanProperty::kAtomic, atomic)) |
| return atomic; |
| |
| // ARIA roles "alert" and "status" should have an implicit aria-atomic value |
| // of true. |
| return RoleValue() == ax::mojom::blink::Role::kAlert || |
| RoleValue() == ax::mojom::blink::Role::kStatus; |
| } |
| |
| const AtomicString& AXObject::ContainerLiveRegionStatus() const { |
| UpdateCachedAttributeValuesIfNeeded(); |
| return cached_live_region_root_ ? cached_live_region_root_->LiveRegionStatus() |
| : g_null_atom; |
| } |
| |
| const AtomicString& AXObject::ContainerLiveRegionRelevant() const { |
| UpdateCachedAttributeValuesIfNeeded(); |
| return cached_live_region_root_ |
| ? cached_live_region_root_->LiveRegionRelevant() |
| : g_null_atom; |
| } |
| |
| bool AXObject::ContainerLiveRegionAtomic() const { |
| UpdateCachedAttributeValuesIfNeeded(); |
| return cached_live_region_root_ && |
| cached_live_region_root_->LiveRegionAtomic(); |
| } |
| |
| bool AXObject::ContainerLiveRegionBusy() const { |
| UpdateCachedAttributeValuesIfNeeded(); |
| return cached_live_region_root_ && |
| cached_live_region_root_->AOMPropertyOrARIAAttributeIsTrue( |
| AOMBooleanProperty::kBusy); |
| } |
| |
| AXObject* AXObject::ElementAccessibilityHitTest(const IntPoint& point) const { |
| // Check if there are any mock elements that need to be handled. |
| for (const auto& child : ChildrenIncludingIgnored()) { |
| if (child->IsMockObject() && |
| child->GetBoundsInFrameCoordinates().Contains(point)) |
| return child->ElementAccessibilityHitTest(point); |
| } |
| |
| return const_cast<AXObject*>(this); |
| } |
| |
| AXObject::AncestorsIterator AXObject::UnignoredAncestorsBegin() const { |
| AXObject* parent = ParentObjectUnignored(); |
| if (parent) |
| return AXObject::AncestorsIterator(*parent); |
| return UnignoredAncestorsEnd(); |
| } |
| |
| AXObject::AncestorsIterator AXObject::UnignoredAncestorsEnd() const { |
| return AXObject::AncestorsIterator(); |
| } |
| |
| AXObject::InOrderTraversalIterator AXObject::GetInOrderTraversalIterator() { |
| return InOrderTraversalIterator(*this); |
| } |
| |
| int AXObject::ChildCountIncludingIgnored() const { |
| return HasIndirectChildren() ? 0 : int{ChildrenIncludingIgnored().size()}; |
| } |
| |
| AXObject* AXObject::ChildAtIncludingIgnored(int index) const { |
| // We need to use "ChildCountIncludingIgnored()" and |
| // "ChildrenIncludingIgnored()" instead of using the "children_" member |
| // directly, because we might need to update children and check for the |
| // presence of indirect children. |
| if (index < 0 || index >= ChildCountIncludingIgnored()) |
| return nullptr; |
| return ChildrenIncludingIgnored()[index]; |
| } |
| |
| const AXObject::AXObjectVector& AXObject::ChildrenIncludingIgnored() const { |
| return const_cast<AXObject*>(this)->ChildrenIncludingIgnored(); |
| } |
| |
| const AXObject::AXObjectVector& AXObject::ChildrenIncludingIgnored() { |
| UpdateChildrenIfNecessary(); |
| return children_; |
| } |
| |
| const AXObject::AXObjectVector AXObject::UnignoredChildren() const { |
| return const_cast<AXObject*>(this)->UnignoredChildren(); |
| } |
| |
| const AXObject::AXObjectVector AXObject::UnignoredChildren() { |
| if (!AccessibilityIsIncludedInTree()) { |
| NOTREACHED() << "We don't support finding the unignored children of " |
| "objects excluded from the accessibility tree."; |
| return {}; |
| } |
| |
| UpdateChildrenIfNecessary(); |
| |
| // Capture only descendants that are not accessibility ignored, and that are |
| // one level deeper than the current object after flattening any accessibility |
| // ignored descendants. |
| // |
| // For example : |
| // ++A |
| // ++++B |
| // ++++C IGNORED |
| // ++++++F |
| // ++++D |
| // ++++++G |
| // ++++E IGNORED |
| // ++++++H IGNORED |
| // ++++++++J |
| // ++++++I |
| // |
| // Objects [B, F, D, I, J] will be returned, since after flattening all |
| // ignored objects ,those are the ones that are one level deep. |
| |
| AXObjectVector unignored_children; |
| AXObject* child = FirstChildIncludingIgnored(); |
| while (child && child != this) { |
| if (child->AccessibilityIsIgnored()) { |
| child = child->NextInPreOrderIncludingIgnored(this); |
| continue; |
| } |
| |
| unignored_children.push_back(child); |
| for (; child != this; child = child->ParentObjectIncludedInTree()) { |
| if (AXObject* sibling = child->NextSiblingIncludingIgnored()) { |
| child = sibling; |
| break; |
| } |
| } |
| } |
| |
| return unignored_children; |
| } |
| |
| AXObject* AXObject::FirstChildIncludingIgnored() const { |
| return ChildCountIncludingIgnored() ? *ChildrenIncludingIgnored().begin() |
| : nullptr; |
| } |
| |
| AXObject* AXObject::LastChildIncludingIgnored() const { |
| return ChildCountIncludingIgnored() ? *(ChildrenIncludingIgnored().end() - 1) |
| : nullptr; |
| } |
| |
| AXObject* AXObject::DeepestFirstChildIncludingIgnored() const { |
| if (!ChildCountIncludingIgnored()) |
| return nullptr; |
| |
| AXObject* deepest_child = FirstChildIncludingIgnored(); |
| while (deepest_child->ChildCountIncludingIgnored()) |
| deepest_child = deepest_child->FirstChildIncludingIgnored(); |
| |
| return deepest_child; |
| } |
| |
| AXObject* AXObject::DeepestLastChildIncludingIgnored() const { |
| if (!ChildCountIncludingIgnored()) |
| return nullptr; |
| |
| AXObject* deepest_child = LastChildIncludingIgnored(); |
| while (deepest_child->ChildCountIncludingIgnored()) |
| deepest_child = deepest_child->LastChildIncludingIgnored(); |
| |
| return deepest_child; |
| } |
| |
| bool AXObject::IsAncestorOf(const AXObject& descendant) const { |
| return descendant.IsDescendantOf(*this); |
| } |
| |
| bool AXObject::IsDescendantOf(const AXObject& ancestor) const { |
| const AXObject* parent = ParentObject(); |
| while (parent && parent != &ancestor) |
| parent = parent->ParentObject(); |
| return !!parent; |
| } |
| |
| AXObject* AXObject::NextSiblingIncludingIgnored() const { |
| if (!AccessibilityIsIncludedInTree()) { |
| NOTREACHED() << "We don't support iterating over objects excluded " |
| "from the accessibility tree."; |
| return nullptr; |
| } |
| |
| const AXObject* parent_in_tree = ParentObjectIncludedInTree(); |
| if (!parent_in_tree) |
| return nullptr; |
| |
| const int index_in_parent = IndexInParent(); |
| if (index_in_parent < parent_in_tree->ChildCountIncludingIgnored() - 1) |
| return parent_in_tree->ChildAtIncludingIgnored(index_in_parent + 1); |
| return nullptr; |
| } |
| |
| AXObject* AXObject::PreviousSiblingIncludingIgnored() const { |
| if (!AccessibilityIsIncludedInTree()) { |
| NOTREACHED() << "We don't support iterating over objects excluded " |
| "from the accessibility tree."; |
| return nullptr; |
| } |
| |
| const AXObject* parent_in_tree = ParentObjectIncludedInTree(); |
| if (!parent_in_tree) |
| return nullptr; |
| |
| const int index_in_parent = IndexInParent(); |
| if (index_in_parent > 0) |
| return parent_in_tree->ChildAtIncludingIgnored(index_in_parent - 1); |
| return nullptr; |
| } |
| |
| AXObject* AXObject::NextInPreOrderIncludingIgnored( |
| const AXObject* within) const { |
| if (!AccessibilityIsIncludedInTree()) { |
| NOTREACHED() << "We don't support iterating over objects excluded " |
| "from the accessibility tree."; |
| return nullptr; |
| } |
| |
| if (ChildCountIncludingIgnored()) |
| return FirstChildIncludingIgnored(); |
| |
| if (within == this) |
| return nullptr; |
| |
| const AXObject* current = this; |
| AXObject* next = current->NextSiblingIncludingIgnored(); |
| for (; !next; next = current->NextSiblingIncludingIgnored()) { |
| current = current->ParentObjectIncludedInTree(); |
| if (!current || within == current) |
| return nullptr; |
| } |
| return next; |
| } |
| |
| AXObject* AXObject::PreviousInPreOrderIncludingIgnored( |
| const AXObject* within) const { |
| if (!AccessibilityIsIncludedInTree()) { |
| NOTREACHED() << "We don't support iterating over objects excluded " |
| "from the accessibility tree."; |
| return nullptr; |
| } |
| if (within == this) |
| return nullptr; |
| |
| if (AXObject* sibling = PreviousSiblingIncludingIgnored()) { |
| if (sibling->ChildCountIncludingIgnored()) |
| return sibling->DeepestLastChildIncludingIgnored(); |
| return sibling; |
| } |
| |
| return ParentObjectIncludedInTree(); |
| } |
| |
| AXObject* AXObject::PreviousInPostOrderIncludingIgnored( |
| const AXObject* within) const { |
| if (!AccessibilityIsIncludedInTree()) { |
| NOTREACHED() << "We don't support iterating over objects excluded " |
| "from the accessibility tree."; |
| return nullptr; |
| } |
| |
| if (ChildCountIncludingIgnored()) |
| return LastChildIncludingIgnored(); |
| |
| if (within == this) |
| return nullptr; |
| |
| const AXObject* current = this; |
| AXObject* previous = current->PreviousSiblingIncludingIgnored(); |
| for (; !previous; previous = current->PreviousSiblingIncludingIgnored()) { |
| current = current->ParentObjectIncludedInTree(); |
| if (!current || within == current) |
| return nullptr; |
| } |
| return previous; |
| } |
| |
| int AXObject::UnignoredChildCount() const { |
| return int{UnignoredChildren().size()}; |
| } |
| |
| AXObject* AXObject::UnignoredChildAt(int index) const { |
| const AXObjectVector unignored_children = UnignoredChildren(); |
| if (index < 0 || index >= int{unignored_children.size()}) |
| return nullptr; |
| return unignored_children[index]; |
| } |
| |
| AXObject* AXObject::UnignoredNextSibling() const { |
| if (AccessibilityIsIgnored()) { |
| NOTREACHED() << "We don't support finding unignored siblings for ignored " |
| "objects because it is not clear whether to search for the " |
| "sibling in the unignored tree or in the whole tree."; |
| return nullptr; |
| } |
| |
| // Find the next sibling for the same unignored parent object, |
| // flattening accessibility ignored objects. |
| // |
| // For example : |
| // ++A |
| // ++++B |
| // ++++C IGNORED |
| // ++++++E |
| // ++++D |
| // Objects [B, E, D] will be siblings since C is ignored. |
| |
| const AXObject* unignored_parent = ParentObjectUnignored(); |
| const AXObject* current_obj = this; |
| while (current_obj) { |
| AXObject* sibling = current_obj->NextSiblingIncludingIgnored(); |
| if (sibling) { |
| // If we found an ignored sibling, walk in next pre-order |
| // until an unignored object is found, flattening the ignored object. |
| while (sibling && sibling->AccessibilityIsIgnored()) { |
| sibling = sibling->NextInPreOrderIncludingIgnored(unignored_parent); |
| } |
| return sibling; |
| } |
| |
| // If a sibling has not been found, try again with the parent object, |
| // until the unignored parent is reached. |
| current_obj = current_obj->ParentObjectIncludedInTree(); |
| if (!current_obj || !current_obj->AccessibilityIsIgnored()) |
| return nullptr; |
| } |
| return nullptr; |
| } |
| |
| AXObject* AXObject::UnignoredPreviousSibling() const { |
| if (AccessibilityIsIgnored()) { |
| NOTREACHED() << "We don't support finding unignored siblings for ignored " |
| "objects because it is not clear whether to search for the " |
| "sibling in the unignored tree or in the whole tree."; |
| return nullptr; |
| } |
| |
| // Find the previous sibling for the same unignored parent object, |
| // flattening accessibility ignored objects. |
| // |
| // For example : |
| // ++A |
| // ++++B |
| // ++++C IGNORED |
| // ++++++E |
| // ++++D |
| // Objects [B, E, D] will be siblings since C is ignored. |
| |
| const AXObject* current_obj = this; |
| while (current_obj) { |
| AXObject* sibling = current_obj->PreviousSiblingIncludingIgnored(); |
| if (sibling) { |
| const AXObject* unignored_parent = ParentObjectUnignored(); |
| // If we found an ignored sibling, walk in previous post-order |
| // until an unignored object is found, flattening the ignored object. |
| while (sibling && sibling->AccessibilityIsIgnored()) { |
| sibling = |
| sibling->PreviousInPostOrderIncludingIgnored(unignored_parent); |
| } |
| return sibling; |
| } |
| |
| // If a sibling has not been found, try again with the parent object, |
| // until the unignored parent is reached. |
| current_obj = current_obj->ParentObjectIncludedInTree(); |
| if (!current_obj || !current_obj->AccessibilityIsIgnored()) |
| return nullptr; |
| } |
| return nullptr; |
| } |
| |
| AXObject* AXObject::UnignoredNextInPreOrder() const { |
| AXObject* next = NextInPreOrderIncludingIgnored(); |
| while (next && next->AccessibilityIsIgnored()) { |
| next = next->NextInPreOrderIncludingIgnored(); |
| } |
| return next; |
| } |
| |
| AXObject* AXObject::UnignoredPreviousInPreOrder() const { |
| AXObject* previous = PreviousInPreOrderIncludingIgnored(); |
| while (previous && previous->AccessibilityIsIgnored()) { |
| previous = previous->PreviousInPreOrderIncludingIgnored(); |
| } |
| return previous; |
| } |
| |
| AXObject* AXObject::ParentObject() const { |
| if (IsDetached()) |
| return nullptr; |
| |
| if (parent_) |
| return parent_; |
| |
| if (AXObjectCache().IsAriaOwned(this)) |
| return AXObjectCache().GetAriaOwnedParent(this); |
| |
| return ComputeParent(); |
| } |
| |
| AXObject* AXObject::ParentObjectIfExists() const { |
| if (IsDetached()) |
| return nullptr; |
| |
| if (parent_) |
| return parent_; |
| |
| return ComputeParentIfExists(); |
| } |
| |
| AXObject* AXObject::ParentObjectUnignored() const { |
| AXObject* parent; |
| for (parent = ParentObject(); parent && parent->AccessibilityIsIgnored(); |
| parent = parent->ParentObject()) { |
| } |
| |
| return parent; |
| } |
| |
| AXObject* AXObject::ParentObjectIncludedInTree() const { |
| AXObject* parent; |
| for (parent = ParentObject(); |
| parent && !parent->AccessibilityIsIncludedInTree(); |
| parent = parent->ParentObject()) { |
| } |
| |
| return parent; |
| } |
| |
| // Container widgets are those that a user tabs into and arrows around |
| // sub-widgets |
| bool AXObject::IsContainerWidget() const { |
| return ui::IsContainerWithSelectableChildren(RoleValue()); |
| } |
| |
| AXObject* AXObject::ContainerWidget() const { |
| AXObject* ancestor = ParentObjectUnignored(); |
| while (ancestor && !ancestor->IsContainerWidget()) |
| ancestor = ancestor->ParentObjectUnignored(); |
| |
| return ancestor; |
| } |
| |
| void AXObject::UpdateChildrenIfNecessary() { |
| if (!HasChildren()) |
| AddChildren(); |
| } |
| |
| void AXObject::ClearChildren() { |
| // Detach all weak pointers from objects to their parents. |
| for (const auto& child : children_) { |
| if (child->parent_ == this) |
| child->DetachFromParent(); |
| } |
| |
| children_.clear(); |
| have_children_ = false; |
| } |
| |
| void AXObject::AddAccessibleNodeChildren() { |
| Element* element = GetElement(); |
| if (!element) |
| return; |
| |
| AccessibleNode* accessible_node = element->ExistingAccessibleNode(); |
| if (!accessible_node) |
| return; |
| |
| for (const auto& child : accessible_node->GetChildren()) |
| children_.push_back(AXObjectCache().GetOrCreate(child)); |
| } |
| |
| Element* AXObject::GetElement() const { |
| return DynamicTo<Element>(GetNode()); |
| } |
| |
| Document* AXObject::GetDocument() const { |
| LocalFrameView* frame_view = DocumentFrameView(); |
| if (!frame_view) |
| return nullptr; |
| |
| return frame_view->GetFrame().GetDocument(); |
| } |
| |
| AXObject* AXObject::RootScroller() const { |
| Node* global_root_scroller = GetDocument() |
| ->GetPage() |
| ->GlobalRootScrollerController() |
| .GlobalRootScroller(); |
| if (!global_root_scroller) |
| return nullptr; |
| |
| // Only return the root scroller if it's part of the same document. |
| if (global_root_scroller->GetDocument() != GetDocument()) |
| return nullptr; |
| |
| return AXObjectCache().GetOrCreate(global_root_scroller); |
| } |
| |
| LocalFrameView* AXObject::DocumentFrameView() const { |
| const AXObject* object = this; |
| while (object && !object->IsAXLayoutObject()) |
| object = object->ParentObject(); |
| |
| if (!object) |
| return nullptr; |
| |
| return object->DocumentFrameView(); |
| } |
| |
| AtomicString AXObject::Language() const { |
| // This method is used when the style engine is either not available on this |
| // object, e.g. for canvas fallback content, or is unable to determine the |
| // document's language. We use the following signals to detect the element's |
| // language, in decreasing priority: |
| // 1. The [language of a node] as defined in HTML, if known. |
| // 2. The list of languages the browser sends in the [Accept-Language] header. |
| // 3. The browser's default language. |
| |
| const AtomicString& lang = GetAttribute(html_names::kLangAttr); |
| if (!lang.IsEmpty()) |
| return lang; |
| |
| AXObject* parent = ParentObject(); |
| if (parent) |
| return parent->Language(); |
| |
| const Document* document = GetDocument(); |
| if (document) { |
| // Fall back to the first content language specified in the meta tag. |
| // This is not part of what the HTML5 Standard suggests but it still appears |
| // to be necessary. |
| if (document->ContentLanguage()) { |
| const String content_languages = document->ContentLanguage(); |
| Vector<String> languages; |
| content_languages.Split(',', languages); |
| if (!languages.IsEmpty()) |
| return AtomicString(languages[0].StripWhiteSpace()); |
| } |
| |
| if (document->GetPage()) { |
| // Use the first accept language preference if present. |
| const String accept_languages = |
| document->GetPage()->GetChromeClient().AcceptLanguages(); |
| Vector<String> languages; |
| accept_languages.Split(',', languages); |
| if (!languages.IsEmpty()) |
| return AtomicString(languages[0].StripWhiteSpace()); |
| } |
| } |
| |
| // As a last resort, return the default language of the browser's UI. |
| AtomicString default_language = DefaultLanguage(); |
| return default_language; |
| } |
| |
| bool AXObject::HasAttribute(const QualifiedName& attribute) const { |
| Element* element = GetElement(); |
| if (!element) |
| return false; |
| if (element->FastHasAttribute(attribute)) |
| return true; |
| return HasInternalsAttribute(*element, attribute); |
| } |
| |
| const AtomicString& AXObject::GetAttribute( |
| const QualifiedName& attribute) const { |
| Element* element = GetElement(); |
| if (!element) |
| return g_null_atom; |
| const AtomicString& value = element->FastGetAttribute(attribute); |
| if (!value.IsNull()) |
| return value; |
| return GetInternalsAttribute(*element, attribute); |
| } |
| |
| bool AXObject::HasInternalsAttribute(Element& element, |
| const QualifiedName& attribute) const { |
| if (!element.DidAttachInternals()) |
| return false; |
| return element.EnsureElementInternals().HasAttribute(attribute); |
| } |
| |
| const AtomicString& AXObject::GetInternalsAttribute( |
| Element& element, |
| const QualifiedName& attribute) const { |
| if (!element.DidAttachInternals()) |
| return g_null_atom; |
| return element.EnsureElementInternals().FastGetAttribute(attribute); |
| } |
| |
| // |
| // Scrollable containers. |
| // |
| |
| bool AXObject::IsScrollableContainer() const { |
| return !!GetScrollableAreaIfScrollable(); |
| } |
| |
| bool AXObject::IsUserScrollable() const { |
| // TODO(accessibility) Actually expose correct info on whether a doc is |
| // is scrollable or not. Unfortunately IsScrollableContainer() always returns |
| // true anyway. For now, just expose as scrollable unless overflow is hidden. |
| if (IsWebArea()) { |
| if (!GetScrollableAreaIfScrollable() || !GetLayoutObject()) |
| return false; |
| |
| const ComputedStyle* style = GetLayoutObject()->Style(); |
| if (!style) |
| return false; |
| |
| return style->ScrollsOverflowY() || style->ScrollsOverflowX(); |
| } |
| |
| return GetLayoutObject() && GetLayoutObject()->IsBox() && |
| ToLayoutBox(GetLayoutObject())->CanBeScrolledAndHasScrollableArea(); |
| } |
| |
| IntPoint AXObject::GetScrollOffset() const { |
| ScrollableArea* area = GetScrollableAreaIfScrollable(); |
| if (!area) |
| return IntPoint(); |
| |
| return IntPoint(area->ScrollOffsetInt().Width(), |
| area->ScrollOffsetInt().Height()); |
| } |
| |
| IntPoint AXObject::MinimumScrollOffset() const { |
| ScrollableArea* area = GetScrollableAreaIfScrollable(); |
| if (!area) |
| return IntPoint(); |
| |
| return IntPoint(area->MinimumScrollOffsetInt().Width(), |
| area->MinimumScrollOffsetInt().Height()); |
| } |
| |
| IntPoint AXObject::MaximumScrollOffset() const { |
| ScrollableArea* area = GetScrollableAreaIfScrollable(); |
| if (!area) |
| return IntPoint(); |
| |
| return IntPoint(area->MaximumScrollOffsetInt().Width(), |
| area->MaximumScrollOffsetInt().Height()); |
| } |
| |
| void AXObject::SetScrollOffset(const IntPoint& offset) const { |
| ScrollableArea* area = GetScrollableAreaIfScrollable(); |
| if (!area) |
| return; |
| |
| // TODO(bokan): This should potentially be a UserScroll. |
| area->SetScrollOffset(ScrollOffset(offset.X(), offset.Y()), |
| mojom::blink::ScrollType::kProgrammatic); |
| } |
| |
| bool AXObject::IsTableLikeRole() const { |
| return ui::IsTableLike(RoleValue()) || |
| RoleValue() == ax::mojom::blink::Role::kLayoutTable; |
| } |
| |
| bool AXObject::IsTableRowLikeRole() const { |
| return ui::IsTableRow(RoleValue()) || |
| RoleValue() == ax::mojom::blink::Role::kLayoutTableRow; |
| } |
| |
| bool AXObject::IsTableCellLikeRole() const { |
| return ui::IsCellOrTableHeader(RoleValue()) || |
| RoleValue() == ax::mojom::blink::Role::kLayoutTableCell; |
| } |
| |
| unsigned AXObject::ColumnCount() const { |
| if (!IsTableLikeRole()) |
| return 0; |
| |
| unsigned max_column_count = 0; |
| for (const auto& row : TableRowChildren()) { |
| unsigned column_count = row->TableCellChildren().size(); |
| max_column_count = std::max(column_count, max_column_count); |
| } |
| |
| return max_column_count; |
| } |
| |
| unsigned AXObject::RowCount() const { |
| if (!IsTableLikeRole()) |
| return 0; |
| |
| return TableRowChildren().size(); |
| } |
| |
| void AXObject::ColumnHeaders(AXObjectVector& headers) const { |
| if (!IsTableLikeRole()) |
| return; |
| |
| for (const auto& row : TableRowChildren()) { |
| for (const auto& cell : row->TableCellChildren()) { |
| if (cell->RoleValue() == ax::mojom::blink::Role::kColumnHeader) |
| headers.push_back(cell); |
| } |
| } |
| } |
| |
| void AXObject::RowHeaders(AXObjectVector& headers) const { |
| if (!IsTableLikeRole()) |
| return; |
| |
| for (const auto& row : TableRowChildren()) { |
| for (const auto& cell : row->TableCellChildren()) { |
| if (cell->RoleValue() == ax::mojom::blink::Role::kRowHeader) |
| headers.push_back(cell); |
| } |
| } |
| } |
| |
| AXObject* AXObject::CellForColumnAndRow(unsigned target_column_index, |
| unsigned target_row_index) const { |
| if (!IsTableLikeRole()) |
| return nullptr; |
| |
| // Note that this code is only triggered if this is not a LayoutTable, |
| // i.e. it's an ARIA grid/table. |
| // |
| // TODO(dmazzoni): delete this code or rename it "for testing only" |
| // since it's only needed for Blink web tests and not for production. |
| unsigned row_index = 0; |
| for (const auto& row : TableRowChildren()) { |
| unsigned column_index = 0; |
| for (const auto& cell : row->TableCellChildren()) { |
| if (target_column_index == column_index && target_row_index == row_index) |
| return cell; |
| column_index++; |
| } |
| row_index++; |
| } |
| |
| return nullptr; |
| } |
| |
| int AXObject::AriaColumnCount() const { |
| if (!IsTableLikeRole()) |
| return 0; |
| |
| int32_t col_count; |
| if (!HasAOMPropertyOrARIAAttribute(AOMIntProperty::kColCount, col_count)) |
| return 0; |
| |
| if (col_count > static_cast<int>(ColumnCount())) |
| return col_count; |
| |
| // Spec says that if all of the columns are present in the DOM, it |
| // is not necessary to set this attribute as the user agent can |
| // automatically calculate the total number of columns. |
| // It returns 0 in order not to set this attribute. |
| if (col_count == static_cast<int>(ColumnCount()) || col_count != -1) |
| return 0; |
| |
| return -1; |
| } |
| |
| int AXObject::AriaRowCount() const { |
| if (!IsTableLikeRole()) |
| return 0; |
| |
| int32_t row_count; |
| if (!HasAOMPropertyOrARIAAttribute(AOMIntProperty::kRowCount, row_count)) |
| return 0; |
| |
| if (row_count > int{RowCount()}) |
| return row_count; |
| |
| // Spec says that if all of the rows are present in the DOM, it is |
| // not necessary to set this attribute as the user agent can |
| // automatically calculate the total number of rows. |
| // It returns 0 in order not to set this attribute. |
| if (row_count == int{RowCount()} || row_count != -1) |
| return 0; |
| |
| // In the spec, -1 explicitly means an unknown number of rows. |
| return -1; |
| } |
| |
| unsigned AXObject::ColumnIndex() const { |
| return 0; |
| } |
| |
| unsigned AXObject::RowIndex() const { |
| return 0; |
| } |
| |
| unsigned AXObject::ColumnSpan() const { |
| return IsTableCellLikeRole() ? 1 : 0; |
| } |
| |
| unsigned AXObject::RowSpan() const { |
| return IsTableCellLikeRole() ? 1 : 0; |
| } |
| |
| unsigned AXObject::AriaColumnIndex() const { |
| UpdateCachedAttributeValuesIfNeeded(); |
| return cached_aria_column_index_; |
| } |
| |
| unsigned AXObject::AriaRowIndex() const { |
| UpdateCachedAttributeValuesIfNeeded(); |
| return cached_aria_row_index_; |
| } |
| |
| unsigned AXObject::ComputeAriaColumnIndex() const { |
| // Return the ARIA column index if it has been set. Otherwise return a default |
| // value of 0. |
| uint32_t col_index = 0; |
| HasAOMPropertyOrARIAAttribute(AOMUIntProperty::kColIndex, col_index); |
| return col_index; |
| } |
| |
| unsigned AXObject::ComputeAriaRowIndex() const { |
| // Return the ARIA row index if it has been set. Otherwise return a default |
| // value of 0. |
| uint32_t row_index = 0; |
| HasAOMPropertyOrARIAAttribute(AOMUIntProperty::kRowIndex, row_index); |
| return row_index; |
| } |
| |
| AXObject::AXObjectVector AXObject::TableRowChildren() const { |
| AXObjectVector result; |
| for (const auto& child : ChildrenIncludingIgnored()) { |
| if (child->IsTableRowLikeRole()) |
| result.push_back(child); |
| else if (child->RoleValue() == ax::mojom::blink::Role::kRowGroup) |
| result.AppendVector(child->TableRowChildren()); |
| } |
| return result; |
| } |
| |
| AXObject::AXObjectVector AXObject::TableCellChildren() const { |
| AXObjectVector result; |
| for (const auto& child : ChildrenIncludingIgnored()) { |
| if (child->IsTableCellLikeRole()) |
| result.push_back(child); |
| else if (child->RoleValue() == ax::mojom::blink::Role::kGenericContainer) |
| result.AppendVector(child->TableCellChildren()); |
| } |
| return result; |
| } |
| |
| const AXObject* AXObject::TableRowParent() const { |
| const AXObject* row = ParentObjectUnignored(); |
| while (row && !row->IsTableRowLikeRole() && |
| row->RoleValue() == ax::mojom::blink::Role::kGenericContainer) |
| row = row->ParentObjectUnignored(); |
| return row; |
| } |
| |
| const AXObject* AXObject::TableParent() const { |
| const AXObject* table = ParentObjectUnignored(); |
| while (table && !table->IsTableLikeRole() && |
| table->RoleValue() == ax::mojom::blink::Role::kGenericContainer) |
| table = table->ParentObjectUnignored(); |
| return table; |
| } |
| |
| int AXObject::GetDOMNodeId() const { |
| Node* node = GetNode(); |
| if (node) |
| return DOMNodeIds::IdForNode(node); |
| return 0; |
| } |
| |
| void AXObject::GetRelativeBounds(AXObject** out_container, |
| FloatRect& out_bounds_in_container, |
| SkMatrix44& out_container_transform, |
| bool* clips_children) const { |
| *out_container = nullptr; |
| out_bounds_in_container = FloatRect(); |
| out_container_transform.setIdentity(); |
| |
| // First check if it has explicit bounds, for example if this element is tied |
| // to a canvas path. When explicit coordinates are provided, the ID of the |
| // explicit container element that the coordinates are relative to must be |
| // provided too. |
| if (!explicit_element_rect_.IsEmpty()) { |
| *out_container = AXObjectCache().ObjectFromAXID(explicit_container_id_); |
| if (*out_container) { |
| out_bounds_in_container = FloatRect(explicit_element_rect_); |
| return; |
| } |
| } |
| |
| LayoutObject* layout_object = LayoutObjectForRelativeBounds(); |
| if (!layout_object) |
| return; |
| |
| if (clips_children) { |
| if (IsWebArea()) |
| *clips_children = true; |
| else |
| *clips_children = layout_object->HasOverflowClip(); |
| } |
| |
| if (IsWebArea()) { |
| if (layout_object->GetFrame()->View()) { |
| out_bounds_in_container.SetSize( |
| FloatSize(layout_object->GetFrame()->View()->Size())); |
| } |
| return; |
| } |
| |
| // First compute the container. The container must be an ancestor in the |
| // accessibility tree, and its LayoutObject must be an ancestor in the layout |
| // tree. Get the first such ancestor that's either scrollable or has a paint |
| // layer. |
| AXObject* container = ParentObjectUnignored(); |
| LayoutObject* container_layout_object = nullptr; |
| if (layout_object->IsFixedPositioned()) { |
| // If it's a fixed position element, the container should simply be the |
| // root web area. |
| container = AXObjectCache().GetOrCreate(GetDocument()); |
| } else { |
| while (container) { |
| container_layout_object = container->GetLayoutObject(); |
| if (container_layout_object && container_layout_object->IsBox() && |
| layout_object->IsDescendantOf(container_layout_object)) { |
| if (container->IsScrollableContainer() || |
| container_layout_object->HasLayer()) { |
| if (layout_object->IsAbsolutePositioned()) { |
| // If it's absolutely positioned, the container must be the |
| // nearest positioned container, or the root. |
| if (container->IsWebArea()) |
| break; |
| if (container_layout_object->IsPositioned()) |
| break; |
| } else { |
| break; |
| } |
| } |
| } |
| |
| container = container->ParentObjectUnignored(); |
| } |
| } |
| |
| if (!container) |
| return; |
| *out_container = container; |
| out_bounds_in_container = |
| layout_object->LocalBoundingBoxRectForAccessibility(); |
| |
| // Frames need to take their border and padding into account so the |
| // child element's computed position will be correct. |
| if (layout_object->IsBox() && layout_object->GetNode() && |
| layout_object->GetNode()->IsFrameOwnerElement()) { |
| out_bounds_in_container = |
| FloatRect(ToLayoutBox(layout_object)->PhysicalContentBoxRect()); |
| } |
| |
| // If the container has a scroll offset, subtract that out because we want our |
| // bounds to be relative to the *unscrolled* position of the container object. |
| if (auto* scrollable_area = container->GetScrollableAreaIfScrollable()) |
| out_bounds_in_container.Move(scrollable_area->GetScrollOffset()); |
| |
| // Compute the transform between the container's coordinate space and this |
| // object. |
| TransformationMatrix transform = layout_object->LocalToAncestorTransform( |
| ToLayoutBoxModelObject(container_layout_object)); |
| |
| // If the transform is just a simple translation, apply that to the |
| // bounding box, but if it's a non-trivial transformation like a rotation, |
| // scaling, etc. then return the full matrix instead. |
| if (transform.IsIdentityOr2DTranslation()) { |
| out_bounds_in_container.Move(transform.To2DTranslation()); |
| } else { |
| out_container_transform = TransformationMatrix::ToSkMatrix44(transform); |
| } |
| } |
| |
| FloatRect AXObject::LocalBoundingBoxRectForAccessibility() { |
| if (!GetLayoutObject()) |
| return FloatRect(); |
| DCHECK(GetLayoutObject()->IsText()); |
| UpdateCachedAttributeValuesIfNeeded(); |
| return cached_local_bounding_box_rect_for_accessibility_; |
| } |
| |
| LayoutRect AXObject::GetBoundsInFrameCoordinates() const { |
| AXObject* container = nullptr; |
| FloatRect bounds; |
| SkMatrix44 transform; |
| GetRelativeBounds(&container, bounds, transform); |
| FloatRect computed_bounds(0, 0, bounds.Width(), bounds.Height()); |
| while (container && container != this) { |
| computed_bounds.Move(bounds.X(), bounds.Y()); |
| if (!container->IsWebArea()) { |
| computed_bounds.Move(-container->GetScrollOffset().X(), |
| -container->GetScrollOffset().Y()); |
| } |
| if (!transform.isIdentity()) { |
| TransformationMatrix transformation_matrix(transform); |
| transformation_matrix.MapRect(computed_bounds); |
| } |
| container->GetRelativeBounds(&container, bounds, transform); |
| } |
| return LayoutRect(computed_bounds); |
| } |
| |
| // |
| // Modify or take an action on an object. |
| // |
| |
| bool AXObject::RequestDecrementAction() { |
| Event* event = |
| Event::CreateCancelable(event_type_names::kAccessibledecrement); |
| if (DispatchEventToAOMEventListeners(*event)) |
| return true; |
| |
| return OnNativeDecrementAction(); |
| } |
| |
| bool AXObject::RequestClickAction() { |
| Event* event = Event::CreateCancelable(event_type_names::kAccessibleclick); |
| if (DispatchEventToAOMEventListeners(*event)) |
| return true; |
| |
| return OnNativeClickAction(); |
| } |
| |
| bool AXObject::OnNativeClickAction() { |
| Document* document = GetDocument(); |
| if (!document) |
| return false; |
| |
| LocalFrame::NotifyUserActivation( |
| document->GetFrame(), |
| mojom::blink::UserActivationNotificationType::kInteraction); |
| |
| Element* element = GetElement(); |
| if (!element && GetNode()) |
| element = GetNode()->parentElement(); |
| |
| if (IsTextControl()) |
| return OnNativeFocusAction(); |
| |
| if (element) { |
| // Always set the sequential focus navigation starting point. |
| // Even if this element isn't focusable, if you press "Tab" it will |
| // start the search from this element. |
| GetDocument()->SetSequentialFocusNavigationStartingPoint(element); |
| |
| // Explicitly focus the element if it's focusable but not currently |
| // the focused element, to be consistent with |
| // EventHandler::HandleMousePressEvent. |
| if (element->IsMouseFocusable() && !element->IsFocusedElementInDocument()) { |
| Page* const page = GetDocument()->GetPage(); |
| if (page) { |
| page->GetFocusController().SetFocusedElement( |
| element, GetDocument()->GetFrame(), |
| FocusParams(SelectionBehaviorOnFocus::kNone, |
| mojom::blink::FocusType::kMouse, nullptr)); |
| } |
| } |
| |
| // For most elements, AccessKeyAction triggers sending a simulated |
| // click, including simulating the mousedown, mouseup, and click events. |
| element->AccessKeyAction(true); |
| return true; |
| } |
| |
| if (CanSetFocusAttribute()) |
| return OnNativeFocusAction(); |
| |
| return false; |
| } |
| |
| bool AXObject::RequestFocusAction() { |
| Event* event = Event::CreateCancelable(event_type_names::kAccessiblefocus); |
| if (DispatchEventToAOMEventListeners(*event)) |
| return true; |
| |
| return OnNativeFocusAction(); |
| } |
| |
| bool AXObject::RequestIncrementAction() { |
| Event* event = |
| Event::CreateCancelable(event_type_names::kAccessibleincrement); |
| if (DispatchEventToAOMEventListeners(*event)) |
| return true; |
| |
| return OnNativeIncrementAction(); |
| } |
| |
| bool AXObject::RequestScrollToGlobalPointAction(const IntPoint& point) { |
| return OnNativeScrollToGlobalPointAction(point); |
| } |
| |
| bool AXObject::RequestScrollToMakeVisibleAction() { |
| Event* event = |
| Event::CreateCancelable(event_type_names::kAccessiblescrollintoview); |
| if (DispatchEventToAOMEventListeners(*event)) |
| return true; |
| |
| return OnNativeScrollToMakeVisibleAction(); |
| } |
| |
| bool AXObject::RequestScrollToMakeVisibleWithSubFocusAction( |
| const IntRect& subfocus, |
| blink::mojom::blink::ScrollAlignment horizontal_scroll_alignment, |
| blink::mojom::blink::ScrollAlignment vertical_scroll_alignment) { |
| return OnNativeScrollToMakeVisibleWithSubFocusAction( |
| subfocus, horizontal_scroll_alignment, vertical_scroll_alignment); |
| } |
| |
| bool AXObject::RequestSetSelectedAction(bool selected) { |
| return OnNativeSetSelectedAction(selected); |
| } |
| |
| bool AXObject::RequestSetSequentialFocusNavigationStartingPointAction() { |
| return OnNativeSetSequentialFocusNavigationStartingPointAction(); |
| } |
| |
| bool AXObject::RequestSetValueAction(const String& value) { |
| return OnNativeSetValueAction(value); |
| } |
| |
| bool AXObject::RequestShowContextMenuAction() { |
| Event* event = |
| Event::CreateCancelable(event_type_names::kAccessiblecontextmenu); |
| if (DispatchEventToAOMEventListeners(*event)) |
| return true; |
| |
| return OnNativeShowContextMenuAction(); |
| } |
| |
| bool AXObject::InternalClearAccessibilityFocusAction() { |
| return false; |
| } |
| |
| bool AXObject::InternalSetAccessibilityFocusAction() { |
| return false; |
| } |
| |
| LayoutObject* AXObject::GetLayoutObjectForNativeScrollAction() const { |
| Node* node = GetNode(); |
| if (!node || !node->isConnected()) |
| return nullptr; |
| |
| // Node might not have a LayoutObject due to the fact that it is in a locked |
| // subtree. Force the update to create the LayoutObject (and update position |
| // information) for this node. |
| GetDocument()->UpdateStyleAndLayoutForNode( |
| node, DocumentUpdateReason::kDisplayLock); |
| return node->GetLayoutObject(); |
| } |
| |
| bool AXObject::OnNativeScrollToMakeVisibleAction() const { |
| LayoutObject* layout_object = GetLayoutObjectForNativeScrollAction(); |
| if (!layout_object) |
| return false; |
| PhysicalRect target_rect(layout_object->AbsoluteBoundingBoxRect()); |
| layout_object->ScrollRectToVisible( |
| target_rect, |
| ScrollAlignment::CreateScrollIntoViewParams( |
| ScrollAlignment::CenterIfNeeded(), ScrollAlignment::CenterIfNeeded(), |
| mojom::blink::ScrollType::kProgrammatic, false, |
| mojom::blink::ScrollBehavior::kAuto)); |
| AXObjectCache().PostNotification( |
| AXObjectCache().GetOrCreate(GetDocument()->GetLayoutView()), |
| ax::mojom::blink::Event::kLocationChanged); |
| return true; |
| } |
| |
| bool AXObject::OnNativeScrollToMakeVisibleWithSubFocusAction( |
| const IntRect& rect, |
| blink::mojom::blink::ScrollAlignment horizontal_scroll_alignment, |
| blink::mojom::blink::ScrollAlignment vertical_scroll_alignment) const { |
| LayoutObject* layout_object = GetLayoutObjectForNativeScrollAction(); |
| if (!layout_object) |
| return false; |
| |
| PhysicalRect target_rect = |
| layout_object->LocalToAbsoluteRect(PhysicalRect(rect)); |
| layout_object->ScrollRectToVisible( |
| target_rect, ScrollAlignment::CreateScrollIntoViewParams( |
| horizontal_scroll_alignment, vertical_scroll_alignment, |
| mojom::blink::ScrollType::kProgrammatic, |
| false /* make_visible_in_visual_viewport */, |
| mojom::blink::ScrollBehavior::kAuto)); |
| AXObjectCache().PostNotification( |
| AXObjectCache().GetOrCreate(GetDocument()->GetLayoutView()), |
| ax::mojom::blink::Event::kLocationChanged); |
| return true; |
| } |
| |
| bool AXObject::OnNativeScrollToGlobalPointAction( |
| const IntPoint& global_point) const { |
| LayoutObject* layout_object = GetLayoutObjectForNativeScrollAction(); |
| if (!layout_object) |
| return false; |
| |
| PhysicalRect target_rect(layout_object->AbsoluteBoundingBoxRect()); |
| target_rect.Move(-PhysicalOffset(global_point)); |
| layout_object->ScrollRectToVisible( |
| target_rect, |
| ScrollAlignment::CreateScrollIntoViewParams( |
| ScrollAlignment::LeftAlways(), ScrollAlignment::TopAlways(), |
| mojom::blink::ScrollType::kProgrammatic, false, |
| mojom::blink::ScrollBehavior::kAuto)); |
| AXObjectCache().PostNotification( |
| AXObjectCache().GetOrCreate(GetDocument()->GetLayoutView()), |
| ax::mojom::blink::Event::kLocationChanged); |
| return true; |
| } |
| |
| bool AXObject::OnNativeSetSequentialFocusNavigationStartingPointAction() { |
| // Call it on the nearest ancestor that overrides this with a specific |
| // implementation. |
| if (ParentObject()) { |
| return ParentObject() |
| ->OnNativeSetSequentialFocusNavigationStartingPointAction(); |
| } |
| return false; |
| } |
| |
| bool AXObject::OnNativeDecrementAction() { |
| return false; |
| } |
| |
| bool AXObject::OnNativeFocusAction() { |
| return false; |
| } |
| |
| bool AXObject::OnNativeIncrementAction() { |
| return false; |
| } |
| |
| bool AXObject::OnNativeSetValueAction(const String&) { |
| return false; |
| } |
| |
| bool AXObject::OnNativeSetSelectedAction(bool) { |
| return false; |
| } |
| |
| bool AXObject::OnNativeShowContextMenuAction() { |
| Element* element = GetElement(); |
| if (!element) |
| element = ParentObject() ? ParentObject()->GetElement() : nullptr; |
| if (!element) |
| return false; |
| |
| Document* document = GetDocument(); |
| if (!document || !document->GetFrame()) |
| return false; |
| |
| ContextMenuAllowedScope scope; |
| document->GetFrame()->GetEventHandler().ShowNonLocatedContextMenu(element); |
| return true; |
| } |
| |
| void AXObject::SelectionChanged() { |
| if (AXObject* parent = ParentObjectIfExists()) |
| parent->SelectionChanged(); |
| } |
| |
| // static |
| bool AXObject::IsARIAControl(ax::mojom::blink::Role aria_role) { |
| return IsARIAInput(aria_role) || |
| aria_role == ax::mojom::blink::Role::kButton || |
| aria_role == ax::mojom::blink::Role::kComboBoxMenuButton || |
| aria_role == ax::mojom::blink::Role::kSlider; |
| } |
| |
| // static |
| bool AXObject::IsARIAInput(ax::mojom::blink::Role aria_role) { |
| return aria_role == ax::mojom::blink::Role::kRadioButton || |
| aria_role == ax::mojom::blink::Role::kCheckBox || |
| aria_role == ax::mojom::blink::Role::kTextField || |
| aria_role == ax::mojom::blink::Role::kSwitch || |
| aria_role == ax::mojom::blink::Role::kSearchBox || |
| aria_role == ax::mojom::blink::Role::kTextFieldWithComboBox; |
| } |
| |
| ax::mojom::blink::Role AXObject::AriaRoleToWebCoreRole(const String& value) { |
| DCHECK(!value.IsEmpty()); |
| |
| static const ARIARoleMap* role_map = CreateARIARoleMap(); |
| |
| Vector<String> role_vector; |
| value.Split(' ', role_vector); |
| ax::mojom::blink::Role role = ax::mojom::blink::Role::kUnknown; |
| for (const auto& child : role_vector) { |
| role = role_map->at(child); |
| if (role != ax::mojom::blink::Role::kUnknown) |
| return role; |
| } |
| |
| return role; |
| } |
| |
| bool AXObject::NameFromSelectedOption(bool recursive) const { |
| switch (RoleValue()) { |
| // Step 2E from: http://www.w3.org/TR/accname-aam-1.1 |
| case ax::mojom::blink::Role::kComboBoxGrouping: |
| case ax::mojom::blink::Role::kComboBoxMenuButton: |
| case ax::mojom::blink::Role::kListBox: |
| return recursive; |
| // This can be either a button widget with a non-false value of |
| // aria-haspopup or a select element with size of 1. |
| case ax::mojom::blink::Role::kPopUpButton: |
| return DynamicTo<HTMLSelectElement>(*GetNode()) ? recursive : false; |
| default: |
| return false; |
| } |
| } |
| |
| bool AXObject::NameFromContents(bool recursive) const { |
| // ARIA 1.1, section 5.2.7.5. |
| bool result = false; |
| |
| switch (RoleValue()) { |
| // ----- NameFrom: contents ------------------------- |
| // Get their own name from contents, or contribute to ancestors |
| case ax::mojom::blink::Role::kAnchor: |
| case ax::mojom::blink::Role::kButton: |
| case ax::mojom::blink::Role::kCell: |
| case ax::mojom::blink::Role::kCheckBox: |
| case ax::mojom::blink::Role::kColumnHeader: |
| case ax::mojom::blink::Role::kDocBackLink: |
| case ax::mojom::blink::Role::kDocBiblioRef: |
| case ax::mojom::blink::Role::kDocNoteRef: |
| case ax::mojom::blink::Role::kDocGlossRef: |
| case ax::mojom::blink::Role::kDisclosureTriangle: |
| case ax::mojom::blink::Role::kHeading: |
| case ax::mojom::blink::Role::kLayoutTableCell: |
| case ax::mojom::blink::Role::kLineBreak: |
| case ax::mojom::blink::Role::kLink: |
| case ax::mojom::blink::Role::kListBoxOption: |
| case ax::mojom::blink::Role::kMenuItem: |
| case ax::mojom::blink::Role::kMenuItemCheckBox: |
| case ax::mojom::blink::Role::kMenuItemRadio: |
| case ax::mojom::blink::Role::kMenuListOption: |
| case ax::mojom::blink::Role::kPopUpButton: |
| case ax::mojom::blink::Role::kPortal: |
| case ax::mojom::blink::Role::kRadioButton: |
| case ax::mojom::blink::Role::kRowHeader: |
| case ax::mojom::blink::Role::kStaticText: |
| case ax::mojom::blink::Role::kSwitch: |
| case ax::mojom::blink::Role::kTab: |
| case ax::mojom::blink::Role::kToggleButton: |
| case ax::mojom::blink::Role::kTreeItem: |
| case ax::mojom::blink::Role::kTooltip: |
| result = true; |
| break; |
| |
| // ----- No name from contents ------------------------- |
| // These never have or contribute a name from contents, as they are |
| // containers for many subobjects. Superset of nameFrom:author ARIA roles. |
| case ax::mojom::blink::Role::kAlert: |
| case ax::mojom::blink::Role::kAlertDialog: |
| case ax::mojom::blink::Role::kApplication: |
| case ax::mojom::blink::Role::kAudio: |
| case ax::mojom::blink::Role::kArticle: |
| case ax::mojom::blink::Role::kBanner: |
| case ax::mojom::blink::Role::kBlockquote: |
| case ax::mojom::blink::Role::kCaret: |
| case ax::mojom::blink::Role::kClient: |
| case ax::mojom::blink::Role::kColorWell: |
| case ax::mojom::blink::Role::kColumn: |
| case ax::mojom::blink::Role::kComboBoxMenuButton: // Only value from |
| // content. |
| case ax::mojom::blink::Role::kComboBoxGrouping: |
| case ax::mojom::blink::Role::kComment: |
| case ax::mojom::blink::Role::kComplementary: |
| case ax::mojom::blink::Role::kContentInfo: |
| case ax::mojom::blink::Role::kDate: |
| case ax::mojom::blink::Role::kDateTime: |
| case ax::mojom::blink::Role::kDesktop: |
| case ax::mojom::blink::Role::kDialog: |
| case ax::mojom::blink::Role::kDirectory: |
| case ax::mojom::blink::Role::kDocCover: |
| case ax::mojom::blink::Role::kDocBiblioEntry: |
| case ax::mojom::blink::Role::kDocEndnote: |
| case ax::mojom::blink::Role::kDocFootnote: |
| case ax::mojom::blink::Role::kDocPageBreak: |
| case ax::mojom::blink::Role::kDocAbstract: |
| case ax::mojom::blink::Role::kDocAcknowledgments: |
| case ax::mojom::blink::Role::kDocAfterword: |
| case ax::mojom::blink::Role::kDocAppendix: |
| case ax::mojom::blink::Role::kDocBibliography: |
| case ax::mojom::blink::Role::kDocChapter: |
| case ax::mojom::blink::Role::kDocColophon: |
| case ax::mojom::blink::Role::kDocConclusion: |
| case ax::mojom::blink::Role::kDocCredit: |
| case ax::mojom::blink::Role::kDocCredits: |
| case ax::mojom::blink::Role::kDocDedication: |
| case ax::mojom::blink::Role::kDocEndnotes: |
| case ax::mojom::blink::Role::kDocEpigraph: |
| case ax::mojom::blink::Role::kDocEpilogue: |
| case ax::mojom::blink::Role::kDocErrata: |
| case ax::mojom::blink::Role::kDocExample: |
| case ax::mojom::blink::Role::kDocForeword: |
| case ax::mojom::blink::Role::kDocGlossary: |
| case ax::mojom::blink::Role::kDocIndex: |
| case ax::mojom::blink::Role::kDocIntroduction: |
| case ax::mojom::blink::Role::kDocNotice: |
| case ax::mojom::blink::Role::kDocPageList: |
| case ax::mojom::blink::Role::kDocPart: |
| case ax::mojom::blink::Role::kDocPreface: |
| case ax::mojom::blink::Role::kDocPrologue: |
| case ax::mojom::blink::Role::kDocPullquote: |
| case ax::mojom::blink::Role::kDocQna: |
| case ax::mojom::blink::Role::kDocSubtitle: |
| case ax::mojom::blink::Role::kDocTip: |
| case ax::mojom::blink::Role::kDocToc: |
| case ax::mojom::blink::Role::kDocument: |
| case ax::mojom::blink::Role::kEmbeddedObject: |
| case ax::mojom::blink::Role::kFeed: |
| case ax::mojom::blink::Role::kFigure: |
| case ax::mojom::blink::Role::kForm: |
| case ax::mojom::blink::Role::kGraphicsDocument: |
| case ax::mojom::blink::Role::kGraphicsObject: |
| case ax::mojom::blink::Role::kGraphicsSymbol: |
| case ax::mojom::blink::Role::kGrid: |
| case ax::mojom::blink::Role::kGroup: |
| case ax::mojom::blink::Role::kHeader: |
| case ax::mojom::blink::Role::kIframePresentational: |
| case ax::mojom::blink::Role::kIframe: |
| case ax::mojom::blink::Role::kImage: |
| case ax::mojom::blink::Role::kImeCandidate: // Internal role, not used on |
| // the web. |
| case ax::mojom::blink::Role::kInputTime: |
| case ax::mojom::blink::Role::kKeyboard: |
| case ax::mojom::blink::Role::kListBox: |
| case ax::mojom::blink::Role::kListGrid: |
| case ax::mojom::blink::Role::kLog: |
| case ax::mojom::blink::Role::kMain: |
| case ax::mojom::blink::Role::kMarquee: |
| case ax::mojom::blink::Role::kMath: |
| case ax::mojom::blink::Role::kMenuListPopup: |
| case ax::mojom::blink::Role::kMenu: |
| case ax::mojom::blink::Role::kMenuBar: |
| case ax::mojom::blink::Role::kMenuButton: // Only value from content, not |
| // name. |
| case ax::mojom::blink::Role::kMeter: |
| case ax::mojom::blink::Role::kNavigation: |
| case ax::mojom::blink::Role::kNote: |
| case ax::mojom::blink::Role::kPane: |
| case ax::mojom::blink::Role::kPluginObject: |
| case ax::mojom::blink::Role::kProgressIndicator: |
| case ax::mojom::blink::Role::kRadioGroup: |
| case ax::mojom::blink::Role::kRowGroup: |
| case ax::mojom::blink::Role::kScrollBar: |
| case ax::mojom::blink::Role::kScrollView: |
| case ax::mojom::blink::Role::kSearch: |
| case ax::mojom::blink::Role::kSearchBox: |
| case ax::mojom::blink::Role::kSplitter: |
| case ax::mojom::blink::Role::kSlider: |
| case ax::mojom::blink::Role::kSpinButton: |
| case ax::mojom::blink::Role::kStatus: |
| case ax::mojom::blink::Role::kSliderThumb: |
| case ax::mojom::blink::Role::kSuggestion: |
| case ax::mojom::blink::Role::kSvgRoot: |
| case ax::mojom::blink::Role::kTable: |
| case ax::mojom::blink::Role::kTableHeaderContainer: |
| case ax::mojom::blink::Role::kTabList: |
| case ax::mojom::blink::Role::kTabPanel: |
| case ax::mojom::blink::Role::kTerm: |
| case ax::mojom::blink::Role::kTextField: |
| case ax::mojom::blink::Role::kTextFieldWithComboBox: |
| case ax::mojom::blink::Role::kTitleBar: |
| case ax::mojom::blink::Role::kTimer: |
| case ax::mojom::blink::Role::kToolbar: |
| case ax::mojom::blink::Role::kTree: |
| case ax::mojom::blink::Role::kTreeGrid: |
| case ax::mojom::blink::Role::kVideo: |
| case ax::mojom::blink::Role::kWebArea: |
| case ax::mojom::blink::Role::kWebView: |
| case ax::mojom::blink::Role::kWindow: |
| result = false; |
| break; |
| |
| // ----- Conditional: contribute to ancestor only, unless focusable ------- |
| // Some objects can contribute their contents to ancestor names, but |
| // only have their own name if they are focusable |
| case ax::mojom::blink::Role::kAbbr: |
| case ax::mojom::blink::Role::kCanvas: |
| case ax::mojom::blink::Role::kCaption: |
| case ax::mojom::blink::Role::kCode: |
| case ax::mojom::blink::Role::kContentDeletion: |
| case ax::mojom::blink::Role::kContentInsertion: |
| case ax::mojom::blink::Role::kDefinition: |
| case ax::mojom::blink::Role::kDescriptionListDetail: |
| case ax::mojom::blink::Role::kDescriptionList: |
| case ax::mojom::blink::Role::kDescriptionListTerm: |
| case ax::mojom::blink::Role::kDetails: |
| case ax::mojom::blink::Role::kEmphasis: |
| case ax::mojom::blink::Role::kFigcaption: |
| case ax::mojom::blink::Role::kFooter: |
| case ax::mojom::blink::Role::kFooterAsNonLandmark: |
| case ax::mojom::blink::Role::kGenericContainer: |
| case ax::mojom::blink::Role::kHeaderAsNonLandmark: |
| case ax::mojom::blink::Role::kIgnored: |
| case ax::mojom::blink::Role::kImageMap: |
| case ax::mojom::blink::Role::kInlineTextBox: |
| case ax::mojom::blink::Role::kLabelText: |
| case ax::mojom::blink::Role::kLayoutTable: |
| case ax::mojom::blink::Role::kLayoutTableRow: |
| case ax::mojom::blink::Role::kLegend: |
| case ax::mojom::blink::Role::kList: |
| case ax::mojom::blink::Role::kListItem: |
| case ax::mojom::blink::Role::kListMarker: |
| case ax::mojom::blink::Role::kMark: |
| case ax::mojom::blink::Role::kNone: |
| case ax::mojom::blink::Role::kParagraph: |
| case ax::mojom::blink::Role::kPre: |
| case ax::mojom::blink::Role::kPresentational: |
| case ax::mojom::blink::Role::kRegion: |
| // Spec says we should always expose the name on rows, |
| // but for performance reasons we only do it |
| // if the row might receive focus |
| case ax::mojom::blink::Role::kRow: |
| case ax::mojom::blink::Role::kRuby: |
| case ax::mojom::blink::Role::kRubyAnnotation: |
| case ax::mojom::blink::Role::kSection: |
| case ax::mojom::blink::Role::kStrong: |
| case ax::mojom::blink::Role::kTime: |
| if (recursive) { |
| // Use contents if part of a recursive name computation. |
| result = true; |
| } else { |
| // Use contents if focusable, so that there is a name in the case |
| // where the author mistakenly forgot to provide one. |
| // Exceptions: |
| // 1.Elements with contenteditable, where using the contents as a name |
| // would cause them to be double-announced. |
| // 2.Containers with aria-activedescendant, where the focus is being |
| // forwarded somewhere else. |
| // TODO(accessibility) Scrollables are currently allowed here in order |
| // to keep the current behavior. In the future, this can be removed |
| // because this code will be handled in IsFocusable(), once |
| // KeyboardFocusableScrollersEnabled is permanently enabled. |
| // Note: this uses the same scrollable check that element.cc uses. |
| bool is_focusable_scrollable = |
| RuntimeEnabledFeatures::KeyboardFocusableScrollersEnabled() && |
| IsUserScrollable(); |
| bool is_focusable = is_focusable_scrollable || CanSetFocusAttribute(); |
| result = is_focusable && !IsEditable() && |
| !GetAOMPropertyOrARIAAttribute( |
| AOMRelationProperty::kActiveDescendant); |
| } |
| break; |
| |
| case ax::mojom::blink::Role::kPdfActionableHighlight: |
| LOG(ERROR) << "PDF specific highlight role, Blink shouldn't generate " |
| "this role type"; |
| NOTREACHED(); |
| break; |
| |
| // A root web area normally only computes its name from the document title, |
| // but a root web area inside a portal's main frame should compute its name |
| // from its contents. This name is used by the portal element that hosts |
| // this portal. |
| case ax::mojom::blink::Role::kRootWebArea: { |
| DCHECK(GetNode()); |
| const Document& document = GetNode()->GetDocument(); |
| bool is_main_frame = |
| document.GetFrame() && document.GetFrame()->IsMainFrame(); |
| bool is_inside_portal = |
| document.GetPage() && document.GetPage()->InsidePortal(); |
| return is_inside_portal && is_main_frame; |
| } |
| |
| case ax::mojom::blink::Role::kUnknown: |
| LOG(ERROR) << "ax::mojom::blink::Role::kUnknown for " << GetNode(); |
| NOTREACHED(); |
| break; |
| } |
| |
| return result; |
| } |
| |
| bool AXObject::SupportsARIAReadOnly() const { |
| if (ui::IsReadOnlySupported(RoleValue())) |
| return true; |
| |
| if (ui::IsCellOrTableHeader(RoleValue())) { |
| // For cells and row/column headers, readonly is supported within a grid. |
| return std::any_of( |
| UnignoredAncestorsBegin(), UnignoredAncestorsEnd(), |
| [](const AXObject& ancestor) { |
| return ancestor.RoleValue() == ax::mojom::blink::Role::kGrid || |
| ancestor.RoleValue() == ax::mojom::blink::Role::kTreeGrid; |
| }); |
| } |
| |
| return false; |
| } |
| |
| ax::mojom::blink::Role AXObject::ButtonRoleType() const { |
| // If aria-pressed is present, then it should be exposed as a toggle button. |
| // http://www.w3.org/TR/wai-aria/states_and_properties#aria-pressed |
| if (AriaPressedIsPresent()) |
| return ax::mojom::blink::Role::kToggleButton; |
| if (HasPopup() != ax::mojom::blink::HasPopup::kFalse) |
| return ax::mojom::blink::Role::kPopUpButton; |
| // We don't contemplate RadioButtonRole, as it depends on the input |
| // type. |
| |
| return ax::mojom::blink::Role::kButton; |
| } |
| |
| // static |
| const AtomicString& AXObject::RoleName(ax::mojom::blink::Role role) { |
| static const Vector<AtomicString>* role_name_vector = CreateRoleNameVector(); |
| |
| return role_name_vector->at(static_cast<wtf_size_t>(role)); |
| } |
| |
| // static |
| const AtomicString& AXObject::InternalRoleName(ax::mojom::blink::Role role) { |
| static const Vector<AtomicString>* internal_role_name_vector = |
| CreateInternalRoleNameVector(); |
| |
| return internal_role_name_vector->at(static_cast<wtf_size_t>(role)); |
| } |
| |
| // static |
| const AXObject* AXObject::LowestCommonAncestor(const AXObject& first, |
| const AXObject& second, |
| int* index_in_ancestor1, |
| int* index_in_ancestor2) { |
| *index_in_ancestor1 = -1; |
| *index_in_ancestor2 = -1; |
| |
| if (first.IsDetached() || second.IsDetached()) |
| return nullptr; |
| |
| if (first == second) |
| return &first; |
| |
| HeapVector<Member<const AXObject>> ancestors1; |
| ancestors1.push_back(&first); |
| while (ancestors1.back()) |
| ancestors1.push_back(ancestors1.back()->ParentObjectIncludedInTree()); |
| |
| HeapVector<Member<const AXObject>> ancestors2; |
| ancestors2.push_back(&second); |
| while (ancestors2.back()) |
| ancestors2.push_back(ancestors2.back()->ParentObjectIncludedInTree()); |
| |
| const AXObject* common_ancestor = nullptr; |
| while (!ancestors1.IsEmpty() && !ancestors2.IsEmpty() && |
| ancestors1.back() == ancestors2.back()) { |
| common_ancestor = ancestors1.back(); |
| ancestors1.pop_back(); |
| ancestors2.pop_back(); |
| } |
| |
| if (common_ancestor) { |
| if (!ancestors1.IsEmpty()) |
| *index_in_ancestor1 = ancestors1.back()->IndexInParent(); |
| if (!ancestors2.IsEmpty()) |
| *index_in_ancestor2 = ancestors2.back()->IndexInParent(); |
| } |
| |
| return common_ancestor; |
| } |
| |
| String AXObject::ToString(bool verbose) const { |
| // Build a friendly name for debugging the object. |
| // If verbose, build a longer name name in the form of: |
| // CheckBox axid#28 <input.someClass#cbox1> name="checkbox" |
| String string_builder = |
| AXObject::InternalRoleName(RoleValue()).GetString().EncodeForDebugging(); |
| |
| if (verbose) { |
| string_builder = string_builder + " axid#" + String::Number(AXObjectID()); |
| // Add useful HTML element info, like <div.myClass#myId>. |
| if (GetElement()) { |
| string_builder = |
| string_builder + " <" + GetElement()->tagName().LowerASCII(); |
| if (GetElement()->FastHasAttribute(html_names::kClassAttr)) { |
| string_builder = string_builder + "." + |
| GetElement()->FastGetAttribute(html_names::kClassAttr); |
| } |
| if (GetElement()->FastHasAttribute(html_names::kIdAttr)) { |
| string_builder = string_builder + "#" + |
| GetElement()->FastGetAttribute(html_names::kIdAttr); |
| } |
| string_builder = string_builder + ">"; |
| } |
| |
| string_builder = string_builder + " name="; |
| } else { |
| string_builder = string_builder + ": "; |
| } |
| |
| // Append name last, in case it is long. |
| return string_builder + ComputedName().EncodeForDebugging(); |
| } |
| |
| bool operator==(const AXObject& first, const AXObject& second) { |
| if (first.IsDetached() || second.IsDetached()) |
| return false; |
| if (&first == &second) { |
| DCHECK_EQ(first.AXObjectID(), second.AXObjectID()); |
| return true; |
| } |
| return false; |
| } |
| |
| bool operator!=(const AXObject& first, const AXObject& second) { |
| return !(first == second); |
| } |
| |
| bool operator<(const AXObject& first, const AXObject& second) { |
| if (first.IsDetached() || second.IsDetached()) |
| return false; |
| |
| int index_in_ancestor1, index_in_ancestor2; |
| const AXObject* ancestor = AXObject::LowestCommonAncestor( |
| first, second, &index_in_ancestor1, &index_in_ancestor2); |
| DCHECK_GE(index_in_ancestor1, -1); |
| DCHECK_GE(index_in_ancestor2, -1); |
| if (!ancestor) |
| return false; |
| return index_in_ancestor1 < index_in_ancestor2; |
| } |
| |
| bool operator<=(const AXObject& first, const AXObject& second) { |
| return first == second || first < second; |
| } |
| |
| bool operator>(const AXObject& first, const AXObject& second) { |
| if (first.IsDetached() || second.IsDetached()) |
| return false; |
| |
| int index_in_ancestor1, index_in_ancestor2; |
| const AXObject* ancestor = AXObject::LowestCommonAncestor( |
| first, second, &index_in_ancestor1, &index_in_ancestor2); |
| DCHECK_GE(index_in_ancestor1, -1); |
| DCHECK_GE(index_in_ancestor2, -1); |
| if (!ancestor) |
| return false; |
| return index_in_ancestor1 > index_in_ancestor2; |
| } |
| |
| bool operator>=(const AXObject& first, const AXObject& second) { |
| return first == second || first > second; |
| } |
| |
| std::ostream& operator<<(std::ostream& stream, const AXObject& obj) { |
| return stream << obj.ToString(true).Utf8(); |
| } |
| |
| void AXObject::Trace(Visitor* visitor) const { |
| visitor->Trace(children_); |
| visitor->Trace(parent_); |
| visitor->Trace(cached_live_region_root_); |
| visitor->Trace(ax_object_cache_); |
| } |
| |
| } // namespace blink |