2015-01-29 17:10:49 -08:00
|
|
|
/**
|
2015-03-23 15:07:33 -07:00
|
|
|
* 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.
|
2015-01-29 17:10:49 -08:00
|
|
|
*
|
|
|
|
* @providesModule ReactIOSDOMIDOperations
|
2015-03-25 18:19:28 -07:00
|
|
|
* @flow
|
2015-01-29 17:10:49 -08:00
|
|
|
*/
|
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
var ReactIOSTagHandles = require('ReactIOSTagHandles');
|
|
|
|
var ReactMultiChildUpdateTypes = require('ReactMultiChildUpdateTypes');
|
2015-03-18 15:57:49 -07:00
|
|
|
var RCTUIManager = require('NativeModules').UIManager;
|
2015-01-29 17:10:49 -08:00
|
|
|
var ReactPerf = require('ReactPerf');
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Updates a component's children by processing a series of updates.
|
|
|
|
* For each of the update/create commands, the `fromIndex` refers to the index
|
|
|
|
* that the item existed at *before* any of the updates are applied, and the
|
|
|
|
* `toIndex` refers to the index after *all* of the updates are applied
|
|
|
|
* (including deletes/moves). TODO: refactor so this can be shared with
|
|
|
|
* DOMChildrenOperations.
|
|
|
|
*
|
|
|
|
* @param {array<object>} updates List of update configurations.
|
|
|
|
* @param {array<string>} markup List of markup strings - in the case of React
|
|
|
|
* IOS, the ids of new components assumed to be already created.
|
|
|
|
*/
|
|
|
|
var dangerouslyProcessChildrenUpdates = function(childrenUpdates, markupList) {
|
|
|
|
if (!childrenUpdates.length) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
var byContainerTag = {};
|
|
|
|
// Group by parent ID - send them across the bridge in separate commands per
|
|
|
|
// containerID.
|
|
|
|
for (var i = 0; i < childrenUpdates.length; i++) {
|
|
|
|
var update = childrenUpdates[i];
|
|
|
|
var containerTag = ReactIOSTagHandles.mostRecentMountedNodeHandleForRootNodeID(update.parentID);
|
|
|
|
var updates = byContainerTag[containerTag] || (byContainerTag[containerTag] = {});
|
|
|
|
if (update.type === ReactMultiChildUpdateTypes.MOVE_EXISTING) {
|
|
|
|
(updates.moveFromIndices || (updates.moveFromIndices = [])).push(update.fromIndex);
|
|
|
|
(updates.moveToIndices || (updates.moveToIndices = [])).push(update.toIndex);
|
|
|
|
} else if (update.type === ReactMultiChildUpdateTypes.REMOVE_NODE) {
|
|
|
|
(updates.removeAtIndices || (updates.removeAtIndices = [])).push(update.fromIndex);
|
|
|
|
} else if (update.type === ReactMultiChildUpdateTypes.INSERT_MARKUP) {
|
|
|
|
var mountImage = markupList[update.markupIndex];
|
|
|
|
var tag = mountImage.tag;
|
|
|
|
var rootNodeID = mountImage.rootNodeID;
|
|
|
|
ReactIOSTagHandles.associateRootNodeIDWithMountedNodeHandle(rootNodeID, tag);
|
|
|
|
(updates.addAtIndices || (updates.addAtIndices = [])).push(update.toIndex);
|
|
|
|
(updates.addChildTags || (updates.addChildTags = [])).push(tag);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Note this enumeration order will be different on V8! Move `byContainerTag`
|
|
|
|
// to a sparse array as soon as we confirm there are not horrible perf
|
|
|
|
// penalties.
|
|
|
|
for (var updateParentTagString in byContainerTag) {
|
|
|
|
var updateParentTagNumber = +updateParentTagString;
|
|
|
|
var childUpdatesToSend = byContainerTag[updateParentTagNumber];
|
2015-03-17 13:42:44 -07:00
|
|
|
RCTUIManager.manageChildren(
|
2015-01-29 17:10:49 -08:00
|
|
|
updateParentTagNumber,
|
|
|
|
childUpdatesToSend.moveFromIndices,
|
|
|
|
childUpdatesToSend.moveToIndices,
|
|
|
|
childUpdatesToSend.addChildTags,
|
|
|
|
childUpdatesToSend.addAtIndices,
|
|
|
|
childUpdatesToSend.removeAtIndices
|
|
|
|
);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Operations used to process updates to DOM nodes. This is made injectable via
|
|
|
|
* `ReactComponent.DOMIDOperations`.
|
|
|
|
*/
|
|
|
|
var ReactIOSDOMIDOperations = {
|
|
|
|
dangerouslyProcessChildrenUpdates: ReactPerf.measure(
|
|
|
|
// FIXME(frantic): #4441289 Hack to avoid modifying react-tools
|
|
|
|
'ReactDOMIDOperations',
|
|
|
|
'dangerouslyProcessChildrenUpdates',
|
|
|
|
dangerouslyProcessChildrenUpdates
|
|
|
|
),
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Replaces a view that exists in the document with markup.
|
|
|
|
*
|
|
|
|
* @param {string} id ID of child to be replaced.
|
|
|
|
* @param {string} markup Mount image to replace child with id.
|
|
|
|
*/
|
|
|
|
dangerouslyReplaceNodeWithMarkupByID: ReactPerf.measure(
|
|
|
|
'ReactDOMIDOperations',
|
|
|
|
'dangerouslyReplaceNodeWithMarkupByID',
|
|
|
|
function(id, mountImage) {
|
|
|
|
var oldTag = ReactIOSTagHandles.mostRecentMountedNodeHandleForRootNodeID(id);
|
2015-03-17 13:42:44 -07:00
|
|
|
RCTUIManager.replaceExistingNonRootView(oldTag, mountImage.tag);
|
2015-01-29 17:10:49 -08:00
|
|
|
ReactIOSTagHandles.associateRootNodeIDWithMountedNodeHandle(id, mountImage.tag);
|
|
|
|
}
|
|
|
|
),
|
|
|
|
};
|
|
|
|
|
|
|
|
module.exports = ReactIOSDOMIDOperations;
|