mirror of
https://github.com/facebook/react.git
synced 2025-11-01 09:12:30 +00:00
defffdbba4
This is a follow up to #31930 and a prerequisite for #31975. With View Transitions, the commit phase becomes async which means that other work can sneak in between. We need to be resilient to that. This PR first refactors the flushMutationEffects and flushLayoutEffects to use module scope variables to track its arguments so we can defer them. It shares these with how we were already doing it for flushPendingEffects. We also track how far along the commit phase we are so we know what we have left to flush. Then callers of flushPassiveEffects become flushPendingEffects. That helper synchronously flushes any remaining phases we've yet to commit. That ensure that things are at least consistent if that happens. Finally, when we are using a scheduled task, we don't do any work. This ensures that we're not flushing any work too early if we could've deferred it. This still ensures that we always do flush it before starting any new work on any root so new roots observe the committed state. There are some unfortunate effects that could happen from allowing things to flush eagerly. Such as if a flushSync sneaks in before startViewTransition, it'll skip the animation. If it's during a suspensey font it'll start the transition before the font has loaded which might be better than breaking flushSync. It'll also potentially flush passive effects inside the startViewTransition which should typically be ok.
335 lines
9.2 KiB
JavaScript
335 lines
9.2 KiB
JavaScript
/**
|
|
* 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.
|
|
*
|
|
* @flow
|
|
*/
|
|
|
|
/* eslint-disable react-internal/prod-error-codes */
|
|
|
|
import type {ReactElement} from 'shared/ReactElementType';
|
|
import type {Fiber, FiberRoot} from './ReactInternalTypes';
|
|
import type {ReactNodeList} from 'shared/ReactTypes';
|
|
|
|
import {
|
|
flushSyncWork,
|
|
scheduleUpdateOnFiber,
|
|
flushPendingEffects,
|
|
} from './ReactFiberWorkLoop';
|
|
import {enqueueConcurrentRenderForLane} from './ReactFiberConcurrentUpdates';
|
|
import {updateContainerSync} from './ReactFiberReconciler';
|
|
import {emptyContextObject} from './ReactFiberContext';
|
|
import {SyncLane} from './ReactFiberLane';
|
|
import {
|
|
ClassComponent,
|
|
FunctionComponent,
|
|
ForwardRef,
|
|
MemoComponent,
|
|
SimpleMemoComponent,
|
|
} from './ReactWorkTags';
|
|
import {
|
|
REACT_FORWARD_REF_TYPE,
|
|
REACT_MEMO_TYPE,
|
|
REACT_LAZY_TYPE,
|
|
} from 'shared/ReactSymbols';
|
|
|
|
export type Family = {
|
|
current: any,
|
|
};
|
|
|
|
export type RefreshUpdate = {
|
|
staleFamilies: Set<Family>,
|
|
updatedFamilies: Set<Family>,
|
|
};
|
|
|
|
// Resolves type to a family.
|
|
type RefreshHandler = any => Family | void;
|
|
|
|
// Used by React Refresh runtime through DevTools Global Hook.
|
|
export type SetRefreshHandler = (handler: RefreshHandler | null) => void;
|
|
export type ScheduleRefresh = (root: FiberRoot, update: RefreshUpdate) => void;
|
|
export type ScheduleRoot = (root: FiberRoot, element: ReactNodeList) => void;
|
|
|
|
let resolveFamily: RefreshHandler | null = null;
|
|
let failedBoundaries: WeakSet<Fiber> | null = null;
|
|
|
|
export const setRefreshHandler = (handler: RefreshHandler | null): void => {
|
|
if (__DEV__) {
|
|
resolveFamily = handler;
|
|
}
|
|
};
|
|
|
|
export function resolveFunctionForHotReloading(type: any): any {
|
|
if (__DEV__) {
|
|
if (resolveFamily === null) {
|
|
// Hot reloading is disabled.
|
|
return type;
|
|
}
|
|
const family = resolveFamily(type);
|
|
if (family === undefined) {
|
|
return type;
|
|
}
|
|
// Use the latest known implementation.
|
|
return family.current;
|
|
} else {
|
|
return type;
|
|
}
|
|
}
|
|
|
|
export function resolveClassForHotReloading(type: any): any {
|
|
// No implementation differences.
|
|
return resolveFunctionForHotReloading(type);
|
|
}
|
|
|
|
export function resolveForwardRefForHotReloading(type: any): any {
|
|
if (__DEV__) {
|
|
if (resolveFamily === null) {
|
|
// Hot reloading is disabled.
|
|
return type;
|
|
}
|
|
const family = resolveFamily(type);
|
|
if (family === undefined) {
|
|
// Check if we're dealing with a real forwardRef. Don't want to crash early.
|
|
if (
|
|
type !== null &&
|
|
type !== undefined &&
|
|
typeof type.render === 'function'
|
|
) {
|
|
// ForwardRef is special because its resolved .type is an object,
|
|
// but it's possible that we only have its inner render function in the map.
|
|
// If that inner render function is different, we'll build a new forwardRef type.
|
|
const currentRender = resolveFunctionForHotReloading(type.render);
|
|
if (type.render !== currentRender) {
|
|
const syntheticType = {
|
|
$$typeof: REACT_FORWARD_REF_TYPE,
|
|
render: currentRender,
|
|
};
|
|
if (type.displayName !== undefined) {
|
|
(syntheticType: any).displayName = type.displayName;
|
|
}
|
|
return syntheticType;
|
|
}
|
|
}
|
|
return type;
|
|
}
|
|
// Use the latest known implementation.
|
|
return family.current;
|
|
} else {
|
|
return type;
|
|
}
|
|
}
|
|
|
|
export function isCompatibleFamilyForHotReloading(
|
|
fiber: Fiber,
|
|
element: ReactElement,
|
|
): boolean {
|
|
if (__DEV__) {
|
|
if (resolveFamily === null) {
|
|
// Hot reloading is disabled.
|
|
return false;
|
|
}
|
|
|
|
const prevType = fiber.elementType;
|
|
const nextType = element.type;
|
|
|
|
// If we got here, we know types aren't === equal.
|
|
let needsCompareFamilies = false;
|
|
|
|
const $$typeofNextType =
|
|
typeof nextType === 'object' && nextType !== null
|
|
? nextType.$$typeof
|
|
: null;
|
|
|
|
switch (fiber.tag) {
|
|
case ClassComponent: {
|
|
if (typeof nextType === 'function') {
|
|
needsCompareFamilies = true;
|
|
}
|
|
break;
|
|
}
|
|
case FunctionComponent: {
|
|
if (typeof nextType === 'function') {
|
|
needsCompareFamilies = true;
|
|
} else if ($$typeofNextType === REACT_LAZY_TYPE) {
|
|
// We don't know the inner type yet.
|
|
// We're going to assume that the lazy inner type is stable,
|
|
// and so it is sufficient to avoid reconciling it away.
|
|
// We're not going to unwrap or actually use the new lazy type.
|
|
needsCompareFamilies = true;
|
|
}
|
|
break;
|
|
}
|
|
case ForwardRef: {
|
|
if ($$typeofNextType === REACT_FORWARD_REF_TYPE) {
|
|
needsCompareFamilies = true;
|
|
} else if ($$typeofNextType === REACT_LAZY_TYPE) {
|
|
needsCompareFamilies = true;
|
|
}
|
|
break;
|
|
}
|
|
case MemoComponent:
|
|
case SimpleMemoComponent: {
|
|
if ($$typeofNextType === REACT_MEMO_TYPE) {
|
|
// TODO: if it was but can no longer be simple,
|
|
// we shouldn't set this.
|
|
needsCompareFamilies = true;
|
|
} else if ($$typeofNextType === REACT_LAZY_TYPE) {
|
|
needsCompareFamilies = true;
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
return false;
|
|
}
|
|
|
|
// Check if both types have a family and it's the same one.
|
|
if (needsCompareFamilies) {
|
|
// Note: memo() and forwardRef() we'll compare outer rather than inner type.
|
|
// This means both of them need to be registered to preserve state.
|
|
// If we unwrapped and compared the inner types for wrappers instead,
|
|
// then we would risk falsely saying two separate memo(Foo)
|
|
// calls are equivalent because they wrap the same Foo function.
|
|
const prevFamily = resolveFamily(prevType);
|
|
// $FlowFixMe[not-a-function] found when upgrading Flow
|
|
if (prevFamily !== undefined && prevFamily === resolveFamily(nextType)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
export function markFailedErrorBoundaryForHotReloading(fiber: Fiber) {
|
|
if (__DEV__) {
|
|
if (resolveFamily === null) {
|
|
// Hot reloading is disabled.
|
|
return;
|
|
}
|
|
if (typeof WeakSet !== 'function') {
|
|
return;
|
|
}
|
|
if (failedBoundaries === null) {
|
|
failedBoundaries = new WeakSet();
|
|
}
|
|
failedBoundaries.add(fiber);
|
|
}
|
|
}
|
|
|
|
export const scheduleRefresh: ScheduleRefresh = (
|
|
root: FiberRoot,
|
|
update: RefreshUpdate,
|
|
): void => {
|
|
if (__DEV__) {
|
|
if (resolveFamily === null) {
|
|
// Hot reloading is disabled.
|
|
return;
|
|
}
|
|
const {staleFamilies, updatedFamilies} = update;
|
|
flushPendingEffects();
|
|
scheduleFibersWithFamiliesRecursively(
|
|
root.current,
|
|
updatedFamilies,
|
|
staleFamilies,
|
|
);
|
|
flushSyncWork();
|
|
}
|
|
};
|
|
|
|
export const scheduleRoot: ScheduleRoot = (
|
|
root: FiberRoot,
|
|
element: ReactNodeList,
|
|
): void => {
|
|
if (__DEV__) {
|
|
if (root.context !== emptyContextObject) {
|
|
// Super edge case: root has a legacy _renderSubtree context
|
|
// but we don't know the parentComponent so we can't pass it.
|
|
// Just ignore. We'll delete this with _renderSubtree code path later.
|
|
return;
|
|
}
|
|
updateContainerSync(element, root, null, null);
|
|
flushSyncWork();
|
|
}
|
|
};
|
|
|
|
function scheduleFibersWithFamiliesRecursively(
|
|
fiber: Fiber,
|
|
updatedFamilies: Set<Family>,
|
|
staleFamilies: Set<Family>,
|
|
): void {
|
|
if (__DEV__) {
|
|
const {alternate, child, sibling, tag, type} = fiber;
|
|
|
|
let candidateType = null;
|
|
switch (tag) {
|
|
case FunctionComponent:
|
|
case SimpleMemoComponent:
|
|
case ClassComponent:
|
|
candidateType = type;
|
|
break;
|
|
case ForwardRef:
|
|
candidateType = type.render;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (resolveFamily === null) {
|
|
throw new Error('Expected resolveFamily to be set during hot reload.');
|
|
}
|
|
|
|
let needsRender = false;
|
|
let needsRemount = false;
|
|
if (candidateType !== null) {
|
|
const family = resolveFamily(candidateType);
|
|
if (family !== undefined) {
|
|
if (staleFamilies.has(family)) {
|
|
needsRemount = true;
|
|
} else if (updatedFamilies.has(family)) {
|
|
if (tag === ClassComponent) {
|
|
needsRemount = true;
|
|
} else {
|
|
needsRender = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (failedBoundaries !== null) {
|
|
if (
|
|
failedBoundaries.has(fiber) ||
|
|
// $FlowFixMe[incompatible-use] found when upgrading Flow
|
|
(alternate !== null && failedBoundaries.has(alternate))
|
|
) {
|
|
needsRemount = true;
|
|
}
|
|
}
|
|
|
|
if (needsRemount) {
|
|
fiber._debugNeedsRemount = true;
|
|
}
|
|
if (needsRemount || needsRender) {
|
|
const root = enqueueConcurrentRenderForLane(fiber, SyncLane);
|
|
if (root !== null) {
|
|
scheduleUpdateOnFiber(root, fiber, SyncLane);
|
|
}
|
|
}
|
|
if (child !== null && !needsRemount) {
|
|
scheduleFibersWithFamiliesRecursively(
|
|
child,
|
|
updatedFamilies,
|
|
staleFamilies,
|
|
);
|
|
}
|
|
if (sibling !== null) {
|
|
scheduleFibersWithFamiliesRecursively(
|
|
sibling,
|
|
updatedFamilies,
|
|
staleFamilies,
|
|
);
|
|
}
|
|
}
|
|
}
|