diff --git a/src/browser/ui/ReactMount.js b/src/browser/ui/ReactMount.js index 581a904427d..38b1e5e7752 100644 --- a/src/browser/ui/ReactMount.js +++ b/src/browser/ui/ReactMount.js @@ -370,8 +370,21 @@ var ReactMount = { var reactRootElement = getReactRootElementInContainer(container); var containerHasReactMarkup = reactRootElement && ReactMount.isRenderedByReact(reactRootElement); + var containerHasNonRootReactChild = + ReactMount.hasNonRootReactChild(container); - var shouldReuseMarkup = containerHasReactMarkup && !prevComponent; + if (__DEV__) { + warning( + !containerHasNonRootReactChild, + 'renderComponent(...): Replacing React-rendered children with a new ' + + 'root component.' + ); + } + + var shouldReuseMarkup = + containerHasReactMarkup && + !prevComponent && + !containerHasNonRootReactChild; var component = ReactMount._renderNewRootComponent( nextDescriptor, @@ -459,6 +472,28 @@ var ReactMount = { var reactRootID = getReactRootID(container); var component = instancesByReactRootID[reactRootID]; if (!component) { + // Check if the node being unmounted was rendered by React, but isn't a + // root node. + var containerHasNonRootReactChild = ReactMount.hasNonRootReactChild( + container); + + // Check if the container itself is a React root node. + var containerID = ReactMount.getID(container); + var containerRootID = ReactInstanceHandles.getReactRootIDFromNodeID( + containerID); + var isContainerReactRoot = + containerID && containerRootID && containerID === containerRootID; + + if (__DEV__) { + warning( + !containerHasNonRootReactChild, + 'unmountComponentAtNode(): The node you\'re attempting to unmount ' + + 'is not a valid React root node, and thus cannot be unmounted.%s', + (isContainerReactRoot ? ' You may have passed in a React root ' + + 'node as argument, rather than its container.' : '') + ); + } + return false; } ReactMount.unmountComponentFromNode(component, container); @@ -560,6 +595,22 @@ var ReactMount = { return id ? id.charAt(0) === SEPARATOR : false; }, + /** + * True if the supplied DOM node has a direct React-rendered child that is + * not a React root element. Useful for warning in renderComponent`, + * `unmountComponentAtNode`, etc. + * + * @param {?DOMElement} node The candidate DOM node. + * @return {boolean} True if the DOM element contains a direct child that was + * rendered by React but is not a root element. + * @internal + */ + hasNonRootReactChild: function(node) { + var reactRootID = getReactRootID(node); + return reactRootID ? reactRootID !== + ReactInstanceHandles.getReactRootIDFromNodeID(reactRootID) : false; + }, + /** * Traverses up the ancestors of the supplied node to find a node that is a * DOM representation of a React component. diff --git a/src/browser/ui/__tests__/ReactMount-test.js b/src/browser/ui/__tests__/ReactMount-test.js index 4dd3448e672..92493cfc106 100644 --- a/src/browser/ui/__tests__/ReactMount-test.js +++ b/src/browser/ui/__tests__/ReactMount-test.js @@ -114,4 +114,24 @@ describe('ReactMount', function() { expect(instance1 === instance2).toBe(true); }); + + it('should warn if render removes React-rendered children', function() { + var container = document.createElement('container'); + var Component = React.createClass({ + render: function() { + return