Source code
Revision control
Copy as Markdown
Other Tools
Test Info: Warnings
- This test has a WPT meta file that expects 2 subtest issues.
- This WPT test may be referenced by the following Test IDs:
- /custom-elements/element-internals-aria-element-reflection.html - WPT Dashboard Interop Dashboard
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8" />
<title>Element Reflection for aria-activedescendant and aria-errormessage on ElementInternals</title>
<link rel=help href="https://whatpr.org/html/3917/common-dom-interfaces.html#reflecting-content-attributes-in-idl-attributes:element">
<link rel="author" title="Alice Boxhall" href="alice@igalia.com">
<script src="/resources/testdriver.js"></script>
<script src="/resources/testdriver-vendor.js"></script>
<script src="/resources/testdriver-actions.js"></script>
<script src="/resources/testharness.js"></script>
<script src="/resources/testharnessreport.js"></script>
</head>
<body>
<script>
class CustomElement extends HTMLElement {
constructor() {
super();
this.i = this.attachInternals();
}
}
customElements.define('custom-element', CustomElement);
</script>
<custom-element id="custom1"></custom-element>
<div id="activedescendant">Active descendant</div>
<div id="controls">Controls</div>
<div id="describedby">Described by</div>
<div id="details">Details</div>
<div id="errormessage">Error message</div>
<div id="flowto">Flow to</div>
<div id="labelledby">Labelled by</div>
<div id="owns">Owns</div>
<div id="labelledby_content">Labelled by from content attribute</div>
<div id="labelledby_element">Labelled by from IDL attribute</div>
<div id="shadowhost">
<template shadowrootmode="open">
<div id="labelledby_element_shadow">Invalid labelled by in shadow root</div>
</template>
</div>
<script>
const element_properties = [
['ariaActiveDescendantElement', 'activedescendant']];
const array_properties = [
['ariaControlsElements', 'controls'],
['ariaDescribedByElements', 'describedby'],
['ariaDetailsElements', 'details'],
['ariaErrorMessageElements', 'errormessage'],
['ariaFlowToElements', 'flowto'],
['ariaLabelledByElements', 'labelledby'],
['ariaOwnsElements', 'owns']];
test(t => {
const custom = document.getElementById('custom1');
for (const [property, id] of element_properties) {
assert_equals(custom.i[property], null);
}
for (const [property, id] of array_properties) {
assert_equals(custom.i[property], null);
}
}, "Getting previously-unset ARIA element reflection properties on ElementInternals should return null.");
test(t => {
const custom = document.getElementById('custom1');
for (const [property, id] of element_properties) {
const related = document.getElementById(id);
custom.i[property] = related;
assert_equals(custom.i[property], related);
}
for (const [property, id] of array_properties) {
const related = document.getElementById(id);
custom.i[property] = [related];
assert_array_equals(custom.i[property], [related]);
}
}, "Getting ARIA element reflection properties on ElementInternals should return the value that was set.");
test(t => {
const custom = document.getElementById('custom1');
for (const [property, id] of array_properties) {
custom.i[property] = [];
assert_array_equals(custom.i[property], []);
}
}, "Setting ARIA element reflection properties to an empty array should work as expected.");
test(t => {
const custom = document.getElementById('custom1');
for (const [property, id] of element_properties) {
const related = document.getElementById(id);
custom.i[property] = related;
assert_equals(custom.i[property], related);
custom.i[property] = null;
assert_equals(custom.i[property], null);
}
for (const [property, id] of array_properties) {
const related = document.getElementById(id);
custom.i[property] = [related];
assert_array_equals(custom.i[property], [related]);
custom.i[property] = null;
assert_equals(custom.i[property], null);
}
}, "Setting ARIA element reflection properties on ElementInternals to null should delete any previous value, and not crash");
promise_test(async t => {
const custom = document.getElementById('custom1');
custom.i.ariaLabelledByElements = null;
const label_before_labelledby_set = await test_driver.get_computed_label(custom);
assert_equals(label_before_labelledby_set, "", "Before ariaLabelledByElements is set, accessible label should be empty.");
const labelledBy = document.getElementById('labelledby');
custom.i.ariaLabelledByElements = [labelledBy];
const label_after_labelledby_set = await test_driver.get_computed_label(custom);
assert_equals(label_after_labelledby_set, "Labelled by", "After ariaLabelledByElements is set, accessible label should be 'Labelled by'");
}, "Setting ariaLabelledByElements on ElementInternals should change the accessible name of the custom element")
promise_test(async t => {
const custom = document.getElementById('custom1');
custom.i.ariaLabelledByElements = null;
const label_before_labelledby_set = await test_driver.get_computed_label(custom);
assert_equals(label_before_labelledby_set, "", "Before ariaLabelledByElements is set, accessible label should be empty.");
const labelledBy = document.getElementById('labelledby');
custom.i.ariaLabelledByElements = [labelledBy];
const label_after_internals_labelledby_set = await test_driver.get_computed_label(custom);
assert_equals(label_after_internals_labelledby_set, "Labelled by", "After ariaLabelledByElements is set, accessible label should be 'Labelled by'");
custom.setAttribute('aria-labelledby', 'labelledby_content');
const label_after_content_labelledby_set = await test_driver.get_computed_label(custom);
assert_equals(label_after_content_labelledby_set, "Labelled by from content attribute", "aria-labelledby content attribute supersedes ariaLabelledByElements on internals");
const labelledby_element = document.getElementById('labelledby_element');
custom.ariaLabelledByElements = [labelledby_element];
const label_after_idl_arialabelledbyelements_set = await test_driver.get_computed_label(custom);
assert_equals(label_after_idl_arialabelledbyelements_set, "Labelled by from IDL attribute", "ariaLabelledByElements on element supersedes ariaLabelledByElements on internals");
custom.setAttribute('aria-labelledby', 'bad_id');
const label_after_content_labelledby_set_to_bad_id = await test_driver.get_computed_label(custom);
assert_equals(label_after_content_labelledby_set_to_bad_id, "", "aria-labelledby content attribute supersedes ariaLabelledByElements on internals even when invalid");
const shadowhost = document.getElementById('shadowhost');
const labelledby_element_invalid = shadowhost.shadowRoot.getElementById('labelledby_element_shadow');
custom.ariaLabelledByElements = [labelledby_element_invalid];
const label_after_idl_arialabelledbyelements_set_to_invalid_element = await test_driver.get_computed_label(custom);
assert_equals(label_after_idl_arialabelledbyelements_set_to_invalid_element, "", "ariaLabelledByElements on element supersedes ariaLabelledByElements on internals even when invalid");
}, "Setting aria-labelledby or ariaLabelledByElements on the custom element should supersede the value of ariaLabelledByElements on ElementInternals");
</script>
<custom-element id="cachingInvariantMain"></custom-element>
<div id="cachingInvariantElement1"></div>
<div id="cachingInvariantElement2"></div>
<div id="cachingInvariantElement3"></div>
<div id="cachingInvariantElement4"></div>
<div id="cachingInvariantElement5"></div>
<script>
test(function(t) {
cachingInvariantMain.i.ariaControlsElements = [cachingInvariantElement1, cachingInvariantElement2];
cachingInvariantMain.i.ariaDescribedByElements = [cachingInvariantElement3, cachingInvariantElement4];
cachingInvariantMain.i.ariaDetailsElements = [cachingInvariantElement5];
cachingInvariantMain.i.ariaFlowToElements = [cachingInvariantElement1, cachingInvariantElement3];
cachingInvariantMain.i.ariaLabelledByElements = [cachingInvariantElement2, cachingInvariantElement4];
cachingInvariantMain.i.ariaOwnsElements = [cachingInvariantElement1, cachingInvariantElement2, cachingInvariantElement3];
let ariaControlsElementsArray = cachingInvariantMain.i.ariaControlsElements;
let ariaDescribedByElementsArray = cachingInvariantMain.i.ariaDescribedByElements;
let ariaDetailsElementsArray = cachingInvariantMain.i.ariaDetailsElements;
let ariaFlowToElementsArray = cachingInvariantMain.i.ariaFlowToElements;
let ariaLabelledByElementsArray = cachingInvariantMain.i.ariaLabelledByElements;
let ariaOwnsElementsArray = cachingInvariantMain.i.ariaOwnsElements;
assert_equals(ariaControlsElementsArray, cachingInvariantMain.i.ariaControlsElements, "Caching invariant for ariaControlsElements");
assert_equals(ariaDescribedByElementsArray, cachingInvariantMain.i.ariaDescribedByElements, "Caching invariant for ariaDescribedByElements");
assert_equals(ariaDetailsElementsArray, cachingInvariantMain.i.ariaDetailsElements, "Caching invariant for ariaDetailsElements");
assert_equals(ariaFlowToElementsArray, cachingInvariantMain.i.ariaFlowToElements, "Caching invariant for ariaFlowToElements");
assert_equals(ariaLabelledByElementsArray, cachingInvariantMain.i.ariaLabelledByElements, "Caching invariant for ariaLabelledByElements");
assert_equals(ariaOwnsElementsArray, cachingInvariantMain.i.ariaOwnsElements, "Caching invariant for ariaOwnsElements");
// Ensure that cached values don't become stale
cachingInvariantMain.i.ariaControlsElements = [cachingInvariantElement4, cachingInvariantElement5];
cachingInvariantMain.i.ariaDescribedByElements = [cachingInvariantElement1, cachingInvariantElement2];
cachingInvariantMain.i.ariaDetailsElements = [cachingInvariantElement3];
cachingInvariantMain.i.ariaFlowToElements = [cachingInvariantElement4, cachingInvariantElement5];
cachingInvariantMain.i.ariaLabelledByElements = [cachingInvariantElement1, cachingInvariantElement2];
cachingInvariantMain.i.ariaOwnsElements = [cachingInvariantElement3, cachingInvariantElement4, cachingInvariantElement1];
ariaControlsElementsArray = cachingInvariantMain.i.ariaControlsElements;
ariaDescribedByElementsArray = cachingInvariantMain.i.ariaDescribedByElements;
ariaDetailsElementsArray = cachingInvariantMain.i.ariaDetailsElements;
ariaFlowToElementsArray = cachingInvariantMain.i.ariaFlowToElements;
ariaLabelledByElementsArray = cachingInvariantMain.i.ariaLabelledByElements;
ariaOwnsElementsArray = cachingInvariantMain.i.ariaOwnsElements;
assert_equals(ariaControlsElementsArray, cachingInvariantMain.i.ariaControlsElements, "Caching invariant for ariaControlsElements");
assert_equals(ariaDescribedByElementsArray, cachingInvariantMain.i.ariaDescribedByElements, "Caching invariant for ariaDescribedByElements");
assert_equals(ariaDetailsElementsArray, cachingInvariantMain.i.ariaDetailsElements, "Caching invariant for ariaDetailsElements");
assert_equals(ariaFlowToElementsArray, cachingInvariantMain.i.ariaFlowToElements, "Caching invariant for ariaFlowToElements");
assert_equals(ariaLabelledByElementsArray, cachingInvariantMain.i.ariaLabelledByElements, "Caching invariant for ariaLabelledByElements");
assert_equals(ariaOwnsElementsArray, cachingInvariantMain.i.ariaOwnsElements, "Caching invariant for ariaOwnsElements");
}, "Caching invariant different attributes.");
</script>
<custom-element id="cachingInvariantMain1"></custom-element>
<custom-element id="cachingInvariantMain2"></custom-element>
<script>
test(function(t) {
cachingInvariantMain1.i.ariaDescribedByElements = [cachingInvariantElement1, cachingInvariantElement2];
cachingInvariantMain2.i.ariaDescribedByElements = [cachingInvariantElement1, cachingInvariantElement2];
let ariaDescribedByElementsArray1 = cachingInvariantMain1.i.ariaDescribedByElements;
let ariaDescribedByElementsArray2 = cachingInvariantMain2.i.ariaDescribedByElements;
assert_equals(ariaDescribedByElementsArray1, cachingInvariantMain1.i.ariaDescribedByElements, "Caching invariant for ariaDescribedByElements in one elemnt");
assert_equals(ariaDescribedByElementsArray2, cachingInvariantMain2.i.ariaDescribedByElements, "Caching invariant for ariaDescribedByElements in onother elemnt");
assert_not_equals(cachingInvariantMain1.i.ariaDescribedByElements, cachingInvariantMain2.i.ariaDescribedByElements);
}, "Caching invariant different elements.");
</script>
</body>
</html>