mirror of
https://github.com/facebook/react-native.git
synced 2025-11-01 09:14:26 +00:00
3724810d21
Summary: This PR adds initial support for Project Catalyst a.k.a. UIKitForMac. This is not yet meant for production, but this is enough for RNTester to successfully compile and mostly work :) Some APIs are not supported on the Mac -- e.g. telephony, and deprecated APIs are removed on Mac ���-- those had to be ifdef'd out via platform checks. The biggest limitation right now is that I couldn't get Web Socket code to successfully compile, and so there are a lot of temporary platform checks for that , and the RCTWebSocket.xcodeproj is marked as not supporting UIKitForMac. Again -- temporary, until someone with more knowledge knows how to fix this. https://github.com/react-native-community/discussions-and-proposals/issues/131 ## Changelog [iOS] [Added] - Fixed compilation for macOS (Project Catalyst) -- not meant for production use yet Pull Request resolved: https://github.com/facebook/react-native/pull/25427 Test Plan: - Open RNTester/RNTester.xcodeproj with Xcode 10.2, run it like a normal iOS app -- make sure it compiles and runs correctly (no regression) - Open the same project with Xcode 11 beta 2 (or higher) on macOS Catalina beta, select "My Mac" as device target, and run -- see that it actually compiles and runs. **Note** there are unfortunately some required steps: - change build configuration to Release (because packager doesn't work correctly yet) - change development team to yours if Xcode tells you to - go to RNTester project → Build phases → Link binary with libraries, and change `platforms` for `libRCTWebSocket.a` to `iOS` (without Mac compatibility). I can't commit that change because it breaks compatibility with earlier Xcode versions The two extra steps for successful compile will disappear once web socket compilation for Catalyst is fixed Reviewed By: mmmulani Differential Revision: D16088263 Pulled By: sammy-SC fbshipit-source-id: 9c0b932b048e50a8e0f336eaa0612851b1909cae
189 lines
6.3 KiB
Objective-C
189 lines
6.3 KiB
Objective-C
/**
|
|
* 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 "RCTLinkingManager.h"
|
|
|
|
#import <React/RCTBridge.h>
|
|
#import <React/RCTEventDispatcher.h>
|
|
#import <React/RCTUtils.h>
|
|
|
|
static NSString *const kOpenURLNotification = @"RCTOpenURLNotification";
|
|
|
|
static void postNotificationWithURL(NSURL *URL, id sender)
|
|
{
|
|
NSDictionary<NSString *, id> *payload = @{@"url": URL.absoluteString};
|
|
[[NSNotificationCenter defaultCenter] postNotificationName:kOpenURLNotification
|
|
object:sender
|
|
userInfo:payload];
|
|
}
|
|
|
|
@implementation RCTLinkingManager
|
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
- (dispatch_queue_t)methodQueue
|
|
{
|
|
return dispatch_get_main_queue();
|
|
}
|
|
|
|
- (void)startObserving
|
|
{
|
|
[[NSNotificationCenter defaultCenter] addObserver:self
|
|
selector:@selector(handleOpenURLNotification:)
|
|
name:kOpenURLNotification
|
|
object:nil];
|
|
}
|
|
|
|
- (void)stopObserving
|
|
{
|
|
[[NSNotificationCenter defaultCenter] removeObserver:self];
|
|
}
|
|
|
|
- (NSArray<NSString *> *)supportedEvents
|
|
{
|
|
return @[@"url"];
|
|
}
|
|
|
|
+ (BOOL)application:(UIApplication *)app
|
|
openURL:(NSURL *)URL
|
|
options:(NSDictionary<UIApplicationOpenURLOptionsKey,id> *)options
|
|
{
|
|
postNotificationWithURL(URL, self);
|
|
return YES;
|
|
}
|
|
|
|
// Corresponding api deprecated in iOS 9
|
|
+ (BOOL)application:(UIApplication *)application
|
|
openURL:(NSURL *)URL
|
|
sourceApplication:(NSString *)sourceApplication
|
|
annotation:(id)annotation
|
|
{
|
|
postNotificationWithURL(URL, self);
|
|
return YES;
|
|
}
|
|
|
|
+ (BOOL)application:(UIApplication *)application
|
|
continueUserActivity:(NSUserActivity *)userActivity
|
|
restorationHandler:
|
|
#if defined(__IPHONE_OS_VERSION_MAX_ALLOWED) && (__IPHONE_OS_VERSION_MAX_ALLOWED >= 12000) /* __IPHONE_12_0 */
|
|
(nonnull void (^)(NSArray<id<UIUserActivityRestoring>> *_Nullable))restorationHandler {
|
|
#else
|
|
(nonnull void (^)(NSArray *_Nullable))restorationHandler {
|
|
#endif
|
|
if ([userActivity.activityType isEqualToString:NSUserActivityTypeBrowsingWeb]) {
|
|
NSDictionary *payload = @{@"url": userActivity.webpageURL.absoluteString};
|
|
[[NSNotificationCenter defaultCenter] postNotificationName:kOpenURLNotification
|
|
object:self
|
|
userInfo:payload];
|
|
}
|
|
return YES;
|
|
}
|
|
|
|
- (void)handleOpenURLNotification:(NSNotification *)notification
|
|
{
|
|
[self sendEventWithName:@"url" body:notification.userInfo];
|
|
}
|
|
|
|
RCT_EXPORT_METHOD(openURL:(NSURL *)URL
|
|
resolve:(RCTPromiseResolveBlock)resolve
|
|
reject:(RCTPromiseRejectBlock)reject)
|
|
{
|
|
if (@available(iOS 10.0, *)) {
|
|
[RCTSharedApplication() openURL:URL options:@{} completionHandler:^(BOOL success) {
|
|
if (success) {
|
|
resolve(@YES);
|
|
} else {
|
|
reject(RCTErrorUnspecified, [NSString stringWithFormat:@"Unable to open URL: %@", URL], nil);
|
|
}
|
|
}];
|
|
} else {
|
|
#if !TARGET_OS_UIKITFORMAC
|
|
// Note: this branch will never be taken on UIKitForMac
|
|
BOOL opened = [RCTSharedApplication() openURL:URL];
|
|
if (opened) {
|
|
resolve(@YES);
|
|
} else {
|
|
reject(RCTErrorUnspecified, [NSString stringWithFormat:@"Unable to open URL: %@", URL], nil);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
}
|
|
|
|
RCT_EXPORT_METHOD(canOpenURL:(NSURL *)URL
|
|
resolve:(RCTPromiseResolveBlock)resolve
|
|
reject:(__unused RCTPromiseRejectBlock)reject)
|
|
{
|
|
if (RCTRunningInAppExtension()) {
|
|
// Technically Today widgets can open urls, but supporting that would require
|
|
// a reference to the NSExtensionContext
|
|
resolve(@NO);
|
|
return;
|
|
}
|
|
|
|
// This can be expensive, so we deliberately don't call on main thread
|
|
BOOL canOpen = [RCTSharedApplication() canOpenURL:URL];
|
|
NSString *scheme = [URL scheme];
|
|
if (canOpen) {
|
|
resolve(@YES);
|
|
} else if (![[scheme lowercaseString] hasPrefix:@"http"] && ![[scheme lowercaseString] hasPrefix:@"https"]) {
|
|
// On iOS 9 and above canOpenURL returns NO without a helpful error.
|
|
// Check if a custom scheme is being used, and if it exists in LSApplicationQueriesSchemes
|
|
NSArray *querySchemes = [[NSBundle mainBundle] objectForInfoDictionaryKey:@"LSApplicationQueriesSchemes"];
|
|
if (querySchemes != nil && ([querySchemes containsObject:scheme] || [querySchemes containsObject:[scheme lowercaseString]])) {
|
|
resolve(@NO);
|
|
} else {
|
|
reject(RCTErrorUnspecified, [NSString stringWithFormat:@"Unable to open URL: %@. Add %@ to LSApplicationQueriesSchemes in your Info.plist.", URL, scheme], nil);
|
|
}
|
|
} else {
|
|
resolve(@NO);
|
|
}
|
|
}
|
|
|
|
RCT_EXPORT_METHOD(getInitialURL:(RCTPromiseResolveBlock)resolve
|
|
reject:(__unused RCTPromiseRejectBlock)reject)
|
|
{
|
|
NSURL *initialURL = nil;
|
|
if (self.bridge.launchOptions[UIApplicationLaunchOptionsURLKey]) {
|
|
initialURL = self.bridge.launchOptions[UIApplicationLaunchOptionsURLKey];
|
|
} else {
|
|
NSDictionary *userActivityDictionary =
|
|
self.bridge.launchOptions[UIApplicationLaunchOptionsUserActivityDictionaryKey];
|
|
if ([userActivityDictionary[UIApplicationLaunchOptionsUserActivityTypeKey] isEqual:NSUserActivityTypeBrowsingWeb]) {
|
|
initialURL = ((NSUserActivity *)userActivityDictionary[@"UIApplicationLaunchOptionsUserActivityKey"]).webpageURL;
|
|
}
|
|
}
|
|
resolve(RCTNullIfNil(initialURL.absoluteString));
|
|
}
|
|
|
|
RCT_EXPORT_METHOD(openSettings:(RCTPromiseResolveBlock)resolve
|
|
reject:(__unused RCTPromiseRejectBlock)reject)
|
|
{
|
|
NSURL *url = [NSURL URLWithString:UIApplicationOpenSettingsURLString];
|
|
if (@available(iOS 10.0, *)) {
|
|
[RCTSharedApplication() openURL:url options:@{} completionHandler:^(BOOL success) {
|
|
if (success) {
|
|
resolve(nil);
|
|
} else {
|
|
reject(RCTErrorUnspecified, @"Unable to open app settings", nil);
|
|
}
|
|
}];
|
|
} else {
|
|
#if !TARGET_OS_UIKITFORMAC
|
|
// Note: This branch will never be taken on UIKitForMac
|
|
BOOL opened = [RCTSharedApplication() openURL:url];
|
|
if (opened) {
|
|
resolve(nil);
|
|
} else {
|
|
reject(RCTErrorUnspecified, @"Unable to open app settings", nil);
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
@end
|