Skip to content

Commit 4f7823e

Browse files
committed
[changed] focus target of the modal to its content
1 parent 281b546 commit 4f7823e

File tree

3 files changed

+47
-14
lines changed

3 files changed

+47
-14
lines changed

package.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -92,6 +92,7 @@
9292
"dependencies": {
9393
"classnames": "^2.1.3",
9494
"dom-helpers": "^2.2.4",
95-
"react-prop-types": "^0.2.1"
95+
"react-prop-types": "^0.2.1",
96+
"warning": "^2.0.0"
9697
}
9798
}

src/Modal.js

+21-8
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
/*eslint-disable react/prop-types */
22
import React, { cloneElement } from 'react';
3+
import warning from 'warning';
34
import elementType from 'react-prop-types/lib/elementType';
45

56
import Portal from './Portal';
@@ -154,11 +155,16 @@ const Modal = React.createClass({
154155
return null;
155156
}
156157

157-
if (dialog.props.role === undefined) {
158-
dialog = cloneElement(dialog, { role: 'document' });
158+
let { role, tabIndex } = dialog.props;
159+
160+
if (role === undefined || tabIndex === undefined) {
161+
dialog = cloneElement(dialog, {
162+
role: role === undefined ? 'document' : role,
163+
tabIndex: tabIndex == null ? '-1' : tabIndex
164+
});
159165
}
160166

161-
if ( Transition ) {
167+
if (Transition) {
162168
dialog = (
163169
<Transition
164170
transitionAppear
@@ -180,7 +186,6 @@ const Modal = React.createClass({
180186
return (
181187
<Portal container={props.container}>
182188
<div
183-
tabIndex='-1'
184189
ref={'modal'}
185190
role={props.role || 'dialog'}
186191
style={props.style}
@@ -318,20 +323,28 @@ const Modal = React.createClass({
318323
}
319324
},
320325

321-
checkForFocus(){
326+
checkForFocus() {
322327
if (canUseDom) {
323328
this.lastFocus = activeElement();
324329
}
325330
},
326331

327332
focus() {
328333
let autoFocus = this.props.autoFocus;
329-
let modalContent = React.findDOMNode(this.refs.modal);
334+
let modalContent = this.getDialogElement();
330335
let current = activeElement(ownerDocument(this));
331336
let focusInModal = current && contains(modalContent, current);
332337

333338
if (modalContent && autoFocus && !focusInModal) {
334339
this.lastFocus = current;
340+
341+
if (!modalContent.hasAttribute('tabIndex')){
342+
modalContent.setAttribute('tabIndex', -1);
343+
warning(false,
344+
'The modal content node does not accept focus. ' +
345+
'For the benefit of assistive technologies, the tabIndex of the node is being set to "-1".');
346+
}
347+
335348
modalContent.focus();
336349
}
337350
},
@@ -352,9 +365,9 @@ const Modal = React.createClass({
352365
}
353366

354367
let active = activeElement(ownerDocument(this));
355-
let modal = React.findDOMNode(this.refs.modal);
368+
let modal = this.getDialogElement();
356369

357-
if ( modal && modal !== active && !contains(modal, active)){
370+
if (modal && modal !== active && !contains(modal, active)) {
358371
modal.focus();
359372
}
360373
},

test/ModalSpec.js

+24-5
Original file line numberDiff line numberDiff line change
@@ -282,8 +282,8 @@ describe('Modal', function () {
282282
document.activeElement.should.equal(focusableContainer);
283283

284284
let instance = render(
285-
<Modal show className='modal'>
286-
<strong>Message</strong>
285+
<Modal show>
286+
<strong className='modal'>Message</strong>
287287
</Modal>
288288
, focusableContainer);
289289

@@ -311,7 +311,9 @@ describe('Modal', function () {
311311

312312
render(
313313
<Modal show>
314-
<input autoFocus />
314+
<div className='modal'>
315+
<input autoFocus/>
316+
</div>
315317
</Modal>
316318
, focusableContainer);
317319

@@ -325,14 +327,31 @@ describe('Modal', function () {
325327
document.activeElement.should.equal(focusableContainer);
326328

327329
render(
328-
<Modal show className='modal'>
329-
<input autoFocus/>
330+
<Modal show>
331+
<div className='modal'>
332+
<input autoFocus/>
333+
</div>
330334
</Modal>
331335
, focusableContainer);
332336

333337
focusableContainer.focus();
334338
document.activeElement.className.should.contain('modal');
335339
});
340+
341+
it('Should warn if the modal content is not focusable', function () {
342+
let Dialog = ()=> ({ render(){ return <div/>; } });
343+
344+
sinon.stub(console, 'error');
345+
346+
render(
347+
<Modal show>
348+
<Dialog />
349+
</Modal>
350+
, focusableContainer);
351+
352+
expect(console.error).to.have.been.calledOnce;
353+
console.error.restore();
354+
});
336355
});
337356

338357
});

0 commit comments

Comments
 (0)