|
8 | 8 | <script src="/resources/testdriver-vendor.js"></script>
|
9 | 9 | <script src="/resources/testdriver-actions.js"></script>
|
10 | 10 | <script src="/wai-aria/scripts/aria-utils.js"></script>
|
| 11 | + <script src="resources/property-reflection-helper.js"></script> |
11 | 12 | </head>
|
12 | 13 | <body>
|
13 | 14 | <div id="host-container"></div>
|
14 | 15 | <script>
|
15 |
| - const Behavior = Object.freeze({ |
16 |
| - ReflectsHost: 'ReflectsHost', |
17 |
| - ReflectsHostInArray: 'ReflectsHostInArray', |
18 |
| - IsNull: 'IsNull', |
19 |
| - ReflectsHostID: 'ReflectsHostID', |
20 |
| - ReflectsHostIDInDOMTokenList: 'ReflectsHostIDInDOMTokenList', |
21 |
| - }); |
22 |
| - |
23 |
| - function test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, attribute, reflected_property, expected_behavior) { |
24 |
| - // There's nothing to test if the referencing element type doesn't have the reflecting |
25 |
| - // property. |
26 |
| - if (!(reflected_property in document.createElement(referencing_element_type))) { |
27 |
| - return; |
28 |
| - } |
29 |
| - |
30 |
| - test(function () { |
31 |
| - const referencing_element = document.createElement(referencing_element_type); |
32 |
| - document.body.appendChild(referencing_element); |
33 |
| - referencing_element.setAttribute(attribute, "host-id"); |
34 |
| - const host_container = document.querySelector("#host-container"); |
35 |
| - const host = element_creation_method(host_container, referenced_element_type); |
36 |
| - if (expected_behavior === Behavior.ReflectsHost) { |
37 |
| - assert_equals(referencing_element[reflected_property], host); |
38 |
| - } else if (expected_behavior === Behavior.ReflectsHostInArray) { |
39 |
| - assert_array_equals(referencing_element[reflected_property], [host]); |
40 |
| - } else if (expected_behavior === Behavior.IsNull) { |
41 |
| - assert_equals(referencing_element[reflected_property], null); |
42 |
| - } else if (expected_behavior === Behavior.ReflectsHostID) { |
43 |
| - assert_equals(referencing_element[reflected_property], "host-id"); |
44 |
| - } else if (expected_behavior === Behavior.ReflectsHostIDInDOMTokenList) { |
45 |
| - assert_true(referencing_element[reflected_property] instanceof DOMTokenList); |
46 |
| - assert_array_equals(Array.from(referencing_element[reflected_property]), ["host-id"]); |
47 |
| - } |
48 |
| - referencing_element.remove(); |
49 |
| - host_container.setHTMLUnsafe(""); |
50 |
| - }, `${referencing_element_type}.${reflected_property} has reflection behavior ${expected_behavior} when pointing to ${referenced_element_type} with reference target${element_creation_method.method_name}`); |
51 |
| - } |
52 |
| - |
53 |
| - function appendTestDeclaratively(host_container, referenced_element_type) { |
54 |
| - host_container.setHTMLUnsafe(` |
55 |
| - <div id="host-id"> |
56 |
| - <template shadowrootmode="open" shadowrootreferencetarget="target"> |
57 |
| - <${referenced_element_type} id="target"></${referenced_element_type}> |
58 |
| - </template> |
59 |
| - </div>`); |
60 |
| - const host = host_container.firstElementChild; |
61 |
| - return host; |
62 |
| - } |
63 |
| - function appendTestWithOptions(host_container, referenced_element_type) { |
64 |
| - host_container.setHTMLUnsafe('<div id="host-id"></div>'); |
65 |
| - const host = host_container.firstElementChild; |
66 |
| - host.attachShadow({ mode: 'open', referenceTarget: 'target' }); |
67 |
| - host.shadowRoot.innerHTML = `<${referenced_element_type} id="target"></${referenced_element_type}>`; |
68 |
| - return host; |
| 16 | + function append_test_declaratively(host_container, referenced_element_type) { |
| 17 | + host_container.setHTMLUnsafe(` |
| 18 | + <div id="host-id"> |
| 19 | + <template shadowrootmode="open" shadowrootreferencetarget="target"> |
| 20 | + <${referenced_element_type} id="target"></${referenced_element_type}> |
| 21 | + </template> |
| 22 | + </div>`); |
| 23 | + const host = host_container.firstElementChild; |
| 24 | + return host; |
69 | 25 | }
|
70 |
| - const element_creation_methods = [ |
71 |
| - appendTestDeclaratively, |
72 |
| - appendTestWithOptions, |
73 |
| - ]; |
74 |
| - element_creation_methods[0].method_name = ''; |
75 |
| - element_creation_methods[1].method_name = ' via options'; |
76 |
| - // We want to test types of elements that are associated with properties that can reflect other |
77 |
| - // elements and can therefore interact with reference target in interesting ways. |
78 |
| - // The HTML5_LABELABLE_ELEMENTS are defined in https://html.spec.whatwg.org/#category-label, |
79 |
| - // while non_labelable_element_types is a manually curated list of other elements with |
80 |
| - // reflecting properties (plus div as representative of more "normal" elements). |
81 |
| - // We'll test all permutations of these element types being both the referencing element |
82 |
| - // pointing into the reference target shadow host, and being the referenced element inside |
83 |
| - // the shadow. |
84 |
| - const non_labelable_element_types = ["div", "object", "label", "fieldset", "legend", "option", "datalist", "form"]; |
85 |
| - const element_types = HTML5_LABELABLE_ELEMENTS.concat(non_labelable_element_types); |
86 |
| - |
87 |
| - for(let element_creation_method of element_creation_methods) { |
88 |
| - for(let referencing_element_type of element_types) { |
89 |
| - for(let referenced_element_type of element_types) { |
90 |
| - test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, "aria-controls", "ariaControlsElements", Behavior.ReflectsHostInArray); |
91 |
| - test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, "aria-activedescendant", "ariaActiveDescendantElement", Behavior.ReflectsHost); |
92 |
| - test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, "aria-describedby", "ariaDescribedByElements", Behavior.ReflectsHostInArray); |
93 |
| - test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, "aria-details", "ariaDetailsElements", Behavior.ReflectsHostInArray); |
94 |
| - test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, "aria-errormessage", "ariaErrorMessageElements", Behavior.ReflectsHostInArray); |
95 |
| - test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, "aria-flowto", "ariaFlowToElements", Behavior.ReflectsHostInArray); |
96 |
| - test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, "aria-labelledby", "ariaLabelledByElements", Behavior.ReflectsHostInArray); |
97 |
| - test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, "aria-owns", "ariaOwnsElements", Behavior.ReflectsHostInArray); |
98 |
| - |
99 |
| - test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, "anchor", "anchorElement", Behavior.ReflectsHost); |
100 |
| - test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, "commandfor", "commandForElement", Behavior.ReflectsHost); |
101 |
| - test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, "popovertarget", "popoverTargetElement", Behavior.ReflectsHost); |
102 |
| - test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, "interesttarget", "interestTargetElement", Behavior.ReflectsHost); |
103 |
| - |
104 |
| - const expected_htmlFor_property_behavior = (referencing_element_type == "output") ? Behavior.ReflectsHostIDInDOMTokenList : Behavior.ReflectsHostID; |
105 |
| - test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, "for", "htmlFor", expected_htmlFor_property_behavior); |
106 | 26 |
|
107 |
| - // The form property of <label>, <legend>, and <option> reflects the form property of the associated labelable element, |
108 |
| - // the associated <fieldset>, and the associated <select>, respectively. Here since we don't have those associated elements, |
109 |
| - // the form property would return null. |
110 |
| - const expected_form_property_behavior = (referenced_element_type == 'form' && |
111 |
| - referencing_element_type != "label" && |
112 |
| - referencing_element_type != "legend" && |
113 |
| - referencing_element_type != "option") ? Behavior.ReflectsHost : Behavior.IsNull; |
114 |
| - test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, "form", "form", expected_form_property_behavior); |
115 |
| - |
116 |
| - const expected_list_property_behavior = (referenced_element_type == 'datalist') ? Behavior.ReflectsHost : Behavior.IsNull; |
117 |
| - test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, "list", "list", expected_list_property_behavior); |
118 |
| - |
119 |
| - const expected_control_property_behavior = HTML5_LABELABLE_ELEMENTS.includes(referenced_element_type) ? Behavior.ReflectsHost : Behavior.IsNull; |
120 |
| - test_property_reflection(element_creation_method, referencing_element_type, referenced_element_type, "for", "control", expected_control_property_behavior); |
121 |
| - } |
122 |
| - } |
123 |
| - } |
| 27 | + run_test_for_all_reflecting_properties(append_test_declaratively, test_property_reflection, ""); |
124 | 28 |
|
125 | 29 | // Test that the corresponding properties return null when the reference target has invalid ID.
|
126 |
| - function appendTestDeclarativelyWithInvalidID(host_container, referenced_element_type) { |
| 30 | + function append_test_declaratively_with_invalid_id(host_container, referenced_element_type) { |
127 | 31 | host_container.setHTMLUnsafe(`
|
128 | 32 | <div id="host-id">
|
129 | 33 | <template shadowrootmode="open" shadowrootreferencetarget="invalid-id">
|
|
135 | 39 | }
|
136 | 40 | for(let referencing_element_type of element_types) {
|
137 | 41 | for(let referenced_element_type of element_types) {
|
138 |
| - test_property_reflection(appendTestDeclarativelyWithInvalidID, referencing_element_type, referenced_element_type, "form", "form", Behavior.IsNull); |
139 |
| - test_property_reflection(appendTestDeclarativelyWithInvalidID, referencing_element_type, referenced_element_type, "list", "list", Behavior.IsNull); |
140 |
| - test_property_reflection(appendTestDeclarativelyWithInvalidID, referencing_element_type, referenced_element_type, "for", "control", Behavior.IsNull); |
| 42 | + test_property_reflection(append_test_declaratively_with_invalid_id, referencing_element_type, referenced_element_type, "form", "form", Behavior.IsNull); |
| 43 | + test_property_reflection(append_test_declaratively_with_invalid_id, referencing_element_type, referenced_element_type, "list", "list", Behavior.IsNull); |
| 44 | + test_property_reflection(append_test_declaratively_with_invalid_id, referencing_element_type, referenced_element_type, "for", "control", Behavior.IsNull); |
141 | 45 | }
|
142 | 46 | }
|
143 | 47 |
|
|
146 | 50 | document.body.appendChild(referencing_element);
|
147 | 51 | referencing_element.setAttribute('for', "host-id");
|
148 | 52 | const host_container = document.querySelector("#host-container");
|
149 |
| - const host = appendTestDeclaratively(host_container, 'input'); |
| 53 | + const host = append_test_declaratively(host_container, 'input'); |
150 | 54 | const referenced_element = host.shadowRoot.getElementById('target');
|
151 | 55 | assert_array_equals(Array.from(referenced_element['labels']), [referencing_element]);
|
152 | 56 | referencing_element.remove();
|
153 | 57 | host_container.setHTMLUnsafe("");
|
154 | 58 | }, `The .labels property of the referenced input element should point to the referencing label element`);
|
155 | 59 | </script>
|
156 | 60 | </body>
|
157 |
| - |
158 | 61 | </html>
|
0 commit comments