From 856bb14b0d60f7d715ff29b93c2c7c4c8fd57cde Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?S=C3=B8ren=20Jacobsen?= Date: Thu, 28 Nov 2024 11:12:04 +0100 Subject: [PATCH] Expose 'allowedAttributes' on alfa Element class Move the logic from 'r18/rule.ts' into the alfa-aria package. --- docs/review/api/alfa-aria.api.md | 3 + packages/alfa-aria/src/node/element.ts | 65 ++++++++++++++++++++- packages/alfa-rules/src/sia-r18/rule.ts | 77 ++----------------------- 3 files changed, 72 insertions(+), 73 deletions(-) diff --git a/docs/review/api/alfa-aria.api.md b/docs/review/api/alfa-aria.api.md index 8c74e17894..14f438f2ee 100644 --- a/docs/review/api/alfa-aria.api.md +++ b/docs/review/api/alfa-aria.api.md @@ -126,6 +126,7 @@ export namespace DOM { // @public (undocumented) export class Element extends Node<"element"> { + allowedAttributes(): readonly Attribute.Name[]; // (undocumented) attribute(refinement: Refinement>): Option>; // (undocumented) @@ -137,6 +138,8 @@ export class Element extends Node<"element"> { // (undocumented) clone(): Element; // (undocumented) + isAttributeAllowed(attribute: Attribute.Name): boolean; + // (undocumented) isIgnored(): boolean; // (undocumented) get name(): Option; diff --git a/packages/alfa-aria/src/node/element.ts b/packages/alfa-aria/src/node/element.ts index 0bb2796785..eb6278b71f 100644 --- a/packages/alfa-aria/src/node/element.ts +++ b/packages/alfa-aria/src/node/element.ts @@ -10,7 +10,11 @@ import type * as dom from "@siteimprove/alfa-dom"; import type { Attribute } from "../attribute.js"; import type { Name } from "../name/index.js"; import { Node } from "../node.js"; -import type { Role } from "../role.js"; +import { Set } from "@siteimprove/alfa-set"; +import { Role } from "../role.js"; +import type { InputType } from "../../../alfa-dom/src/node/element/input-type.js"; +import { Element as DomElement } from "@siteimprove/alfa-dom"; +import { Selective } from "@siteimprove/alfa-selective"; /** * @public @@ -115,6 +119,65 @@ export class Element extends Node<"element"> { ...this._children.map((child) => String.indent(child.toString())), ].join("\n"); } + + private static allowedAttributesForInputType( + inputType: InputType + ): readonly Attribute.Name[] { + switch (inputType) { + // https://www.w3.org/TR/html-aria/#el-input-color + case "color": + return ["aria-disabled"]; + // https://www.w3.org/TR/html-aria/#el-input-date + case "date": + // https://www.w3.org/TR/html-aria/#el-input-datetime-local + case "datetime-local": + // https://www.w3.org/TR/html-aria/#el-input-email + case "email": + // https://www.w3.org/TR/html-aria/#el-input-month + case "month": + // https://www.w3.org/TR/html-aria/#el-input-password + case "password": + // https://www.w3.org/TR/html-aria/#el-input-time + case "time": + // https://www.w3.org/TR/html-aria/#el-input-week + case "week": + return Role.of("textbox").supportedAttributes; + // https://www.w3.org/TR/html-aria/#el-input-file + case "file": + return ["aria-disabled", "aria-invalid", "aria-required"]; + default: + return []; + } + } + + /** + * The attributes that are allowed on this element, taking into consideration ARIA in HTML conformance requirements. + * See {@link https://w3c.github.io/html-aria/#docconformance} + */ + public allowedAttributes(): readonly Attribute.Name[] { + const global = Role.of("roletype").supportedAttributes; + const fromRole = this.role.map(role => role.supportedAttributes).getOr([]); + const additional = Selective.of(this.node) + .if(DomElement.hasName("input"), input => + Element.allowedAttributesForInputType(input.inputType()) + ) + // https://www.w3.org/TR/html-aria/#el-select + .if( + DomElement.hasName("select"), + select => + DomElement.hasDisplaySize((size: Number) => size !== 1)(select) + ? Role.of("combobox").supportedAttributes + : Role.of("menu").supportedAttributes + ) + .else(() => []) + .get(); + + return Array.from(Set.from([... global, ...fromRole, ...additional])); + } + + public isAttributeAllowed(attribute: Attribute.Name): boolean { + return this.allowedAttributes().includes(attribute); + } } /** diff --git a/packages/alfa-rules/src/sia-r18/rule.ts b/packages/alfa-rules/src/sia-r18/rule.ts index 4a7c0793b3..1315207c74 100644 --- a/packages/alfa-rules/src/sia-r18/rule.ts +++ b/packages/alfa-rules/src/sia-r18/rule.ts @@ -1,12 +1,10 @@ import { Diagnostic, Rule } from "@siteimprove/alfa-act"; -import { DOM, Role } from "@siteimprove/alfa-aria"; +import { DOM } from "@siteimprove/alfa-aria"; import type { Attribute } from "@siteimprove/alfa-dom"; -import { Element, Node, Query } from "@siteimprove/alfa-dom"; +import { Node, Query } from "@siteimprove/alfa-dom"; import { Predicate } from "@siteimprove/alfa-predicate"; import { Err, Ok } from "@siteimprove/alfa-result"; -import { Selective } from "@siteimprove/alfa-selective"; import { Sequence } from "@siteimprove/alfa-sequence"; -import { Set } from "@siteimprove/alfa-set"; import { Technique } from "@siteimprove/alfa-wcag"; import type { Page } from "@siteimprove/alfa-web"; @@ -18,7 +16,6 @@ import { ARIA } from "../requirements/index.js"; import { Scope, Stability, Version } from "../tags/index.js"; const { hasRole, isIncludedInTheAccessibilityTree } = DOM; -const { hasDisplaySize, hasInputType, hasName } = Element; const { test, property } = Predicate; const { getElementDescendants } = Query; @@ -30,8 +27,6 @@ export default Rule.Atomic.of({ ], tags: [Scope.Component, Stability.Stable, Version.of(2)], evaluate({ device, document }) { - const global = Set.from(Role.of("roletype").supportedAttributes); - return { applicability() { return getElementDescendants(document, Node.fullTree) @@ -44,19 +39,13 @@ export default Rule.Atomic.of({ }, expectations(target) { - // Since the attribute was found on a element, it has a owner. + // Since the attribute was found on a element, it has an owner. const owner = target.owner.getUnsafe(); + const ariaNode = aria.Node.from(owner, device) as aria.Element; return { 1: expectation( - global.has(target.name as aria.Attribute.Name) || - test( - hasRole(device, (role) => - role.isAttributeSupported(target.name as aria.Attribute.Name), - ), - owner, - ) || - ariaHtmlAllowed(target), + ariaNode.isAttributeAllowed(target.name as aria.Attribute.Name), () => Outcomes.IsAllowed, () => Outcomes.IsNotAllowed, ), @@ -76,62 +65,6 @@ export default Rule.Atomic.of({ }, }); -function allowedForInputType( - attributeName: aria.Attribute.Name, -): Predicate { - return hasInputType((inputType) => { - switch (inputType) { - // https://www.w3.org/TR/html-aria/#el-input-color - case "color": - return attributeName === "aria-disabled"; - // https://www.w3.org/TR/html-aria/#el-input-date - case "date": - // https://www.w3.org/TR/html-aria/#el-input-datetime-local - case "datetime-local": - // https://www.w3.org/TR/html-aria/#el-input-email - case "email": - // https://www.w3.org/TR/html-aria/#el-input-month - case "month": - // https://www.w3.org/TR/html-aria/#el-input-password - case "password": - // https://www.w3.org/TR/html-aria/#el-input-time - case "time": - // https://www.w3.org/TR/html-aria/#el-input-week - case "week": - return Role.of("textbox").isAttributeSupported(attributeName); - // https://www.w3.org/TR/html-aria/#el-input-file - case "file": - return ( - attributeName === "aria-disabled" || - attributeName === "aria-invalid" || - attributeName === "aria-required" - ); - default: - return false; - } - }); -} - -function ariaHtmlAllowed(target: Attribute): boolean { - const attributeName = target.name as aria.Attribute.Name; - return target.owner - .map((element) => - Selective.of(element) - .if(hasName("input"), allowedForInputType(attributeName)) - // https://www.w3.org/TR/html-aria/#el-select - .if( - hasName("select"), - (select) => - (hasDisplaySize((size: Number) => size !== 1)(select) && - Role.of("combobox").isAttributeSupported(attributeName)) || - Role.of("menu").isAttributeSupported(attributeName), - ) - .else(() => false) - .get(), - ) - .getOr(false); -} - /** * @public */