334 lines
9.9 KiB
Plaintext
334 lines
9.9 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 "RCTSurfacePresenter.h"
|
|
|
|
#import <objc/runtime.h>
|
|
#import <mutex>
|
|
#import <jsi/jsi.h>
|
|
#import <cxxreact/MessageQueueThread.h>
|
|
|
|
#import <React/RCTAssert.h>
|
|
#import <React/RCTBridge+Private.h>
|
|
#import <React/RCTComponentViewRegistry.h>
|
|
#import <React/RCTFabricSurface.h>
|
|
#import <React/RCTImageLoader.h>
|
|
#import <React/RCTMountingManager.h>
|
|
#import <React/RCTMountingManagerDelegate.h>
|
|
#import <React/RCTScheduler.h>
|
|
#import <React/RCTSurfaceRegistry.h>
|
|
#import <React/RCTSurfaceView.h>
|
|
#import <React/RCTSurfaceView+Internal.h>
|
|
#import <React/RCTUtils.h>
|
|
#import <fabric/core/LayoutContext.h>
|
|
#import <fabric/core/LayoutConstraints.h>
|
|
#import <fabric/imagemanager/ImageManager.h>
|
|
#import <fabric/uimanager/ContextContainer.h>
|
|
|
|
#import "MainRunLoopEventBeat.h"
|
|
#import "MessageQueueEventBeat.h"
|
|
#import "RCTConversions.h"
|
|
|
|
using namespace facebook::react;
|
|
|
|
@interface RCTBridge ()
|
|
- (std::shared_ptr<facebook::react::MessageQueueThread>)jsMessageThread;
|
|
@end
|
|
|
|
@interface RCTSurfacePresenter () <RCTSchedulerDelegate, RCTMountingManagerDelegate>
|
|
@end
|
|
|
|
@implementation RCTSurfacePresenter {
|
|
std::mutex _schedulerMutex;
|
|
RCTScheduler *_Nullable _scheduler; // Thread-safe. Mutation of the instance variable is protected by `_schedulerMutex`.
|
|
RCTMountingManager *_mountingManager; // Thread-safe.
|
|
RCTSurfaceRegistry *_surfaceRegistry; // Thread-safe.
|
|
RCTBridge *_bridge; // Unsafe. We are moving away from Bridge.
|
|
RCTBridge *_batchedBridge;
|
|
}
|
|
|
|
- (instancetype)initWithBridge:(RCTBridge *)bridge
|
|
{
|
|
if (self = [super init]) {
|
|
_bridge = bridge;
|
|
_batchedBridge = [_bridge batchedBridge] ?: _bridge;
|
|
|
|
_surfaceRegistry = [[RCTSurfaceRegistry alloc] init];
|
|
|
|
_mountingManager = [[RCTMountingManager alloc] init];
|
|
_mountingManager.delegate = self;
|
|
|
|
[[NSNotificationCenter defaultCenter] addObserver:self
|
|
selector:@selector(handleBridgeWillReloadNotification:)
|
|
name:RCTBridgeWillReloadNotification
|
|
object:_bridge];
|
|
[[NSNotificationCenter defaultCenter] addObserver:self
|
|
selector:@selector(handleJavaScriptDidLoadNotification:)
|
|
name:RCTJavaScriptDidLoadNotification
|
|
object:_bridge];
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
[[NSNotificationCenter defaultCenter] removeObserver:self];
|
|
}
|
|
|
|
#pragma mark - Internal Surface-dedicated Interface
|
|
|
|
- (void)registerSurface:(RCTFabricSurface *)surface
|
|
{
|
|
[_surfaceRegistry registerSurface:surface];
|
|
[self _startSurface:surface];
|
|
}
|
|
|
|
- (void)unregisterSurface:(RCTFabricSurface *)surface
|
|
{
|
|
[self _stopSurface:surface];
|
|
[_surfaceRegistry unregisterSurface:surface];
|
|
}
|
|
|
|
- (void)setProps:(NSDictionary *)props
|
|
surface:(RCTFabricSurface *)surface
|
|
{
|
|
// This implementation is suboptimal indeed but still better than nothing for now.
|
|
[self _stopSurface:surface];
|
|
[self _startSurface:surface];
|
|
}
|
|
|
|
- (RCTFabricSurface *)surfaceForRootTag:(ReactTag)rootTag
|
|
{
|
|
return [_surfaceRegistry surfaceForRootTag:rootTag];
|
|
}
|
|
|
|
- (CGSize)sizeThatFitsMinimumSize:(CGSize)minimumSize
|
|
maximumSize:(CGSize)maximumSize
|
|
surface:(RCTFabricSurface *)surface
|
|
{
|
|
LayoutContext layoutContext = {
|
|
.pointScaleFactor = RCTScreenScale()
|
|
};
|
|
|
|
LayoutConstraints layoutConstraints = {
|
|
.minimumSize = RCTSizeFromCGSize(minimumSize),
|
|
.maximumSize = RCTSizeFromCGSize(maximumSize)
|
|
};
|
|
|
|
return [self._scheduler measureSurfaceWithLayoutConstraints:layoutConstraints
|
|
layoutContext:layoutContext
|
|
surfaceId:surface.rootTag];
|
|
}
|
|
|
|
- (void)setMinimumSize:(CGSize)minimumSize
|
|
maximumSize:(CGSize)maximumSize
|
|
surface:(RCTFabricSurface *)surface
|
|
{
|
|
LayoutContext layoutContext = {
|
|
.pointScaleFactor = RCTScreenScale()
|
|
};
|
|
|
|
LayoutConstraints layoutConstraints = {
|
|
.minimumSize = RCTSizeFromCGSize(minimumSize),
|
|
.maximumSize = RCTSizeFromCGSize(maximumSize)
|
|
};
|
|
|
|
[self._scheduler constraintSurfaceLayoutWithLayoutConstraints:layoutConstraints
|
|
layoutContext:layoutContext
|
|
surfaceId:surface.rootTag];
|
|
}
|
|
|
|
#pragma mark - Private
|
|
|
|
- (RCTScheduler *)_scheduler
|
|
{
|
|
std::lock_guard<std::mutex> lock(_schedulerMutex);
|
|
|
|
if (_scheduler) {
|
|
return _scheduler;
|
|
}
|
|
|
|
auto contextContainer = std::make_shared<ContextContainer>();
|
|
|
|
auto messageQueueThread = _batchedBridge.jsMessageThread;
|
|
auto runtime = (facebook::jsi::Runtime *)((RCTCxxBridge *)_batchedBridge).runtime;
|
|
|
|
RuntimeExecutor runtimeExecutor =
|
|
[runtime, messageQueueThread](std::function<void(facebook::jsi::Runtime &runtime)> &&callback) {
|
|
messageQueueThread->runOnQueue([runtime, callback = std::move(callback)]() {
|
|
callback(*runtime);
|
|
});
|
|
};
|
|
|
|
EventBeatFactory synchronousBeatFactory = [messageQueueThread]() {
|
|
return std::make_unique<MainRunLoopEventBeat>(messageQueueThread);
|
|
};
|
|
|
|
EventBeatFactory asynchronousBeatFactory = [messageQueueThread]() {
|
|
return std::make_unique<MessageQueueEventBeat>(messageQueueThread);
|
|
};
|
|
|
|
contextContainer->registerInstance<EventBeatFactory>(synchronousBeatFactory, "synchronous");
|
|
contextContainer->registerInstance<EventBeatFactory>(asynchronousBeatFactory, "asynchronous");
|
|
|
|
contextContainer->registerInstance(runtimeExecutor, "runtime-executor");
|
|
|
|
contextContainer->registerInstance(std::make_shared<ImageManager>((__bridge void *)[_bridge imageLoader]));
|
|
|
|
_scheduler = [[RCTScheduler alloc] initWithContextContainer:contextContainer];
|
|
_scheduler.delegate = self;
|
|
|
|
return _scheduler;
|
|
}
|
|
|
|
- (void)_startSurface:(RCTFabricSurface *)surface
|
|
{
|
|
[_mountingManager.componentViewRegistry dequeueComponentViewWithName:@"Root" tag:surface.rootTag];
|
|
|
|
LayoutContext layoutContext = {
|
|
.pointScaleFactor = RCTScreenScale()
|
|
};
|
|
|
|
LayoutConstraints layoutConstraints = {
|
|
.minimumSize = RCTSizeFromCGSize(surface.minimumSize),
|
|
.maximumSize = RCTSizeFromCGSize(surface.maximumSize)
|
|
};
|
|
|
|
[self._scheduler startSurfaceWithSurfaceId:surface.rootTag
|
|
moduleName:surface.moduleName
|
|
initailProps:surface.properties
|
|
layoutConstraints:layoutConstraints
|
|
layoutContext:layoutContext];
|
|
}
|
|
|
|
- (void)_stopSurface:(RCTFabricSurface *)surface
|
|
{
|
|
[self._scheduler stopSurfaceWithSurfaceId:surface.rootTag];
|
|
|
|
UIView<RCTComponentViewProtocol> *rootView = [_mountingManager.componentViewRegistry componentViewByTag:surface.rootTag];
|
|
[_mountingManager.componentViewRegistry enqueueComponentViewWithName:@"Root" tag:surface.rootTag componentView:rootView];
|
|
|
|
[surface _unsetStage:(RCTSurfaceStagePrepared | RCTSurfaceStageMounted)];
|
|
}
|
|
|
|
- (void)_startAllSurfaces
|
|
{
|
|
for (RCTFabricSurface *surface in _surfaceRegistry.enumerator) {
|
|
[self _startSurface:surface];
|
|
}
|
|
}
|
|
|
|
- (void)_stopAllSurfaces
|
|
{
|
|
for (RCTFabricSurface *surface in _surfaceRegistry.enumerator) {
|
|
[self _stopSurface:surface];
|
|
}
|
|
}
|
|
|
|
#pragma mark - RCTSchedulerDelegate
|
|
|
|
- (void)schedulerDidFinishTransaction:(facebook::react::ShadowViewMutationList)mutations
|
|
rootTag:(ReactTag)rootTag
|
|
{
|
|
RCTFabricSurface *surface = [_surfaceRegistry surfaceForRootTag:rootTag];
|
|
|
|
[surface _setStage:RCTSurfaceStagePrepared];
|
|
|
|
[_mountingManager performTransactionWithMutations:mutations
|
|
rootTag:rootTag];
|
|
}
|
|
|
|
- (void)schedulerDidRequestPreliminaryViewAllocationWithComponentName:(NSString *)componentName
|
|
{
|
|
[_mountingManager preliminaryCreateComponentViewWithName:componentName];
|
|
}
|
|
|
|
#pragma mark - RCTMountingManagerDelegate
|
|
|
|
- (void)mountingManager:(RCTMountingManager *)mountingManager willMountComponentsWithRootTag:(ReactTag)rootTag
|
|
{
|
|
RCTAssertMainQueue();
|
|
|
|
// Does nothing.
|
|
}
|
|
|
|
- (void)mountingManager:(RCTMountingManager *)mountingManager didMountComponentsWithRootTag:(ReactTag)rootTag
|
|
{
|
|
RCTAssertMainQueue();
|
|
|
|
RCTFabricSurface *surface = [_surfaceRegistry surfaceForRootTag:rootTag];
|
|
RCTSurfaceStage stage = surface.stage;
|
|
if (stage & RCTSurfaceStagePrepared) {
|
|
// We have to progress the stage only if the preparing phase is done.
|
|
if ([surface _setStage:RCTSurfaceStageMounted]) {
|
|
UIView *rootComponentView = [_mountingManager.componentViewRegistry componentViewByTag:rootTag];
|
|
surface.view.rootView = (RCTSurfaceRootView *)rootComponentView;
|
|
}
|
|
}
|
|
}
|
|
|
|
#pragma mark - Bridge events
|
|
|
|
- (void)handleBridgeWillReloadNotification:(NSNotification *)notification
|
|
{
|
|
{
|
|
std::lock_guard<std::mutex> lock(_schedulerMutex);
|
|
if (!_scheduler) {
|
|
// Seems we are already in the realoding process.
|
|
return;
|
|
}
|
|
}
|
|
|
|
[self _stopAllSurfaces];
|
|
|
|
{
|
|
std::lock_guard<std::mutex> lock(_schedulerMutex);
|
|
_scheduler = nil;
|
|
}
|
|
}
|
|
|
|
- (void)handleJavaScriptDidLoadNotification:(NSNotification *)notification
|
|
{
|
|
RCTBridge *bridge = notification.userInfo[@"bridge"];
|
|
if (bridge != _batchedBridge) {
|
|
_batchedBridge = bridge;
|
|
|
|
[self _startAllSurfaces];
|
|
}
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation RCTSurfacePresenter (Deprecated)
|
|
|
|
- (std::shared_ptr<FabricUIManager>)uiManager_DO_NOT_USE
|
|
{
|
|
return _scheduler.uiManager_DO_NOT_USE;
|
|
}
|
|
|
|
- (RCTBridge *)bridge_DO_NOT_USE
|
|
{
|
|
return _bridge;
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation RCTBridge (Deprecated)
|
|
|
|
- (void)setSurfacePresenter:(RCTSurfacePresenter *)surfacePresenter
|
|
{
|
|
objc_setAssociatedObject(self, @selector(surfacePresenter), surfacePresenter, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
|
|
}
|
|
|
|
- (RCTSurfacePresenter *)surfacePresenter
|
|
{
|
|
return objc_getAssociatedObject(self, @selector(surfacePresenter));
|
|
}
|
|
|
|
@end
|