/** * 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. */ #import "RCTRootView.h" #import #import "RCTBridge.h" #import "RCTContextExecutor.h" #import "RCTDevMenu.h" #import "RCTEventDispatcher.h" #import "RCTKeyCommands.h" #import "RCTLog.h" #import "RCTSourceCode.h" #import "RCTTouchHandler.h" #import "RCTUIManager.h" #import "RCTUtils.h" #import "RCTWebViewExecutor.h" #import "UIView+React.h" NSString *const RCTJavaScriptDidLoadNotification = @"RCTJavaScriptDidLoadNotification"; NSString *const RCTReloadNotification = @"RCTReloadNotification"; NSString *const RCTReloadViewsNotification = @"RCTReloadViewsNotification"; /** * HACK(t6568049) This should be removed soon, hiding to prevent people from * relying on it */ @interface RCTBridge (RCTRootView) - (void)setJavaScriptExecutor:(id)executor; @end @interface RCTUIManager (RCTRootView) - (NSNumber *)allocateRootTag; @end @implementation RCTRootView { RCTDevMenu *_devMenu; RCTBridge *_bridge; RCTTouchHandler *_touchHandler; NSString *_moduleName; BOOL _registered; NSDictionary *_launchOptions; UIView *_contentView; } - (instancetype)initWithBridge:(RCTBridge *)bridge moduleName:(NSString *)moduleName { RCTAssert(bridge, @"A bridge instance is required to create an RCTRootView"); RCTAssert(moduleName, @"A moduleName is required to create an RCTRootView"); if ((self = [super init])) { #ifdef DEBUG _enableDevMenu = YES; #endif _bridge = bridge; _moduleName = moduleName; self.backgroundColor = [UIColor whiteColor]; [self setUp]; } return self; } - (instancetype)initWithBundleURL:(NSURL *)bundleURL moduleName:(NSString *)moduleName launchOptions:(NSDictionary *)launchOptions { RCTBridge *bridge = [[RCTBridge alloc] initWithBundlePath:bundleURL.absoluteString moduleProvider:nil launchOptions:launchOptions]; return [self initWithBridge:bridge moduleName:moduleName]; } - (void)dealloc { [self tearDown]; } - (void)setUp { if (!_registered) { /** * Every root view that is created must have a unique react tag. * Numbering of these tags goes from 1, 11, 21, 31, etc * * NOTE: Since the bridge persists, the RootViews might be reused, so now * the react tag is assigned every time we load new content. */ _contentView = [[UIView alloc] init]; _contentView.reactTag = [_bridge.uiManager allocateRootTag]; _touchHandler = [[RCTTouchHandler alloc] initWithBridge:_bridge]; [_contentView addGestureRecognizer:_touchHandler]; [self addSubview:_contentView]; [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(reload) name:RCTReloadViewsNotification object:_bridge]; if (_bridge.loaded) { [self bundleFinishedLoading]; } else { [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(bundleFinishedLoading) name:RCTJavaScriptDidLoadNotification object:_bridge]; } } } - (void)tearDown { if (_registered) { _registered = NO; [[NSNotificationCenter defaultCenter] removeObserver:self]; [_contentView removeGestureRecognizer:_touchHandler]; [_contentView removeFromSuperview]; [_touchHandler invalidate]; [_bridge enqueueJSCall:@"ReactIOS.unmountComponentAtNodeAndRemoveContainer" args:@[_contentView.reactTag]]; } } - (BOOL)isValid { return _registered; } - (void)invalidate { [self tearDown]; } - (UIViewController *)backingViewController { return _backingViewController ?: [super backingViewController]; } - (BOOL)canBecomeFirstResponder { return YES; } - (void)motionEnded:(UIEventSubtype)motion withEvent:(UIEvent *)event { if (motion == UIEventSubtypeMotionShake && self.enableDevMenu) { if (!_devMenu) { _devMenu = [[RCTDevMenu alloc] initWithBridge:self.bridge]; } [_devMenu show]; } } + (NSArray *)JSMethods { return @[ @"AppRegistry.runApplication", @"ReactIOS.unmountComponentAtNodeAndRemoveContainer" ]; } - (void)bundleFinishedLoading { dispatch_async(dispatch_get_main_queue(), ^{ _registered = YES; NSString *moduleName = _moduleName ?: @""; NSDictionary *appParameters = @{ @"rootTag": _contentView.reactTag, @"initialProps": self.initialProperties ?: @{}, }; [_bridge.uiManager registerRootView:_contentView]; [_bridge enqueueJSCall:@"AppRegistry.runApplication" args:@[moduleName, appParameters]]; }); } - (void)layoutSubviews { [super layoutSubviews]; _contentView.frame = self.bounds; if (_registered) { [_bridge.uiManager setFrame:self.frame forRootView:_contentView]; } } - (void)setFrame:(CGRect)frame { [super setFrame:frame]; _contentView.frame = self.bounds; } - (void)reload { [self tearDown]; [self setUp]; } + (void)reloadAll { [[NSNotificationCenter defaultCenter] postNotificationName:RCTReloadNotification object:self]; } - (NSNumber *)reactTag { return _contentView.reactTag; } - (void)startOrResetInteractionTiming { [_touchHandler startOrResetInteractionTiming]; } - (NSDictionary *)endAndResetInteractionTiming { return [_touchHandler endAndResetInteractionTiming]; } @end @implementation RCTUIManager (RCTRootView) - (NSNumber *)allocateRootTag { NSNumber *rootTag = objc_getAssociatedObject(self, _cmd) ?: @1; objc_setAssociatedObject(self, _cmd, @(rootTag.integerValue + 10), OBJC_ASSOCIATION_RETAIN_NONATOMIC); return rootTag; } @end