diff --git a/packages/react-reconciler/src/ReactFiberRootScheduler.js b/packages/react-reconciler/src/ReactFiberRootScheduler.js index be14c4695089c..db60ab8be5360 100644 --- a/packages/react-reconciler/src/ReactFiberRootScheduler.js +++ b/packages/react-reconciler/src/ReactFiberRootScheduler.js @@ -119,6 +119,15 @@ export function ensureRootIsScheduled(root: FiberRoot): void { // unblock additional features we have planned. scheduleTaskForRootDuringMicrotask(root, now()); } + + if ( + __DEV__ && + ReactCurrentActQueue.isBatchingLegacy && + root.tag === LegacyRoot + ) { + // Special `act` case: Record whenever a legacy update is scheduled. + ReactCurrentActQueue.didScheduleLegacyUpdate = true; + } } export function flushSyncWorkOnAllRoots() { diff --git a/packages/react-reconciler/src/ReactFiberWorkLoop.js b/packages/react-reconciler/src/ReactFiberWorkLoop.js index 1715eab466c9d..401defb280fe4 100644 --- a/packages/react-reconciler/src/ReactFiberWorkLoop.js +++ b/packages/react-reconciler/src/ReactFiberWorkLoop.js @@ -828,7 +828,6 @@ export function scheduleUpdateOnFiber( ) { if (__DEV__ && ReactCurrentActQueue.isBatchingLegacy) { // Treat `act` as if it's inside `batchedUpdates`, even in legacy mode. - ReactCurrentActQueue.didScheduleLegacyUpdate = true; } else { // Flush the synchronous work now, unless we're already working or inside // a batch. This is intentionally inside scheduleUpdateOnFiber instead of diff --git a/packages/react-reconciler/src/__tests__/ReactIsomorphicAct-test.js b/packages/react-reconciler/src/__tests__/ReactIsomorphicAct-test.js index e64a208c54dc1..52d08a2a8005c 100644 --- a/packages/react-reconciler/src/__tests__/ReactIsomorphicAct-test.js +++ b/packages/react-reconciler/src/__tests__/ReactIsomorphicAct-test.js @@ -17,10 +17,13 @@ let Suspense; let DiscreteEventPriority; let startTransition; let waitForMicrotasks; +let Scheduler; +let assertLog; describe('isomorphic act()', () => { beforeEach(() => { React = require('react'); + Scheduler = require('scheduler'); ReactNoop = require('react-noop-renderer'); DiscreteEventPriority = @@ -31,6 +34,7 @@ describe('isomorphic act()', () => { startTransition = React.startTransition; waitForMicrotasks = require('internal-test-utils').waitForMicrotasks; + assertLog = require('internal-test-utils').assertLog; }); beforeEach(() => { @@ -41,6 +45,11 @@ describe('isomorphic act()', () => { jest.restoreAllMocks(); }); + function Text({text}) { + Scheduler.log(text); + return text; + } + // @gate __DEV__ test('bypasses queueMicrotask', async () => { const root = ReactNoop.createRoot(); @@ -131,20 +140,70 @@ describe('isomorphic act()', () => { test('in legacy mode, in an async scope, updates are batched until the first `await`', async () => { const root = ReactNoop.createLegacyRoot(); + await act(async () => { + queueMicrotask(() => { + Scheduler.log('Current tree in microtask: ' + root.getChildrenAsJSX()); + root.render(); + }); + root.render(); + root.render(); + + await null; + assertLog([ + // A and B should render in a single batch _before_ the microtask queue + // has run. This replicates the behavior of the original `act` + // implementation, for compatibility. + 'B', + 'Current tree in microtask: B', + + // C isn't scheduled until a microtask, so it's rendered separately. + 'C', + ]); + + // Subsequent updates should also render in separate batches. + root.render(); + root.render(); + assertLog(['D', 'E']); + }); + }); + + // @gate __DEV__ + test('in legacy mode, in an async scope, updates are batched until the first `await` (regression test: batchedUpdates)', async () => { + const root = ReactNoop.createLegacyRoot(); + await act(async () => { // These updates are batched. This replicates the behavior of the original // `act` implementation, for compatibility. - root.render('A'); - root.render('B'); - // Nothing has rendered yet. - expect(root).toMatchRenderedOutput(null); - await null; - // Updates are flushed after the first await. - expect(root).toMatchRenderedOutput('B'); + queueMicrotask(() => { + Scheduler.log('Current tree in microtask: ' + root.getChildrenAsJSX()); + root.render(); + }); - // Subsequent updates in the same scope aren't batched. - root.render('C'); - expect(root).toMatchRenderedOutput('C'); + // This is a regression test. The presence of `batchedUpdates` would cause + // these updates to not flush until a microtask. The correct behavior is + // that they flush before the microtask queue, regardless of whether + // they are wrapped with `batchedUpdates`. + ReactNoop.batchedUpdates(() => { + root.render(); + root.render(); + }); + + await null; + assertLog([ + // A and B should render in a single batch _before_ the microtask queue + // has run. This replicates the behavior of the original `act` + // implementation, for compatibility. + 'B', + 'Current tree in microtask: B', + + // C isn't scheduled until a microtask, so it's rendered separately. + 'C', + ]); + + // Subsequent updates should also render in separate batches. + root.render(); + root.render(); + assertLog(['D', 'E']); }); }); diff --git a/scripts/jest/matchers/reactTestMatchers.js b/scripts/jest/matchers/reactTestMatchers.js index ecf0329a0407d..63d03c5d70936 100644 --- a/scripts/jest/matchers/reactTestMatchers.js +++ b/scripts/jest/matchers/reactTestMatchers.js @@ -20,13 +20,13 @@ function captureAssertion(fn) { return {pass: true}; } -function assertYieldsWereCleared(Scheduler) { +function assertYieldsWereCleared(Scheduler, caller) { const actualYields = Scheduler.unstable_clearLog(); if (actualYields.length !== 0) { const error = Error( 'The event log is not empty. Call assertLog(...) first.' ); - Error.captureStackTrace(error, assertYieldsWereCleared); + Error.captureStackTrace(error, caller); throw error; } } @@ -34,7 +34,7 @@ function assertYieldsWereCleared(Scheduler) { function toMatchRenderedOutput(ReactNoop, expectedJSX) { if (typeof ReactNoop.getChildrenAsJSX === 'function') { const Scheduler = ReactNoop._Scheduler; - assertYieldsWereCleared(Scheduler); + assertYieldsWereCleared(Scheduler, toMatchRenderedOutput); return captureAssertion(() => { expect(ReactNoop.getChildrenAsJSX()).toEqual(expectedJSX); });