mirror of
https://github.com/facebook/react.git
synced 2025-11-01 09:12:30 +00:00
2e0d86d221
* Pass children to hydration root constructor I already made this change for the concurrent root API in #23309. This does the same thing for the legacy API. Doesn't change any behavior, but I will use this in the next steps. * Add isRootDehydrated function Currently this does nothing except read a boolean field, but I'm about to change this logic. Since this is accessed by React DOM, too, I put the function in a separate module that can be deep imported. Previously, it was accessing the FiberRoot directly. The reason it's a separate module is to break a circular dependency between React DOM and the reconciler. * Allow updates at lower pri without forcing client render Currently, if a root is updated before the shell has finished hydrating (for example, due to a top-level navigation), we immediately revert to client rendering. This is rare because the root is expected is finish quickly, but not exceedingly rare because the root may be suspended. This adds support for updating the root without forcing a client render as long as the update has lower priority than the initial hydration, i.e. if the update is wrapped in startTransition. To implement this, I had to do some refactoring. The main idea here is to make it closer to how we implement hydration in Suspense boundaries: - I moved isDehydrated from the shared FiberRoot object to the HostRoot's state object. - In the begin phase, I check if the root has received an by comparing the new children to the initial children. If they are different, we revert to client rendering, and set isDehydrated to false using a derived state update (a la getDerivedStateFromProps). - There are a few places where we used to set root.isDehydrated to false as a way to force a client render. Instead, I set the ForceClientRender flag on the root work-in-progress fiber. - Whenever we fall back to client rendering, I log a recoverable error. The overall code structure is almost identical to the corresponding logic for Suspense components. The reason this works is because if the update has lower priority than the initial hydration, it won't be processed during the hydration render, so the children will be the same. We can go even further and allow updates at _higher_ priority (though not sync) by implementing selective hydration at the root, like we do for Suspense boundaries: interrupt the current render, attempt hydration at slightly higher priority than the update, then continue rendering the update. I haven't implemented this yet, but I've structured the code in anticipation of adding this later. * Wrap useMutableSource logic in feature flag
301 lines
8.9 KiB
JavaScript
301 lines
8.9 KiB
JavaScript
/**
|
|
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*
|
|
* @flow
|
|
*/
|
|
|
|
import type {HostComponent} from './ReactNativeTypes';
|
|
import type {ReactNodeList} from 'shared/ReactTypes';
|
|
import type {ElementRef, Element, ElementType} from 'react';
|
|
|
|
import './ReactNativeInjection';
|
|
|
|
import {
|
|
findHostInstance,
|
|
findHostInstanceWithWarning,
|
|
batchedUpdates as batchedUpdatesImpl,
|
|
discreteUpdates,
|
|
createContainer,
|
|
updateContainer,
|
|
injectIntoDevTools,
|
|
getPublicRootInstance,
|
|
} from 'react-reconciler/src/ReactFiberReconciler';
|
|
// TODO: direct imports like some-package/src/* are bad. Fix me.
|
|
import {getStackByFiberInDevAndProd} from 'react-reconciler/src/ReactFiberComponentStack';
|
|
import {createPortal as createPortalImpl} from 'react-reconciler/src/ReactPortal';
|
|
import {
|
|
setBatchingImplementation,
|
|
batchedUpdates,
|
|
} from './legacy-events/ReactGenericBatching';
|
|
import ReactVersion from 'shared/ReactVersion';
|
|
// Modules provided by RN:
|
|
import {
|
|
UIManager,
|
|
legacySendAccessibilityEvent,
|
|
} from 'react-native/Libraries/ReactPrivate/ReactNativePrivateInterface';
|
|
import {getInspectorDataForInstance} from './ReactNativeFiberInspector';
|
|
|
|
import {getClosestInstanceFromNode} from './ReactNativeComponentTree';
|
|
import {
|
|
getInspectorDataForViewTag,
|
|
getInspectorDataForViewAtPoint,
|
|
} from './ReactNativeFiberInspector';
|
|
import {LegacyRoot} from 'react-reconciler/src/ReactRootTags';
|
|
import ReactSharedInternals from 'shared/ReactSharedInternals';
|
|
import getComponentNameFromType from 'shared/getComponentNameFromType';
|
|
|
|
const ReactCurrentOwner = ReactSharedInternals.ReactCurrentOwner;
|
|
|
|
function findHostInstance_DEPRECATED(
|
|
componentOrHandle: any,
|
|
): ?React$ElementRef<HostComponent<mixed>> {
|
|
if (__DEV__) {
|
|
const owner = ReactCurrentOwner.current;
|
|
if (owner !== null && owner.stateNode !== null) {
|
|
if (!owner.stateNode._warnedAboutRefsInRender) {
|
|
console.error(
|
|
'%s is accessing findNodeHandle inside its render(). ' +
|
|
'render() should be a pure function of props and state. It should ' +
|
|
'never access something that requires stale data from the previous ' +
|
|
'render, such as refs. Move this logic to componentDidMount and ' +
|
|
'componentDidUpdate instead.',
|
|
getComponentNameFromType(owner.type) || 'A component',
|
|
);
|
|
}
|
|
|
|
owner.stateNode._warnedAboutRefsInRender = true;
|
|
}
|
|
}
|
|
if (componentOrHandle == null) {
|
|
return null;
|
|
}
|
|
if (componentOrHandle._nativeTag) {
|
|
return componentOrHandle;
|
|
}
|
|
if (componentOrHandle.canonical && componentOrHandle.canonical._nativeTag) {
|
|
return componentOrHandle.canonical;
|
|
}
|
|
let hostInstance;
|
|
if (__DEV__) {
|
|
hostInstance = findHostInstanceWithWarning(
|
|
componentOrHandle,
|
|
'findHostInstance_DEPRECATED',
|
|
);
|
|
} else {
|
|
hostInstance = findHostInstance(componentOrHandle);
|
|
}
|
|
|
|
if (hostInstance == null) {
|
|
return hostInstance;
|
|
}
|
|
if ((hostInstance: any).canonical) {
|
|
// Fabric
|
|
return (hostInstance: any).canonical;
|
|
}
|
|
// $FlowFixMe[incompatible-return]
|
|
return hostInstance;
|
|
}
|
|
|
|
function findNodeHandle(componentOrHandle: any): ?number {
|
|
if (__DEV__) {
|
|
const owner = ReactCurrentOwner.current;
|
|
if (owner !== null && owner.stateNode !== null) {
|
|
if (!owner.stateNode._warnedAboutRefsInRender) {
|
|
console.error(
|
|
'%s is accessing findNodeHandle inside its render(). ' +
|
|
'render() should be a pure function of props and state. It should ' +
|
|
'never access something that requires stale data from the previous ' +
|
|
'render, such as refs. Move this logic to componentDidMount and ' +
|
|
'componentDidUpdate instead.',
|
|
getComponentNameFromType(owner.type) || 'A component',
|
|
);
|
|
}
|
|
|
|
owner.stateNode._warnedAboutRefsInRender = true;
|
|
}
|
|
}
|
|
if (componentOrHandle == null) {
|
|
return null;
|
|
}
|
|
if (typeof componentOrHandle === 'number') {
|
|
// Already a node handle
|
|
return componentOrHandle;
|
|
}
|
|
if (componentOrHandle._nativeTag) {
|
|
return componentOrHandle._nativeTag;
|
|
}
|
|
if (componentOrHandle.canonical && componentOrHandle.canonical._nativeTag) {
|
|
return componentOrHandle.canonical._nativeTag;
|
|
}
|
|
let hostInstance;
|
|
if (__DEV__) {
|
|
hostInstance = findHostInstanceWithWarning(
|
|
componentOrHandle,
|
|
'findNodeHandle',
|
|
);
|
|
} else {
|
|
hostInstance = findHostInstance(componentOrHandle);
|
|
}
|
|
|
|
if (hostInstance == null) {
|
|
return hostInstance;
|
|
}
|
|
if ((hostInstance: any).canonical) {
|
|
// Fabric
|
|
return (hostInstance: any).canonical._nativeTag;
|
|
}
|
|
return hostInstance._nativeTag;
|
|
}
|
|
|
|
function dispatchCommand(handle: any, command: string, args: Array<any>) {
|
|
if (handle._nativeTag == null) {
|
|
if (__DEV__) {
|
|
console.error(
|
|
"dispatchCommand was called with a ref that isn't a " +
|
|
'native component. Use React.forwardRef to get access to the underlying native component',
|
|
);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (handle._internalInstanceHandle != null) {
|
|
const {stateNode} = handle._internalInstanceHandle;
|
|
if (stateNode != null) {
|
|
nativeFabricUIManager.dispatchCommand(stateNode.node, command, args);
|
|
}
|
|
} else {
|
|
UIManager.dispatchViewManagerCommand(handle._nativeTag, command, args);
|
|
}
|
|
}
|
|
|
|
function sendAccessibilityEvent(handle: any, eventType: string) {
|
|
if (handle._nativeTag == null) {
|
|
if (__DEV__) {
|
|
console.error(
|
|
"sendAccessibilityEvent was called with a ref that isn't a " +
|
|
'native component. Use React.forwardRef to get access to the underlying native component',
|
|
);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (handle._internalInstanceHandle != null) {
|
|
const {stateNode} = handle._internalInstanceHandle;
|
|
if (stateNode != null) {
|
|
nativeFabricUIManager.sendAccessibilityEvent(stateNode.node, eventType);
|
|
}
|
|
} else {
|
|
legacySendAccessibilityEvent(handle._nativeTag, eventType);
|
|
}
|
|
}
|
|
|
|
function onRecoverableError(error) {
|
|
// TODO: Expose onRecoverableError option to userspace
|
|
// eslint-disable-next-line react-internal/no-production-logging, react-internal/warning-args
|
|
console.error(error);
|
|
}
|
|
|
|
function render(
|
|
element: Element<ElementType>,
|
|
containerTag: number,
|
|
callback: ?() => void,
|
|
): ?ElementRef<ElementType> {
|
|
let root = roots.get(containerTag);
|
|
|
|
if (!root) {
|
|
// TODO (bvaughn): If we decide to keep the wrapper component,
|
|
// We could create a wrapper for containerTag as well to reduce special casing.
|
|
root = createContainer(
|
|
containerTag,
|
|
LegacyRoot,
|
|
null,
|
|
false,
|
|
null,
|
|
'',
|
|
onRecoverableError,
|
|
null,
|
|
);
|
|
roots.set(containerTag, root);
|
|
}
|
|
updateContainer(element, root, null, callback);
|
|
|
|
// $FlowIssue Flow has hardcoded values for React DOM that don't work with RN
|
|
return getPublicRootInstance(root);
|
|
}
|
|
|
|
function unmountComponentAtNode(containerTag: number) {
|
|
const root = roots.get(containerTag);
|
|
if (root) {
|
|
// TODO: Is it safe to reset this now or should I wait since this unmount could be deferred?
|
|
updateContainer(null, root, null, () => {
|
|
roots.delete(containerTag);
|
|
});
|
|
}
|
|
}
|
|
|
|
function unmountComponentAtNodeAndRemoveContainer(containerTag: number) {
|
|
unmountComponentAtNode(containerTag);
|
|
|
|
// Call back into native to remove all of the subviews from this container
|
|
UIManager.removeRootView(containerTag);
|
|
}
|
|
|
|
function createPortal(
|
|
children: ReactNodeList,
|
|
containerTag: number,
|
|
key: ?string = null,
|
|
) {
|
|
return createPortalImpl(children, containerTag, null, key);
|
|
}
|
|
|
|
setBatchingImplementation(batchedUpdatesImpl, discreteUpdates);
|
|
|
|
function computeComponentStackForErrorReporting(reactTag: number): string {
|
|
const fiber = getClosestInstanceFromNode(reactTag);
|
|
if (!fiber) {
|
|
return '';
|
|
}
|
|
return getStackByFiberInDevAndProd(fiber);
|
|
}
|
|
|
|
const roots = new Map();
|
|
|
|
const Internals = {
|
|
computeComponentStackForErrorReporting,
|
|
};
|
|
|
|
export {
|
|
// This is needed for implementation details of TouchableNativeFeedback
|
|
// Remove this once TouchableNativeFeedback doesn't use cloneElement
|
|
findHostInstance_DEPRECATED,
|
|
findNodeHandle,
|
|
dispatchCommand,
|
|
sendAccessibilityEvent,
|
|
render,
|
|
unmountComponentAtNode,
|
|
unmountComponentAtNodeAndRemoveContainer,
|
|
createPortal,
|
|
batchedUpdates as unstable_batchedUpdates,
|
|
Internals as __SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED,
|
|
// This export is typically undefined in production builds.
|
|
// See the "enableGetInspectorDataForInstanceInProduction" flag.
|
|
getInspectorDataForInstance,
|
|
};
|
|
|
|
injectIntoDevTools({
|
|
findFiberByHostInstance: getClosestInstanceFromNode,
|
|
bundleType: __DEV__ ? 1 : 0,
|
|
version: ReactVersion,
|
|
rendererPackageName: 'react-native-renderer',
|
|
rendererConfig: {
|
|
getInspectorDataForViewTag: getInspectorDataForViewTag,
|
|
getInspectorDataForViewAtPoint: getInspectorDataForViewAtPoint.bind(
|
|
null,
|
|
findNodeHandle,
|
|
),
|
|
},
|
|
});
|