blob: 0dcaf54d78003e7b5b8a3d44fa258e54e6d3fa5b [file] [log] [blame]
/*
* Copyright (C) 2012, Google 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.
*/
#ifndef THIRD_PARTY_BLINK_RENDERER_MODULES_ACCESSIBILITY_AX_NODE_OBJECT_H_
#define THIRD_PARTY_BLINK_RENDERER_MODULES_ACCESSIBILITY_AX_NODE_OBJECT_H_
#include "base/dcheck_is_on.h"
#include "third_party/blink/renderer/core/editing/markers/document_marker.h"
#include "third_party/blink/renderer/core/html/forms/html_form_control_element.h"
#include "third_party/blink/renderer/modules/accessibility/ax_object.h"
#include "third_party/blink/renderer/modules/modules_export.h"
namespace blink {
class AXObjectCacheImpl;
class Element;
class HTMLElement;
class HTMLLabelElement;
class Node;
class MODULES_EXPORT AXNodeObject : public AXObject {
public:
AXNodeObject(Node*, AXObjectCacheImpl&);
AXNodeObject(const AXNodeObject&) = delete;
AXNodeObject& operator=(const AXNodeObject&) = delete;
~AXNodeObject() override;
static std::optional<String> GetCSSAltText(const Element*);
void Trace(Visitor*) const override;
// Call to force-load inline text boxes for the current subtree.
void LoadInlineTextBoxes() override;
// Should inline text boxes be considered when adding chldren to this node.
bool ShouldLoadInlineTextBoxes() const override;
protected:
#if DCHECK_IS_ON()
bool initialized_ = false;
mutable bool getting_bounds_ = false;
#endif
// The accessibility role, not taking the ARIA role into account.
ax::mojom::blink::Role native_role_;
// The ARIA role, not taking the native role into account.
ax::mojom::blink::Role aria_role_;
AXObjectInclusion ShouldIncludeBasedOnSemantics(
IgnoredReasons* = nullptr) const;
bool ComputeAccessibilityIsIgnored(IgnoredReasons* = nullptr) const override;
ax::mojom::blink::Role DetermineTableSectionRole() const;
ax::mojom::blink::Role DetermineTableCellRole() const;
ax::mojom::blink::Role DetermineTableRowRole() const;
bool IsDataTable() const override;
ax::mojom::blink::Role DetermineAccessibilityRole() override;
ax::mojom::blink::Role NativeRoleIgnoringAria() const override;
void AlterSliderOrSpinButtonValue(bool increase);
AXObject* ActiveDescendant() override;
String AriaAccessibilityDescription() const;
String AutoComplete() const override;
void AccessibilityChildrenFromAOMProperty(AOMRelationListProperty,
AXObject::AXObjectVector&) const;
Element* MenuItemElementForMenu() const;
HTMLElement* CorrespondingControlForLabelElement() const;
//
// Overridden from AXObject.
//
void Init(AXObject* parent) override;
void Detach() override;
bool IsAXNodeObject() const final;
// Check object role or purpose.
bool IsAutofillAvailable() const override;
bool IsDefault() const final;
bool IsFieldset() const final;
bool IsHovered() const final;
bool IsImageButton() const;
bool IsInputImage() const final;
bool IsLineBreakingObject() const override;
bool IsLoaded() const override;
bool IsMultiSelectable() const override;
bool IsNativeImage() const final;
bool IsOffScreen() const override;
bool IsProgressIndicator() const override;
bool IsSlider() const override;
bool IsSpinButton() const override;
bool IsNativeSlider() const override;
bool IsNativeSpinButton() const override;
bool IsEmbeddingElement() const override;
// Check object state.
bool IsClickable() const final;
bool IsFocused() const override;
AccessibilityExpanded IsExpanded() const override;
AccessibilitySelectedState IsSelected() const override;
bool IsSelectedFromFocusSupported() const override;
bool IsSelectedFromFocus() const override;
bool IsRequired() const final;
bool IsControl() const override;
AXRestriction Restriction() const override;
// Properties of static elements.
const AtomicString& AccessKey() const override;
RGBA32 ColorValue() const final;
RGBA32 GetColor() const final;
RGBA32 BackgroundColor() const override;
const AtomicString& ComputedFontFamily() const final;
String FontFamilyForSerialization() const final;
// Font size is in pixels.
float FontSize() const final;
float FontWeight() const final;
bool CanvasHasFallbackContent() const final;
int HeadingLevel() const final;
unsigned HierarchicalLevel() const final;
void SerializeMarkerAttributes(ui::AXNodeData* node_data) const override;
AXObject* InPageLinkTarget() const override;
const AtomicString& EffectiveTarget() const override;
AccessibilityOrientation Orientation() const override;
AXObject* GetChildFigcaption() const override;
bool IsDescendantOfLandmarkDisallowedElement() const override;
// Is a redundant label of a radio button or checkbox.
static bool IsRedundantLabel(HTMLLabelElement* label);
// Used to compute kRadioGroupIds, which is only used on Mac.
// TODO(accessibility) Consider computing on browser side and removing here.
AXObjectVector RadioButtonsInGroup() const override;
static HeapVector<Member<HTMLInputElement>> FindAllRadioButtonsWithSameName(
HTMLInputElement* radio_button);
ax::mojom::blink::WritingDirection GetTextDirection() const final;
ax::mojom::blink::TextPosition GetTextPosition() const final;
void GetTextStyleAndTextDecorationStyle(
int32_t* text_style,
ax::mojom::blink::TextDecorationStyle* text_overline_style,
ax::mojom::blink::TextDecorationStyle* text_strikethrough_style,
ax::mojom::blink::TextDecorationStyle* text_underline_style) const final;
String ImageDataUrl(const gfx::Size& max_size) const final;
int TextOffsetInFormattingContext(int offset) const override;
// Object attributes.
ax::mojom::blink::TextAlign GetTextAlign() const final;
float GetTextIndent() const final;
// Properties of interactive elements.
ax::mojom::blink::AriaCurrentState GetAriaCurrentState() const final;
ax::mojom::blink::InvalidState GetInvalidState() const final;
bool IsValidFormControl(ListedElement* form_control) const;
bool ValueForRange(float* out_value) const override;
bool MaxValueForRange(float* out_value) const override;
bool MinValueForRange(float* out_value) const override;
bool StepValueForRange(float* out_value) const override;
KURL Url() const override;
AXObject* ChooserPopup() const override;
String GetValueForControl() const override;
String SlowGetValueForControlIncludingContentEditable() const override;
String TextFromDescendants(AXObjectSet& visited,
const AXObject* aria_label_or_description_root,
bool recursive) const override;
// ARIA attributes.
ax::mojom::blink::Role AriaRoleAttribute() const final;
void AriaDescribedbyElements(AXObjectVector&) const override;
void AriaOwnsElements(AXObjectVector&) const override;
void Dropeffects(
Vector<ax::mojom::blink::Dropeffect>& dropeffects) const override;
ax::mojom::blink::HasPopup HasPopup() const override;
ax::mojom::blink::IsPopup IsPopup() const override;
bool IsEditableRoot() const override;
bool HasContentEditableAttributeSet() const override;
// Modify or take an action on an object.
bool OnNativeSetValueAction(const String&) override;
// AX name calculation.
String GetName(ax::mojom::blink::NameFrom&,
AXObjectVector* name_objects) const override;
String TextAlternative(bool recursive,
const AXObject* aria_label_or_description_root,
AXObjectSet& visited,
ax::mojom::blink::NameFrom&,
AXRelatedObjectVector*,
NameSources*) const override;
String Description(ax::mojom::blink::NameFrom,
ax::mojom::blink::DescriptionFrom&,
AXObjectVector* description_objects) const override;
String Description(ax::mojom::blink::NameFrom,
ax::mojom::blink::DescriptionFrom&,
DescriptionSources*,
AXRelatedObjectVector*) const override;
String SVGDescription(ax::mojom::blink::NameFrom,
ax::mojom::blink::DescriptionFrom&,
DescriptionSources*,
AXRelatedObjectVector*) const;
String Placeholder(ax::mojom::blink::NameFrom) const override;
String Title(ax::mojom::blink::NameFrom) const override;
// Location
void GetRelativeBounds(AXObject** out_container,
gfx::RectF& out_bounds_in_container,
gfx::Transform& out_container_transform,
bool* clips_children = nullptr) const override;
void AddChildren() override;
bool CanHaveChildren() const override;
// Set is_from_aria_owns to true if the child is being added because it was
// pointed to from aria-owns.
void AddChild(AXObject*, bool is_from_aria_owns = false);
// Add a child that must be included in tree, enforced via DCHECK.
void AddChildAndCheckIncluded(AXObject*, bool is_from_aria_owns = false);
// If node is non-null, GetOrCreate an AXObject for it and add as a child.
void AddNodeChild(Node*);
// Set is_from_aria_owns to true if the child is being insert because it was
// pointed to from aria-owns.
void InsertChild(AXObject*, unsigned index, bool is_from_aria_owns = false);
void SelectedOptions(AXObjectVector&) const override;
// Properties of the object's owning document or page.
double EstimatedLoadingProgress() const override;
// DOM and Render tree access.
Element* ActionElement() const override;
Element* AnchorElement() const override;
Document* GetDocument() const override;
Node* GetNode() const final;
// DOM and layout tree access.
AtomicString Language() const override;
bool HasAttribute(const QualifiedName&) const override;
const AtomicString& GetAttribute(const QualifiedName&) const override;
// Modify or take an action on an object.
bool OnNativeBlurAction() final;
bool OnNativeFocusAction() final;
bool OnNativeIncrementAction() final;
bool OnNativeDecrementAction() final;
bool OnNativeSetSequentialFocusNavigationStartingPointAction() final;
// Notifications that this object may have changed.
void HandleAriaExpandedChanged() override;
void HandleActiveDescendantChanged() override;
// Gets a list of nodes that form an error message for this node, if it
// exists. Error messages from ARIA will always override native error
// messages.
AXObjectVector ErrorMessage() const override;
// Gets a list of nodes specified by `aria-errormessage` that form an error
// message for this node, if any exist.
AXObjectVector ErrorMessageFromAria() const override;
// Gets a list of nodes created from HTML validation that form an error
// message for this node, if any exist.
AXObjectVector ErrorMessageFromHTML() const override;
// Position in set and Size of set
int PosInSet() const override;
int SetSize() const override;
// Aria-owns.
void ComputeAriaOwnsChildren(
HeapVector<Member<AXObject>>& owned_children) const;
// Helper method for LoadInlineTextBoxes().
void LoadInlineTextBoxesHelper() override;
//
// Layout object specific methods.
//
// If we can't determine a useful role from the DOM node, attempt to determine
// a role from the layout object.
virtual ax::mojom::blink::Role RoleFromLayoutObjectOrNode() const;
private:
bool HasInternalsAttribute(Element&, const QualifiedName&) const;
const AtomicString& GetInternalsAttribute(Element&,
const QualifiedName&) const;
bool IsNativeCheckboxInMixedState() const;
// This function returns the text of a tooltip associated with the element.
// Although there are two ways of doing this, it is unlikely that an author
// would provide 2 overlapping types of tooltips. Order of precedence:
// 1. The title attribute is currently preferred if present.
// 2. The contents of a plain hint, which has no interesting semantic or
// interactive content, is used next.
// TODO(accessibility): Follow-up with standards discussion to determine
// whether a different order of precedence makes sense.
String TextAlternativeFromTooltip(
ax::mojom::blink::NameFrom& name_from,
NameSources* name_sources,
bool* found_text_alternative,
String* text_alternative,
AXRelatedObjectVector* related_objects) const;
String TextAlternativeFromTitleAttribute(
const AtomicString& title,
ax::mojom::blink::NameFrom& name_from,
NameSources* name_sources,
bool* found_text_alternative) const;
String NativeTextAlternative(AXObjectSet& visited,
ax::mojom::blink::NameFrom&,
AXRelatedObjectVector*,
NameSources*,
bool* found_text_alternative) const;
String MaybeAppendFileDescriptionToName(const String& name) const;
String PlaceholderFromNativeAttribute() const;
String GetValueContributionToName(AXObjectSet& visited) const;
bool UseNameFromSelectedOption() const;
virtual bool IsTabItemSelected() const;
void AddChildrenImpl();
void AddNodeChildren();
void AddPseudoElementChildrenFromLayoutTree();
bool CanAddLayoutChild(LayoutObject& child);
void AddInlineTextBoxChildren();
void AddImageMapChildren();
void AddPopupChildren();
bool HasValidHTMLTableStructureAndLayout() const;
void AddTableChildren();
void AddValidationMessageChild();
void AddAccessibleNodeChildren();
void AddOwnedChildren();
#if DCHECK_IS_ON()
void CheckValidChild(AXObject* child);
#endif
ax::mojom::blink::TextPosition GetTextPositionFromRole() const;
ax::mojom::blink::Dropeffect ParseDropeffect(String& dropeffect) const;
static bool IsNameFromLabelElement(HTMLElement* control);
#if defined(REDUCE_AX_INLINE_TEXTBOXES)
bool always_load_inline_text_boxes_ = false;
#endif
Member<Node> node_;
};
} // namespace blink
#endif // THIRD_PARTY_BLINK_RENDERER_MODULES_ACCESSIBILITY_AX_NODE_OBJECT_H_