Skip to content

Commit f2c3b68

Browse files
jquensetaion
authored andcommitted
[changed] tab keyboard navigation to be more inline with ARIA spec
http://www.w3.org/TR/wai-aria-practices/#tabpanel
1 parent 06ba2dc commit f2c3b68

File tree

5 files changed

+179
-6
lines changed

5 files changed

+179
-6
lines changed

src/NavItem.js

+2
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,7 @@ const NavItem = React.createClass({
3636
title,
3737
target,
3838
children,
39+
tabIndex, //eslint-disable-line
3940
'aria-controls': ariaControls,
4041
...props } = this.props;
4142
let classes = {
@@ -47,6 +48,7 @@ const NavItem = React.createClass({
4748
href,
4849
title,
4950
target,
51+
tabIndex,
5052
id: linkId,
5153
onClick: this.handleClick
5254
};

src/Tabs.js

+92-6
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,19 @@
11
import classNames from 'classnames';
2-
import React, { cloneElement } from 'react';
2+
import React, { cloneElement, findDOMNode } from 'react';
33

44
import Col from './Col';
55
import Nav from './Nav';
66
import NavItem from './NavItem';
77
import styleMaps from './styleMaps';
8-
8+
import keycode from 'keycode';
9+
import createChainedFunction from './utils/createChainedFunction';
910
import ValidComponentChildren from './utils/ValidComponentChildren';
1011

1112
let paneId = (props, child) => child.props.id ? child.props.id : props.id && (props.id + '___pane___' + child.props.eventKey);
1213
let tabId = (props, child) => child.props.id ? child.props.id + '___tab' : props.id && (props.id + '___tab___' + child.props.eventKey);
1314

15+
let findChild = ValidComponentChildren.find;
16+
1417
function getDefaultActiveKeyFromChildren(children) {
1518
let defaultActiveKey;
1619

@@ -23,6 +26,30 @@ function getDefaultActiveKeyFromChildren(children) {
2326
return defaultActiveKey;
2427
}
2528

29+
function move(children, currentKey, keys, moveNext) {
30+
let lastIdx = keys.length - 1;
31+
let stopAt = keys[moveNext ? Math.max(lastIdx, 0) : 0];
32+
let nextKey = currentKey;
33+
34+
function getNext() {
35+
let idx = keys.indexOf(nextKey);
36+
nextKey = moveNext
37+
? keys[Math.min(lastIdx, idx + 1)]
38+
: keys[Math.max(0, idx - 1)];
39+
40+
return findChild(children,
41+
_child => _child.props.eventKey === nextKey);
42+
}
43+
44+
let next = getNext();
45+
46+
while (next.props.eventKey !== stopAt && next.props.disabled) {
47+
next = getNext();
48+
}
49+
50+
return next.props.disabled ? currentKey : next.props.eventKey;
51+
}
52+
2653
const Tabs = React.createClass({
2754
propTypes: {
2855
activeKey: React.PropTypes.any,
@@ -103,6 +130,22 @@ const Tabs = React.createClass({
103130
}
104131
},
105132

133+
componentDidUpdate() {
134+
let tabs = this._tabs;
135+
let tabIdx = this._eventKeys().indexOf(this.getActiveKey());
136+
137+
if (this._needsRefocus) {
138+
this._needsRefocus = false;
139+
if (tabs && tabIdx !== -1) {
140+
let tabNode = findDOMNode(tabs[tabIdx]);
141+
142+
if (tabNode) {
143+
tabNode.firstChild.focus();
144+
}
145+
}
146+
}
147+
},
148+
106149
handlePaneAnimateOutEnd() {
107150
this.setState({
108151
previousActiveKey: null
@@ -223,20 +266,23 @@ const Tabs = React.createClass({
223266
);
224267
},
225268

226-
renderTab(child) {
269+
renderTab(child, index) {
227270
if (child.props.title == null) {
228271
return null;
229272
}
230273

231-
let {eventKey, title, disabled} = child.props;
274+
let { eventKey, title, disabled, onKeyDown, tabIndex = 0 } = child.props;
275+
let isActive = this.getActiveKey() === eventKey;
232276

233277
return (
234278
<NavItem
235279
linkId={tabId(this.props, child)}
236-
ref={'tab' + eventKey}
280+
ref={ref => (this._tabs || (this._tabs = []))[index] = ref}
237281
aria-controls={paneId(this.props, child)}
282+
onKeyDown={createChainedFunction(this.handleKeyDown, onKeyDown)}
238283
eventKey={eventKey}
239-
disabled={disabled}>
284+
tabIndex={isActive ? tabIndex : -1}
285+
disabled={disabled }>
240286
{title}
241287
</NavItem>
242288
);
@@ -286,6 +332,46 @@ const Tabs = React.createClass({
286332
previousActiveKey
287333
});
288334
}
335+
},
336+
337+
handleKeyDown(event) {
338+
let keys = this._eventKeys();
339+
let currentKey = this.getActiveKey() || keys[0];
340+
let next;
341+
342+
switch (event.keyCode) {
343+
344+
case keycode.codes.left:
345+
case keycode.codes.up:
346+
next = move(this.props.children, currentKey, keys, false);
347+
348+
if (next && next !== currentKey) {
349+
event.preventDefault();
350+
this.handleSelect(next);
351+
this._needsRefocus = true;
352+
}
353+
break;
354+
case keycode.codes.right:
355+
case keycode.codes.down:
356+
next = move(this.props.children, currentKey, keys, true);
357+
358+
if (next && next !== currentKey) {
359+
event.preventDefault();
360+
this.handleSelect(next);
361+
this._needsRefocus = true;
362+
}
363+
break;
364+
default:
365+
}
366+
},
367+
368+
_eventKeys() {
369+
let keys = [];
370+
371+
ValidComponentChildren.forEach(this.props.children,
372+
({props: { eventKey }}) => keys.push(eventKey));
373+
374+
return keys;
289375
}
290376
});
291377

src/utils/ValidComponentChildren.js

+13
Original file line numberDiff line numberDiff line change
@@ -82,9 +82,22 @@ function hasValidComponent(children) {
8282
return hasValid;
8383
}
8484

85+
function find(children, finder) {
86+
let child;
87+
88+
forEachValidComponents(children, (c, idx)=> {
89+
if (!child && finder(c, idx, children)) {
90+
child = c;
91+
}
92+
});
93+
94+
return child;
95+
}
96+
8597
export default {
8698
map: mapValidComponents,
8799
forEach: forEachValidComponents,
88100
numberOf: numberOfValidComponents,
101+
find,
89102
hasValidComponent
90103
};

test/NavItemSpec.js

+14
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,20 @@ describe('NavItem', function () {
4343
assert.ok(!React.findDOMNode(instance).hasAttribute('title'));
4444
});
4545

46+
it('Should pass tabIndex to the anchor', () => {
47+
let instance = ReactTestUtils.renderIntoDocument(
48+
<NavItem href='/hi' tabIndex='3' title='boom!'>
49+
Item content
50+
</NavItem>
51+
);
52+
53+
let node = React.findDOMNode(instance);
54+
55+
expect(node.hasAttribute('tabindex')).to.equal(false);
56+
expect(node.firstChild.getAttribute('tabindex')).to.equal('3');
57+
58+
});
59+
4660
it('Should call `onSelect` when item is selected', function (done) {
4761
function handleSelect(key) {
4862
assert.equal(key, '2');

test/TabsSpec.js

+58
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import React from 'react';
22
import ReactTestUtils from 'react/lib/ReactTestUtils';
3+
import keycode from 'keycode';
34

45
import Col from '../src/Col';
56
import Nav from '../src/Nav';
@@ -432,6 +433,63 @@ describe('Tabs', function () {
432433
checkTabRemovingWithAnimation(false);
433434
});
434435

436+
describe('keyboard navigation', function() {
437+
let instance;
438+
439+
beforeEach(function() {
440+
instance = render(
441+
<Tabs defaultActiveKey={1} id='tabs'>
442+
<Tab id='pane-1' title="Tab 1" eventKey={1}>Tab 1 content</Tab>
443+
<Tab id='pane-2' title="Tab 2" eventKey={2} disabled>Tab 2 content</Tab>
444+
<Tab id='pane-2' title="Tab 3" eventKey={3}>Tab 3 content</Tab>
445+
</Tabs>
446+
, document.body);
447+
});
448+
449+
afterEach(function() {
450+
instance = React.unmountComponentAtNode(document.body);
451+
});
452+
453+
it('only the active tab should be focusable', () => {
454+
let tabs = ReactTestUtils.scryRenderedComponentsWithType(instance, NavItem);
455+
456+
expect(React.findDOMNode(tabs[0]).firstChild.getAttribute('tabindex')).to.equal('0');
457+
458+
expect(React.findDOMNode(tabs[1]).firstChild.getAttribute('tabindex')).to.equal('-1');
459+
expect(React.findDOMNode(tabs[2]).firstChild.getAttribute('tabindex')).to.equal('-1');
460+
});
461+
462+
it('should focus the next tab on arrow key', () => {
463+
let tabs = ReactTestUtils.scryRenderedComponentsWithType(instance, NavItem);
464+
465+
let firstAnchor = React.findDOMNode(tabs[0]).firstChild;
466+
let lastAnchor = React.findDOMNode(tabs[2]).firstChild; // skip disabled
467+
468+
firstAnchor.focus();
469+
470+
ReactTestUtils.Simulate.keyDown(firstAnchor, { keyCode: keycode('right') });
471+
472+
expect(instance.getActiveKey() === 2);
473+
expect(document.activeElement).to.equal(lastAnchor);
474+
});
475+
476+
it('should focus the previous tab on arrow key', () => {
477+
instance.setState({ activeKey: 3 });
478+
479+
let tabs = ReactTestUtils.scryRenderedComponentsWithType(instance, NavItem);
480+
481+
let firstAnchor = React.findDOMNode(tabs[0]).firstChild;
482+
let lastAnchor = React.findDOMNode(tabs[2]).firstChild;
483+
484+
lastAnchor.focus();
485+
486+
ReactTestUtils.Simulate.keyDown(lastAnchor, { keyCode: keycode('left') });
487+
488+
expect(instance.getActiveKey() === 2);
489+
expect(document.activeElement).to.equal(firstAnchor);
490+
});
491+
});
492+
435493
describe('Web Accessibility', function() {
436494
let instance;
437495
beforeEach(function() {

0 commit comments

Comments
 (0)