@@ -31,6 +31,28 @@ function getContainer(context){
31
31
domUtils . ownerDocument ( context ) . body ;
32
32
}
33
33
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 ;
34
56
35
57
if ( domUtils . canUseDom ) {
36
58
let scrollDiv = document . createElement ( 'div' ) ;
@@ -60,7 +82,8 @@ const Modal = React.createClass({
60
82
closeButton : React . PropTypes . bool ,
61
83
animation : React . PropTypes . bool ,
62
84
onRequestHide : React . PropTypes . func . isRequired ,
63
- dialogClassName : React . PropTypes . string
85
+ dialogClassName : React . PropTypes . string ,
86
+ enforceFocus : React . PropTypes . bool
64
87
} ,
65
88
66
89
getDefaultProps ( ) {
@@ -69,10 +92,15 @@ const Modal = React.createClass({
69
92
backdrop : true ,
70
93
keyboard : true ,
71
94
animation : true ,
72
- closeButton : true
95
+ closeButton : true ,
96
+ enforceFocus : true
73
97
} ;
74
98
} ,
75
99
100
+ getInitialState ( ) {
101
+ return { } ;
102
+ } ,
103
+
76
104
render ( ) {
77
105
let state = this . state ;
78
106
let modalStyle = { ...state . dialogStyles , display : 'block' } ;
@@ -107,7 +135,7 @@ const Modal = React.createClass({
107
135
) ;
108
136
109
137
return this . props . backdrop ?
110
- this . renderBackdrop ( modal ) : modal ;
138
+ this . renderBackdrop ( modal , state . backdropStyles ) : modal ;
111
139
} ,
112
140
113
141
renderBackdrop ( modal ) {
@@ -132,8 +160,8 @@ const Modal = React.createClass({
132
160
let closeButton ;
133
161
if ( this . props . closeButton ) {
134
162
closeButton = (
135
- < button type = "button" className = "close" aria-hidden = "true" onClick = { this . props . onRequestHide } > ×</ button >
136
- ) ;
163
+ < button type = "button" className = "close" aria-hidden = "true" onClick = { this . props . onRequestHide } > ×</ button >
164
+ ) ;
137
165
}
138
166
139
167
return (
@@ -169,6 +197,10 @@ const Modal = React.createClass({
169
197
this . _onWindowResizeListener =
170
198
EventListener . listen ( win , 'resize' , this . handleWindowResize ) ;
171
199
200
+ if ( this . props . enforceFocus ) {
201
+ this . _onFocusinListener = onFocus ( this , this . enforceFocus ) ;
202
+ }
203
+
172
204
let container = getContainer ( this ) ;
173
205
174
206
container . className += container . className . length ? ' modal-open' : 'modal-open' ;
@@ -199,6 +231,10 @@ const Modal = React.createClass({
199
231
this . _onDocumentKeyupListener . remove ( ) ;
200
232
this . _onWindowResizeListener . remove ( ) ;
201
233
234
+ if ( this . _onFocusinListener ) {
235
+ this . _onFocusinListener . remove ( ) ;
236
+ }
237
+
202
238
let container = getContainer ( this ) ;
203
239
204
240
container . className = container . className . replace ( / ? m o d a l - o p e n / , '' ) ;
@@ -237,6 +273,19 @@ const Modal = React.createClass({
237
273
}
238
274
} ,
239
275
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
+
240
289
_getStyles ( ) {
241
290
if ( ! domUtils . canUseDom ) { return { } ; }
242
291
0 commit comments