Skip to content

Commit 7175431

Browse files
[fixed] Don't try to access .ownerDocument on null
We previously did not take into account that React.findDOMNode(this) can return null, (for example if you have a modal and only want to render the overlay). We now fallback to the root document in that case. react-bootstrap@ee0382e
1 parent a58cff9 commit 7175431

7 files changed

+51
-10
lines changed

src/AffixMixin.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -104,7 +104,7 @@ const AffixMixin = {
104104
this._onWindowScrollListener =
105105
EventListener.listen(window, 'scroll', this.checkPosition);
106106
this._onDocumentClickListener =
107-
EventListener.listen(React.findDOMNode(this).ownerDocument, 'click', this.checkPositionWithEventLoop);
107+
EventListener.listen(domUtils.ownerDocument(this), 'click', this.checkPositionWithEventLoop);
108108
},
109109

110110
componentWillUnmount() {

src/DropdownStateMixin.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
import React from 'react';
2+
import domUtils from './utils/domUtils';
23
import EventListener from './utils/EventListener';
34

45
/**
@@ -56,7 +57,7 @@ const DropdownStateMixin = {
5657
},
5758

5859
bindRootCloseHandlers() {
59-
let doc = React.findDOMNode(this).ownerDocument;
60+
let doc = domUtils.ownerDocument(this);
6061

6162
this._onDocumentClickListener =
6263
EventListener.listen(doc, 'click', this.handleDocumentClick);

src/FadeMixin.js

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
import React from 'react';
2+
import domUtils from './utils/domUtils';
23

34
// TODO: listen for onTransitionEnd to remove el
45
function getElementsAndSelf (root, classes){
@@ -57,7 +58,8 @@ export default {
5758

5859
componentWillUnmount: function () {
5960
let els = getElementsAndSelf(React.findDOMNode(this), ['fade']),
60-
container = (this.props.container && React.findDOMNode(this.props.container)) || React.findDOMNode(this).ownerDocument.body;
61+
container = (this.props.container && React.findDOMNode(this.props.container)) ||
62+
domUtils.ownerDocument(this).body;
6163

6264
if (els.length) {
6365
this._fadeOutEl = document.createElement('div');

src/Modal.js

+6-3
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ import React from 'react';
22
import classSet from 'classnames';
33
import BootstrapMixin from './BootstrapMixin';
44
import FadeMixin from './FadeMixin';
5+
import domUtils from './utils/domUtils';
56
import EventListener from './utils/EventListener';
67

78

@@ -128,9 +129,10 @@ const Modal = React.createClass({
128129

129130
componentDidMount() {
130131
this._onDocumentKeyupListener =
131-
EventListener.listen(React.findDOMNode(this).ownerDocument, 'keyup', this.handleDocumentKeyUp);
132+
EventListener.listen(domUtils.ownerDocument(this), 'keyup', this.handleDocumentKeyUp);
132133

133-
let container = (this.props.container && React.findDOMNode(this.props.container)) || React.findDOMNode(this).ownerDocument.body;
134+
let container = (this.props.container && React.findDOMNode(this.props.container)) ||
135+
domUtils.ownerDocument(this).body;
134136
container.className += container.className.length ? ' modal-open' : 'modal-open';
135137

136138
if (this.props.backdrop) {
@@ -146,7 +148,8 @@ const Modal = React.createClass({
146148

147149
componentWillUnmount() {
148150
this._onDocumentKeyupListener.remove();
149-
let container = (this.props.container && React.findDOMNode(this.props.container)) || React.findDOMNode(this).ownerDocument.body;
151+
let container = (this.props.container && React.findDOMNode(this.props.container)) ||
152+
domUtils.ownerDocument(this).body;
150153
container.className = container.className.replace(/ ?modal-open/, '');
151154
},
152155

src/OverlayMixin.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import React from 'react';
22
import CustomPropTypes from './utils/CustomPropTypes';
3+
import domUtils from './utils/domUtils';
34

45
export default {
56
propTypes: {
@@ -63,6 +64,6 @@ export default {
6364
},
6465

6566
getContainerDOMNode() {
66-
return React.findDOMNode(this.props.container || React.findDOMNode(this).ownerDocument.body);
67+
return React.findDOMNode(this.props.container) || domUtils.ownerDocument(this).body;
6768
}
6869
};

src/utils/domUtils.js

+17-3
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,24 @@
1+
import React from 'react';
2+
3+
/**
4+
* Get elements owner document
5+
*
6+
* @param {ReactComponent|HTMLElement} componentOrElement
7+
* @returns {HTMLElement}
8+
*/
9+
function ownerDocument(componentOrElement) {
10+
let elem = React.findDOMNode(componentOrElement);
11+
return (elem && elem.ownerDocument) || document;
12+
}
13+
114
/**
215
* Shortcut to compute element style
316
*
417
* @param {HTMLElement} elem
518
* @returns {CssStyle}
619
*/
720
function getComputedStyles(elem) {
8-
return elem.ownerDocument.defaultView.getComputedStyle(elem, null);
21+
return ownerDocument(elem).defaultView.getComputedStyle(elem, null);
922
}
1023

1124
/**
@@ -21,7 +34,7 @@ function getOffset(DOMNode) {
2134
return window.jQuery(DOMNode).offset();
2235
}
2336

24-
let docElem = DOMNode.ownerDocument.documentElement;
37+
let docElem = ownerDocument(DOMNode).documentElement;
2538
let box = { top: 0, left: 0 };
2639

2740
// If we don't have gBCR, just use 0,0 rather than error
@@ -89,7 +102,7 @@ function getPosition(elem, offsetParent) {
89102
* @returns {HTMLElement}
90103
*/
91104
function offsetParentFunc(elem) {
92-
let docElem = elem.ownerDocument.documentElement;
105+
let docElem = ownerDocument(elem).documentElement;
93106
let offsetParent = elem.offsetParent || docElem;
94107

95108
while ( offsetParent && ( offsetParent.nodeName !== 'HTML' &&
@@ -101,6 +114,7 @@ function offsetParentFunc(elem) {
101114
}
102115

103116
export default {
117+
ownerDocument: ownerDocument,
104118
getComputedStyles: getComputedStyles,
105119
getOffset: getOffset,
106120
getPosition: getPosition,

test/OverlayMixinSpec.js

+20
Original file line numberDiff line numberDiff line change
@@ -60,4 +60,24 @@ describe('OverlayMixin', function () {
6060

6161
assert.equal(instance.refs.overlay.getOverlayDOMNode(), null);
6262
});
63+
64+
it('Should render only an overlay', function() {
65+
let OnlyOverlay = React.createClass({
66+
mixins: [OverlayMixin],
67+
68+
render: function() {
69+
return null;
70+
},
71+
72+
renderOverlay: function() {
73+
return this.props.overlay;
74+
}
75+
});
76+
77+
let overlayInstance = ReactTestUtils.renderIntoDocument(
78+
<OnlyOverlay overlay={<div id="test1" />} />
79+
);
80+
81+
assert.equal(overlayInstance.getOverlayDOMNode().nodeName, 'DIV');
82+
});
6383
});

0 commit comments

Comments
 (0)