353 lines
10 KiB
Objective-C
353 lines
10 KiB
Objective-C
/**
|
|
* 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 "RCTBridge.h"
|
|
#import "RCTContextExecutor.h"
|
|
#import "RCTDevMenu.h"
|
|
#import "RCTEventDispatcher.h"
|
|
#import "RCTKeyCommands.h"
|
|
#import "RCTLog.h"
|
|
#import "RCTRedBox.h"
|
|
#import "RCTSourceCode.h"
|
|
#import "RCTTouchHandler.h"
|
|
#import "RCTUIManager.h"
|
|
#import "RCTUtils.h"
|
|
#import "RCTWebViewExecutor.h"
|
|
#import "UIView+ReactKit.h"
|
|
|
|
NSString *const RCTReloadNotification = @"RCTReloadNotification";
|
|
|
|
/**
|
|
* HACK(t6568049) This should be removed soon, hiding to prevent people from
|
|
* relying on it
|
|
*/
|
|
@interface RCTBridge (RCTRootView)
|
|
|
|
- (void)setJavaScriptExecutor:(id<RCTJavaScriptExecutor>)executor;
|
|
|
|
@end
|
|
|
|
@implementation RCTRootView
|
|
{
|
|
RCTDevMenu *_devMenu;
|
|
RCTBridge *_bridge;
|
|
RCTTouchHandler *_touchHandler;
|
|
id<RCTJavaScriptExecutor> _executor;
|
|
BOOL _registered;
|
|
NSDictionary *_launchOptions;
|
|
}
|
|
|
|
static Class _globalExecutorClass;
|
|
|
|
+ (void)initialize
|
|
{
|
|
|
|
#if TARGET_IPHONE_SIMULATOR
|
|
|
|
// Register Cmd-R as a global refresh key
|
|
[[RCTKeyCommands sharedInstance] registerKeyCommandWithInput:@"r"
|
|
modifierFlags:UIKeyModifierCommand
|
|
action:^(UIKeyCommand *command) {
|
|
[self reloadAll];
|
|
}];
|
|
|
|
// Cmd-D reloads using the web view executor, allows attaching from Safari dev tools.
|
|
[[RCTKeyCommands sharedInstance] registerKeyCommandWithInput:@"d"
|
|
modifierFlags:UIKeyModifierCommand
|
|
action:^(UIKeyCommand *command) {
|
|
_globalExecutorClass = NSClassFromString(@"RCTWebSocketExecutor");
|
|
if (!_globalExecutorClass) {
|
|
RCTLogError(@"WebSocket debugger is not available. Did you forget to include RCTWebSocketExecutor?");
|
|
}
|
|
[self reloadAll];
|
|
}];
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
- (instancetype)initWithBundleURL:(NSURL *)bundleURL
|
|
moduleName:(NSString *)moduleName
|
|
launchOptions:(NSDictionary *)launchOptions
|
|
{
|
|
if ((self = [super init])) {
|
|
RCTAssert(bundleURL, @"A bundleURL is required to create an RCTRootView");
|
|
RCTAssert(moduleName, @"A bundleURL is required to create an RCTRootView");
|
|
_moduleName = moduleName;
|
|
_launchOptions = launchOptions;
|
|
[self setUp];
|
|
[self setScriptURL:bundleURL];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
/**
|
|
* HACK(t6568049) Private constructor for testing purposes
|
|
*/
|
|
- (instancetype)_initWithBundleURL:(NSURL *)bundleURL
|
|
moduleName:(NSString *)moduleName
|
|
launchOptions:(NSDictionary *)launchOptions
|
|
moduleProvider:(RCTBridgeModuleProviderBlock)moduleProvider
|
|
{
|
|
if ((self = [super init])) {
|
|
_moduleProvider = moduleProvider;
|
|
_moduleName = moduleName;
|
|
_launchOptions = launchOptions;
|
|
[self setUp];
|
|
[self setScriptURL:bundleURL];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (void)setUp
|
|
{
|
|
// Every root view that is created must have a unique react tag.
|
|
// Numbering of these tags goes from 1, 11, 21, 31, etc
|
|
static NSInteger rootViewTag = 1;
|
|
self.reactTag = @(rootViewTag);
|
|
#ifdef DEBUG
|
|
self.enableDevMenu = YES;
|
|
#endif
|
|
self.backgroundColor = [UIColor whiteColor];
|
|
rootViewTag += 10;
|
|
|
|
// Add reload observer
|
|
[[NSNotificationCenter defaultCenter] addObserver:self
|
|
selector:@selector(reload)
|
|
name:RCTReloadNotification
|
|
object:nil];
|
|
}
|
|
|
|
- (BOOL)canBecomeFirstResponder
|
|
{
|
|
return YES;
|
|
}
|
|
|
|
- (void)motionEnded:(UIEventSubtype)motion withEvent:(UIEvent *)event
|
|
{
|
|
if (motion == UIEventSubtypeMotionShake && self.enableDevMenu) {
|
|
if (!_devMenu) {
|
|
_devMenu = [[RCTDevMenu alloc] initWithRootView:self];
|
|
}
|
|
[_devMenu show];
|
|
}
|
|
}
|
|
|
|
+ (NSArray *)JSMethods
|
|
{
|
|
return @[
|
|
@"AppRegistry.runApplication",
|
|
@"ReactIOS.unmountComponentAtNodeAndRemoveContainer"
|
|
];
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
[[NSNotificationCenter defaultCenter] removeObserver:self];
|
|
|
|
[_bridge enqueueJSCall:@"ReactIOS.unmountComponentAtNodeAndRemoveContainer"
|
|
args:@[self.reactTag]];
|
|
[self invalidate];
|
|
}
|
|
|
|
#pragma mark - RCTInvalidating
|
|
|
|
- (BOOL)isValid
|
|
{
|
|
return [_bridge isValid];
|
|
}
|
|
|
|
- (void)invalidate
|
|
{
|
|
// Clear view
|
|
[self.subviews makeObjectsPerformSelector:@selector(removeFromSuperview)];
|
|
|
|
[self removeGestureRecognizer:_touchHandler];
|
|
[_touchHandler invalidate];
|
|
[_executor invalidate];
|
|
|
|
// TODO: eventually we'll want to be able to share the bridge between
|
|
// multiple rootviews, in which case we'll need to move this elsewhere
|
|
[_bridge invalidate];
|
|
}
|
|
|
|
#pragma mark Bundle loading
|
|
|
|
- (void)bundleFinishedLoading:(NSError *)error
|
|
{
|
|
if (error != nil) {
|
|
NSArray *stack = [[error userInfo] objectForKey:@"stack"];
|
|
if (stack) {
|
|
[[RCTRedBox sharedInstance] showErrorMessage:[error localizedDescription] withStack:stack];
|
|
} else {
|
|
[[RCTRedBox sharedInstance] showErrorMessage:[error localizedDescription] withDetails:[error localizedFailureReason]];
|
|
}
|
|
} else {
|
|
|
|
[_bridge.uiManager registerRootView:self];
|
|
_registered = YES;
|
|
|
|
NSString *moduleName = _moduleName ?: @"";
|
|
NSDictionary *appParameters = @{
|
|
@"rootTag": self.reactTag,
|
|
@"initialProps": self.initialProperties ?: @{},
|
|
};
|
|
[_bridge enqueueJSCall:@"AppRegistry.runApplication"
|
|
args:@[moduleName, appParameters]];
|
|
}
|
|
}
|
|
|
|
- (void)loadBundle
|
|
{
|
|
[self invalidate];
|
|
|
|
if (!_scriptURL) {
|
|
return;
|
|
}
|
|
|
|
// Clean up
|
|
[self removeGestureRecognizer:_touchHandler];
|
|
[_touchHandler invalidate];
|
|
[_executor invalidate];
|
|
[_bridge invalidate];
|
|
|
|
_registered = NO;
|
|
|
|
// Choose local executor if specified, followed by global, followed by default
|
|
_executor = [[_executorClass ?: _globalExecutorClass ?: [RCTContextExecutor class] alloc] init];
|
|
|
|
/**
|
|
* HACK(t6568049) Most of the properties passed into the bridge are not used
|
|
* right now but it'll be changed soon so it's here for convenience.
|
|
*/
|
|
_bridge = [[RCTBridge alloc] initWithBundlePath:_scriptURL.absoluteString
|
|
moduleProvider:_moduleProvider
|
|
launchOptions:_launchOptions];
|
|
[_bridge setJavaScriptExecutor:_executor];
|
|
|
|
_touchHandler = [[RCTTouchHandler alloc] initWithBridge:_bridge];
|
|
[self addGestureRecognizer:_touchHandler];
|
|
|
|
// Load the bundle
|
|
NSURLSessionDataTask *task = [[NSURLSession sharedSession] dataTaskWithURL:_scriptURL completionHandler:
|
|
^(NSData *data, NSURLResponse *response, NSError *error) {
|
|
|
|
// Handle general request errors
|
|
if (error) {
|
|
if ([[error domain] isEqualToString:NSURLErrorDomain]) {
|
|
NSDictionary *userInfo = @{
|
|
NSLocalizedDescriptionKey: @"Could not connect to development server. Ensure node server is running - run 'npm start' from ReactKit root",
|
|
NSLocalizedFailureReasonErrorKey: [error localizedDescription],
|
|
NSUnderlyingErrorKey: error,
|
|
};
|
|
error = [NSError errorWithDomain:@"JSServer"
|
|
code:error.code
|
|
userInfo:userInfo];
|
|
}
|
|
[self bundleFinishedLoading:error];
|
|
return;
|
|
}
|
|
|
|
// Parse response as text
|
|
NSStringEncoding encoding = NSUTF8StringEncoding;
|
|
if (response.textEncodingName != nil) {
|
|
CFStringEncoding cfEncoding = CFStringConvertIANACharSetNameToEncoding((CFStringRef)response.textEncodingName);
|
|
if (cfEncoding != kCFStringEncodingInvalidId) {
|
|
encoding = CFStringConvertEncodingToNSStringEncoding(cfEncoding);
|
|
}
|
|
}
|
|
NSString *rawText = [[NSString alloc] initWithData:data encoding:encoding];
|
|
|
|
// Handle HTTP errors
|
|
if ([response isKindOfClass:[NSHTTPURLResponse class]] && [(NSHTTPURLResponse *)response statusCode] != 200) {
|
|
NSDictionary *userInfo;
|
|
NSDictionary *errorDetails = RCTJSONParse(rawText, nil);
|
|
if ([errorDetails isKindOfClass:[NSDictionary class]]) {
|
|
userInfo = @{
|
|
NSLocalizedDescriptionKey: errorDetails[@"message"] ?: @"No message provided",
|
|
@"stack": @[@{
|
|
@"methodName": errorDetails[@"description"] ?: @"",
|
|
@"file": errorDetails[@"filename"] ?: @"",
|
|
@"lineNumber": errorDetails[@"lineNumber"] ?: @0
|
|
}]
|
|
};
|
|
} else {
|
|
userInfo = @{NSLocalizedDescriptionKey: rawText};
|
|
}
|
|
error = [NSError errorWithDomain:@"JSServer"
|
|
code:[(NSHTTPURLResponse *)response statusCode]
|
|
userInfo:userInfo];
|
|
|
|
[self bundleFinishedLoading:error];
|
|
return;
|
|
}
|
|
if (!_bridge.isValid) {
|
|
return; // Bridge was invalidated in the meanwhile
|
|
}
|
|
|
|
// Success!
|
|
RCTSourceCode *sourceCodeModule = _bridge.modules[NSStringFromClass([RCTSourceCode class])];
|
|
sourceCodeModule.scriptURL = _scriptURL;
|
|
sourceCodeModule.scriptText = rawText;
|
|
|
|
[_bridge enqueueApplicationScript:rawText url:_scriptURL onComplete:^(NSError *_error) {
|
|
dispatch_async(dispatch_get_main_queue(), ^{
|
|
if (_bridge.isValid) {
|
|
[self bundleFinishedLoading:_error];
|
|
}
|
|
});
|
|
}];
|
|
|
|
}];
|
|
|
|
[task resume];
|
|
}
|
|
|
|
- (void)setScriptURL:(NSURL *)scriptURL
|
|
{
|
|
if ([_scriptURL isEqual:scriptURL]) {
|
|
return;
|
|
}
|
|
|
|
_scriptURL = scriptURL;
|
|
[self loadBundle];
|
|
}
|
|
|
|
- (void)layoutSubviews
|
|
{
|
|
[super layoutSubviews];
|
|
if (_registered) {
|
|
[_bridge.uiManager setFrame:self.frame forRootView:self];
|
|
}
|
|
}
|
|
|
|
- (void)reload
|
|
{
|
|
[self loadBundle];
|
|
}
|
|
|
|
+ (void)reloadAll
|
|
{
|
|
[[NSNotificationCenter defaultCenter] postNotificationName:RCTReloadNotification object:nil];
|
|
}
|
|
|
|
- (void)startOrResetInteractionTiming
|
|
{
|
|
[_touchHandler startOrResetInteractionTiming];
|
|
}
|
|
|
|
- (NSDictionary *)endAndResetInteractionTiming
|
|
{
|
|
return [_touchHandler endAndResetInteractionTiming];
|
|
}
|
|
|
|
@end
|