mirror of
https://github.com/status-im/react-native.git
synced 2025-01-18 05:23:26 +00:00
20a8673b48
Summary: @public There are some race conditions between VM objects getting deallocated and the instanceHandle held by the eventEmitter can point to deallocated memory space, causing undefined behavior like a crash. For now, keep a strong ref to the eventTarget inside EventEmitter to avoid that scenario. This is a temporary workaround. Reviewed By: shergin Differential Revision: D8576785 fbshipit-source-id: 87ef36f716270ceca906b32bb86e0046ceaca19e
261 lines
11 KiB
C++
261 lines
11 KiB
C++
/**
|
|
* Copyright (c) 2015-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.
|
|
*/
|
|
|
|
#include "FabricUIManager.h"
|
|
|
|
#include <glog/logging.h>
|
|
|
|
#include <fabric/core/componentDescriptor.h>
|
|
#include <fabric/core/LayoutContext.h>
|
|
#include <fabric/debug/DebugStringConvertible.h>
|
|
#include <fabric/debug/DebugStringConvertibleItem.h>
|
|
#include <fabric/view/ViewComponentDescriptor.h>
|
|
#include <fabric/view/ViewProps.h>
|
|
#include <fabric/view/ViewShadowNode.h>
|
|
|
|
namespace facebook {
|
|
namespace react {
|
|
|
|
// TODO: Kill this flag and remove debug logging.
|
|
const bool isLoggingEnabled = false;
|
|
|
|
static const RawProps rawPropsFromDynamic(const folly::dynamic object) {
|
|
// TODO: Convert this to something smarter, probably returning `std::iterator`.
|
|
RawProps result;
|
|
|
|
if (object.isNull()) {
|
|
return result;
|
|
}
|
|
|
|
assert(object.isObject());
|
|
|
|
for (const auto &pair : object.items()) {
|
|
assert(pair.first.isString());
|
|
result[pair.first.asString()] = pair.second;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
static const std::string componentNameByReactViewName(std::string viewName) {
|
|
// We need this function only for the transition period;
|
|
// eventually, all names will be unified.
|
|
|
|
std::string rctPrefix("RCT");
|
|
if (std::mismatch(rctPrefix.begin(), rctPrefix.end(), viewName.begin()).first == rctPrefix.end()) {
|
|
// If `viewName` has "RCT" prefix, remove it.
|
|
viewName.erase(0, rctPrefix.length());
|
|
}
|
|
|
|
// Fabric uses slightly new names for Text components because of differences
|
|
// in semantic.
|
|
if (viewName == "Text") {
|
|
return "Paragraph";
|
|
}
|
|
if (viewName == "VirtualText") {
|
|
return "Text";
|
|
}
|
|
|
|
// We need this temporarly for testing purposes until we have proper
|
|
// implementation of core components: <Image>, <ScrollContentView>.
|
|
if (
|
|
viewName == "ImageView" ||
|
|
viewName == "ScrollContentView"
|
|
) {
|
|
return "View";
|
|
}
|
|
|
|
return viewName;
|
|
}
|
|
|
|
FabricUIManager::FabricUIManager(SharedComponentDescriptorRegistry componentDescriptorRegistry) {
|
|
componentDescriptorRegistry_ = componentDescriptorRegistry;
|
|
}
|
|
|
|
FabricUIManager::~FabricUIManager() {
|
|
if (eventHandler_) {
|
|
releaseEventHandlerFunction_(eventHandler_);
|
|
}
|
|
}
|
|
|
|
void FabricUIManager::setDelegate(UIManagerDelegate *delegate) {
|
|
delegate_ = delegate;
|
|
}
|
|
|
|
UIManagerDelegate *FabricUIManager::getDelegate() {
|
|
return delegate_;
|
|
}
|
|
|
|
void FabricUIManager::setCreateEventTargetFunction(std::function<CreateEventTargetFunction> createEventTargetFunction) {
|
|
createEventTargetFunction_ = createEventTargetFunction;
|
|
}
|
|
|
|
void FabricUIManager::setDispatchEventFunction(std::function<DispatchEventFunction> dispatchEventFunction) {
|
|
dispatchEventFunction_ = dispatchEventFunction;
|
|
}
|
|
|
|
void FabricUIManager::setReleaseEventHandlerFunction(std::function<ReleaseEventHandlerFunction> releaseEventHandlerFunction) {
|
|
releaseEventHandlerFunction_ = releaseEventHandlerFunction;
|
|
}
|
|
|
|
void FabricUIManager::setReleaseEventTargetFunction(std::function<ReleaseEventTargetFunction> releaseEventTargetFunction) {
|
|
releaseEventTargetFunction_ = releaseEventTargetFunction;
|
|
}
|
|
|
|
EventTarget FabricUIManager::createEventTarget(const InstanceHandle &instanceHandle) const {
|
|
return createEventTargetFunction_(instanceHandle);
|
|
}
|
|
|
|
void FabricUIManager::dispatchEvent(const EventTarget &eventTarget, const std::string &type, const folly::dynamic &payload) const {
|
|
dispatchEventFunction_(
|
|
eventHandler_,
|
|
eventTarget,
|
|
const_cast<std::string &>(type),
|
|
const_cast<folly::dynamic &>(payload)
|
|
);
|
|
}
|
|
|
|
void FabricUIManager::releaseEventTarget(const EventTarget &eventTarget) const {
|
|
releaseEventTargetFunction_(eventTarget);
|
|
}
|
|
|
|
SharedShadowNode FabricUIManager::createNode(int tag, std::string viewName, int rootTag, folly::dynamic props, InstanceHandle instanceHandle) {
|
|
isLoggingEnabled && LOG(INFO) << "FabricUIManager::createNode(tag: " << tag << ", name: " << viewName << ", rootTag: " << rootTag << ", props: " << props << ")";
|
|
|
|
ComponentName componentName = componentNameByReactViewName(viewName);
|
|
const SharedComponentDescriptor &componentDescriptor = (*componentDescriptorRegistry_)[componentName];
|
|
RawProps rawProps = rawPropsFromDynamic(props);
|
|
|
|
SharedShadowNode shadowNode =
|
|
componentDescriptor->createShadowNode(
|
|
tag,
|
|
rootTag,
|
|
componentDescriptor->createEventEmitter(instanceHandle, tag),
|
|
componentDescriptor->cloneProps(nullptr, rawProps)
|
|
);
|
|
|
|
isLoggingEnabled && LOG(INFO) << "FabricUIManager::createNode() -> " << shadowNode->getDebugDescription(DebugStringConvertibleOptions {.format = false});
|
|
|
|
if (delegate_) {
|
|
delegate_->uiManagerDidCreateShadowNode(shadowNode);
|
|
}
|
|
|
|
return shadowNode;
|
|
}
|
|
|
|
SharedShadowNode FabricUIManager::cloneNode(const SharedShadowNode &shadowNode, InstanceHandle instanceHandle) {
|
|
isLoggingEnabled && LOG(INFO) << "FabricUIManager::cloneNode(shadowNode: " << shadowNode->getDebugDescription(DebugStringConvertibleOptions {.format = false}) << ")";
|
|
const SharedComponentDescriptor &componentDescriptor = (*componentDescriptorRegistry_)[shadowNode];
|
|
|
|
SharedShadowNode clonedShadowNode =
|
|
componentDescriptor->cloneShadowNode(
|
|
shadowNode,
|
|
nullptr,
|
|
componentDescriptor->createEventEmitter(instanceHandle, shadowNode->getTag()),
|
|
nullptr
|
|
);
|
|
|
|
isLoggingEnabled && LOG(INFO) << "FabricUIManager::cloneNode() -> " << clonedShadowNode->getDebugDescription(DebugStringConvertibleOptions {.format = false});
|
|
return clonedShadowNode;
|
|
}
|
|
|
|
SharedShadowNode FabricUIManager::cloneNodeWithNewChildren(const SharedShadowNode &shadowNode, InstanceHandle instanceHandle) {
|
|
isLoggingEnabled && LOG(INFO) << "FabricUIManager::cloneNodeWithNewChildren(shadowNode: " << shadowNode->getDebugDescription(DebugStringConvertibleOptions {.format = false}) << ")";
|
|
// Assuming semantic: Cloning with same props but empty children.
|
|
const SharedComponentDescriptor &componentDescriptor = (*componentDescriptorRegistry_)[shadowNode];
|
|
|
|
SharedShadowNode clonedShadowNode =
|
|
componentDescriptor->cloneShadowNode(
|
|
shadowNode,
|
|
nullptr,
|
|
componentDescriptor->createEventEmitter(instanceHandle, shadowNode->getTag()),
|
|
ShadowNode::emptySharedShadowNodeSharedList()
|
|
);
|
|
|
|
isLoggingEnabled && LOG(INFO) << "FabricUIManager::cloneNodeWithNewChildren() -> " << clonedShadowNode->getDebugDescription(DebugStringConvertibleOptions {.format = false});
|
|
return clonedShadowNode;
|
|
}
|
|
|
|
SharedShadowNode FabricUIManager::cloneNodeWithNewProps(const SharedShadowNode &shadowNode, folly::dynamic props, InstanceHandle instanceHandle) {
|
|
isLoggingEnabled && LOG(INFO) << "FabricUIManager::cloneNodeWithNewProps(shadowNode: " << shadowNode->getDebugDescription(DebugStringConvertibleOptions {.format = false}) << ", props: " << props << ")";
|
|
// Assuming semantic: Cloning with same children and specified props.
|
|
const SharedComponentDescriptor &componentDescriptor = (*componentDescriptorRegistry_)[shadowNode];
|
|
RawProps rawProps = rawPropsFromDynamic(props);
|
|
|
|
SharedShadowNode clonedShadowNode =
|
|
componentDescriptor->cloneShadowNode(
|
|
shadowNode,
|
|
componentDescriptor->cloneProps(shadowNode->getProps(), rawProps),
|
|
componentDescriptor->createEventEmitter(instanceHandle, shadowNode->getTag()),
|
|
nullptr
|
|
);
|
|
|
|
isLoggingEnabled && LOG(INFO) << "FabricUIManager::cloneNodeWithNewProps() -> " << clonedShadowNode->getDebugDescription(DebugStringConvertibleOptions {.format = false});
|
|
return clonedShadowNode;
|
|
}
|
|
|
|
SharedShadowNode FabricUIManager::cloneNodeWithNewChildrenAndProps(const SharedShadowNode &shadowNode, folly::dynamic props, InstanceHandle instanceHandle) {
|
|
isLoggingEnabled && LOG(INFO) << "FabricUIManager::cloneNodeWithNewChildrenAndProps(shadowNode: " << shadowNode->getDebugDescription(DebugStringConvertibleOptions {.format = false}) << ", props: " << props << ")";
|
|
// Assuming semantic: Cloning with empty children and specified props.
|
|
const SharedComponentDescriptor &componentDescriptor = (*componentDescriptorRegistry_)[shadowNode];
|
|
RawProps rawProps = rawPropsFromDynamic(props);
|
|
|
|
SharedShadowNode clonedShadowNode =
|
|
componentDescriptor->cloneShadowNode(
|
|
shadowNode,
|
|
componentDescriptor->cloneProps(shadowNode->getProps(), rawProps),
|
|
componentDescriptor->createEventEmitter(instanceHandle, shadowNode->getTag()),
|
|
ShadowNode::emptySharedShadowNodeSharedList()
|
|
);
|
|
|
|
isLoggingEnabled && LOG(INFO) << "FabricUIManager::cloneNodeWithNewChildrenAndProps() -> " << clonedShadowNode->getDebugDescription(DebugStringConvertibleOptions {.format = false});
|
|
return clonedShadowNode;
|
|
}
|
|
|
|
void FabricUIManager::appendChild(const SharedShadowNode &parentShadowNode, const SharedShadowNode &childShadowNode) {
|
|
isLoggingEnabled && LOG(INFO) << "FabricUIManager::appendChild(parentShadowNode: " << parentShadowNode->getDebugDescription(DebugStringConvertibleOptions {.format = false}) << ", childShadowNode: " << childShadowNode->getDebugDescription(DebugStringConvertibleOptions {.format = false}) << ")";
|
|
const SharedComponentDescriptor &componentDescriptor = (*componentDescriptorRegistry_)[parentShadowNode];
|
|
|
|
// TODO: Remove this after we move this to JS side.
|
|
if (childShadowNode->getSealed()) {
|
|
auto childComponentDescriptor = (*componentDescriptorRegistry_)[childShadowNode];
|
|
auto clonedChildShadowNode = childComponentDescriptor->cloneShadowNode(childShadowNode);
|
|
auto nonConstClonedChildShadowNode = std::const_pointer_cast<ShadowNode>(clonedChildShadowNode);
|
|
nonConstClonedChildShadowNode->shallowSourceNode();
|
|
componentDescriptor->appendChild(parentShadowNode, clonedChildShadowNode);
|
|
return;
|
|
}
|
|
|
|
componentDescriptor->appendChild(parentShadowNode, childShadowNode);
|
|
}
|
|
|
|
SharedShadowNodeUnsharedList FabricUIManager::createChildSet(int rootTag) {
|
|
isLoggingEnabled && LOG(INFO) << "FabricUIManager::createChildSet(rootTag: " << rootTag << ")";
|
|
return std::make_shared<SharedShadowNodeList>(SharedShadowNodeList({}));
|
|
}
|
|
|
|
void FabricUIManager::appendChildToSet(const SharedShadowNodeUnsharedList &shadowNodeList, const SharedShadowNode &shadowNode) {
|
|
isLoggingEnabled && LOG(INFO) << "FabricUIManager::appendChildToSet(shadowNodeList: " << shadowNodeList << ", shadowNode: " << shadowNode->getDebugDescription(DebugStringConvertibleOptions {.format = false}) << ")";
|
|
shadowNodeList->push_back(shadowNode);
|
|
}
|
|
|
|
void FabricUIManager::completeRoot(int rootTag, const SharedShadowNodeUnsharedList &children) {
|
|
isLoggingEnabled && LOG(INFO) << "FabricUIManager::completeRoot(rootTag: " << rootTag << ", shadowNodeList: " << children << ")";
|
|
|
|
if (delegate_) {
|
|
delegate_->uiManagerDidFinishTransaction(rootTag, children);
|
|
}
|
|
}
|
|
|
|
void FabricUIManager::registerEventHandler(const EventHandler &eventHandler) {
|
|
isLoggingEnabled && LOG(INFO) << "FabricUIManager::registerEventHandler(eventHandler: " << eventHandler << ")";
|
|
eventHandler_ = eventHandler;
|
|
}
|
|
|
|
} // namespace react
|
|
} // namespace facebook
|