diff --git a/packages/react-devtools-shared/src/__tests__/TimelineProfiler-test.js b/packages/react-devtools-shared/src/__tests__/TimelineProfiler-test.js index 32e90276825e4..cd9081d8b95b2 100644 --- a/packages/react-devtools-shared/src/__tests__/TimelineProfiler-test.js +++ b/packages/react-devtools-shared/src/__tests__/TimelineProfiler-test.js @@ -1243,10 +1243,9 @@ describe('Timeline profiler', () => { function Example() { const setHigh = React.useState(0)[1]; const setLow = React.useState(0)[1]; - const startTransition = React.useTransition()[1]; updaterFn = () => { - startTransition(() => { + React.startTransition(() => { setLow(prevLow => prevLow + 1); }); setHigh(prevHigh => prevHigh + 1); @@ -1265,24 +1264,6 @@ describe('Timeline profiler', () => { const timelineData = stopProfilingAndGetTimelineData(); expect(timelineData.schedulingEvents).toMatchInlineSnapshot(` [ - { - "componentName": "Example", - "componentStack": " - in Example (at **)", - "lanes": "0b0000000000000000000000000001000", - "timestamp": 10, - "type": "schedule-state-update", - "warning": null, - }, - { - "componentName": "Example", - "componentStack": " - in Example (at **)", - "lanes": "0b0000000000000000000000010000000", - "timestamp": 10, - "type": "schedule-state-update", - "warning": null, - }, { "componentName": "Example", "componentStack": " diff --git a/packages/react-reconciler/src/ReactFiberHooks.js b/packages/react-reconciler/src/ReactFiberHooks.js index 221515c248de3..63ce2a8e655b5 100644 --- a/packages/react-reconciler/src/ReactFiberHooks.js +++ b/packages/react-reconciler/src/ReactFiberHooks.js @@ -149,11 +149,13 @@ import type {ThenableState} from './ReactFiberThenable'; import type {BatchConfigTransition} from './ReactFiberTracingMarkerComponent'; import {requestAsyncActionContext} from './ReactFiberAsyncAction'; import {HostTransitionContext} from './ReactFiberHostContext'; +import {requestTransitionLane} from './ReactFiberRootScheduler'; const {ReactCurrentDispatcher, ReactCurrentBatchConfig} = ReactSharedInternals; export type Update = { lane: Lane, + revertLane: Lane, action: A, hasEagerState: boolean, eagerState: S | null, @@ -1136,6 +1138,14 @@ function updateReducer( init?: I => S, ): [S, Dispatch] { const hook = updateWorkInProgressHook(); + return updateReducerImpl(hook, ((currentHook: any): Hook), reducer); +} + +function updateReducerImpl( + hook: Hook, + current: Hook, + reducer: (S, A) => S, +): [S, Dispatch] { const queue = hook.queue; if (queue === null) { @@ -1146,10 +1156,8 @@ function updateReducer( queue.lastRenderedReducer = reducer; - const current: Hook = (currentHook: any); - // The last rebase update that is NOT part of the base state. - let baseQueue = current.baseQueue; + let baseQueue = hook.baseQueue; // The last pending update that hasn't been processed yet. const pendingQueue = queue.pending; @@ -1180,7 +1188,7 @@ function updateReducer( if (baseQueue !== null) { // We have a queue to process. const first = baseQueue.next; - let newState = current.baseState; + let newState = hook.baseState; let newBaseState = null; let newBaseQueueFirst = null; @@ -1206,6 +1214,7 @@ function updateReducer( // update/state. const clone: Update = { lane: updateLane, + revertLane: update.revertLane, action: update.action, hasEagerState: update.hasEagerState, eagerState: update.eagerState, @@ -1228,18 +1237,68 @@ function updateReducer( } else { // This update does have sufficient priority. - if (newBaseQueueLast !== null) { - const clone: Update = { - // This update is going to be committed so we never want uncommit - // it. Using NoLane works because 0 is a subset of all bitmasks, so - // this will never be skipped by the check above. - lane: NoLane, - action: update.action, - hasEagerState: update.hasEagerState, - eagerState: update.eagerState, - next: (null: any), - }; - newBaseQueueLast = newBaseQueueLast.next = clone; + // Check if this is an optimistic update. + const revertLane = update.revertLane; + if (!enableAsyncActions || revertLane === NoLane) { + // This is not an optimistic update, and we're going to apply it now. + // But, if there were earlier updates that were skipped, we need to + // leave this update in the queue so it can be rebased later. + if (newBaseQueueLast !== null) { + const clone: Update = { + // This update is going to be committed so we never want uncommit + // it. Using NoLane works because 0 is a subset of all bitmasks, so + // this will never be skipped by the check above. + lane: NoLane, + revertLane: NoLane, + action: update.action, + hasEagerState: update.hasEagerState, + eagerState: update.eagerState, + next: (null: any), + }; + newBaseQueueLast = newBaseQueueLast.next = clone; + } + } else { + // This is an optimistic update. If the "revert" priority is + // sufficient, don't apply the update. Otherwise, apply the update, + // but leave it in the queue so it can be either reverted or + // rebased in a subsequent render. + if (isSubsetOfLanes(renderLanes, revertLane)) { + // The transition that this optimistic update is associated with + // has finished. Pretend the update doesn't exist by skipping + // over it. + update = update.next; + continue; + } else { + const clone: Update = { + // Once we commit an optimistic update, we shouldn't uncommit it + // until the transition it is associated with has finished + // (represented by revertLane). Using NoLane here works because 0 + // is a subset of all bitmasks, so this will never be skipped by + // the check above. + lane: NoLane, + // Reuse the same revertLane so we know when the transition + // has finished. + revertLane: update.revertLane, + action: update.action, + hasEagerState: update.hasEagerState, + eagerState: update.eagerState, + next: (null: any), + }; + if (newBaseQueueLast === null) { + newBaseQueueFirst = newBaseQueueLast = clone; + newBaseState = newState; + } else { + newBaseQueueLast = newBaseQueueLast.next = clone; + } + // Update the remaining priority in the queue. + // TODO: Don't need to accumulate this. Instead, we can remove + // renderLanes from the original lanes. + currentlyRenderingFiber.lanes = mergeLanes( + currentlyRenderingFiber.lanes, + revertLane, + ); + markSkippedUpdateLanes(revertLane); + } } // Process this update. @@ -1899,12 +1958,6 @@ function mountStateImpl(initialState: (() => S) | S): Hook { lastRenderedState: (initialState: any), }; hook.queue = queue; - const dispatch: Dispatch> = (dispatchSetState.bind( - null, - currentlyRenderingFiber, - queue, - ): any); - queue.dispatch = dispatch; return hook; } @@ -1912,43 +1965,99 @@ function mountState( initialState: (() => S) | S, ): [S, Dispatch>] { const hook = mountStateImpl(initialState); - return [hook.memoizedState, hook.queue.dispatch]; + const queue = hook.queue; + const dispatch: Dispatch> = (dispatchSetState.bind( + null, + currentlyRenderingFiber, + queue, + ): any); + queue.dispatch = dispatch; + return [hook.memoizedState, dispatch]; } function updateState( initialState: (() => S) | S, ): [S, Dispatch>] { - return updateReducer(basicStateReducer, (initialState: any)); + return updateReducer(basicStateReducer, initialState); } function rerenderState( initialState: (() => S) | S, ): [S, Dispatch>] { - return rerenderReducer(basicStateReducer, (initialState: any)); + return rerenderReducer(basicStateReducer, initialState); } function mountOptimisticState( passthrough: S, reducer: ?(S, A) => S, ): [S, (A) => void] { - // $FlowFixMe - TODO: Actual implementation - return mountState(passthrough); + const hook = mountWorkInProgressHook(); + hook.memoizedState = hook.baseState = passthrough; + const queue: UpdateQueue = { + pending: null, + lanes: NoLanes, + dispatch: null, + // Optimistic state does not use the eager update optimization. + lastRenderedReducer: null, + lastRenderedState: null, + }; + hook.queue = queue; + // This is different than the normal setState function. + const dispatch: A => void = (dispatchOptimisticSetState.bind( + null, + currentlyRenderingFiber, + true, + queue, + ): any); + queue.dispatch = dispatch; + return [passthrough, dispatch]; } function updateOptimisticState( passthrough: S, reducer: ?(S, A) => S, ): [S, (A) => void] { - // $FlowFixMe - TODO: Actual implementation - return updateState(passthrough); + const hook = updateWorkInProgressHook(); + + // Optimistic updates are always rebased on top of the latest value passed in + // as an argument. It's called a passthrough because if there are no pending + // updates, it will be returned as-is. + // + // Reset the base state and memoized state to the passthrough. Future + // updates will be applied on top of this. + hook.baseState = hook.memoizedState = passthrough; + + // If a reducer is not provided, default to the same one used by useState. + const resolvedReducer: (S, A) => S = + typeof reducer === 'function' ? reducer : (basicStateReducer: any); + + return updateReducerImpl(hook, ((currentHook: any): Hook), resolvedReducer); } function rerenderOptimisticState( passthrough: S, reducer: ?(S, A) => S, ): [S, (A) => void] { - // $FlowFixMe - TODO: Actual implementation - return rerenderState(passthrough); + // Unlike useState, useOptimisticState doesn't support render phase updates. + // Also unlike useState, we need to replay all pending updates again in case + // the passthrough value changed. + // + // So instead of a forked re-render implementation that knows how to handle + // render phase udpates, we can use the same implementation as during a + // regular mount or update. + + if (currentHook !== null) { + // This is an update. Process the update queue. + return updateOptimisticState(passthrough, reducer); + } + + // This is a mount. No updates to process. + const hook = updateWorkInProgressHook(); + // Reset the base state and memoized state to the passthrough. Future + // updates will be applied on top of this. + hook.baseState = hook.memoizedState = passthrough; + const dispatch = hook.queue.dispatch; + return [passthrough, dispatch]; } function pushEffect( @@ -2491,8 +2600,20 @@ function startTransition( ); const prevTransition = ReactCurrentBatchConfig.transition; - ReactCurrentBatchConfig.transition = null; - dispatchSetState(fiber, queue, pendingState); + + if (enableAsyncActions) { + // We don't really need to use an optimistic update here, because we + // schedule a second "revert" update below (which we use to suspend the + // transition until the async action scope has finished). But we'll use an + // optimistic update anyway to make it less likely the behavior accidentally + // diverges; for example, both an optimistic update and this one should + // share the same lane. + dispatchOptimisticSetState(fiber, false, queue, pendingState); + } else { + ReactCurrentBatchConfig.transition = null; + dispatchSetState(fiber, queue, pendingState); + } + const currentTransition = (ReactCurrentBatchConfig.transition = ({}: BatchConfigTransition)); @@ -2827,6 +2948,7 @@ function dispatchReducerAction( const update: Update = { lane, + revertLane: NoLane, action, hasEagerState: false, eagerState: null, @@ -2865,6 +2987,7 @@ function dispatchSetState( const update: Update = { lane, + revertLane: NoLane, action, hasEagerState: false, eagerState: null, @@ -2928,6 +3051,58 @@ function dispatchSetState( markUpdateInDevTools(fiber, lane, action); } +function dispatchOptimisticSetState( + fiber: Fiber, + throwIfDuringRender: boolean, + queue: UpdateQueue, + action: A, +): void { + const update: Update = { + // An optimistic update commits synchronously. + lane: SyncLane, + // After committing, the optimistic update is "reverted" using the same + // lane as the transition it's associated with. + // + // TODO: Warn if there's no transition/action associated with this + // optimistic update. + revertLane: requestTransitionLane(), + action, + hasEagerState: false, + eagerState: null, + next: (null: any), + }; + + if (isRenderPhaseUpdate(fiber)) { + // When calling startTransition during render, this warns instead of + // throwing because throwing would be a breaking change. setOptimisticState + // is a new API so it's OK to throw. + if (throwIfDuringRender) { + throw new Error('Cannot update optimistic state while rendering.'); + } else { + // startTransition was called during render. We don't need to do anything + // besides warn here because the render phase update would be overidden by + // the second update, anyway. We can remove this branch and make it throw + // in a future release. + if (__DEV__) { + console.error('Cannot call startTransition while rendering.'); + } + } + } else { + const root = enqueueConcurrentHookUpdate(fiber, queue, update, SyncLane); + if (root !== null) { + // NOTE: The optimistic update implementation assumes that the transition + // will never be attempted before the optimistic update. This currently + // holds because the optimistic update is always synchronous. If we ever + // change that, we'll need to account for this. + scheduleUpdateOnFiber(root, fiber, SyncLane); + // Optimistic updates are always synchronous, so we don't need to call + // entangleTransitionUpdate here. + } + } + + markUpdateInDevTools(fiber, SyncLane, action); +} + function isRenderPhaseUpdate(fiber: Fiber): boolean { const alternate = fiber.alternate; return ( diff --git a/packages/react-reconciler/src/__tests__/ReactAsyncActions-test.js b/packages/react-reconciler/src/__tests__/ReactAsyncActions-test.js index 159a4a821834c..7937067d1b2ec 100644 --- a/packages/react-reconciler/src/__tests__/ReactAsyncActions-test.js +++ b/packages/react-reconciler/src/__tests__/ReactAsyncActions-test.js @@ -648,17 +648,430 @@ describe('ReactAsyncActions', () => { }); // @gate enableAsyncActions - test('useOptimisticState exists', async () => { - // This API isn't implemented yet. This just tests that it's wired - // up correctly. + test('useOptimisticState can be used to implement a pending state', async () => { + const startTransition = React.startTransition; + + let setIsPending; + function App({text}) { + const [isPending, _setIsPending] = useOptimisticState(false); + setIsPending = _setIsPending; + return ( + <> + + + + ); + } + + // Initial render + const root = ReactNoop.createRoot(); + resolveText('A'); + await act(() => root.render()); + assertLog(['Pending: false', 'A']); + expect(root).toMatchRenderedOutput('Pending: falseA'); + + // Start a transition + await act(() => + startTransition(() => { + setIsPending(true); + root.render(); + }), + ); + assertLog([ + // Render the pending state immediately + 'Pending: true', + 'A', + + // Then attempt to render the transition. The pending state will be + // automatically reverted. + 'Pending: false', + 'Suspend! [B]', + ]); + + // Resolve the transition + await act(() => resolveText('B')); + assertLog([ + // Render the pending state immediately + 'Pending: false', + 'B', + ]); + }); + + // @gate enableAsyncActions + test('useOptimisticState rebases pending updates on top of passthrough value', async () => { + let serverCart = ['A']; + + async function submitNewItem(item) { + await getText('Adding item ' + item); + serverCart = [...serverCart, item]; + React.startTransition(() => { + root.render(); + }); + } + + let addItemToCart; + function App({cart}) { + const [isPending, startTransition] = useTransition(); + + const savedCartSize = cart.length; + const [optimisticCartSize, setOptimisticCartSize] = + useOptimisticState(savedCartSize); + + addItemToCart = item => { + startTransition(async () => { + setOptimisticCartSize(n => n + 1); + await submitNewItem(item); + }); + }; + + return ( + <> +
+ +
+
+ +
+
    + {cart.map(item => ( +
  • + +
  • + ))} +
+ + ); + } + + // Initial render + const root = ReactNoop.createRoot(); + await act(() => root.render()); + assertLog(['Pending: false', 'Items in cart: 1', 'Item A']); + expect(root).toMatchRenderedOutput( + <> +
Pending: false
+
Items in cart: 1
+
    +
  • Item A
  • +
+ , + ); + + // The cart size is incremented even though B hasn't been added yet. + await act(() => addItemToCart('B')); + assertLog(['Pending: true', 'Items in cart: 2', 'Item A']); + expect(root).toMatchRenderedOutput( + <> +
Pending: true
+
Items in cart: 2
+
    +
  • Item A
  • +
+ , + ); + + // While B is still pending, another item gets added to the cart + // out-of-band. + serverCart = [...serverCart, 'C']; + // NOTE: This is a synchronous update only because we don't yet support + // parallel transitions; all transitions are entangled together. Once we add + // support for parallel transitions, we can update this test. + ReactNoop.flushSync(() => root.render()); + assertLog([ + 'Pending: true', + // Note that the optimistic cart size is still correct, because the + // pending update was rebased on top new value. + 'Items in cart: 3', + 'Item A', + 'Item C', + ]); + expect(root).toMatchRenderedOutput( + <> +
Pending: true
+
Items in cart: 3
+
    +
  • Item A
  • +
  • Item C
  • +
+ , + ); + + // Finish loading B. The optimistic state is reverted. + await act(() => resolveText('Adding item B')); + assertLog([ + 'Pending: false', + 'Items in cart: 3', + 'Item A', + 'Item C', + 'Item B', + ]); + expect(root).toMatchRenderedOutput( + <> +
Pending: false
+
Items in cart: 3
+
    +
  • Item A
  • +
  • Item C
  • +
  • Item B
  • +
+ , + ); + }); + + // @gate enableAsyncActions + test('useOptimisticState accepts a custom reducer', async () => { + let serverCart = ['A']; + + async function submitNewItem(item) { + await getText('Adding item ' + item); + serverCart = [...serverCart, item]; + React.startTransition(() => { + root.render(); + }); + } + + let addItemToCart; + function App({cart}) { + const [isPending, startTransition] = useTransition(); + + const savedCartSize = cart.length; + const [optimisticCartSize, addToOptimisticCart] = useOptimisticState( + savedCartSize, + (prevSize, newItem) => { + Scheduler.log('Increment optimistic cart size for ' + newItem); + return prevSize + 1; + }, + ); + + addItemToCart = item => { + startTransition(async () => { + addToOptimisticCart(item); + await submitNewItem(item); + }); + }; + + return ( + <> +
+ +
+
+ +
+
    + {cart.map(item => ( +
  • + +
  • + ))} +
+ + ); + } + + // Initial render + const root = ReactNoop.createRoot(); + await act(() => root.render()); + assertLog(['Pending: false', 'Items in cart: 1', 'Item A']); + expect(root).toMatchRenderedOutput( + <> +
Pending: false
+
Items in cart: 1
+
    +
  • Item A
  • +
+ , + ); + + // The cart size is incremented even though B hasn't been added yet. + await act(() => addItemToCart('B')); + assertLog([ + 'Increment optimistic cart size for B', + 'Pending: true', + 'Items in cart: 2', + 'Item A', + ]); + expect(root).toMatchRenderedOutput( + <> +
Pending: true
+
Items in cart: 2
+
    +
  • Item A
  • +
+ , + ); + + // While B is still pending, another item gets added to the cart + // out-of-band. + serverCart = [...serverCart, 'C']; + // NOTE: This is a synchronous update only because we don't yet support + // parallel transitions; all transitions are entangled together. Once we add + // support for parallel transitions, we can update this test. + ReactNoop.flushSync(() => root.render()); + assertLog([ + 'Increment optimistic cart size for B', + 'Pending: true', + // Note that the optimistic cart size is still correct, because the + // pending update was rebased on top new value. + 'Items in cart: 3', + 'Item A', + 'Item C', + ]); + expect(root).toMatchRenderedOutput( + <> +
Pending: true
+
Items in cart: 3
+
    +
  • Item A
  • +
  • Item C
  • +
+ , + ); + + // Finish loading B. The optimistic state is reverted. + await act(() => resolveText('Adding item B')); + assertLog([ + 'Pending: false', + 'Items in cart: 3', + 'Item A', + 'Item C', + 'Item B', + ]); + expect(root).toMatchRenderedOutput( + <> +
Pending: false
+
Items in cart: 3
+
    +
  • Item A
  • +
  • Item C
  • +
  • Item B
  • +
+ , + ); + }); + + // @gate enableAsyncActions + test('useOptimisticState rebases if the passthrough is updated during a render phase update', async () => { + // This is kind of an esoteric case where it's hard to come up with a + // realistic real-world scenario but it should still work. + let increment; + let setCount; function App() { - const [text] = useOptimisticState('Hi'); - return ; + const [isPending, startTransition] = useTransition(2); + const [count, _setCount] = useState(0); + setCount = _setCount; + + const [optimisticCount, setOptimisticCount] = useOptimisticState( + count, + prev => { + Scheduler.log('Increment optimistic count'); + return prev + 1; + }, + ); + + if (count === 1) { + Scheduler.log('Render phase update count from 1 to 2'); + setCount(2); + } + + increment = () => + startTransition(async () => { + setOptimisticCount(n => n + 1); + await getText('Wait to increment'); + React.startTransition(() => setCount(n => n + 1)); + }); + + return ( + <> +
+ +
+ {isPending ? ( +
+ +
+ ) : null} + + ); } const root = ReactNoop.createRoot(); await act(() => root.render()); - assertLog(['Hi']); - expect(root).toMatchRenderedOutput('Hi'); + assertLog(['Count: 0']); + expect(root).toMatchRenderedOutput(
Count: 0
); + + await act(() => increment()); + assertLog([ + 'Increment optimistic count', + 'Count: 0', + 'Optimistic count: 1', + ]); + expect(root).toMatchRenderedOutput( + <> +
Count: 0
+
Optimistic count: 1
+ , + ); + + await act(() => setCount(1)); + assertLog([ + 'Increment optimistic count', + 'Render phase update count from 1 to 2', + // The optimistic update is rebased on top of the new passthrough value. + 'Increment optimistic count', + 'Count: 2', + 'Optimistic count: 3', + ]); + expect(root).toMatchRenderedOutput( + <> +
Count: 2
+
Optimistic count: 3
+ , + ); + + // Finish the action + await act(() => resolveText('Wait to increment')); + assertLog(['Count: 3']); + expect(root).toMatchRenderedOutput(
Count: 3
); + }); + + // @gate enableAsyncActions + test('useOptimisticState rebases if the passthrough is updated during a render phase update (initial mount)', async () => { + // This is kind of an esoteric case where it's hard to come up with a + // realistic real-world scenario but it should still work. + function App() { + const [count, setCount] = useState(0); + const [optimisticCount] = useOptimisticState(count); + + if (count === 0) { + Scheduler.log('Render phase update count from 1 to 2'); + setCount(1); + } + + return ( + <> +
+ +
+
+ +
+ + ); + } + + const root = ReactNoop.createRoot(); + await act(() => root.render()); + assertLog([ + 'Render phase update count from 1 to 2', + 'Count: 1', + 'Optimistic count: 1', + ]); + expect(root).toMatchRenderedOutput( + <> +
Count: 1
+
Optimistic count: 1
+ , + ); }); });