mirror of
https://github.com/facebook/react.git
synced 2025-11-01 09:12:30 +00:00
313611572b
* Move files and tests to more meaningful places * Fix the build Now that we import reconciler via react-reconciler, I needed to make a few tweaks. * Update sizes * Move @preventMunge directive to FB header * Revert unintentional change * Fix Flow coverage I forgot to @flow-ify those files. This uncovered some issues. * Prettier, I love you but you're bringing me down Prettier, I love you but you're bringing me down Like a rat in a cage Pulling minimum wage Prettier, I love you but you're bringing me down Prettier, you're safer and you're wasting my time Our records all show you were filthy but fine But they shuttered your stores When you opened the doors To the cops who were bored once they'd run out of crime Prettier, you're perfect, oh, please don't change a thing Your mild billionaire mayor's now convinced he's a king So the boring collect I mean all disrespect In the neighborhood bars I'd once dreamt I would drink Prettier, I love you but you're freaking me out There's a ton of the twist but we're fresh out of shout Like a death in the hall That you hear through your wall Prettier, I love you but you're freaking me out Prettier, I love you but you're bringing me down Prettier, I love you but you're bringing me down Like a death of the heart Jesus, where do I start? But you're still the one pool where I'd happily drown And oh! Take me off your mailing list For kids who think it still exists Yes, for those who think it still exists Maybe I'm wrong and maybe you're right Maybe I'm wrong and maybe you're right Maybe you're right, maybe I'm wrong And just maybe you're right And oh! Maybe mother told you true And there'll always be somebody there for you And you'll never be alone But maybe she's wrong and maybe I'm right And just maybe she's wrong Maybe she's wrong and maybe I'm right And if so, here's this song!
396 lines
11 KiB
JavaScript
396 lines
11 KiB
JavaScript
/**
|
|
* Copyright (c) 2013-present, Facebook, Inc.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*
|
|
* @providesModule ReactNativeFiberRenderer
|
|
* @flow
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
const ReactFiberReconciler = require('react-reconciler');
|
|
const ReactNativeAttributePayload = require('ReactNativeAttributePayload');
|
|
const ReactNativeComponentTree = require('ReactNativeComponentTree');
|
|
const ReactNativeFiberHostComponent = require('ReactNativeFiberHostComponent');
|
|
const ReactNativeTagHandles = require('ReactNativeTagHandles');
|
|
const ReactNativeViewConfigRegistry = require('ReactNativeViewConfigRegistry');
|
|
const UIManager = require('UIManager');
|
|
|
|
const deepFreezeAndThrowOnMutationInDev = require('deepFreezeAndThrowOnMutationInDev');
|
|
const emptyObject = require('fbjs/lib/emptyObject');
|
|
const invariant = require('fbjs/lib/invariant');
|
|
|
|
import type {ReactNativeBaseComponentViewConfig} from 'ReactNativeTypes';
|
|
|
|
export type Container = number;
|
|
export type Instance = {
|
|
_children: Array<Instance | number>,
|
|
_nativeTag: number,
|
|
viewConfig: ReactNativeBaseComponentViewConfig,
|
|
};
|
|
export type Props = Object;
|
|
export type TextInstance = number;
|
|
|
|
const {
|
|
precacheFiberNode,
|
|
uncacheFiberNode,
|
|
updateFiberProps,
|
|
} = ReactNativeComponentTree;
|
|
|
|
function recursivelyUncacheFiberNode(node: Instance | TextInstance) {
|
|
if (typeof node === 'number') {
|
|
// Leaf node (eg text)
|
|
uncacheFiberNode(node);
|
|
} else {
|
|
uncacheFiberNode((node: any)._nativeTag);
|
|
|
|
(node: any)._children.forEach(recursivelyUncacheFiberNode);
|
|
}
|
|
}
|
|
|
|
const NativeRenderer = ReactFiberReconciler({
|
|
appendInitialChild(
|
|
parentInstance: Instance,
|
|
child: Instance | TextInstance,
|
|
): void {
|
|
parentInstance._children.push(child);
|
|
},
|
|
|
|
createInstance(
|
|
type: string,
|
|
props: Props,
|
|
rootContainerInstance: Container,
|
|
hostContext: {},
|
|
internalInstanceHandle: Object,
|
|
): Instance {
|
|
const tag = ReactNativeTagHandles.allocateTag();
|
|
const viewConfig = ReactNativeViewConfigRegistry.get(type);
|
|
|
|
if (__DEV__) {
|
|
for (const key in viewConfig.validAttributes) {
|
|
if (props.hasOwnProperty(key)) {
|
|
deepFreezeAndThrowOnMutationInDev(props[key]);
|
|
}
|
|
}
|
|
}
|
|
|
|
const updatePayload = ReactNativeAttributePayload.create(
|
|
props,
|
|
viewConfig.validAttributes,
|
|
);
|
|
|
|
UIManager.createView(
|
|
tag, // reactTag
|
|
viewConfig.uiViewClassName, // viewName
|
|
rootContainerInstance, // rootTag
|
|
updatePayload, // props
|
|
);
|
|
|
|
const component = new ReactNativeFiberHostComponent(tag, viewConfig);
|
|
|
|
precacheFiberNode(internalInstanceHandle, tag);
|
|
updateFiberProps(tag, props);
|
|
|
|
// Not sure how to avoid this cast. Flow is okay if the component is defined
|
|
// in the same file but if it's external it can't see the types.
|
|
return ((component: any): Instance);
|
|
},
|
|
|
|
createTextInstance(
|
|
text: string,
|
|
rootContainerInstance: Container,
|
|
hostContext: {},
|
|
internalInstanceHandle: Object,
|
|
): TextInstance {
|
|
const tag = ReactNativeTagHandles.allocateTag();
|
|
|
|
UIManager.createView(
|
|
tag, // reactTag
|
|
'RCTRawText', // viewName
|
|
rootContainerInstance, // rootTag
|
|
{text: text}, // props
|
|
);
|
|
|
|
precacheFiberNode(internalInstanceHandle, tag);
|
|
|
|
return tag;
|
|
},
|
|
|
|
finalizeInitialChildren(
|
|
parentInstance: Instance,
|
|
type: string,
|
|
props: Props,
|
|
rootContainerInstance: Container,
|
|
): boolean {
|
|
// Don't send a no-op message over the bridge.
|
|
if (parentInstance._children.length === 0) {
|
|
return false;
|
|
}
|
|
|
|
// Map from child objects to native tags.
|
|
// Either way we need to pass a copy of the Array to prevent it from being frozen.
|
|
const nativeTags = parentInstance._children.map(
|
|
child =>
|
|
(typeof child === 'number'
|
|
? child // Leaf node (eg text)
|
|
: child._nativeTag),
|
|
);
|
|
|
|
UIManager.setChildren(
|
|
parentInstance._nativeTag, // containerTag
|
|
nativeTags, // reactTags
|
|
);
|
|
|
|
return false;
|
|
},
|
|
|
|
getRootHostContext(): {} {
|
|
return emptyObject;
|
|
},
|
|
|
|
getChildHostContext(): {} {
|
|
return emptyObject;
|
|
},
|
|
|
|
getPublicInstance(instance) {
|
|
return instance;
|
|
},
|
|
|
|
prepareForCommit(): void {
|
|
// Noop
|
|
},
|
|
|
|
prepareUpdate(
|
|
instance: Instance,
|
|
type: string,
|
|
oldProps: Props,
|
|
newProps: Props,
|
|
rootContainerInstance: Container,
|
|
hostContext: {},
|
|
): null | Object {
|
|
return emptyObject;
|
|
},
|
|
|
|
resetAfterCommit(): void {
|
|
// Noop
|
|
},
|
|
|
|
shouldDeprioritizeSubtree(type: string, props: Props): boolean {
|
|
return false;
|
|
},
|
|
|
|
scheduleDeferredCallback: global.requestIdleCallback,
|
|
|
|
shouldSetTextContent(type: string, props: Props): boolean {
|
|
// TODO (bvaughn) Revisit this decision.
|
|
// Always returning false simplifies the createInstance() implementation,
|
|
// But creates an additional child Fiber for raw text children.
|
|
// No additional native views are created though.
|
|
// It's not clear to me which is better so I'm deferring for now.
|
|
// More context @ github.com/facebook/react/pull/8560#discussion_r92111303
|
|
return false;
|
|
},
|
|
|
|
useSyncScheduling: true,
|
|
|
|
now(): number {
|
|
// TODO: Enable expiration by implementing this method.
|
|
return 0;
|
|
},
|
|
|
|
mutation: {
|
|
appendChild(
|
|
parentInstance: Instance,
|
|
child: Instance | TextInstance,
|
|
): void {
|
|
const childTag = typeof child === 'number' ? child : child._nativeTag;
|
|
const children = parentInstance._children;
|
|
const index = children.indexOf(child);
|
|
|
|
if (index >= 0) {
|
|
children.splice(index, 1);
|
|
children.push(child);
|
|
|
|
UIManager.manageChildren(
|
|
parentInstance._nativeTag, // containerTag
|
|
[index], // moveFromIndices
|
|
[children.length - 1], // moveToIndices
|
|
[], // addChildReactTags
|
|
[], // addAtIndices
|
|
[], // removeAtIndices
|
|
);
|
|
} else {
|
|
children.push(child);
|
|
|
|
UIManager.manageChildren(
|
|
parentInstance._nativeTag, // containerTag
|
|
[], // moveFromIndices
|
|
[], // moveToIndices
|
|
[childTag], // addChildReactTags
|
|
[children.length - 1], // addAtIndices
|
|
[], // removeAtIndices
|
|
);
|
|
}
|
|
},
|
|
|
|
appendChildToContainer(
|
|
parentInstance: Container,
|
|
child: Instance | TextInstance,
|
|
): void {
|
|
const childTag = typeof child === 'number' ? child : child._nativeTag;
|
|
UIManager.setChildren(
|
|
parentInstance, // containerTag
|
|
[childTag], // reactTags
|
|
);
|
|
},
|
|
|
|
commitTextUpdate(
|
|
textInstance: TextInstance,
|
|
oldText: string,
|
|
newText: string,
|
|
): void {
|
|
UIManager.updateView(
|
|
textInstance, // reactTag
|
|
'RCTRawText', // viewName
|
|
{text: newText}, // props
|
|
);
|
|
},
|
|
|
|
commitMount(
|
|
instance: Instance,
|
|
type: string,
|
|
newProps: Props,
|
|
internalInstanceHandle: Object,
|
|
): void {
|
|
// Noop
|
|
},
|
|
|
|
commitUpdate(
|
|
instance: Instance,
|
|
updatePayloadTODO: Object,
|
|
type: string,
|
|
oldProps: Props,
|
|
newProps: Props,
|
|
internalInstanceHandle: Object,
|
|
): void {
|
|
const viewConfig = instance.viewConfig;
|
|
|
|
updateFiberProps(instance._nativeTag, newProps);
|
|
|
|
const updatePayload = ReactNativeAttributePayload.diff(
|
|
oldProps,
|
|
newProps,
|
|
viewConfig.validAttributes,
|
|
);
|
|
|
|
// Avoid the overhead of bridge calls if there's no update.
|
|
// This is an expensive no-op for Android, and causes an unnecessary
|
|
// view invalidation for certain components (eg RCTTextInput) on iOS.
|
|
if (updatePayload != null) {
|
|
UIManager.updateView(
|
|
instance._nativeTag, // reactTag
|
|
viewConfig.uiViewClassName, // viewName
|
|
updatePayload, // props
|
|
);
|
|
}
|
|
},
|
|
|
|
insertBefore(
|
|
parentInstance: Instance,
|
|
child: Instance | TextInstance,
|
|
beforeChild: Instance | TextInstance,
|
|
): void {
|
|
const children = (parentInstance: any)._children;
|
|
const index = children.indexOf(child);
|
|
|
|
// Move existing child or add new child?
|
|
if (index >= 0) {
|
|
children.splice(index, 1);
|
|
const beforeChildIndex = children.indexOf(beforeChild);
|
|
children.splice(beforeChildIndex, 0, child);
|
|
|
|
UIManager.manageChildren(
|
|
(parentInstance: any)._nativeTag, // containerID
|
|
[index], // moveFromIndices
|
|
[beforeChildIndex], // moveToIndices
|
|
[], // addChildReactTags
|
|
[], // addAtIndices
|
|
[], // removeAtIndices
|
|
);
|
|
} else {
|
|
const beforeChildIndex = children.indexOf(beforeChild);
|
|
children.splice(beforeChildIndex, 0, child);
|
|
|
|
const childTag = typeof child === 'number' ? child : child._nativeTag;
|
|
|
|
UIManager.manageChildren(
|
|
(parentInstance: any)._nativeTag, // containerID
|
|
[], // moveFromIndices
|
|
[], // moveToIndices
|
|
[childTag], // addChildReactTags
|
|
[beforeChildIndex], // addAtIndices
|
|
[], // removeAtIndices
|
|
);
|
|
}
|
|
},
|
|
|
|
insertInContainerBefore(
|
|
parentInstance: Container,
|
|
child: Instance | TextInstance,
|
|
beforeChild: Instance | TextInstance,
|
|
): void {
|
|
// TODO (bvaughn): Remove this check when...
|
|
// We create a wrapper object for the container in ReactNative render()
|
|
// Or we refactor to remove wrapper objects entirely.
|
|
// For more info on pros/cons see PR #8560 description.
|
|
invariant(
|
|
typeof parentInstance !== 'number',
|
|
'Container does not support insertBefore operation',
|
|
);
|
|
},
|
|
|
|
removeChild(
|
|
parentInstance: Instance,
|
|
child: Instance | TextInstance,
|
|
): void {
|
|
recursivelyUncacheFiberNode(child);
|
|
const children = parentInstance._children;
|
|
const index = children.indexOf(child);
|
|
|
|
children.splice(index, 1);
|
|
|
|
UIManager.manageChildren(
|
|
parentInstance._nativeTag, // containerID
|
|
[], // moveFromIndices
|
|
[], // moveToIndices
|
|
[], // addChildReactTags
|
|
[], // addAtIndices
|
|
[index], // removeAtIndices
|
|
);
|
|
},
|
|
|
|
removeChildFromContainer(
|
|
parentInstance: Container,
|
|
child: Instance | TextInstance,
|
|
): void {
|
|
recursivelyUncacheFiberNode(child);
|
|
UIManager.manageChildren(
|
|
parentInstance, // containerID
|
|
[], // moveFromIndices
|
|
[], // moveToIndices
|
|
[], // addChildReactTags
|
|
[], // addAtIndices
|
|
[0], // removeAtIndices
|
|
);
|
|
},
|
|
|
|
resetTextContent(instance: Instance): void {
|
|
// Noop
|
|
},
|
|
},
|
|
});
|
|
|
|
module.exports = NativeRenderer;
|