forked from facebook/react-native
-
Notifications
You must be signed in to change notification settings - Fork 167
Expand file tree
/
Copy pathRCTLinkingManager.mm
More file actions
123 lines (99 loc) · 3.5 KB
/
RCTLinkingManager.mm
File metadata and controls
123 lines (99 loc) · 3.5 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
/**
* Copyright (c) 2015-present, Facebook, Inc.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*/
// [macOS]
#import "RCTLinkingManager.h"
#import <FBReactNativeSpec/FBReactNativeSpec.h>
#import <React/RCTBridge.h>
#import <React/RCTEventDispatcher.h>
#import <React/RCTUtils.h>
#import "RCTLinkingPlugins.h"
NSString *const RCTOpenURLNotification = @"RCTOpenURLNotification";
static NSString *initialURL = nil;
static BOOL moduleInitalized = NO;
static BOOL alwaysForegroundLastWindow = YES;
static void postNotificationWithURL(NSString *url, id sender)
{
NSDictionary<NSString *, id> *payload = @{@"url": url};
[[NSNotificationCenter defaultCenter] postNotificationName:RCTOpenURLNotification
object:sender
userInfo:payload];
}
@implementation RCTLinkingManager
RCT_EXPORT_MODULE()
- (dispatch_queue_t)methodQueue
{
return dispatch_get_main_queue();
}
- (void)startObserving
{
moduleInitalized = YES;
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(handleOpenURLNotification:)
name:RCTOpenURLNotification
object:nil];
}
- (void)stopObserving
{
moduleInitalized = NO;
[[NSNotificationCenter defaultCenter] removeObserver:self];
}
- (NSArray<NSString *> *)supportedEvents
{
return @[@"url"];
}
+ (void)setAlwaysForegroundLastWindow:(BOOL)alwaysForeground
{
alwaysForegroundLastWindow = alwaysForeground;
}
+ (void)getUrlEventHandler:(NSAppleEventDescriptor *)event withReplyEvent:(NSAppleEventDescriptor *)replyEvent
{
// extract url value from the event
NSString* url = [[event paramDescriptorForKeyword:keyDirectObject] stringValue];
// If the application was launched via URL, this handler will be called before
// the module is initialized by the bridge. Store the initial URL, becase we are not listening to the notification yet.
if (!moduleInitalized && initialURL == nil) {
initialURL = url;
}
postNotificationWithURL(url, self);
}
- (void)handleOpenURLNotification:(NSNotification *)notification
{
// Activate app, because [NSApp mainWindow] returns nil when the app is hidden and another app is maximized
[NSApp activateIgnoringOtherApps:YES];
// foreground top level window
if (alwaysForegroundLastWindow) {
NSWindow *lastWindow = [[NSApp windows] lastObject];
[lastWindow makeKeyAndOrderFront:nil];
}
[self sendEventWithName:@"url" body:notification.userInfo];
}
RCT_EXPORT_METHOD(openURL:(NSURL *)URL
resolve:(RCTPromiseResolveBlock)resolve
reject:(RCTPromiseRejectBlock)reject)
{
BOOL result = [[NSWorkspace sharedWorkspace] openURL:URL];
if (result) {
resolve(@YES);
} else {
reject(RCTErrorUnspecified, [NSString stringWithFormat:@"Unable to open URL: %@", URL], nil);
}
}
RCT_EXPORT_METHOD(canOpenURL:(NSURL *)URL
resolve:(RCTPromiseResolveBlock)resolve
reject:(__unused RCTPromiseRejectBlock)reject)
{
resolve(@YES);
}
RCT_EXPORT_METHOD(getInitialURL:(RCTPromiseResolveBlock)resolve
reject:(__unused RCTPromiseRejectBlock)reject)
{
resolve(RCTNullIfNil(initialURL));
}
@end
Class RCTLinkingManagerCls(void) {
return RCTLinkingManager.class;
}