blob: 06ff2a0eba74720db28af5dcce1fce6fcedc8290 [file] [log] [blame]
/*
* Copyright (C) 2016 Igalia, S.L.
* 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.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND 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 THE COPYRIGHT
* OWNER OR 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 "config.h"
#include "AccessibilitySVGObject.h"
#include "AXObjectCache.h"
#include "ElementChildIteratorInlines.h"
#include "ElementInlines.h"
#include "EventTargetInlines.h"
#include "HTMLNames.h"
#include "RenderIterator.h"
#include "RenderObject.h"
#include "RenderObjectInlines.h"
#include "RenderText.h"
#include "SVGAElement.h"
#include "SVGDescElement.h"
#include "SVGElementTypeHelpers.h"
#include "SVGGElement.h"
#include "SVGNames.h"
#include "SVGTitleElement.h"
#include "SVGUseElement.h"
#include "TypedElementDescendantIteratorInlines.h"
#include "XLinkNames.h"
#include <wtf/Language.h>
namespace WebCore {
AccessibilitySVGObject::AccessibilitySVGObject(AXID axID, RenderObject& renderer, AXObjectCache& cache, bool isSVGRoot)
: AccessibilityRenderObject(axID, renderer, cache)
{
m_isSVGRoot = isSVGRoot;
}
AccessibilitySVGObject::~AccessibilitySVGObject() = default;
Ref<AccessibilitySVGObject> AccessibilitySVGObject::create(AXID axID, RenderObject& renderer, AXObjectCache& cache, bool isSVGRoot)
{
return adoptRef(*new AccessibilitySVGObject(axID, renderer, cache, isSVGRoot));
}
AccessibilityObject* AccessibilitySVGObject::targetForUseElement() const
{
RefPtr use = dynamicDowncast<SVGUseElement>(element());
if (!use)
return nullptr;
auto href = use->href();
if (href.isEmpty())
href = getAttribute(HTMLNames::hrefAttr);
auto target = SVGURIReference::targetElementFromIRIString(href, use->treeScopeForSVGReferences());
CheckedPtr cache = axObjectCache();
return cache ? cache->getOrCreate(target.element.get()) : nullptr;
}
template <typename ChildrenType>
Element* AccessibilitySVGObject::childElementWithMatchingLanguage(ChildrenType& children) const
{
String languageCode = languageIncludingAncestors();
if (languageCode.isEmpty())
languageCode = defaultLanguage();
// The best match for a group of child SVG2 'title' or 'desc' elements may be the one
// which lacks a 'lang' attribute value. However, indexOfBestMatchingLanguageInList()
// currently bases its decision on non-empty strings. Furthermore, we cannot count on
// that child element having a given position. So we'll look for such an element while
// building the language list and save it as our fallback.
RefPtr<Element> fallback;
Vector<String> childLanguageCodes;
Vector<Element*> elements;
for (Ref child : children) {
auto& lang = child->attributeWithoutSynchronization(SVGNames::langAttr);
childLanguageCodes.append(lang);
elements.append(child.ptr());
// The current draft of the SVG2 spec states if there are multiple equally-valid
// matches, the first match should be used.
if (lang.isEmpty() && !fallback)
fallback = child.ptr();
}
bool exactMatch;
size_t index = indexOfBestMatchingLanguageInList(languageCode, childLanguageCodes, exactMatch);
if (index < childLanguageCodes.size())
return elements[index];
return fallback.get();
}
void AccessibilitySVGObject::accessibilityText(Vector<AccessibilityText>& textOrder) const
{
String description = this->description();
if (!description.isEmpty())
textOrder.append(AccessibilityText(WTFMove(description), AccessibilityTextSource::Alternative));
String helptext = helpText();
if (!helptext.isEmpty())
textOrder.append(AccessibilityText(WTFMove(helptext), AccessibilityTextSource::Help));
}
String AccessibilitySVGObject::description() const
{
// According to the SVG Accessibility API Mappings spec, the order of priority is:
// 1. aria-labelledby
// 2. aria-label
// 3. a direct child title element (selected according to language)
// 4. xlink:title attribute
// 5. for a use element, the accessible name calculated for the re-used content
// 6. for text container elements, the text content
String ariaDescription = ariaAccessibilityDescription();
if (!ariaDescription.isEmpty())
return ariaDescription;
RefPtr element = this->element();
if (element) {
auto titleElements = childrenOfType<SVGTitleElement>(*element);
if (RefPtr titleChild = childElementWithMatchingLanguage(titleElements))
return titleChild->textContent();
}
if (is<SVGAElement>(element.get())) {
const auto& xlinkTitle = element->attributeWithoutSynchronization(XLinkNames::titleAttr);
if (!xlinkTitle.isEmpty())
return xlinkTitle;
}
if (RefPtr target = targetForUseElement())
return target->description();
// FIXME: This is here to not break the svg-image.html test. But 'alt' is not
// listed as a supported attribute of the 'image' element in the SVG spec:
// https://www.w3.org/TR/SVG/struct.html#ImageElement
if (m_renderer && m_renderer->isRenderOrLegacyRenderSVGImage()) {
const auto& alt = getAttribute(HTMLNames::altAttr);
if (!alt.isNull())
return alt;
}
return { };
}
String AccessibilitySVGObject::helpText() const
{
RefPtr element = this->element();
if (!element)
return { };
// According to the SVG Accessibility API Mappings spec, the order of priority is:
// 1. aria-describedby
// 2. a direct child desc element
// 3. for a use element, the accessible description calculated for the re-used content
// 4. for text container elements, the text content, if not used for the name
// 5. a direct child title element that provides a tooltip, if not used for the name
String describedBy = ariaDescribedByAttribute();
if (!describedBy.isEmpty())
return describedBy;
auto descriptionElements = childrenOfType<SVGDescElement>(*element);
if (RefPtr descriptionChild = childElementWithMatchingLanguage(descriptionElements))
return descriptionChild->textContent();
if (RefPtr target = targetForUseElement())
return target->helpText();
auto titleElements = childrenOfType<SVGTitleElement>(*element);
if (RefPtr titleChild = childElementWithMatchingLanguage(titleElements)) {
if (titleChild->textContent() != description())
return titleChild->textContent();
}
return { };
}
bool AccessibilitySVGObject::hasTitleOrDescriptionChild() const
{
RefPtr element = this->element();
if (!element)
return false;
for (const Ref child : childrenOfType<SVGElement>(*element)) {
if (is<SVGTitleElement>(child) || is<SVGDescElement>(child))
return true;
}
return false;
}
bool AccessibilitySVGObject::computeIsIgnored() const
{
// According to the SVG Accessibility API Mappings spec, items should be excluded if:
// * They would be excluded according to the Core Accessibility API Mappings.
// * They are neither perceivable nor interactive.
// * Their first mappable role is presentational, unless they have a global ARIA
// attribute (covered by Core AAM) or at least one 'title' or 'desc' child element.
// * They have an ancestor with Children Presentational: True (covered by Core AAM)
AccessibilityObjectInclusion decision = defaultObjectInclusion();
if (decision == AccessibilityObjectInclusion::IgnoreObject)
return true;
if (!m_renderer || m_renderer->isLegacyRenderSVGHiddenContainer() || m_renderer->isRenderSVGHiddenContainer())
return true;
// The SVG AAM states objects with at least one 'title' or 'desc' element MUST be included.
// At this time, the presence of a matching 'lang' attribute is not mentioned in the spec.
if (hasTitleOrDescriptionChild())
return false;
if (ignoredFromPresentationalRole())
return true;
if (ariaRoleAttribute() != AccessibilityRole::Unknown)
return false;
// The SVG AAM states text elements should also be included, if they have content.
if (m_renderer->isRenderSVGText() || m_renderer->isRenderSVGTextPath()) {
for (auto& child : childrenOfType<RenderText>(downcast<RenderElement>(*m_renderer))) {
if (!child.containsOnlyCollapsibleWhitespace())
return false;
}
}
// SVG shapes should not be included unless there's a concrete reason for inclusion.
// https://rawgit.com/w3c/aria/master/svg-aam/svg-aam.html#exclude_elements
if (m_renderer->isRenderOrLegacyRenderSVGShape()) {
if (canSetFocusAttribute() || element()->hasEventListeners())
return false;
if (RefPtr svgParent = Accessibility::findAncestor<AccessibilityObject>(*this, true, [] (const AccessibilityObject& object) {
return object.hasAttributesRequiredForInclusion() || object.isAccessibilitySVGRoot();
}))
return !svgParent->hasAttributesRequiredForInclusion();
return true;
}
return AccessibilityRenderObject::computeIsIgnored();
}
bool AccessibilitySVGObject::inheritsPresentationalRole() const
{
if (canSetFocusAttribute())
return false;
auto role = this->role();
if (role != AccessibilityRole::SVGTextPath && role != AccessibilityRole::SVGTSpan)
return false;
for (RefPtr parent = parentObject(); parent; parent = parent->parentObject()) {
if (is<AccessibilityRenderObject>(*parent) && parent->hasElementName(ElementName::SVG_text))
return parent->role() == AccessibilityRole::Presentational;
}
return false;
}
AccessibilityRole AccessibilitySVGObject::determineAriaRoleAttribute() const
{
auto role = AccessibilityRenderObject::determineAriaRoleAttribute();
if (role != AccessibilityRole::Presentational)
return role;
// The presence of a 'title' or 'desc' child element trumps PresentationalRole.
// https://lists.w3.org/Archives/Public/public-svg-a11y/2016Apr/0016.html
// At this time, the presence of a matching 'lang' attribute is not mentioned.
return hasTitleOrDescriptionChild() ? AccessibilityRole::Unknown : role;
}
AccessibilityRole AccessibilitySVGObject::determineAccessibilityRole()
{
if ((m_ariaRole = determineAriaRoleAttribute()) != AccessibilityRole::Unknown)
return m_ariaRole;
if (!m_renderer)
return AccessibilityRole::Unknown;
if (isAccessibilitySVGRoot())
return AccessibilityRole::Generic;
RefPtr element = this->element();
if (m_renderer->isRenderOrLegacyRenderSVGShape() || m_renderer->isRenderOrLegacyRenderSVGPath() || m_renderer->isRenderOrLegacyRenderSVGImage() || is<SVGUseElement>(element))
return AccessibilityRole::Image;
if (m_renderer->isRenderOrLegacyRenderSVGForeignObject())
return AccessibilityRole::Generic;
if (is<SVGGElement>(element)) {
// https://w3c.github.io/svg-aam/#include_elements
// g elements are generic (like a div) unless they have a name or is focusable.
if (WebCore::hasAccNameAttribute(*element) || hasTitleOrDescriptionChild() || canSetFocusAttribute())
return AccessibilityRole::Group;
return AccessibilityRole::Generic;
}
if (m_renderer->isRenderSVGInlineText())
return AccessibilityRole::StaticText;
if (m_renderer->isRenderSVGText())
return AccessibilityRole::SVGText;
if (m_renderer->isRenderSVGTextPath())
return AccessibilityRole::SVGTextPath;
if (m_renderer->isRenderSVGTSpan())
return AccessibilityRole::SVGTSpan;
if (is<SVGAElement>(element))
return AccessibilityRole::Link;
return AccessibilityRenderObject::determineAccessibilityRole();
}
AccessibilityObject* AccessibilitySVGObject::parentObject() const
{
if (m_parent) {
// If a parent was set because this is a remote SVG resource, use that.
ASSERT(m_isSVGRoot);
return m_parent.get();
}
// Otherwise, we should rely on the standard render tree for the parent.
return AccessibilityRenderObject::parentObject();
}
bool AccessibilitySVGObject::isRootWithAccessibleContent() const
{
if (!isAccessibilitySVGRoot())
return false;
RefPtr rootElement = this->element();
if (!rootElement)
return false;
auto isAccessibleSVGElement = [] (const SVGElement& element) -> bool {
// The presence of an SVGTitle or SVGDesc element is enough to deem the SVG hierarchy as accessible.
if (is<SVGTitleElement>(element)
|| is<SVGDescElement>(element))
return true;
// Text content is accessible.
if (element.isTextContent())
return true;
// If the role or aria-label attributes are specified, this is accessible.
if (!element.attributeWithoutSynchronization(HTMLNames::roleAttr).isEmpty()
|| !element.attributeWithoutSynchronization(HTMLNames::aria_labelAttr).isEmpty())
return true;
return false;
};
RefPtr svgRootElement = dynamicDowncast<SVGElement>(*rootElement);
if (svgRootElement && isAccessibleSVGElement(*svgRootElement))
return true;
// This SVG hierarchy is accessible if any of its descendants is accessible.
for (const Ref descendant : descendantsOfType<SVGElement>(*rootElement)) {
if (isAccessibleSVGElement(descendant.get()))
return true;
}
return false;
}
} // namespace WebCore