Skip to content

Commit 5313abe

Browse files
committed
[fixed] Modal is focused when opened, for improved accessibility
Adds test case for focusing on the modal when opening it
1 parent ed3f641 commit 5313abe

File tree

2 files changed

+78
-0
lines changed

2 files changed

+78
-0
lines changed

src/Modal.js

+17
Original file line numberDiff line numberDiff line change
@@ -126,6 +126,8 @@ const Modal = React.createClass({
126126
domUtils.ownerDocument(this).body;
127127
container.className += container.className.length ? ' modal-open' : 'modal-open';
128128

129+
this.focusModalContent();
130+
129131
if (this.props.backdrop) {
130132
this.iosClickHack();
131133
}
@@ -142,6 +144,8 @@ const Modal = React.createClass({
142144
let container = (this.props.container && React.findDOMNode(this.props.container)) ||
143145
domUtils.ownerDocument(this).body;
144146
container.className = container.className.replace(/ ?modal-open/, '');
147+
148+
this.restoreLastFocus();
145149
},
146150

147151
handleBackdropClick(e) {
@@ -156,6 +160,19 @@ const Modal = React.createClass({
156160
if (this.props.keyboard && e.keyCode === 27) {
157161
this.props.onRequestHide();
158162
}
163+
},
164+
165+
focusModalContent () {
166+
this.lastFocus = document.activeElement;
167+
let modalContent = React.findDOMNode(this.refs.modal);
168+
modalContent.focus();
169+
},
170+
171+
restoreLastFocus () {
172+
if (this.lastFocus) {
173+
this.lastFocus.focus();
174+
this.lastFocus = null;
175+
}
159176
}
160177
});
161178

test/ModalSpec.js

+61
Original file line numberDiff line numberDiff line change
@@ -97,4 +97,65 @@ describe('Modal', function () {
9797
assert.match(dialog.props.className, /\btestCss\b/);
9898
});
9999

100+
describe('Focused state', function () {
101+
let focusableContainer = null;
102+
beforeEach(function () {
103+
focusableContainer = document.createElement('div');
104+
focusableContainer.tabIndex = 0;
105+
document.body.appendChild(focusableContainer);
106+
focusableContainer.focus();
107+
});
108+
109+
afterEach(function () {
110+
React.unmountComponentAtNode(focusableContainer);
111+
document.body.removeChild(focusableContainer);
112+
});
113+
114+
it('Should focus on the Modal when it is opened', function (done) {
115+
document.activeElement.should.equal(focusableContainer);
116+
117+
let doneOp = function () {
118+
// focus should be back on the previous element when modal closed
119+
setTimeout(function () {
120+
document.activeElement.should.equal(focusableContainer);
121+
done();
122+
}, 0);
123+
};
124+
125+
let Container = React.createClass({
126+
getInitialState() {
127+
return {modalOpen: true};
128+
},
129+
handleCloseModal() {
130+
this.setState({modalOpen: false});
131+
doneOp();
132+
},
133+
render() {
134+
if (this.state.modalOpen) {
135+
return (
136+
<Modal onRequestHide={this.handleCloseModal} container={this}>
137+
<strong>Message</strong>
138+
</Modal>
139+
);
140+
} else {
141+
return <span/>;
142+
}
143+
}
144+
});
145+
146+
let instance = React.render(<Container />, focusableContainer);
147+
148+
setTimeout(function () {
149+
// modal should be focused when opened
150+
let modal = instance.getDOMNode().getElementsByClassName('modal')[0];
151+
document.activeElement.should.equal(modal);
152+
153+
// close the modal
154+
let backdrop = instance.getDOMNode().getElementsByClassName('modal-backdrop')[0];
155+
ReactTestUtils.Simulate.click(backdrop);
156+
}, 0);
157+
});
158+
159+
});
160+
100161
});

0 commit comments

Comments
 (0)