Skip to content

Commit 1e552cc

Browse files
committedJun 18, 2015
[added] Accessibility: use appropriate ARIA's when an id is given to the tabbed area
1 parent 8752754 commit 1e552cc

File tree

4 files changed

+85
-5
lines changed

4 files changed

+85
-5
lines changed
 

‎src/TabPane.js

+5-1
Original file line numberDiff line numberDiff line change
@@ -78,7 +78,11 @@ const TabPane = React.createClass({
7878
};
7979

8080
return (
81-
<div {...this.props} className={classNames(this.props.className, classes)}>
81+
<div {...this.props}
82+
role='tabpanel'
83+
aria-hidden={!this.props.active}
84+
className={classNames(this.props.className, classes)}
85+
>
8286
{this.props.children}
8387
</div>
8488
);

‎src/TabbedArea.js

+16-4
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,9 @@ import ValidComponentChildren from './utils/ValidComponentChildren';
55
import Nav from './Nav';
66
import NavItem from './NavItem';
77

8+
let panelId = (props, child) => child.props.id ? child.props.id : props.id && (props.id + '___panel___' + child.props.eventKey);
9+
let tabId = (props, child) => child.props.id ? child.props.id + '___tab' : props.id && (props.id + '___tab___' + child.props.eventKey);
10+
811
function getDefaultActiveKeyFromChildren(children) {
912
let defaultActiveKey;
1013

@@ -61,6 +64,8 @@ const TabbedArea = React.createClass({
6164
},
6265

6366
render() {
67+
let { id, ...props } = this.props; // eslint-disable-line object-shorthand
68+
6469
let activeKey =
6570
this.props.activeKey != null ? this.props.activeKey : this.state.activeKey;
6671

@@ -69,15 +74,15 @@ const TabbedArea = React.createClass({
6974
}
7075

7176
let nav = (
72-
<Nav {...this.props} activeKey={activeKey} onSelect={this.handleSelect} ref="tabs">
77+
<Nav {...props} activeKey={activeKey} onSelect={this.handleSelect} ref="tabs">
7378
{ValidComponentChildren.map(this.props.children, renderTabIfSet, this)}
7479
</Nav>
7580
);
7681

7782
return (
7883
<div>
7984
{nav}
80-
<div id={this.props.id} className="tab-content" ref="panes">
85+
<div id={id} className="tab-content" ref="panes">
8186
{ValidComponentChildren.map(this.props.children, this.renderPane)}
8287
</div>
8388
</div>
@@ -91,11 +96,15 @@ const TabbedArea = React.createClass({
9196
renderPane(child, index) {
9297
let activeKey = this.getActiveKey();
9398

99+
let active = (child.props.eventKey === activeKey &&
100+
(this.state.previousActiveKey == null || !this.props.animation));
101+
94102
return cloneElement(
95103
child,
96104
{
97-
active: (child.props.eventKey === activeKey &&
98-
(this.state.previousActiveKey == null || !this.props.animation)),
105+
active,
106+
id: panelId(this.props, child),
107+
'aria-labelledby': tabId(this.props, child),
99108
key: child.key ? child.key : index,
100109
animation: this.props.animation,
101110
onAnimateOutEnd: (this.state.previousActiveKey != null &&
@@ -106,9 +115,12 @@ const TabbedArea = React.createClass({
106115

107116
renderTab(child) {
108117
let {eventKey, className, tab, disabled } = child.props;
118+
109119
return (
110120
<NavItem
121+
linkId={tabId(this.props, child)}
111122
ref={'tab' + eventKey}
123+
aria-controls={panelId(this.props, child)}
112124
eventKey={eventKey}
113125
className={className}
114126
disabled={disabled}>

‎test/TabPaneSpec.js

+19
Original file line numberDiff line numberDiff line change
@@ -16,4 +16,23 @@ describe('TabPane', function () {
1616
);
1717
assert.ok(ReactTestUtils.findRenderedDOMComponentWithClass(instance, 'active'));
1818
});
19+
20+
describe('Web Accessibility', function(){
21+
22+
it('Should have aria-hidden', function () {
23+
let instance = ReactTestUtils.renderIntoDocument(
24+
<TabPane active>Item content</TabPane>
25+
);
26+
27+
assert.equal(React.findDOMNode(instance).getAttribute('aria-hidden'), 'false');
28+
});
29+
30+
it('Should have role', function () {
31+
let instance = ReactTestUtils.renderIntoDocument(
32+
<TabPane active>Item content</TabPane>
33+
);
34+
35+
assert.equal(React.findDOMNode(instance).getAttribute('role'), 'tabpanel');
36+
});
37+
});
1938
});

‎test/TabbedAreaSpec.js

+45
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import React from 'react';
22
import ReactTestUtils from 'react/lib/ReactTestUtils';
33
import TabbedArea from '../src/TabbedArea';
4+
import NavItem from '../src/NavItem';
45
import TabPane from '../src/TabPane';
56
import ValidComponentChildren from '../src/utils/ValidComponentChildren';
67

@@ -229,5 +230,49 @@ describe('TabbedArea', function () {
229230
assert.equal(tabbedArea.refs.tabs.props.activeKey, 2);
230231
});
231232

233+
describe('Web Accessibility', function(){
232234

235+
it('Should generate ids from parent id', function () {
236+
let instance = ReactTestUtils.renderIntoDocument(
237+
<TabbedArea defaultActiveKey={2} id='tabs'>
238+
<TabPane tab="Tab 1" eventKey={1}>Tab 1 content</TabPane>
239+
<TabPane tab="Tab 2" eventKey={2}>Tab 2 content</TabPane>
240+
</TabbedArea>
241+
);
242+
243+
let tabs = ReactTestUtils.scryRenderedComponentsWithType(instance, NavItem);
244+
245+
tabs.every(tab =>
246+
assert.ok(tab.props['aria-controls'] && tab.props.linkId));
247+
});
248+
249+
it('Should add aria-controls', function () {
250+
let instance = ReactTestUtils.renderIntoDocument(
251+
<TabbedArea defaultActiveKey={2} id='tabs'>
252+
<TabPane id='pane-1' tab="Tab 1" eventKey={1}>Tab 1 content</TabPane>
253+
<TabPane id='pane-2' tab="Tab 2" eventKey={2}>Tab 2 content</TabPane>
254+
</TabbedArea>
255+
);
256+
257+
let panes = ReactTestUtils.scryRenderedComponentsWithType(instance, TabPane);
258+
259+
assert.equal(panes[0].props['aria-labelledby'], 'pane-1___tab');
260+
assert.equal(panes[1].props['aria-labelledby'], 'pane-2___tab');
261+
});
262+
263+
it('Should add aria-controls', function () {
264+
let instance = ReactTestUtils.renderIntoDocument(
265+
<TabbedArea defaultActiveKey={2} id='tabs'>
266+
<TabPane id='pane-1' tab="Tab 1" eventKey={1}>Tab 1 content</TabPane>
267+
<TabPane id='pane-2' tab="Tab 2" eventKey={2}>Tab 2 content</TabPane>
268+
</TabbedArea>
269+
);
270+
271+
let tabs = ReactTestUtils.scryRenderedComponentsWithType(instance, NavItem);
272+
273+
assert.equal(tabs[0].props['aria-controls'], 'pane-1');
274+
assert.equal(tabs[1].props['aria-controls'], 'pane-2');
275+
});
276+
277+
});
233278
});

0 commit comments

Comments
 (0)
Please sign in to comment.