mirror of
https://github.com/facebook/react-native.git
synced 2025-11-01 09:14:26 +00:00
80ce3c7fe2
Summary: In development, we can specify a debug name to force the whole React tree to have a meaningful name. This is useful for debugging: React DevTools, component stack traces, etc. More context in D26637787 (https://github.com/facebook/react-native/commit/eeb36f470929c2fdd8e1ed69898a5ba9144b8715). This worked fine until now because `renderApplication` is never called more than once per React Native root. With pre-rendering, this won't be true as we'll be able to re-render the whole React tree with new props (after navigating to the screen). The problem with the current implementation is that we generate a new component every time `renderApplication` is called, which makes the React reconciliation consider them different components and discards the whole sub-tree. This fixes it by caching those components so we always return a component with the same identity (exactly the same function) when we call `renderApplication` repeatedly. This won't have any effect in production where these components aren't created at all. Changelog: [Internal] Reviewed By: ShikaSD Differential Revision: D28061384 fbshipit-source-id: 95833a5b74bb622896cb47426f74324d8740e5d7
77 lines
2.5 KiB
JavaScript
77 lines
2.5 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.
|
|
*
|
|
* @format
|
|
* @flow
|
|
*/
|
|
|
|
const AppContainer = require('./AppContainer');
|
|
import GlobalPerformanceLogger from '../Utilities/GlobalPerformanceLogger';
|
|
import type {IPerformanceLogger} from '../Utilities/createPerformanceLogger';
|
|
import PerformanceLoggerContext from '../Utilities/PerformanceLoggerContext';
|
|
import type {DisplayModeType} from './DisplayMode';
|
|
import getCachedComponentWithDebugName from './getCachedComponentWithDebugName';
|
|
const React = require('react');
|
|
|
|
const invariant = require('invariant');
|
|
|
|
// require BackHandler so it sets the default handler that exits the app if no listeners respond
|
|
require('../Utilities/BackHandler');
|
|
|
|
function renderApplication<Props: Object>(
|
|
RootComponent: React.ComponentType<Props>,
|
|
initialProps: Props,
|
|
rootTag: any,
|
|
WrapperComponent?: ?React.ComponentType<*>,
|
|
fabric?: boolean,
|
|
showArchitectureIndicator?: boolean,
|
|
scopedPerformanceLogger?: IPerformanceLogger,
|
|
isLogBox?: boolean,
|
|
debugName?: string,
|
|
displayMode?: ?DisplayModeType,
|
|
) {
|
|
invariant(rootTag, 'Expect to have a valid rootTag, instead got ', rootTag);
|
|
|
|
const performanceLogger = scopedPerformanceLogger ?? GlobalPerformanceLogger;
|
|
|
|
let renderable = (
|
|
<PerformanceLoggerContext.Provider value={performanceLogger}>
|
|
<AppContainer
|
|
rootTag={rootTag}
|
|
fabric={fabric}
|
|
showArchitectureIndicator={showArchitectureIndicator}
|
|
WrapperComponent={WrapperComponent}
|
|
initialProps={initialProps ?? Object.freeze({})}
|
|
internal_excludeLogBox={isLogBox}>
|
|
<RootComponent {...initialProps} rootTag={rootTag} />
|
|
</AppContainer>
|
|
</PerformanceLoggerContext.Provider>
|
|
);
|
|
|
|
if (__DEV__ && debugName) {
|
|
const RootComponentWithMeaningfulName = getCachedComponentWithDebugName(
|
|
`${debugName}(RootComponent)`,
|
|
);
|
|
renderable = (
|
|
<RootComponentWithMeaningfulName>
|
|
{renderable}
|
|
</RootComponentWithMeaningfulName>
|
|
);
|
|
}
|
|
|
|
performanceLogger.startTimespan('renderApplication_React_render');
|
|
performanceLogger.setExtra('usedReactFabric', fabric ? '1' : '0');
|
|
|
|
if (fabric) {
|
|
require('../Renderer/shims/ReactFabric').render(renderable, rootTag);
|
|
} else {
|
|
require('../Renderer/shims/ReactNative').render(renderable, rootTag);
|
|
}
|
|
performanceLogger.stopTimespan('renderApplication_React_render');
|
|
}
|
|
|
|
module.exports = renderApplication;
|