mirror of
https://github.com/zhigang1992/react-native.git
synced 2026-02-06 22:44:22 +08:00
Summary: public We're seeing related crashes. The diff has no tests, the perf tests weren't conclusive, and the person who'd be supporting it no longer is available to work on it. We can try this again later in a less rushed manner with proper perf testing. Reviewed By: davidaurelio Differential Revision: D2696615 fb-gh-sync-id: 3b6814ac12af19516146d5c42d2add8321b10db5
252 lines
7.6 KiB
JavaScript
252 lines
7.6 KiB
JavaScript
/**
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This source code is licensed under the BSD-style license found in the
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
*
|
|
* @providesModule ReactNativeBaseComponent
|
|
* @flow
|
|
*/
|
|
'use strict';
|
|
|
|
var NativeMethodsMixin = require('NativeMethodsMixin');
|
|
var ReactNativeAttributePayload = require('ReactNativeAttributePayload');
|
|
var ReactNativeEventEmitter = require('ReactNativeEventEmitter');
|
|
var ReactNativeStyleAttributes = require('ReactNativeStyleAttributes');
|
|
var ReactNativeTagHandles = require('ReactNativeTagHandles');
|
|
var ReactMultiChild = require('ReactMultiChild');
|
|
var RCTUIManager = require('NativeModules').UIManager;
|
|
|
|
var deepFreezeAndThrowOnMutationInDev = require('deepFreezeAndThrowOnMutationInDev');
|
|
var warning = require('warning');
|
|
|
|
var registrationNames = ReactNativeEventEmitter.registrationNames;
|
|
var putListener = ReactNativeEventEmitter.putListener;
|
|
var deleteListener = ReactNativeEventEmitter.deleteListener;
|
|
var deleteAllListeners = ReactNativeEventEmitter.deleteAllListeners;
|
|
|
|
type ReactNativeBaseComponentViewConfig = {
|
|
validAttributes: Object;
|
|
uiViewClassName: string;
|
|
}
|
|
|
|
// require('UIManagerStatTracker').install(); // uncomment to enable
|
|
|
|
/**
|
|
* @constructor ReactNativeBaseComponent
|
|
* @extends ReactComponent
|
|
* @extends ReactMultiChild
|
|
* @param {!object} UIKit View Configuration.
|
|
*/
|
|
var ReactNativeBaseComponent = function(
|
|
viewConfig: ReactNativeBaseComponentViewConfig
|
|
) {
|
|
this.viewConfig = viewConfig;
|
|
};
|
|
|
|
/**
|
|
* Generates and caches arrays of the form:
|
|
*
|
|
* [0, 1, 2, 3]
|
|
* [0, 1, 2, 3, 4]
|
|
* [0, 1]
|
|
*
|
|
* @param {number} size Size of array to generate.
|
|
* @return {Array<number>} Array with values that mirror the index.
|
|
*/
|
|
var cachedIndexArray = function(size) {
|
|
var cachedResult = cachedIndexArray._cache[size];
|
|
if (!cachedResult) {
|
|
var arr = [];
|
|
for (var i = 0; i < size; i++) {
|
|
arr[i] = i;
|
|
}
|
|
cachedIndexArray._cache[size] = arr;
|
|
return arr;
|
|
} else {
|
|
return cachedResult;
|
|
}
|
|
};
|
|
cachedIndexArray._cache = {};
|
|
|
|
/**
|
|
* Mixin for containers that contain UIViews. NOTE: markup is rendered markup
|
|
* which is a `viewID` ... see the return value for `mountComponent` !
|
|
*/
|
|
ReactNativeBaseComponent.Mixin = {
|
|
getPublicInstance: function() {
|
|
// TODO: This should probably use a composite wrapper
|
|
return this;
|
|
},
|
|
|
|
construct: function(element) {
|
|
this._currentElement = element;
|
|
},
|
|
|
|
unmountComponent: function() {
|
|
deleteAllListeners(this._rootNodeID);
|
|
this.unmountChildren();
|
|
this._rootNodeID = null;
|
|
},
|
|
|
|
/**
|
|
* Every native component is responsible for allocating its own `tag`, and
|
|
* issuing the native `createView` command. But it is not responsible for
|
|
* recording the fact that its own `rootNodeID` is associated with a
|
|
* `nodeHandle`. Only the code that actually adds its `nodeHandle` (`tag`) as
|
|
* a child of a container can confidently record that in
|
|
* `ReactNativeTagHandles`.
|
|
*/
|
|
initializeChildren: function(children, containerTag, transaction, context) {
|
|
var mountImages = this.mountChildren(children, transaction, context);
|
|
// In a well balanced tree, half of the nodes are in the bottom row and have
|
|
// no children - let's avoid calling out to the native bridge for a large
|
|
// portion of the children.
|
|
if (mountImages.length) {
|
|
var indexes = cachedIndexArray(mountImages.length);
|
|
// TODO: Pool these per platform view class. Reusing the `mountImages`
|
|
// array would likely be a jit deopt.
|
|
var createdTags = [];
|
|
for (var i = 0; i < mountImages.length; i++) {
|
|
var mountImage = mountImages[i];
|
|
var childTag = mountImage.tag;
|
|
var childID = mountImage.rootNodeID;
|
|
warning(
|
|
mountImage && mountImage.rootNodeID && mountImage.tag,
|
|
'Mount image returned does not have required data'
|
|
);
|
|
ReactNativeTagHandles.associateRootNodeIDWithMountedNodeHandle(
|
|
childID,
|
|
childTag
|
|
);
|
|
createdTags[i] = mountImage.tag;
|
|
}
|
|
RCTUIManager
|
|
.manageChildren(containerTag, null, null, createdTags, indexes, null);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Updates the component's currently mounted representation.
|
|
*
|
|
* @param {object} nextElement
|
|
* @param {ReactReconcileTransaction} transaction
|
|
* @param {object} context
|
|
* @internal
|
|
*/
|
|
receiveComponent: function(nextElement, transaction, context) {
|
|
var prevElement = this._currentElement;
|
|
this._currentElement = nextElement;
|
|
|
|
if (__DEV__) {
|
|
deepFreezeAndThrowOnMutationInDev(this._currentElement.props);
|
|
}
|
|
|
|
var updatePayload = ReactNativeAttributePayload.diff(
|
|
prevElement.props,
|
|
nextElement.props,
|
|
this.viewConfig.validAttributes
|
|
);
|
|
|
|
if (updatePayload) {
|
|
RCTUIManager.updateView(
|
|
ReactNativeTagHandles.mostRecentMountedNodeHandleForRootNodeID(this._rootNodeID),
|
|
this.viewConfig.uiViewClassName,
|
|
updatePayload
|
|
);
|
|
}
|
|
|
|
this._reconcileListenersUponUpdate(
|
|
prevElement.props,
|
|
nextElement.props
|
|
);
|
|
this.updateChildren(nextElement.props.children, transaction, context);
|
|
},
|
|
|
|
/**
|
|
* @param {object} initialProps Native component props.
|
|
*/
|
|
_registerListenersUponCreation: function(initialProps) {
|
|
for (var key in initialProps) {
|
|
// NOTE: The check for `!props[key]`, is only possible because this method
|
|
// registers listeners the *first* time a component is created.
|
|
if (registrationNames[key] && initialProps[key]) {
|
|
var listener = initialProps[key];
|
|
putListener(this._rootNodeID, key, listener);
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Reconciles event listeners, adding or removing if necessary.
|
|
* @param {object} prevProps Native component props including events.
|
|
* @param {object} nextProps Next native component props including events.
|
|
*/
|
|
_reconcileListenersUponUpdate: function(prevProps, nextProps) {
|
|
for (var key in nextProps) {
|
|
if (registrationNames[key] && (nextProps[key] !== prevProps[key])) {
|
|
if (nextProps[key]) {
|
|
putListener(this._rootNodeID, key, nextProps[key]);
|
|
} else {
|
|
deleteListener(this._rootNodeID, key);
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* @param {string} rootID Root ID of this subtree.
|
|
* @param {Transaction} transaction For creating/updating.
|
|
* @return {string} Unique iOS view tag.
|
|
*/
|
|
mountComponent: function(rootID, transaction, context) {
|
|
this._rootNodeID = rootID;
|
|
|
|
var tag = ReactNativeTagHandles.allocateTag();
|
|
|
|
if (__DEV__) {
|
|
deepFreezeAndThrowOnMutationInDev(this._currentElement.props);
|
|
}
|
|
|
|
var updatePayload = ReactNativeAttributePayload.create(
|
|
this._currentElement.props,
|
|
this.viewConfig.validAttributes
|
|
);
|
|
|
|
var nativeTopRootID = ReactNativeTagHandles.getNativeTopRootIDFromNodeID(rootID);
|
|
RCTUIManager.createView(
|
|
tag,
|
|
this.viewConfig.uiViewClassName,
|
|
nativeTopRootID ? ReactNativeTagHandles.rootNodeIDToTag[nativeTopRootID] : null,
|
|
updatePayload
|
|
);
|
|
|
|
this._registerListenersUponCreation(this._currentElement.props);
|
|
this.initializeChildren(
|
|
this._currentElement.props.children,
|
|
tag,
|
|
transaction,
|
|
context
|
|
);
|
|
return {
|
|
rootNodeID: rootID,
|
|
tag: tag
|
|
};
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Order of mixins is important. ReactNativeBaseComponent overrides methods in
|
|
* ReactMultiChild.
|
|
*/
|
|
Object.assign(
|
|
ReactNativeBaseComponent.prototype,
|
|
ReactMultiChild.Mixin,
|
|
ReactNativeBaseComponent.Mixin,
|
|
NativeMethodsMixin
|
|
);
|
|
|
|
module.exports = ReactNativeBaseComponent;
|