Skip to content

Commit 66f0f92

Browse files
committed
[added] enforceFocus prop to Modal
Allows you to configure whether the modal should enforce focus when open. Ideally this doesn't need to be a public API, but in the case of static models (like in the docs) you need to turn it off, because you can't assume its the only one open on the page.
1 parent 3869ca2 commit 66f0f92

File tree

4 files changed

+71
-6
lines changed

4 files changed

+71
-6
lines changed

docs/examples/ModalStatic.js

+1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
const modalInstance = (
22
<div className='static-modal'>
33
<Modal title='Modal title'
4+
enforceFocus={false}
45
backdrop={false}
56
animation={false}
67
container={mountNode}

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@
1313
"test": "npm run lint && npm run build && karma start --single-run && _mocha --compilers js:babel-core/register test/server/*Spec.js",
1414
"lint": "eslint ./",
1515
"docs-build": "babel-node tools/build-cli.js --docs-only",
16-
"docs": "docs/dev-run",
16+
"docs": "babel-node docs/dev-run",
1717
"docs-prod": "npm run docs-build && NODE_ENV=production babel-node docs/server.js",
1818
"docs-prod-unoptimized": "npm run docs-build -- --dev && NODE_ENV=production babel-node docs/server.js"
1919
},

src/Modal.js

+54-5
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,28 @@ function getContainer(context){
3131
domUtils.ownerDocument(context).body;
3232
}
3333

34+
/**
35+
* Firefox doesn't have a focusin event so using capture is easiest way to get bubbling
36+
* IE8 can't do addEventListener, but does have onfocus in, so we use that in ie8
37+
* @param {ReactElement|HTMLElement} context
38+
* @param {Function} handler
39+
*/
40+
function onFocus(context, handler) {
41+
let doc = domUtils.ownerDocument(context);
42+
let useFocusin = !doc.addEventListener
43+
, remove;
44+
45+
if (useFocusin) {
46+
document.attachEvent('onfocusin', handler);
47+
remove = () => document.detachEvent('onfocusin', handler);
48+
} else {
49+
document.addEventListener('focus', handler, true);
50+
remove = () => document.removeEventListener('focus', handler, true);
51+
}
52+
return { remove };
53+
}
54+
55+
let scrollbarSize;
3456

3557
if ( domUtils.canUseDom) {
3658
let scrollDiv = document.createElement('div');
@@ -60,7 +82,8 @@ const Modal = React.createClass({
6082
closeButton: React.PropTypes.bool,
6183
animation: React.PropTypes.bool,
6284
onRequestHide: React.PropTypes.func.isRequired,
63-
dialogClassName: React.PropTypes.string
85+
dialogClassName: React.PropTypes.string,
86+
enforceFocus: React.PropTypes.bool
6487
},
6588

6689
getDefaultProps() {
@@ -69,10 +92,15 @@ const Modal = React.createClass({
6992
backdrop: true,
7093
keyboard: true,
7194
animation: true,
72-
closeButton: true
95+
closeButton: true,
96+
enforceFocus: true
7397
};
7498
},
7599

100+
getInitialState(){
101+
return { };
102+
},
103+
76104
render() {
77105
let state = this.state;
78106
let modalStyle = { ...state.dialogStyles, display: 'block'};
@@ -107,7 +135,7 @@ const Modal = React.createClass({
107135
);
108136

109137
return this.props.backdrop ?
110-
this.renderBackdrop(modal) : modal;
138+
this.renderBackdrop(modal, state.backdropStyles) : modal;
111139
},
112140

113141
renderBackdrop(modal) {
@@ -132,8 +160,8 @@ const Modal = React.createClass({
132160
let closeButton;
133161
if (this.props.closeButton) {
134162
closeButton = (
135-
<button type="button" className="close" aria-hidden="true" onClick={this.props.onRequestHide}>&times;</button>
136-
);
163+
<button type="button" className="close" aria-hidden="true" onClick={this.props.onRequestHide}>&times;</button>
164+
);
137165
}
138166

139167
return (
@@ -169,6 +197,10 @@ const Modal = React.createClass({
169197
this._onWindowResizeListener =
170198
EventListener.listen(win, 'resize', this.handleWindowResize);
171199

200+
if (this.props.enforceFocus) {
201+
this._onFocusinListener = onFocus(this, this.enforceFocus);
202+
}
203+
172204
let container = getContainer(this);
173205

174206
container.className += container.className.length ? ' modal-open' : 'modal-open';
@@ -199,6 +231,10 @@ const Modal = React.createClass({
199231
this._onDocumentKeyupListener.remove();
200232
this._onWindowResizeListener.remove();
201233

234+
if (this._onFocusinListener) {
235+
this._onFocusinListener.remove();
236+
}
237+
202238
let container = getContainer(this);
203239

204240
container.className = container.className.replace(/ ?modal-open/, '');
@@ -237,6 +273,19 @@ const Modal = React.createClass({
237273
}
238274
},
239275

276+
enforceFocus() {
277+
if ( !this.isMounted() ) {
278+
return;
279+
}
280+
281+
let active = domUtils.activeElement(this)
282+
, modal = React.findDOMNode(this.refs.modal);
283+
284+
if (modal !== active && !domUtils.contains(modal, active)){
285+
modal.focus();
286+
}
287+
},
288+
240289
_getStyles() {
241290
if ( !domUtils.canUseDom ) { return {}; }
242291

src/utils/domUtils.js

+15
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,20 @@ function ownerWindow(componentOrElement) {
2626
: doc.parentWindow;
2727
}
2828

29+
/**
30+
* get the active element, safe in IE
31+
* @return {HTMLElement}
32+
*/
33+
function getActiveElement(componentOrElement){
34+
let doc = ownerDocument(componentOrElement);
35+
36+
try {
37+
return doc.activeElement || doc.body;
38+
} catch (e) {
39+
return doc.body;
40+
}
41+
}
42+
2943
/**
3044
* Shortcut to compute element style
3145
*
@@ -160,5 +174,6 @@ export default {
160174
getComputedStyles,
161175
getOffset,
162176
getPosition,
177+
activeElement: getActiveElement,
163178
offsetParent: offsetParentFunc
164179
};

0 commit comments

Comments
 (0)