diff --git a/packages/react-dom/src/__tests__/ReactDOMServerIntegrationElements-test.js b/packages/react-dom/src/__tests__/ReactDOMServerIntegrationElements-test.js index 859647b39c1ff..c41170aa54e91 100644 --- a/packages/react-dom/src/__tests__/ReactDOMServerIntegrationElements-test.js +++ b/packages/react-dom/src/__tests__/ReactDOMServerIntegrationElements-test.js @@ -924,33 +924,37 @@ describe('ReactDOMServerIntegration', () => { ); }); - describe('components that throw errors', function() { - itThrowsWhenRendering( - 'a function returning undefined', - async render => { - const UndefinedComponent = () => undefined; - await render(, 1); - }, - 'UndefinedComponent(...): Nothing was returned from render. ' + - 'This usually means a return statement is missing. Or, to ' + - 'render nothing, return null.', - ); + describe('components that render nullish', function() { + itRenders('a function returning null', async render => { + const NullComponent = () => null; + await render(); + }); - itThrowsWhenRendering( - 'a class returning undefined', - async render => { - class UndefinedComponent extends React.Component { - render() { - return undefined; - } + itRenders('a class returning null', async render => { + class NullComponent extends React.Component { + render() { + return null; } - await render(, 1); - }, - 'UndefinedComponent(...): Nothing was returned from render. ' + - 'This usually means a return statement is missing. Or, to ' + - 'render nothing, return null.', - ); + } + await render(); + }); + + itRenders('a function returning undefined', async render => { + const UndefinedComponent = () => undefined; + await render(); + }); + + itRenders('a class returning undefined', async render => { + class UndefinedComponent extends React.Component { + render() { + return undefined; + } + } + await render(); + }); + }); + describe('components that throw errors', function() { itThrowsWhenRendering( 'a function returning an object', async render => { diff --git a/packages/react-dom/src/__tests__/ReactEmptyComponent-test.js b/packages/react-dom/src/__tests__/ReactEmptyComponent-test.js index 0c0b1d1a92739..12146e9b7edc5 100644 --- a/packages/react-dom/src/__tests__/ReactEmptyComponent-test.js +++ b/packages/react-dom/src/__tests__/ReactEmptyComponent-test.js @@ -45,145 +45,117 @@ describe('ReactEmptyComponent', () => { }; }); - it('should not produce child DOM nodes for null and false', () => { - class Component1 extends React.Component { - render() { - return null; - } - } - - class Component2 extends React.Component { - render() { - return false; + describe.each([null, undefined])('when %s', nullORUndefined => { + it('should not throw when rendering', () => { + class Component extends React.Component { + render() { + return nullORUndefined; + } } - } - - const container1 = document.createElement('div'); - ReactDOM.render(, container1); - expect(container1.children.length).toBe(0); - - const container2 = document.createElement('div'); - ReactDOM.render(, container2); - expect(container2.children.length).toBe(0); - }); - - it('should still throw when rendering to undefined', () => { - class Component extends React.Component { - render() {} - } - - expect(function() { - ReactTestUtils.renderIntoDocument(); - }).toThrowError( - 'Component(...): Nothing was returned from render. This usually means a return statement is missing. ' + - 'Or, to render nothing, return null.', - ); - }); - it('should be able to switch between rendering null and a normal tag', () => { - const instance1 = ( - - ); - const instance2 = ( - - ); + expect(function() { + ReactTestUtils.renderIntoDocument(); + }).not.toThrowError(); + }); - ReactTestUtils.renderIntoDocument(instance1); - ReactTestUtils.renderIntoDocument(instance2); + it('should not produce child DOM nodes for nullish and false', () => { + class Component1 extends React.Component { + render() { + return nullORUndefined; + } + } - expect(log).toHaveBeenCalledTimes(4); - expect(log).toHaveBeenNthCalledWith(1, null); - expect(log).toHaveBeenNthCalledWith( - 2, - expect.objectContaining({tagName: 'DIV'}), - ); - expect(log).toHaveBeenNthCalledWith( - 3, - expect.objectContaining({tagName: 'DIV'}), - ); - expect(log).toHaveBeenNthCalledWith(4, null); - }); + class Component2 extends React.Component { + render() { + return false; + } + } - it('should be able to switch in a list of children', () => { - const instance1 = ( - - ); + const container1 = document.createElement('div'); + ReactDOM.render(, container1); + expect(container1.children.length).toBe(0); - ReactTestUtils.renderIntoDocument( -
- {instance1} - {instance1} - {instance1} -
, - ); + const container2 = document.createElement('div'); + ReactDOM.render(, container2); + expect(container2.children.length).toBe(0); + }); - expect(log).toHaveBeenCalledTimes(6); - expect(log).toHaveBeenNthCalledWith(1, null); - expect(log).toHaveBeenNthCalledWith(2, null); - expect(log).toHaveBeenNthCalledWith(3, null); - expect(log).toHaveBeenNthCalledWith( - 4, - expect.objectContaining({tagName: 'DIV'}), - ); - expect(log).toHaveBeenNthCalledWith( - 5, - expect.objectContaining({tagName: 'DIV'}), - ); - expect(log).toHaveBeenNthCalledWith( - 6, - expect.objectContaining({tagName: 'DIV'}), - ); - }); - - it('should distinguish between a script placeholder and an actual script tag', () => { - const instance1 = ( - - ); - const instance2 = ( - - ); + it('should be able to switch between rendering nullish and a normal tag', () => { + const instance1 = ( + + ); + const instance2 = ( + + ); - expect(function() { ReactTestUtils.renderIntoDocument(instance1); - }).not.toThrow(); - expect(function() { ReactTestUtils.renderIntoDocument(instance2); - }).not.toThrow(); - expect(log).toHaveBeenCalledTimes(4); - expect(log).toHaveBeenNthCalledWith(1, null); - expect(log).toHaveBeenNthCalledWith( - 2, - expect.objectContaining({tagName: 'SCRIPT'}), - ); - expect(log).toHaveBeenNthCalledWith( - 3, - expect.objectContaining({tagName: 'SCRIPT'}), - ); - expect(log).toHaveBeenNthCalledWith(4, null); - }); + expect(log).toHaveBeenCalledTimes(4); + expect(log).toHaveBeenNthCalledWith(1, null); + expect(log).toHaveBeenNthCalledWith( + 2, + expect.objectContaining({tagName: 'DIV'}), + ); + expect(log).toHaveBeenNthCalledWith( + 3, + expect.objectContaining({tagName: 'DIV'}), + ); + expect(log).toHaveBeenNthCalledWith(4, null); + }); - it( - 'should have findDOMNode return null when multiple layers of composite ' + - 'components render to the same null placeholder', - () => { - class GrandChild extends React.Component { - render() { - return null; - } - } + it('should be able to switch in a list of children', () => { + const instance1 = ( + + ); - class Child extends React.Component { - render() { - return ; - } - } + ReactTestUtils.renderIntoDocument( +
+ {instance1} + {instance1} + {instance1} +
, + ); + + expect(log).toHaveBeenCalledTimes(6); + expect(log).toHaveBeenNthCalledWith(1, null); + expect(log).toHaveBeenNthCalledWith(2, null); + expect(log).toHaveBeenNthCalledWith(3, null); + expect(log).toHaveBeenNthCalledWith( + 4, + expect.objectContaining({tagName: 'DIV'}), + ); + expect(log).toHaveBeenNthCalledWith( + 5, + expect.objectContaining({tagName: 'DIV'}), + ); + expect(log).toHaveBeenNthCalledWith( + 6, + expect.objectContaining({tagName: 'DIV'}), + ); + }); + it('should distinguish between a script placeholder and an actual script tag', () => { const instance1 = ( - + ); const instance2 = ( - + ); expect(function() { @@ -194,146 +166,192 @@ describe('ReactEmptyComponent', () => { }).not.toThrow(); expect(log).toHaveBeenCalledTimes(4); + expect(log).toHaveBeenNthCalledWith(1, null); expect(log).toHaveBeenNthCalledWith( - 1, - expect.objectContaining({tagName: 'DIV'}), + 2, + expect.objectContaining({tagName: 'SCRIPT'}), ); - expect(log).toHaveBeenNthCalledWith(2, null); - expect(log).toHaveBeenNthCalledWith(3, null); expect(log).toHaveBeenNthCalledWith( - 4, - expect.objectContaining({tagName: 'DIV'}), + 3, + expect.objectContaining({tagName: 'SCRIPT'}), ); - }, - ); + expect(log).toHaveBeenNthCalledWith(4, null); + }); + + it( + 'should have findDOMNode return null when multiple layers of composite ' + + 'components render to the same nullish placeholder', + () => { + class GrandChild extends React.Component { + render() { + return nullORUndefined; + } + } - it('works when switching components', () => { - let assertions = 0; + class Child extends React.Component { + render() { + return ; + } + } - class Inner extends React.Component { - render() { - return ; - } + const instance1 = ( + + ); + const instance2 = ( + + ); - componentDidMount() { - // Make sure the DOM node resolves properly even if we're replacing a - // `null` component - expect(ReactDOM.findDOMNode(this)).not.toBe(null); - assertions++; - } + expect(function() { + ReactTestUtils.renderIntoDocument(instance1); + }).not.toThrow(); + expect(function() { + ReactTestUtils.renderIntoDocument(instance2); + }).not.toThrow(); + + expect(log).toHaveBeenCalledTimes(4); + expect(log).toHaveBeenNthCalledWith( + 1, + expect.objectContaining({tagName: 'DIV'}), + ); + expect(log).toHaveBeenNthCalledWith(2, null); + expect(log).toHaveBeenNthCalledWith(3, null); + expect(log).toHaveBeenNthCalledWith( + 4, + expect.objectContaining({tagName: 'DIV'}), + ); + }, + ); - componentWillUnmount() { - // Even though we're getting replaced by `null`, we haven't been - // replaced yet! - expect(ReactDOM.findDOMNode(this)).not.toBe(null); - assertions++; + it('works when switching components', () => { + let assertions = 0; + + class Inner extends React.Component { + render() { + return ; + } + + componentDidMount() { + // Make sure the DOM node resolves properly even if we're replacing a + // `null` component + expect(ReactDOM.findDOMNode(this)).not.toBe(null); + assertions++; + } + + componentWillUnmount() { + // Even though we're getting replaced by `null`, we haven't been + // replaced yet! + expect(ReactDOM.findDOMNode(this)).not.toBe(null); + assertions++; + } } - } - class Wrapper extends React.Component { - render() { - return this.props.showInner ? : null; + class Wrapper extends React.Component { + render() { + return this.props.showInner ? : nullORUndefined; + } } - } - const el = document.createElement('div'); - let component; + const el = document.createElement('div'); + let component; - // Render the component... - component = ReactDOM.render(, el); - expect(ReactDOM.findDOMNode(component)).not.toBe(null); + // Render the component... + component = ReactDOM.render(, el); + expect(ReactDOM.findDOMNode(component)).not.toBe(null); - // Switch to null... - component = ReactDOM.render(, el); - expect(ReactDOM.findDOMNode(component)).toBe(null); + // Switch to null... + component = ReactDOM.render(, el); + expect(ReactDOM.findDOMNode(component)).toBe(null); - // ...then switch back. - component = ReactDOM.render(, el); - expect(ReactDOM.findDOMNode(component)).not.toBe(null); + // ...then switch back. + component = ReactDOM.render(, el); + expect(ReactDOM.findDOMNode(component)).not.toBe(null); - expect(assertions).toBe(3); - }); + expect(assertions).toBe(3); + }); - it('can render null at the top level', () => { - const div = document.createElement('div'); - ReactDOM.render(null, div); - expect(div.innerHTML).toBe(''); - }); + it('can render nullish at the top level', () => { + const div = document.createElement('div'); + ReactDOM.render(nullORUndefined, div); + expect(div.innerHTML).toBe(''); + }); - it('does not break when updating during mount', () => { - class Child extends React.Component { - componentDidMount() { - if (this.props.onMount) { - this.props.onMount(); + it('does not break when updating during mount', () => { + class Child extends React.Component { + componentDidMount() { + if (this.props.onMount) { + this.props.onMount(); + } } - } - render() { - if (!this.props.visible) { - return null; - } + render() { + if (!this.props.visible) { + return nullORUndefined; + } - return
hello world
; + return
hello world
; + } } - } - class Parent extends React.Component { - update = () => { - this.forceUpdate(); - }; + class Parent extends React.Component { + update = () => { + this.forceUpdate(); + }; - render() { - return ( -
- - - -
- ); + render() { + return ( +
+ + + +
+ ); + } } - } - expect(function() { - ReactTestUtils.renderIntoDocument(); - }).not.toThrow(); - }); + expect(function() { + ReactTestUtils.renderIntoDocument(); + }).not.toThrow(); + }); - it('preserves the dom node during updates', () => { - class Empty extends React.Component { - render() { - return null; + it('preserves the dom node during updates', () => { + class Empty extends React.Component { + render() { + return nullORUndefined; + } } - } - const container = document.createElement('div'); + const container = document.createElement('div'); - ReactDOM.render(, container); - const noscript1 = container.firstChild; - expect(noscript1).toBe(null); + ReactDOM.render(, container); + const noscript1 = container.firstChild; + expect(noscript1).toBe(null); - // This update shouldn't create a DOM node - ReactDOM.render(, container); - const noscript2 = container.firstChild; - expect(noscript2).toBe(null); - }); + // This update shouldn't create a DOM node + ReactDOM.render(, container); + const noscript2 = container.firstChild; + expect(noscript2).toBe(null); + }); - it('should warn about React.forwardRef that returns undefined', () => { - const Empty = () => {}; - const EmptyForwardRef = React.forwardRef(Empty); + it('should not warn about React.forwardRef that returns nullish', () => { + const Empty = () => { + return nullORUndefined; + }; + const EmptyForwardRef = React.forwardRef(Empty); - expect(() => { - ReactTestUtils.renderIntoDocument(); - }).toThrowError( - 'ForwardRef(Empty)(...): Nothing was returned from render.', - ); - }); + expect(() => { + ReactTestUtils.renderIntoDocument(); + }).not.toThrowError(); + }); - it('should warn about React.memo that returns undefined', () => { - const Empty = () => {}; - const EmptyMemo = React.memo(Empty); + it('should not warn about React.memo that returns nullish', () => { + const Empty = () => { + return nullORUndefined; + }; + const EmptyMemo = React.memo(Empty); - expect(() => { - ReactTestUtils.renderIntoDocument(); - }).toThrowError('Empty(...): Nothing was returned from render.'); + expect(() => { + ReactTestUtils.renderIntoDocument(); + }).not.toThrowError(); + }); }); }); diff --git a/packages/react-dom/src/__tests__/ReactFunctionComponent-test.js b/packages/react-dom/src/__tests__/ReactFunctionComponent-test.js index 0aa9aa69285fa..9228316f884ee 100644 --- a/packages/react-dom/src/__tests__/ReactFunctionComponent-test.js +++ b/packages/react-dom/src/__tests__/ReactFunctionComponent-test.js @@ -136,7 +136,7 @@ describe('ReactFunctionComponent', () => { ); }); - it('should throw when stateless component returns undefined', () => { + it('should not throw when stateless component returns undefined', () => { function NotAComponent() {} expect(function() { ReactTestUtils.renderIntoDocument( @@ -144,10 +144,7 @@ describe('ReactFunctionComponent', () => { , ); - }).toThrowError( - 'NotAComponent(...): Nothing was returned from render. ' + - 'This usually means a return statement is missing. Or, to render nothing, return null.', - ); + }).not.toThrowError(); }); it('should throw on string refs in pure functions', () => { diff --git a/packages/react-dom/src/__tests__/utils/ReactDOMServerIntegrationTestUtils.js b/packages/react-dom/src/__tests__/utils/ReactDOMServerIntegrationTestUtils.js index 81be4cb30d72d..8714059bb3861 100644 --- a/packages/react-dom/src/__tests__/utils/ReactDOMServerIntegrationTestUtils.js +++ b/packages/react-dom/src/__tests__/utils/ReactDOMServerIntegrationTestUtils.js @@ -242,7 +242,7 @@ module.exports = function(initModules) { await asyncReactDOMRender(element, cleanContainer, true); // This gives us the expected text content. const cleanTextContent = - cleanContainer.lastChild && cleanContainer.lastChild.textContent; + (cleanContainer.lastChild && cleanContainer.lastChild.textContent) || ''; // The only guarantee is that text content has been patched up if needed. expect(hydratedTextContent).toBe(cleanTextContent); diff --git a/packages/react-dom/src/server/ReactPartialRenderer.js b/packages/react-dom/src/server/ReactPartialRenderer.js index 752af3b2c7377..c19b8b78576b2 100644 --- a/packages/react-dom/src/server/ReactPartialRenderer.js +++ b/packages/react-dom/src/server/ReactPartialRenderer.js @@ -391,18 +391,6 @@ function createOpenTagMarkup( return ret; } -function validateRenderResult(child, type) { - if (child === undefined) { - invariant( - false, - '%s(...): Nothing was returned from render. This usually means a ' + - 'return statement is missing. Or, to render nothing, ' + - 'return null.', - getComponentNameFromType(type) || 'Component', - ); - } -} - function resolve( child: mixed, context: Object, @@ -631,7 +619,6 @@ function resolve( inst.render == null ) { child = inst; - validateRenderResult(child, Component); return; } } @@ -720,15 +707,6 @@ function resolve( } child = inst.render(); - if (__DEV__) { - if (child === undefined && inst.render._isMockFunction) { - // This is probably bad practice. Consider warning here and - // deprecating this convenience. - child = null; - } - } - validateRenderResult(child, Component); - let childContext; if (disableLegacyContext) { if (__DEV__) { diff --git a/packages/react-reconciler/src/ReactChildFiber.new.js b/packages/react-reconciler/src/ReactChildFiber.new.js index 024a903dbd915..dbe298337ee7b 100644 --- a/packages/react-reconciler/src/ReactChildFiber.new.js +++ b/packages/react-reconciler/src/ReactChildFiber.new.js @@ -1302,36 +1302,6 @@ function ChildReconciler(shouldTrackSideEffects) { warnOnFunctionType(returnFiber); } } - if (typeof newChild === 'undefined' && !isUnkeyedTopLevelFragment) { - // If the new child is undefined, and the return fiber is a composite - // component, throw an error. If Fiber return types are disabled, - // we already threw above. - switch (returnFiber.tag) { - case ClassComponent: { - if (__DEV__) { - const instance = returnFiber.stateNode; - if (instance.render._isMockFunction) { - // We allow auto-mocks to proceed as if they're returning null. - break; - } - } - } - // Intentionally fall through to the next case, which handles both - // functions and classes - // eslint-disable-next-lined no-fallthrough - case FunctionComponent: - case ForwardRef: - case SimpleMemoComponent: { - invariant( - false, - '%s(...): Nothing was returned from render. This usually means a ' + - 'return statement is missing. Or, to render nothing, ' + - 'return null.', - getComponentNameFromFiber(returnFiber) || 'Component', - ); - } - } - } // Remaining cases are all treated as empty. return deleteRemainingChildren(returnFiber, currentFirstChild); diff --git a/packages/react-reconciler/src/ReactChildFiber.old.js b/packages/react-reconciler/src/ReactChildFiber.old.js index 80b4fc35512d0..b36f14d318e06 100644 --- a/packages/react-reconciler/src/ReactChildFiber.old.js +++ b/packages/react-reconciler/src/ReactChildFiber.old.js @@ -1302,36 +1302,6 @@ function ChildReconciler(shouldTrackSideEffects) { warnOnFunctionType(returnFiber); } } - if (typeof newChild === 'undefined' && !isUnkeyedTopLevelFragment) { - // If the new child is undefined, and the return fiber is a composite - // component, throw an error. If Fiber return types are disabled, - // we already threw above. - switch (returnFiber.tag) { - case ClassComponent: { - if (__DEV__) { - const instance = returnFiber.stateNode; - if (instance.render._isMockFunction) { - // We allow auto-mocks to proceed as if they're returning null. - break; - } - } - } - // Intentionally fall through to the next case, which handles both - // functions and classes - // eslint-disable-next-lined no-fallthrough - case FunctionComponent: - case ForwardRef: - case SimpleMemoComponent: { - invariant( - false, - '%s(...): Nothing was returned from render. This usually means a ' + - 'return statement is missing. Or, to render nothing, ' + - 'return null.', - getComponentNameFromFiber(returnFiber) || 'Component', - ); - } - } - } // Remaining cases are all treated as empty. return deleteRemainingChildren(returnFiber, currentFirstChild); diff --git a/packages/react-server/src/ReactFizzServer.js b/packages/react-server/src/ReactFizzServer.js index 9e2874eea4493..95074071da1d6 100644 --- a/packages/react-server/src/ReactFizzServer.js +++ b/packages/react-server/src/ReactFizzServer.js @@ -554,16 +554,6 @@ function shouldConstruct(Component) { return Component.prototype && Component.prototype.isReactComponent; } -function invalidRenderResult(type: any): void { - invariant( - false, - '%s(...): Nothing was returned from render. This usually means a ' + - 'return statement is missing. Or, to render nothing, ' + - 'return null.', - getComponentNameFromType(type) || 'Component', - ); -} - function renderWithHooks( request: Request, task: Task, @@ -574,11 +564,7 @@ function renderWithHooks( const componentIdentity = {}; prepareToUseHooks(componentIdentity); const result = Component(props, secondArg); - const children = finishHooks(Component, props, result, secondArg); - if (children === undefined) { - invalidRenderResult(Component); - } - return children; + return finishHooks(Component, props, result, secondArg); } function finishClassComponent( @@ -589,13 +575,6 @@ function finishClassComponent( props: any, ): ReactNodeList { const nextChildren = instance.render(); - if (nextChildren === undefined) { - if (__DEV__ && instance.render._isMockFunction) { - // We allow auto-mocks to proceed as if they're returning null. - } else { - invalidRenderResult(Component); - } - } if (__DEV__) { if (instance.props !== props) {