mirror of
https://github.com/status-im/react-native.git
synced 2025-01-26 09:19:10 +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
142 lines
3.7 KiB
Objective-C
142 lines
3.7 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 "RCTAlertManager.h"
|
|
|
|
#import "RCTAssert.h"
|
|
#import "RCTLog.h"
|
|
#import "RCTUtils.h"
|
|
|
|
@interface RCTAlertManager() <UIAlertViewDelegate>
|
|
|
|
@end
|
|
|
|
@implementation RCTAlertManager
|
|
{
|
|
NSMutableArray *_alerts;
|
|
NSMutableArray *_alertCallbacks;
|
|
NSMutableArray *_alertButtonKeys;
|
|
}
|
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
- (instancetype)init
|
|
{
|
|
if ((self = [super init])) {
|
|
_alerts = [NSMutableArray new];
|
|
_alertCallbacks = [NSMutableArray new];
|
|
_alertButtonKeys = [NSMutableArray new];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (dispatch_queue_t)methodQueue
|
|
{
|
|
return dispatch_get_main_queue();
|
|
}
|
|
|
|
- (void)invalidate
|
|
{
|
|
for (UIAlertView *alert in _alerts) {
|
|
[alert dismissWithClickedButtonIndex:0 animated:YES];
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {NSDictionary} args Dictionary of the form
|
|
*
|
|
* @{
|
|
* @"message": @"<Alert message>",
|
|
* @"buttons": @[
|
|
* @{@"<key1>": @"<title1>"},
|
|
* @{@"<key2>": @"<cancelButtonTitle>"},
|
|
* ]
|
|
* }
|
|
* The key from the `buttons` dictionary is passed back in the callback on click.
|
|
* Buttons are displayed in the order they are specified. If "cancel" is used as
|
|
* the button key, it will be differently highlighted, according to iOS UI conventions.
|
|
*/
|
|
RCT_EXPORT_METHOD(alertWithArgs:(NSDictionary *)args
|
|
callback:(RCTResponseSenderBlock)callback)
|
|
{
|
|
NSString *title = args[@"title"];
|
|
NSString *message = args[@"message"];
|
|
NSString *type = args[@"type"];
|
|
NSArray *buttons = args[@"buttons"];
|
|
|
|
if (!title && !message) {
|
|
RCTLogError(@"Must specify either an alert title, or message, or both");
|
|
return;
|
|
} else if (buttons.count == 0) {
|
|
RCTLogError(@"Must have at least one button.");
|
|
return;
|
|
}
|
|
|
|
if (RCTRunningInAppExtension()) {
|
|
return;
|
|
}
|
|
|
|
UIAlertView *alertView = RCTAlertView(title, nil, self, nil, nil);
|
|
NSMutableArray *buttonKeys = [[NSMutableArray alloc] initWithCapacity:buttons.count];
|
|
|
|
if ([type isEqualToString:@"plain-text"]) {
|
|
alertView.alertViewStyle = UIAlertViewStylePlainTextInput;
|
|
[alertView textFieldAtIndex:0].text = message;
|
|
} else {
|
|
alertView.message = message;
|
|
}
|
|
|
|
NSInteger index = 0;
|
|
for (NSDictionary *button in buttons) {
|
|
if (button.count != 1) {
|
|
RCTLogError(@"Button definitions should have exactly one key.");
|
|
}
|
|
NSString *buttonKey = button.allKeys.firstObject;
|
|
NSString *buttonTitle = [button[buttonKey] description];
|
|
[alertView addButtonWithTitle:buttonTitle];
|
|
if ([buttonKey isEqualToString: @"cancel"]) {
|
|
alertView.cancelButtonIndex = index;
|
|
}
|
|
[buttonKeys addObject:buttonKey];
|
|
index ++;
|
|
}
|
|
|
|
[_alerts addObject:alertView];
|
|
[_alertCallbacks addObject:callback ?: ^(__unused id unused) {}];
|
|
[_alertButtonKeys addObject:buttonKeys];
|
|
|
|
[alertView show];
|
|
}
|
|
|
|
#pragma mark - UIAlertViewDelegate
|
|
|
|
- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex
|
|
{
|
|
NSUInteger index = [_alerts indexOfObject:alertView];
|
|
RCTAssert(index != NSNotFound, @"Dismissed alert was not recognised");
|
|
|
|
RCTResponseSenderBlock callback = _alertCallbacks[index];
|
|
NSArray *buttonKeys = _alertButtonKeys[index];
|
|
NSArray *args;
|
|
|
|
if (alertView.alertViewStyle == UIAlertViewStylePlainTextInput) {
|
|
args = @[buttonKeys[buttonIndex], [alertView textFieldAtIndex:0].text];
|
|
} else {
|
|
args = @[buttonKeys[buttonIndex]];
|
|
}
|
|
|
|
callback(args);
|
|
|
|
[_alerts removeObjectAtIndex:index];
|
|
[_alertCallbacks removeObjectAtIndex:index];
|
|
[_alertButtonKeys removeObjectAtIndex:index];
|
|
}
|
|
|
|
@end
|