diff --git a/packages/react-dom/src/__tests__/ReactCompositeComponent-test.js b/packages/react-dom/src/__tests__/ReactCompositeComponent-test.js index 2e56a911a0c38..ba3118bc651b0 100644 --- a/packages/react-dom/src/__tests__/ReactCompositeComponent-test.js +++ b/packages/react-dom/src/__tests__/ReactCompositeComponent-test.js @@ -261,29 +261,17 @@ describe('ReactCompositeComponent', () => { await act(() => { root.render(); }); - if (gate(flags => flags.enableRefAsProp)) { - expect(instance1.props).toEqual({prop: 'testKey', ref: refFn1}); - } else { - expect(instance1.props).toEqual({prop: 'testKey'}); - } + expect(instance1.props).toEqual({prop: 'testKey'}); await act(() => { root.render(); }); - if (gate(flags => flags.enableRefAsProp)) { - expect(instance2.props).toEqual({prop: 'testKey', ref: refFn2}); - } else { - expect(instance2.props).toEqual({prop: 'testKey'}); - } + expect(instance2.props).toEqual({prop: 'testKey'}); await act(() => { root.render(); }); - if (gate(flags => flags.enableRefAsProp)) { - expect(instance3.props).toEqual({prop: null, ref: refFn3}); - } else { - expect(instance3.props).toEqual({prop: null}); - } + expect(instance3.props).toEqual({prop: null}); }); it('should not mutate passed-in props object', async () => { diff --git a/packages/react-reconciler/src/ReactFiberBeginWork.js b/packages/react-reconciler/src/ReactFiberBeginWork.js index 7a12ec9250677..1aba7480adcac 100644 --- a/packages/react-reconciler/src/ReactFiberBeginWork.js +++ b/packages/react-reconciler/src/ReactFiberBeginWork.js @@ -245,6 +245,7 @@ import { mountClassInstance, resumeMountClassInstance, updateClassInstance, + resolveClassComponentProps, } from './ReactFiberClassComponent'; import {resolveDefaultProps} from './ReactFiberLazyComponent'; import { @@ -1763,9 +1764,9 @@ function mountLazyComponent( // Store the unwrapped component in the type. workInProgress.type = Component; - const resolvedProps = resolveDefaultProps(Component, props); if (typeof Component === 'function') { if (isFunctionClassComponent(Component)) { + const resolvedProps = resolveClassComponentProps(Component, props, false); workInProgress.tag = ClassComponent; if (__DEV__) { workInProgress.type = Component = @@ -1779,6 +1780,7 @@ function mountLazyComponent( renderLanes, ); } else { + const resolvedProps = resolveDefaultProps(Component, props); workInProgress.tag = FunctionComponent; if (__DEV__) { validateFunctionComponentInDev(workInProgress, Component); @@ -1796,6 +1798,7 @@ function mountLazyComponent( } else if (Component !== undefined && Component !== null) { const $$typeof = Component.$$typeof; if ($$typeof === REACT_FORWARD_REF_TYPE) { + const resolvedProps = resolveDefaultProps(Component, props); workInProgress.tag = ForwardRef; if (__DEV__) { workInProgress.type = Component = @@ -1809,6 +1812,7 @@ function mountLazyComponent( renderLanes, ); } else if ($$typeof === REACT_MEMO_TYPE) { + const resolvedProps = resolveDefaultProps(Component, props); workInProgress.tag = MemoComponent; return updateMemoComponent( null, @@ -3939,10 +3943,11 @@ function beginWork( case ClassComponent: { const Component = workInProgress.type; const unresolvedProps = workInProgress.pendingProps; - const resolvedProps = - workInProgress.elementType === Component - ? unresolvedProps - : resolveDefaultProps(Component, unresolvedProps); + const resolvedProps = resolveClassComponentProps( + Component, + unresolvedProps, + workInProgress.elementType === Component, + ); return updateClassComponent( current, workInProgress, @@ -4025,10 +4030,11 @@ function beginWork( } const Component = workInProgress.type; const unresolvedProps = workInProgress.pendingProps; - const resolvedProps = - workInProgress.elementType === Component - ? unresolvedProps - : resolveDefaultProps(Component, unresolvedProps); + const resolvedProps = resolveClassComponentProps( + Component, + unresolvedProps, + workInProgress.elementType === Component, + ); return mountIncompleteClassComponent( current, workInProgress, @@ -4043,10 +4049,11 @@ function beginWork( } const Component = workInProgress.type; const unresolvedProps = workInProgress.pendingProps; - const resolvedProps = - workInProgress.elementType === Component - ? unresolvedProps - : resolveDefaultProps(Component, unresolvedProps); + const resolvedProps = resolveClassComponentProps( + Component, + unresolvedProps, + workInProgress.elementType === Component, + ); return mountIncompleteFunctionComponent( current, workInProgress, diff --git a/packages/react-reconciler/src/ReactFiberClassComponent.js b/packages/react-reconciler/src/ReactFiberClassComponent.js index 47d1c3cfee476..697be25eee678 100644 --- a/packages/react-reconciler/src/ReactFiberClassComponent.js +++ b/packages/react-reconciler/src/ReactFiberClassComponent.js @@ -23,6 +23,7 @@ import { enableDebugTracing, enableSchedulingProfiler, enableLazyContextPropagation, + enableRefAsProp, } from 'shared/ReactFeatureFlags'; import ReactStrictModeWarnings from './ReactStrictModeWarnings'; import {isMounted} from './ReactFiberTreeReflection'; @@ -34,7 +35,6 @@ import assign from 'shared/assign'; import isArray from 'shared/isArray'; import {REACT_CONTEXT_TYPE, REACT_CONSUMER_TYPE} from 'shared/ReactSymbols'; -import {resolveDefaultProps} from './ReactFiberLazyComponent'; import { DebugTracingMode, NoMode, @@ -904,7 +904,12 @@ function resumeMountClassInstance( ): boolean { const instance = workInProgress.stateNode; - const oldProps = workInProgress.memoizedProps; + const unresolvedOldProps = workInProgress.memoizedProps; + const oldProps = resolveClassComponentProps( + ctor, + unresolvedOldProps, + workInProgress.type === workInProgress.elementType, + ); instance.props = oldProps; const oldContext = instance.context; @@ -926,6 +931,13 @@ function resumeMountClassInstance( typeof getDerivedStateFromProps === 'function' || typeof instance.getSnapshotBeforeUpdate === 'function'; + // When comparing whether props changed, we should compare using the + // unresolved props object that is stored on the fiber, rather than the + // one that gets assigned to the instance, because that object may have been + // cloned to resolve default props and/or remove `ref`. + const unresolvedNewProps = workInProgress.pendingProps; + const didReceiveNewProps = unresolvedNewProps !== unresolvedOldProps; + // Note: During these life-cycles, instance.props/instance.state are what // ever the previously attempted to render - not the "current". However, // during componentDidUpdate we pass the "current" props. @@ -937,7 +949,7 @@ function resumeMountClassInstance( (typeof instance.UNSAFE_componentWillReceiveProps === 'function' || typeof instance.componentWillReceiveProps === 'function') ) { - if (oldProps !== newProps || oldContext !== nextContext) { + if (didReceiveNewProps || oldContext !== nextContext) { callComponentWillReceiveProps( workInProgress, instance, @@ -955,7 +967,7 @@ function resumeMountClassInstance( suspendIfUpdateReadFromEntangledAsyncAction(); newState = workInProgress.memoizedState; if ( - oldProps === newProps && + !didReceiveNewProps && oldState === newState && !hasContextChanged() && !checkHasForceUpdateAfterProcessing() @@ -1052,10 +1064,11 @@ function updateClassInstance( cloneUpdateQueue(current, workInProgress); const unresolvedOldProps = workInProgress.memoizedProps; - const oldProps = - workInProgress.type === workInProgress.elementType - ? unresolvedOldProps - : resolveDefaultProps(workInProgress.type, unresolvedOldProps); + const oldProps = resolveClassComponentProps( + ctor, + unresolvedOldProps, + workInProgress.type === workInProgress.elementType, + ); instance.props = oldProps; const unresolvedNewProps = workInProgress.pendingProps; @@ -1225,6 +1238,42 @@ function updateClassInstance( return shouldUpdate; } +export function resolveClassComponentProps( + Component: any, + baseProps: Object, + // Only resolve default props if this is a lazy component. Otherwise, they + // would have already been resolved by the JSX runtime. + // TODO: We're going to remove default prop resolution from the JSX runtime + // and keep it only for class components. As part of that change, we should + // remove this extra check. + alreadyResolvedDefaultProps: boolean, +): Object { + let newProps = baseProps; + + // Resolve default props. Taken from old JSX runtime, where this used to live. + const defaultProps = Component.defaultProps; + if (defaultProps && !alreadyResolvedDefaultProps) { + newProps = assign({}, newProps, baseProps); + for (const propName in defaultProps) { + if (newProps[propName] === undefined) { + newProps[propName] = defaultProps[propName]; + } + } + } + + if (enableRefAsProp) { + // Remove ref from the props object, if it exists. + if ('ref' in newProps) { + if (newProps === baseProps) { + newProps = assign({}, newProps); + } + delete newProps.ref; + } + } + + return newProps; +} + export { constructClassInstance, mountClassInstance, diff --git a/packages/react-reconciler/src/ReactFiberCommitWork.js b/packages/react-reconciler/src/ReactFiberCommitWork.js index ba11d5b6149b2..213ff26d82e2c 100644 --- a/packages/react-reconciler/src/ReactFiberCommitWork.js +++ b/packages/react-reconciler/src/ReactFiberCommitWork.js @@ -104,7 +104,7 @@ import { setCurrentFiber as setCurrentDebugFiberInDEV, getCurrentFiber as getCurrentDebugFiberInDEV, } from './ReactCurrentFiber'; -import {resolveDefaultProps} from './ReactFiberLazyComponent'; +import {resolveClassComponentProps} from './ReactFiberClassComponent'; import { isCurrentUpdateNested, getCommitTime, @@ -244,7 +244,11 @@ function shouldProfile(current: Fiber): boolean { } function callComponentWillUnmountWithTimer(current: Fiber, instance: any) { - instance.props = current.memoizedProps; + instance.props = resolveClassComponentProps( + current.type, + current.memoizedProps, + current.elementType === current.type, + ); instance.state = current.memoizedState; if (shouldProfile(current)) { try { @@ -471,7 +475,8 @@ function commitBeforeMutationEffectsOnFiber(finishedWork: Fiber) { // TODO: revisit this when we implement resuming. if (__DEV__) { if ( - finishedWork.type === finishedWork.elementType && + !finishedWork.type.defaultProps && + !('ref' in finishedWork.memoizedProps) && !didWarnAboutReassigningProps ) { if (instance.props !== finishedWork.memoizedProps) { @@ -497,9 +502,11 @@ function commitBeforeMutationEffectsOnFiber(finishedWork: Fiber) { } } const snapshot = instance.getSnapshotBeforeUpdate( - finishedWork.elementType === finishedWork.type - ? prevProps - : resolveDefaultProps(finishedWork.type, prevProps), + resolveClassComponentProps( + finishedWork.type, + prevProps, + finishedWork.elementType === finishedWork.type, + ), prevState, ); if (__DEV__) { @@ -807,7 +814,8 @@ function commitClassLayoutLifecycles( // TODO: revisit this when we implement resuming. if (__DEV__) { if ( - finishedWork.type === finishedWork.elementType && + !finishedWork.type.defaultProps && + !('ref' in finishedWork.memoizedProps) && !didWarnAboutReassigningProps ) { if (instance.props !== finishedWork.memoizedProps) { @@ -848,17 +856,19 @@ function commitClassLayoutLifecycles( } } } else { - const prevProps = - finishedWork.elementType === finishedWork.type - ? current.memoizedProps - : resolveDefaultProps(finishedWork.type, current.memoizedProps); + const prevProps = resolveClassComponentProps( + finishedWork.type, + current.memoizedProps, + finishedWork.elementType === finishedWork.type, + ); const prevState = current.memoizedState; // We could update instance props and state here, // but instead we rely on them being set during last render. // TODO: revisit this when we implement resuming. if (__DEV__) { if ( - finishedWork.type === finishedWork.elementType && + !finishedWork.type.defaultProps && + !('ref' in finishedWork.memoizedProps) && !didWarnAboutReassigningProps ) { if (instance.props !== finishedWork.memoizedProps) { @@ -918,7 +928,8 @@ function commitClassCallbacks(finishedWork: Fiber) { const instance = finishedWork.stateNode; if (__DEV__) { if ( - finishedWork.type === finishedWork.elementType && + !finishedWork.type.defaultProps && + !('ref' in finishedWork.memoizedProps) && !didWarnAboutReassigningProps ) { if (instance.props !== finishedWork.memoizedProps) { diff --git a/packages/react-reconciler/src/ReactFiberLazyComponent.js b/packages/react-reconciler/src/ReactFiberLazyComponent.js index b66f8efe4ed1b..b6c28f0ed5cce 100644 --- a/packages/react-reconciler/src/ReactFiberLazyComponent.js +++ b/packages/react-reconciler/src/ReactFiberLazyComponent.js @@ -10,6 +10,10 @@ import assign from 'shared/assign'; export function resolveDefaultProps(Component: any, baseProps: Object): Object { + // TODO: Remove support for default props for everything except class + // components, including setting default props on a lazy wrapper around a + // class type. + if (Component && Component.defaultProps) { // Resolve default props. Taken from ReactElement const props = assign({}, baseProps); diff --git a/packages/react-reconciler/src/__tests__/ReactClassComponentPropResolution-test.js b/packages/react-reconciler/src/__tests__/ReactClassComponentPropResolution-test.js new file mode 100644 index 0000000000000..53f3a3d04b564 --- /dev/null +++ b/packages/react-reconciler/src/__tests__/ReactClassComponentPropResolution-test.js @@ -0,0 +1,133 @@ +/** + * Copyright (c) Meta Platforms, Inc. and affiliates. + * + * This source code is licensed under the MIT license found in the + * LICENSE file in the root directory of this source tree. + * + * @emails react-core + */ + +'use strict'; + +let React; +let ReactNoop; +let Scheduler; +let act; +let assertLog; + +describe('ReactClassComponentPropResolution', () => { + beforeEach(() => { + jest.resetModules(); + + React = require('react'); + ReactNoop = require('react-noop-renderer'); + Scheduler = require('scheduler'); + act = require('internal-test-utils').act; + assertLog = require('internal-test-utils').assertLog; + }); + + function Text({text}) { + Scheduler.log(text); + return text; + } + + test('resolves ref and default props before calling lifecycle methods', async () => { + const root = ReactNoop.createRoot(); + + function getPropKeys(props) { + return Object.keys(props).join(', '); + } + + class Component extends React.Component { + constructor(props) { + super(props); + Scheduler.log('constructor: ' + getPropKeys(props)); + } + shouldComponentUpdate(props) { + Scheduler.log( + 'shouldComponentUpdate (prev props): ' + getPropKeys(this.props), + ); + Scheduler.log( + 'shouldComponentUpdate (next props): ' + getPropKeys(props), + ); + return true; + } + componentDidUpdate(props) { + Scheduler.log('componentDidUpdate (prev props): ' + getPropKeys(props)); + Scheduler.log( + 'componentDidUpdate (next props): ' + getPropKeys(this.props), + ); + return true; + } + componentDidMount() { + Scheduler.log('componentDidMount: ' + getPropKeys(this.props)); + return true; + } + UNSAFE_componentWillMount() { + Scheduler.log('componentWillMount: ' + getPropKeys(this.props)); + } + UNSAFE_componentWillReceiveProps(nextProps) { + Scheduler.log( + 'componentWillReceiveProps (prev props): ' + getPropKeys(this.props), + ); + Scheduler.log( + 'componentWillReceiveProps (next props): ' + getPropKeys(nextProps), + ); + } + UNSAFE_componentWillUpdate(nextProps) { + Scheduler.log( + 'componentWillUpdate (prev props): ' + getPropKeys(this.props), + ); + Scheduler.log( + 'componentWillUpdate (next props): ' + getPropKeys(nextProps), + ); + } + componentWillUnmount() { + Scheduler.log('componentWillUnmount: ' + getPropKeys(this.props)); + } + render() { + return ; + } + } + + Component.defaultProps = { + default: 'yo', + }; + + // `ref` should never appear as a prop. `default` always should. + + // Mount + const ref = React.createRef(); + await act(async () => { + root.render(); + }); + assertLog([ + 'constructor: text, default', + 'componentWillMount: text, default', + 'render: text, default', + 'componentDidMount: text, default', + ]); + + // Update + await act(async () => { + root.render(); + }); + assertLog([ + 'componentWillReceiveProps (prev props): text, default', + 'componentWillReceiveProps (next props): text, default', + 'shouldComponentUpdate (prev props): text, default', + 'shouldComponentUpdate (next props): text, default', + 'componentWillUpdate (prev props): text, default', + 'componentWillUpdate (next props): text, default', + 'render: text, default', + 'componentDidUpdate (prev props): text, default', + 'componentDidUpdate (next props): text, default', + ]); + + // Unmount + await act(async () => { + root.render(null); + }); + assertLog(['componentWillUnmount: text, default']); + }); +}); diff --git a/packages/react/src/__tests__/ReactCreateElement-test.js b/packages/react/src/__tests__/ReactCreateElement-test.js index cf372a33e35f0..c4a2d66310492 100644 --- a/packages/react/src/__tests__/ReactCreateElement-test.js +++ b/packages/react/src/__tests__/ReactCreateElement-test.js @@ -90,7 +90,7 @@ describe('ReactCreateElement', () => { ); }); - // @gate !enableRefAsProp + // @gate !enableRefAsProp || !__DEV__ it('should warn when `ref` is being accessed', async () => { class Child extends React.Component { render() { diff --git a/packages/react/src/__tests__/ReactJSXRuntime-test.js b/packages/react/src/__tests__/ReactJSXRuntime-test.js index 1c57954b205c4..d250792583271 100644 --- a/packages/react/src/__tests__/ReactJSXRuntime-test.js +++ b/packages/react/src/__tests__/ReactJSXRuntime-test.js @@ -244,7 +244,7 @@ describe('ReactJSXRuntime', () => { ); }); - // @gate !enableRefAsProp + // @gate !enableRefAsProp || !__DEV__ it('should warn when `ref` is being accessed', async () => { const container = document.createElement('div'); class Child extends React.Component {