Skip to content

Commit 67d4e4f

Browse files
dbjorgestraker
authored andcommitted
fix(target-size): do not treat focusable tabpanels as targets (#4702)
See #4421 for context Closes: #4701
1 parent 46ca544 commit 67d4e4f

File tree

9 files changed

+46
-16
lines changed

9 files changed

+46
-16
lines changed

lib/checks/aria/valid-scrollable-semantics-evaluate.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,8 @@ const VALID_ROLES_FOR_SCROLLABLE_REGIONS = {
3030
navigation: true,
3131
region: true,
3232
search: false,
33-
status: true
33+
status: true,
34+
tabpanel: true
3435
};
3536

3637
/**

lib/standards/aria-roles.js

+4-1
Original file line numberDiff line numberDiff line change
@@ -781,7 +781,10 @@ const ariaRoles = {
781781
superclassRole: ['composite']
782782
},
783783
tabpanel: {
784-
type: 'widget',
784+
// Spec ambiguity: Aria 1.1 and 1.2 both say that tabpanel rolls up to
785+
// structure via its section superclass, but also include it as a widget
786+
// in §5.3.2 Widget Roles.
787+
type: 'structure',
785788
// Spec difference: Aria-expanded removed in 1.2
786789
allowedAttrs: ['aria-expanded'],
787790
superclassRole: ['section'],

test/checks/aria/has-widget-role.js

+9-9
Original file line numberDiff line numberDiff line change
@@ -172,15 +172,6 @@ describe('has-widget-role', function () {
172172
);
173173
});
174174

175-
it('should return true for role=tabpanel', function () {
176-
node = document.createElement('div');
177-
node.setAttribute('role', 'tabpanel');
178-
fixture.appendChild(node);
179-
assert.isTrue(
180-
axe.testUtils.getCheckEvaluate('has-widget-role').call(checkContext, node)
181-
);
182-
});
183-
184175
it('should return true for role=textbox', function () {
185176
node = document.createElement('div');
186177
node.setAttribute('role', 'textbox');
@@ -479,6 +470,15 @@ describe('has-widget-role', function () {
479470
);
480471
});
481472

473+
it('should return false for role=tabpanel', function () {
474+
node = document.createElement('div');
475+
node.setAttribute('role', 'tabpanel');
476+
fixture.appendChild(node);
477+
assert.isFalse(
478+
axe.testUtils.getCheckEvaluate('has-widget-role').call(checkContext, node)
479+
);
480+
});
481+
482482
it('should return false for role=term', function () {
483483
node = document.createElement('div');
484484
node.setAttribute('role', 'term');

test/checks/aria/valid-scrollable-semantics.js

+12
Original file line numberDiff line numberDiff line change
@@ -186,6 +186,18 @@ describe('valid-scrollable-semantics', function () {
186186
);
187187
});
188188

189+
it('should return true for role=tabpanel', function () {
190+
var node = document.createElement('div');
191+
node.setAttribute('role', 'tabpanel');
192+
fixture.appendChild(node);
193+
flatTreeSetup(fixture);
194+
assert.isTrue(
195+
axe.testUtils
196+
.getCheckEvaluate('valid-scrollable-semantics')
197+
.call(checkContext, node)
198+
);
199+
});
200+
189201
it('should return true for role=tooltip', function () {
190202
var node = document.createElement('div');
191203
node.setAttribute('role', 'tooltip');

test/integration/full/target-size/too-many-rects.html

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@
2424
<section id="incomplete">
2525
<div
2626
id="incomplete-many-rects"
27-
role="tabpanel"
27+
role="button"
2828
tabindex="0"
2929
style="display: inline-block"
3030
>

test/integration/rules/focus-order-semantics/focus-order-semantics.html

+3-2
Original file line numberDiff line numberDiff line change
@@ -36,11 +36,12 @@ <h4>Valid landmark roles for scrollable containers</h4>
3636
<div id="pass11" role="alert" tabindex="0"></div>
3737
<div id="pass12" role="log" tabindex="0"></div>
3838
<div id="pass13" role="status" tabindex="0"></div>
39+
<div id="pass14" role="tabpanel" tabindex="0"></div>
3940
</div>
4041
<h4>Valid window roles for scrollable containers</h4>
4142
<div>
42-
<div id="pass14" role="alertdialog" tabindex="0"></div>
43-
<div id="pass15" role="dialog" tabindex="0"></div>
43+
<div id="pass15" role="alertdialog" tabindex="0"></div>
44+
<div id="pass16" role="dialog" tabindex="0"></div>
4445
</div>
4546
<h4>
4647
Valid scrollable HTML tags for scrollable regions, not selected by this

test/integration/rules/focus-order-semantics/focus-order-semantics.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,8 @@
1616
["#pass12"],
1717
["#pass13"],
1818
["#pass14"],
19-
["#pass15"]
19+
["#pass15"],
20+
["#pass16"]
2021
],
2122
"violations": [
2223
["#violation1"],

test/integration/rules/target-size/target-size.html

+11-1
Original file line numberDiff line numberDiff line change
@@ -60,7 +60,12 @@
6060

6161
<div>
6262
<button id="pass15">x</button>
63-
<div role="timer" tabindex="0" id="ignore6">Timer</div>
63+
<div role="tabpanel" tabindex="0" id="ignore6">Tabpanel</div>
64+
</div>
65+
66+
<div>
67+
<button id="pass16">x</button>
68+
<div role="timer" tabindex="0" id="ignore7">Timer</div>
6469
</div>
6570

6671
<!-- Nested controls-->
@@ -88,6 +93,11 @@
8893
</a>
8994
</p>
9095

96+
<div role="tabpanel" tabindex="0">
97+
Focusable tabpanel containing small control
98+
<button id="pass17">x</button>
99+
</div>
100+
91101
<!-- Failed examples -->
92102
<p>
93103
<button id="fail1">x</button

test/integration/rules/target-size/target-size.json

+2
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,8 @@
2727
["#pass13"],
2828
["#pass14"],
2929
["#pass15"],
30+
["#pass16"],
31+
["#pass17"],
3032
["#pass-adjacent"],
3133
["#pass-fixed"]
3234
],

0 commit comments

Comments
 (0)