Skip to content

Commit

Permalink
Remove _ctor field from Lazy components (#18217)
Browse files Browse the repository at this point in the history
* This type is all wrong and nothing cares because it's all any

* Refine Flow types of Lazy Components

We can type each condition.

* Remove _ctor field from Lazy components

This field is not needed because it's only used before we've initialized,
and we don't have anything else to store before we've initialized.

* Check for _ctor in case it's an older isomorphic that created it

We try not to break across minors but it's no guarantee.

* Move types and constants from shared to isomorphic

The "react" package owns the data structure of the Lazy component. It
creates it and decides how any downstream renderer may use it.

* Move constants to shared

Apparently we can't depend on react/src/ because the whole package is
considered "external" as far as rollup is concerned.
  • Loading branch information
sebmarkbage authored Mar 5, 2020
1 parent 2fe0fbb commit fa03206
Show file tree
Hide file tree
Showing 6 changed files with 95 additions and 48 deletions.
10 changes: 3 additions & 7 deletions packages/react-dom/src/server/ReactPartialRenderer.js
Original file line number Diff line number Diff line change
Expand Up @@ -9,20 +9,16 @@

import type {ThreadID} from './ReactThreadIDAllocator';
import type {ReactElement} from 'shared/ReactElementType';
import type {LazyComponent} from 'shared/ReactLazyComponent';
import type {LazyComponent} from 'react/src/ReactLazy';
import type {ReactProvider, ReactContext} from 'shared/ReactTypes';

import * as React from 'react';
import invariant from 'shared/invariant';
import getComponentName from 'shared/getComponentName';
import describeComponentFrame from 'shared/describeComponentFrame';
import ReactSharedInternals from 'shared/ReactSharedInternals';
import {
Resolved,
Rejected,
Pending,
initializeLazyComponentType,
} from 'shared/ReactLazyComponent';
import {initializeLazyComponentType} from 'shared/ReactLazyComponent';
import {Resolved, Rejected, Pending} from 'shared/ReactLazyStatusTags';
import {
warnAboutDeprecatedLifecycles,
disableLegacyContext,
Expand Down
5 changes: 3 additions & 2 deletions packages/react-reconciler/src/ReactFiberLazyComponent.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,9 +7,10 @@
* @flow
*/

import type {LazyComponent} from 'shared/ReactLazyComponent';
import type {LazyComponent} from 'react/src/ReactLazy';

import {Resolved, initializeLazyComponentType} from 'shared/ReactLazyComponent';
import {Resolved} from 'shared/ReactLazyStatusTags';
import {initializeLazyComponentType} from 'shared/ReactLazyComponent';

export function resolveDefaultProps(Component: any, baseProps: Object): Object {
if (Component && Component.defaultProps) {
Expand Down
50 changes: 44 additions & 6 deletions packages/react/src/ReactLazy.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,25 +3,61 @@
*
* 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 {LazyComponent, Thenable} from 'shared/ReactLazyComponent';

import {REACT_LAZY_TYPE} from 'shared/ReactSymbols';

export function lazy<T, R>(ctor: () => Thenable<T, R>): LazyComponent<T> {
let lazyType = {
type Thenable<T, R> = {
then(resolve: (T) => mixed, reject: (mixed) => mixed): R,
};

export type UninitializedLazyComponent<T> = {
$$typeof: Symbol | number,
_status: -1,
_result: () => Thenable<{default: T, ...} | T, mixed>,
};

export type PendingLazyComponent<T> = {
$$typeof: Symbol | number,
_status: 0,
_result: Thenable<{default: T, ...} | T, mixed>,
};

export type ResolvedLazyComponent<T> = {
$$typeof: Symbol | number,
_status: 1,
_result: T,
};

export type RejectedLazyComponent = {
$$typeof: Symbol | number,
_status: 2,
_result: mixed,
};

export type LazyComponent<T> =
| UninitializedLazyComponent<T>
| PendingLazyComponent<T>
| ResolvedLazyComponent<T>
| RejectedLazyComponent;

export function lazy<T>(
ctor: () => Thenable<{default: T, ...} | T, mixed>,
): LazyComponent<T> {
let lazyType: LazyComponent<T> = {
$$typeof: REACT_LAZY_TYPE,
_ctor: ctor,
// React uses these fields to store the result.
_status: -1,
_result: null,
_result: ctor,
};

if (__DEV__) {
// In production, this would just set it on the object.
let defaultProps;
let propTypes;
// $FlowFixMe
Object.defineProperties(lazyType, {
defaultProps: {
configurable: true,
Expand All @@ -36,6 +72,7 @@ export function lazy<T, R>(ctor: () => Thenable<T, R>): LazyComponent<T> {
);
defaultProps = newDefaultProps;
// Match production behavior more closely:
// $FlowFixMe
Object.defineProperty(lazyType, 'defaultProps', {
enumerable: true,
});
Expand All @@ -54,6 +91,7 @@ export function lazy<T, R>(ctor: () => Thenable<T, R>): LazyComponent<T> {
);
propTypes = newPropTypes;
// Match production behavior more closely:
// $FlowFixMe
Object.defineProperty(lazyType, 'propTypes', {
enumerable: true,
});
Expand Down
62 changes: 30 additions & 32 deletions packages/shared/ReactLazyComponent.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,46 +7,40 @@
* @flow
*/

export type Thenable<T, R> = {
then(resolve: (T) => mixed, reject: (mixed) => mixed): R,
...
};
import type {
PendingLazyComponent,
ResolvedLazyComponent,
RejectedLazyComponent,
LazyComponent,
} from 'react/src/ReactLazy';

export type LazyComponent<T> = {
$$typeof: Symbol | number,
_ctor: () => Thenable<{default: T, ...}, mixed>,
_status: 0 | 1 | 2,
_result: any,
...
};

type ResolvedLazyComponent<T> = {
$$typeof: Symbol | number,
_ctor: () => Thenable<{default: T, ...}, mixed>,
_status: 1,
_result: any,
...
};

export const Uninitialized = -1;
export const Pending = 0;
export const Resolved = 1;
export const Rejected = 2;
import {
Uninitialized,
Pending,
Resolved,
Rejected,
} from './ReactLazyStatusTags';

export function refineResolvedLazyComponent<T>(
lazyComponent: LazyComponent<T>,
): ResolvedLazyComponent<T> | null {
): T | null {
return lazyComponent._status === Resolved ? lazyComponent._result : null;
}

export function initializeLazyComponentType(
lazyComponent: LazyComponent<any>,
): void {
if (lazyComponent._status === Uninitialized) {
lazyComponent._status = Pending;
const ctor = lazyComponent._ctor;
let ctor = lazyComponent._result;
if (!ctor) {
// TODO: Remove this later. THis only exists in case you use an older "react" package.
ctor = ((lazyComponent: any)._ctor: typeof ctor);
}
const thenable = ctor();
lazyComponent._result = thenable;
// Transition to the next state.
const pending: PendingLazyComponent<any> = (lazyComponent: any);
pending._status = Pending;
pending._result = thenable;
thenable.then(
moduleObject => {
if (lazyComponent._status === Pending) {
Expand All @@ -61,14 +55,18 @@ export function initializeLazyComponentType(
);
}
}
lazyComponent._status = Resolved;
lazyComponent._result = defaultExport;
// Transition to the next state.
const resolved: ResolvedLazyComponent<any> = (lazyComponent: any);
resolved._status = Resolved;
resolved._result = defaultExport;
}
},
error => {
if (lazyComponent._status === Pending) {
lazyComponent._status = Rejected;
lazyComponent._result = error;
// Transition to the next state.
const rejected: RejectedLazyComponent = (lazyComponent: any);
rejected._status = Rejected;
rejected._result = error;
}
},
);
Expand Down
14 changes: 14 additions & 0 deletions packages/shared/ReactLazyStatusTags.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
/**
* 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
*/

// TODO: Move this to "react" once we can import from externals.
export const Uninitialized = -1;
export const Pending = 0;
export const Resolved = 1;
export const Rejected = 2;
2 changes: 1 addition & 1 deletion packages/shared/getComponentName.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
* @flow
*/

import type {LazyComponent} from 'shared/ReactLazyComponent';
import type {LazyComponent} from 'react/src/ReactLazy';

import {
REACT_CONTEXT_TYPE,
Expand Down

0 comments on commit fa03206

Please sign in to comment.