mirror of
https://github.com/zhigang1992/react-native.git
synced 2026-05-13 12:26:45 +08:00
Summary: Put `prepareForRecycle` to last before enqueue to recycle pool, ensure only call it when count lower than RCTComponentViewRegistryRecyclePoolMaxSize. cc. shergin . [General] [Changed] - Reorder prepareForRecycle before adding recycle pool Pull Request resolved: https://github.com/facebook/react-native/pull/24025 Differential Revision: D14536843 Pulled By: shergin fbshipit-source-id: 82a816e2c0afb5a6bb72637d7d55d6a4fda708af
229 lines
7.6 KiB
Plaintext
229 lines
7.6 KiB
Plaintext
/**
|
|
* 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.
|
|
*/
|
|
|
|
#import "RCTComponentViewRegistry.h"
|
|
|
|
#import <Foundation/NSMapTable.h>
|
|
#import <React/RCTAssert.h>
|
|
|
|
#import "RCTImageComponentView.h"
|
|
#import "RCTParagraphComponentView.h"
|
|
#import "RCTViewComponentView.h"
|
|
|
|
using namespace facebook::react;
|
|
|
|
#define LEGACY_UIMANAGER_INTEGRATION_ENABLED 1
|
|
|
|
#ifdef LEGACY_UIMANAGER_INTEGRATION_ENABLED
|
|
|
|
#import <React/RCTBridge+Private.h>
|
|
#import <React/RCTUIManager.h>
|
|
|
|
/**
|
|
* Warning: This is a total hack and temporary solution.
|
|
* Unless we have a pure Fabric-based implementation of UIManager commands
|
|
* delivery pipeline, we have to leverage existing infra. This code tricks
|
|
* legacy UIManager by registering all Fabric-managed views in it,
|
|
* hence existing command-delivery infra can reach "foreign" views using
|
|
* the old pipeline.
|
|
*/
|
|
@interface RCTUIManager ()
|
|
- (NSMutableDictionary<NSNumber *, UIView *> *)viewRegistry;
|
|
@end
|
|
|
|
@interface RCTUIManager (Hack)
|
|
|
|
+ (void)registerView:(UIView *)view;
|
|
+ (void)unregisterView:(UIView *)view;
|
|
|
|
@end
|
|
|
|
@implementation RCTUIManager (Hack)
|
|
|
|
+ (void)registerView:(UIView *)view
|
|
{
|
|
if (!view) {
|
|
return;
|
|
}
|
|
|
|
RCTUIManager *uiManager = [[RCTBridge currentBridge] uiManager];
|
|
view.reactTag = @(view.tag);
|
|
[uiManager.viewRegistry setObject:view forKey:@(view.tag)];
|
|
}
|
|
|
|
+ (void)unregisterView:(UIView *)view
|
|
{
|
|
if (!view) {
|
|
return;
|
|
}
|
|
|
|
RCTUIManager *uiManager = [[RCTBridge currentBridge] uiManager];
|
|
view.reactTag = nil;
|
|
[uiManager.viewRegistry removeObjectForKey:@(view.tag)];
|
|
}
|
|
|
|
@end
|
|
|
|
#endif
|
|
|
|
const NSInteger RCTComponentViewRegistryRecyclePoolMaxSize = 1024;
|
|
|
|
@implementation RCTComponentViewRegistry {
|
|
NSMapTable<id /* ReactTag */, UIView<RCTComponentViewProtocol> *> *_registry;
|
|
NSMapTable<id /* ComponentHandle */, NSHashTable<UIView<RCTComponentViewProtocol> *> *> *_recyclePool;
|
|
}
|
|
|
|
- (instancetype)init
|
|
{
|
|
if (self = [super init]) {
|
|
_registry = [NSMapTable mapTableWithKeyOptions:NSPointerFunctionsIntegerPersonality | NSPointerFunctionsOpaqueMemory
|
|
valueOptions:NSPointerFunctionsObjectPersonality];
|
|
_recyclePool =
|
|
[NSMapTable mapTableWithKeyOptions:NSPointerFunctionsOpaquePersonality | NSPointerFunctionsOpaqueMemory
|
|
valueOptions:NSPointerFunctionsObjectPersonality];
|
|
_componentViewFactory = [RCTComponentViewFactory standardComponentViewFactory];
|
|
|
|
[[NSNotificationCenter defaultCenter] addObserver:self
|
|
selector:@selector(handleApplicationDidReceiveMemoryWarningNotification)
|
|
name:UIApplicationDidReceiveMemoryWarningNotification
|
|
object:nil];
|
|
|
|
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.1 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
|
|
// Calling this a bit later, when the main thread is probably idle while JavaScript thread is busy.
|
|
[self preallocateViewComponents];
|
|
});
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
- (void)preallocateViewComponents
|
|
{
|
|
// This data is based on empirical evidence which should represent the reality pretty well.
|
|
// Regular `<View>` has magnitude equals to `1` by definition.
|
|
std::vector<std::pair<ComponentHandle, float>> componentMagnitudes = {
|
|
{[RCTViewComponentView componentHandle], 1},
|
|
{[RCTImageComponentView componentHandle], 0.3},
|
|
{[RCTParagraphComponentView componentHandle], 0.3},
|
|
};
|
|
|
|
// `complexity` represents the complexity of a typical surface in a number of `<View>` components (with Flattening
|
|
// enabled).
|
|
float complexity = 100;
|
|
|
|
// The whole process should not take more than 10ms in the worst case, so there is no need to split it up.
|
|
for (const auto &componentMagnitude : componentMagnitudes) {
|
|
for (int i = 0; i < complexity * componentMagnitude.second; i++) {
|
|
[self optimisticallyCreateComponentViewWithComponentHandle:componentMagnitude.first];
|
|
}
|
|
}
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
[[NSNotificationCenter defaultCenter] removeObserver:self];
|
|
}
|
|
|
|
- (UIView<RCTComponentViewProtocol> *)dequeueComponentViewWithComponentHandle:(ComponentHandle)componentHandle
|
|
tag:(ReactTag)tag
|
|
{
|
|
RCTAssertMainQueue();
|
|
|
|
RCTAssert(
|
|
![_registry objectForKey:(__bridge id)(void *)tag],
|
|
@"RCTComponentViewRegistry: Attempt to dequeue already registered component.");
|
|
|
|
UIView<RCTComponentViewProtocol> *componentView = [self _dequeueComponentViewWithComponentHandle:componentHandle];
|
|
componentView.tag = tag;
|
|
[_registry setObject:componentView forKey:(__bridge id)(void *)tag];
|
|
|
|
#ifdef LEGACY_UIMANAGER_INTEGRATION_ENABLED
|
|
[RCTUIManager registerView:componentView];
|
|
#endif
|
|
|
|
return componentView;
|
|
}
|
|
|
|
- (void)enqueueComponentViewWithComponentHandle:(ComponentHandle)componentHandle
|
|
tag:(ReactTag)tag
|
|
componentView:(UIView<RCTComponentViewProtocol> *)componentView
|
|
{
|
|
RCTAssertMainQueue();
|
|
|
|
RCTAssert(
|
|
[_registry objectForKey:(__bridge id)(void *)tag],
|
|
@"RCTComponentViewRegistry: Attempt to enqueue unregistered component.");
|
|
|
|
#ifdef LEGACY_UIMANAGER_INTEGRATION_ENABLED
|
|
[RCTUIManager unregisterView:componentView];
|
|
#endif
|
|
|
|
[_registry removeObjectForKey:(__bridge id)(void *)tag];
|
|
componentView.tag = 0;
|
|
[self _enqueueComponentViewWithComponentHandle:componentHandle componentView:componentView];
|
|
}
|
|
|
|
- (void)optimisticallyCreateComponentViewWithComponentHandle:(ComponentHandle)componentHandle
|
|
{
|
|
RCTAssertMainQueue();
|
|
[self _enqueueComponentViewWithComponentHandle:componentHandle
|
|
componentView:[self.componentViewFactory
|
|
createComponentViewWithComponentHandle:componentHandle]];
|
|
}
|
|
|
|
- (UIView<RCTComponentViewProtocol> *)componentViewByTag:(ReactTag)tag
|
|
{
|
|
RCTAssertMainQueue();
|
|
return [_registry objectForKey:(__bridge id)(void *)tag];
|
|
}
|
|
|
|
- (ReactTag)tagByComponentView:(UIView<RCTComponentViewProtocol> *)componentView
|
|
{
|
|
RCTAssertMainQueue();
|
|
return componentView.tag;
|
|
}
|
|
|
|
- (nullable UIView<RCTComponentViewProtocol> *)_dequeueComponentViewWithComponentHandle:(ComponentHandle)componentHandle
|
|
{
|
|
RCTAssertMainQueue();
|
|
NSHashTable<UIView<RCTComponentViewProtocol> *> *componentViews =
|
|
[_recyclePool objectForKey:(__bridge id)(void *)componentHandle];
|
|
if (!componentViews || componentViews.count == 0) {
|
|
return [self.componentViewFactory createComponentViewWithComponentHandle:componentHandle];
|
|
}
|
|
|
|
UIView<RCTComponentViewProtocol> *componentView = [componentViews anyObject];
|
|
[componentViews removeObject:componentView];
|
|
return componentView;
|
|
}
|
|
|
|
- (void)_enqueueComponentViewWithComponentHandle:(ComponentHandle)componentHandle
|
|
componentView:(UIView<RCTComponentViewProtocol> *)componentView
|
|
{
|
|
RCTAssertMainQueue();
|
|
NSHashTable<UIView<RCTComponentViewProtocol> *> *componentViews =
|
|
[_recyclePool objectForKey:(__bridge id)(void *)componentHandle];
|
|
if (!componentViews) {
|
|
componentViews = [NSHashTable hashTableWithOptions:NSPointerFunctionsObjectPersonality];
|
|
[_recyclePool setObject:componentViews forKey:(__bridge id)(void *)componentHandle];
|
|
}
|
|
|
|
if (componentViews.count >= RCTComponentViewRegistryRecyclePoolMaxSize) {
|
|
return;
|
|
}
|
|
|
|
[componentView prepareForRecycle];
|
|
[componentViews addObject:componentView];
|
|
}
|
|
|
|
- (void)handleApplicationDidReceiveMemoryWarningNotification
|
|
{
|
|
[_recyclePool removeAllObjects];
|
|
}
|
|
|
|
@end
|