* 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/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(LayoutObject*, AXObjectCacheImpl&);
AXNodeObject(const AXNodeObject&) = delete;
AXNodeObject& operator=(const AXNodeObject&) = delete;
~AXNodeObject() override;
static std::optional<String> GetCSSAltText(const Element*);
static std::optional<String> GetCSSContentText(const Element*);
void Trace(Visitor*) const override;
void LoadInlineTextBoxes() override;
ScrollableArea* GetScrollableAreaIfScrollable() const final;
protected:
#if DCHECK_IS_ON()
bool initialized_ = false;
mutable bool getting_bounds_ = false;
#endif
ax::mojom::blink::Role native_role_ = ax::mojom::blink::Role::kUnknown;
ax::mojom::blink::Role aria_role_ = ax::mojom::blink::Role::kUnknown;
bool HasCustomElementTreeProcessing() const;
bool ShouldIncludeCustomElement() const;
AXObjectInclusion ShouldIncludeBasedOnSemantics(
IgnoredReasons* = nullptr) const;
bool ComputeIsIgnored(IgnoredReasons*) const override;
bool ComputeIsIgnoredAsInsideInactiveScrollMarkerTab() override;
ax::mojom::blink::Role DetermineRoleValue() override;
ax::mojom::blink::Role NativeRoleIgnoringAria() const override;
void AlterSliderOrSpinButtonValue(bool increase);
AXObject* ActiveDescendant() const override;
String AriaAccessibilityDescription() const;
String AutoComplete() const override;
bool IsDataTable() const override;
unsigned ColumnCount() const override;
unsigned RowCount() const override;
void ColumnHeaders(AXObjectVector&) const override;
void RowHeaders(AXObjectVector&) const override;
AXObject* CellForColumnAndRow(unsigned column, unsigned row) const override;
unsigned ColumnIndex() const override;
unsigned RowIndex() const override;
unsigned ColumnSpan() const override;
unsigned RowSpan() const override;
AXObject* HeaderObject() const override;
ax::mojom::blink::SortDirection GetSortDirection() const override;
ax::mojom::blink::Role DetermineTableSectionRole() const;
ax::mojom::blink::Role DetermineTableCellRole() const;
ax::mojom::blink::Role DetermineTableRowRole() const;
Element* MenuItemElementForMenu() const;
HTMLElement* CorrespondingControlForLabelElement() const;
void Init(AXObject* parent) override;
void Detach() override;
bool IsAXNodeObject() const final;
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 IsProgressIndicator() const override;
bool IsSlider() const override;
bool IsSpinButton() const override;
bool IsNativeSlider() const override;
bool IsNativeSpinButton() const override;
bool IsEmbeddingElement() const override;
bool IsLinked() const override;
bool IsVisible() const override;
bool IsVisited() const override;
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 IsNotUserSelectable() const override;
bool IsRequired() const final;
bool IsControl() const override;
AXRestriction Restriction() const override;
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;
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;
ax::mojom::blink::ListStyle GetListStyle() const final;
AXObject* InPageLinkTarget() const override;
const AtomicString& EffectiveTarget() const override;
AccessibilityOrientation Orientation() const override;
AXObject* GetChildFigcaption() const override;
bool IsDescendantOfLandmarkDisallowedElement() const override;
static bool IsRedundantLabel(HTMLLabelElement* label);
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;
ax::mojom::blink::TextAlign GetTextAlign() const final;
float GetTextIndent() const final;
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 GetValueForControl(AXObjectSet& visited) const override;
String SlowGetValueForControlIncludingContentEditable() const override;
String SlowGetValueForControlIncludingContentEditable(
AXObjectSet& visited) const override;
String TextFromDescendants(AXObjectSet& visited,
const AXObject* aria_label_or_description_root,
bool recursive) const override;
ax::mojom::blink::Role RawAriaRole() const final;
ax::mojom::blink::HasPopup HasPopup() const override;
ax::mojom::blink::IsPopup IsPopup() const override;
bool IsEditableRoot() const override;
bool HasContentEditableAttributeSet() const override;
bool OnNativeSetSelectedAction(bool selected) override;
bool OnNativeSetValueAction(const String&) override;
String GetName(ax::mojom::blink::NameFrom&,
AXObjectVector* name_objects,
NameSources* name_sources) const override;
String TextAlternative(bool recursive,
const AXObject* aria_label_or_description_root,
AXObjectSet& visited,
ax::mojom::blink::NameFrom&,
AXRelatedObjectVector*,
NameSources*) const override;
static String GetSavedTextAlternativeFromNameSource(
bool found_text_alternative,
ax::mojom::NameFrom& name_from,
AXRelatedObjectVector* related_objects,
NameSources* name_sources);
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;
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;
void AddChild(AXObject*, bool is_from_aria_owns = false);
void AddChildAndCheckIncluded(AXObject*, bool is_from_aria_owns = false);
void AddNodeChild(Node*);
void InsertChild(AXObject*, unsigned index, bool is_from_aria_owns = false);
void SelectedOptions(AXObjectVector&) const override;
double EstimatedLoadingProgress() const override;
Element* ActionElement() const override;
Element* AnchorElement() const override;
Document* GetDocument() const override;
ALWAYS_INLINE Node* GetNode() const;
LayoutObject* GetLayoutObject() const final;
bool OnNativeBlurAction() final;
bool OnNativeFocusAction() final;
bool OnNativeIncrementAction() final;
bool OnNativeDecrementAction() final;
bool OnNativeSetSequentialFocusNavigationStartingPointAction() final;
void HandleAriaExpandedChanged() override;
void HandleActiveDescendantChanged() override;
AXObjectVector ErrorMessage() const override;
AXObjectVector ErrorMessageFromHTML() const override;
AXObjectVector RelationVectorFromAria(
const QualifiedName& attr_name) const override;
int PosInSet() const override;
int SetSize() const override;
void ComputeAriaOwnsChildren(
HeapVector<Member<AXObject>>& owned_children) const;
void LoadInlineTextBoxesHelper() override;
AXObject* AccessibilityHitTest(const gfx::Point&) const override;
ax::mojom::blink::Role RoleFromLayoutObjectOrNode() const;
void HandleAutofillSuggestionAvailabilityChanged(
WebAXAutofillSuggestionAvailability suggestion_availability) override;
void GetWordBoundaries(Vector<int>& word_starts,
Vector<int>& word_ends) const override;
private:
struct GenerationalCache : public GarbageCollected<GenerationalCache> {
virtual void Trace(Visitor*) const;
uint64_t generation = 0;
Member<AXObject> next_on_line;
Member<AXObject> previous_on_line;
};
mutable Member<GenerationalCache> generational_cache_;
void MaybeResetCache() const;
AXObject* SetNextOnLine(AXObject* next_on_line) const;
AXObject* SetPreviousOnLine(AXObject* previous_on_line) const;
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;
bool ShouldIncludeContentInTextAlternative(
bool recursive,
const AXObject* aria_label_or_description_root,
AXObjectSet& visited) const;
String PlaceholderFromNativeAttribute() const;
String GetValueContributionToName(AXObjectSet& visited) const;
bool UseNameFromSelectedOption() const;
virtual bool IsTabItemSelected() const;
void AddNodeChildImpl(Node*);
void AddChildrenImpl();
void AddNodeChildren();
void AddPseudoElementChildrenFromLayoutTree();
bool CanAddLayoutChild(LayoutObject& child);
void AddInlineTextBoxChildren();
void AddInlineTextBoxChildrenWithBlockFlowIterator();
void AddImageMapChildren();
void AddPopupChildren();
bool HasValidHTMLTableStructureAndLayout() const;
void AddTableChildren();
bool FindAllTableCellsWithRole(ax::mojom::blink::Role, AXObjectVector&) const;
void AddValidationMessageChild();
void AddAccessibleNodeChildren();
void AddOwnedChildren();
void AddScrollMarkerGroupChildren();
#if DCHECK_IS_ON()
void CheckValidChild(AXObject* child);
#endif
ax::mojom::blink::TextPosition GetTextPositionFromRole() const;
static bool IsNameFromLabelElement(HTMLElement* control);
AXObject* AccessibilityImageMapHitTest(HTMLAreaElement*,
const gfx::Point&) const;
AXObject* GetFirstInlineBlockOrDeepestInlineAXChildInLayoutTree(
AXObject* start_object,
bool first) const;
AXObject* NextOnLine() const override;
AXObject* PreviousOnLine() const override;
Member<Node> node_;
Member<LayoutObject> layout_object_;
friend class AXObject;
};
template <>
struct DowncastTraits<AXNodeObject> {
static bool AllowFrom(const AXObject& object) {
return object.IsNodeObject();
}
};
}
#endif