mirror of
https://github.com/status-im/react-native.git
synced 2025-01-25 00:39:03 +00:00
2f9bd1f62f
Summary: This adds workarounds for the code that was preventing React from compiling when linked against an iOS App Extension target. Some iOS APIs are unavailable to App Extensions, and Xcode's static analysis will catch attempts to use methods that have been flagged as unavailable. React currently uses two APIs that are off limits to extensions: `[UIApplication sharedApplication]` and `[UIAlertView initWith ...]`. This commit adds a helper function to `RCTUtils.[hm]` called `RCTRunningInAppExtension()`, which returns `YES` if, at runtime, it can be determined that we're running in an app extension (by checking whether the path to `[NSBundle mainBundle]` has the `"appex"` path extension). It also adds a `RCTSharedApplication()` function, which will return `nil` if running in an App Extension. If running in an App, `RCTSharedApplication()` calls `sharedApplication` by calling `performSelector:` on the `UIApplication` class. This passes the static analysis check, and, in my opinion, obeys the "spirit of th Closes https://github.com/facebook/react-native/pull/1895 Reviewed By: @svcscm Differential Revision: D2224128 Pulled By: @nicklockwood
85 lines
2.4 KiB
Objective-C
85 lines
2.4 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 "RCTLinkingManager.h"
|
|
|
|
#import "RCTBridge.h"
|
|
#import "RCTEventDispatcher.h"
|
|
#import "RCTUtils.h"
|
|
|
|
NSString *const RCTOpenURLNotification = @"RCTOpenURLNotification";
|
|
|
|
@implementation RCTLinkingManager
|
|
|
|
@synthesize bridge = _bridge;
|
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
- (instancetype)init
|
|
{
|
|
if ((self = [super init])) {
|
|
[[NSNotificationCenter defaultCenter] addObserver:self
|
|
selector:@selector(handleOpenURLNotification:)
|
|
name:RCTOpenURLNotification
|
|
object:nil];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
[[NSNotificationCenter defaultCenter] removeObserver:self];
|
|
}
|
|
|
|
+ (BOOL)application:(UIApplication *)application
|
|
openURL:(NSURL *)URL
|
|
sourceApplication:(NSString *)sourceApplication
|
|
annotation:(id)annotation
|
|
{
|
|
NSDictionary *payload = @{@"url": URL.absoluteString};
|
|
[[NSNotificationCenter defaultCenter] postNotificationName:RCTOpenURLNotification
|
|
object:self
|
|
userInfo:payload];
|
|
return YES;
|
|
}
|
|
|
|
- (void)handleOpenURLNotification:(NSNotification *)notification
|
|
{
|
|
[_bridge.eventDispatcher sendDeviceEventWithName:@"openURL"
|
|
body:notification.userInfo];
|
|
}
|
|
|
|
RCT_EXPORT_METHOD(openURL:(NSURL *)URL)
|
|
{
|
|
// Doesn't really matter what thread we call this on since it exits the app
|
|
[RCTSharedApplication() openURL:URL];
|
|
}
|
|
|
|
RCT_EXPORT_METHOD(canOpenURL:(NSURL *)URL
|
|
callback:(RCTResponseSenderBlock)callback)
|
|
{
|
|
if (RCTRunningInAppExtension()) {
|
|
// Technically Today widgets can open urls, but supporting that would require
|
|
// a reference to the NSExtensionContext
|
|
callback(@[@(NO)]);
|
|
}
|
|
|
|
// This can be expensive, so we deliberately don't call on main thread
|
|
BOOL canOpen = [RCTSharedApplication() canOpenURL:URL];
|
|
callback(@[@(canOpen)]);
|
|
}
|
|
|
|
- (NSDictionary *)constantsToExport
|
|
{
|
|
NSURL *initialURL = _bridge.launchOptions[UIApplicationLaunchOptionsURLKey];
|
|
return @{@"initialURL": RCTNullIfNil(initialURL.absoluteString)};
|
|
}
|
|
|
|
@end
|