/** * 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 "RCTConvert.h" #import "RCTLog.h" #import "RCTUtils.h" @interface RCTAlertManager() @end @implementation RCTAlertManager { NSMutableArray *_alerts; NSMutableArray *_alertControllers; NSMutableArray *_alertCallbacks; NSMutableArray *> *_alertButtonKeys; } RCT_EXPORT_MODULE() - (dispatch_queue_t)methodQueue { return dispatch_get_main_queue(); } - (void)invalidate { for (UIAlertView *alert in _alerts) { [alert dismissWithClickedButtonIndex:0 animated:YES]; } for (UIAlertController *alertController in _alertControllers) { [alertController.presentingViewController dismissViewControllerAnimated:YES completion:nil]; } } /** * @param {NSDictionary} args Dictionary of the form * * @{ * @"message": @"", * @"buttons": @[ * @{@"": @""}, * @{@"": @""}, * ] * } * 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 = [RCTConvert NSString:args[@"title"]]; NSString *message = [RCTConvert NSString:args[@"message"]]; NSString *type = [RCTConvert NSString:args[@"type"]]; NSDictionaryArray *buttons = [RCTConvert NSDictionaryArray:args[@"buttons"]]; BOOL allowsTextInput = [type isEqual:@"plain-text"]; 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 __IPHONE_OS_VERSION_MIN_REQUIRED < __IPHONE_8_0 // TODO: we've encountered some bug when presenting alerts on top of a window that is subsequently // dismissed. As a temporary solution to this, we'll use UIAlertView preferentially if it's available. BOOL preferAlertView = (!RCTRunningInAppExtension() && UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone); if (preferAlertView || [UIAlertController class] == nil) { UIAlertView *alertView = RCTAlertView(title, nil, self, nil, nil); NSMutableArray *buttonKeys = [[NSMutableArray alloc] initWithCapacity:buttons.count]; if (allowsTextInput) { 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 ++; } if (!_alerts) { _alerts = [NSMutableArray new]; _alertCallbacks = [NSMutableArray new]; _alertButtonKeys = [NSMutableArray new]; } [_alerts addObject:alertView]; [_alertCallbacks addObject:callback ?: ^(__unused id unused) {}]; [_alertButtonKeys addObject:buttonKeys]; [alertView show]; } else #endif { UIViewController *presentingController = RCTKeyWindow().rootViewController; if (presentingController == nil) { RCTLogError(@"Tried to display alert view but there is no application window. args: %@", args); return; } // Walk the chain up to get the topmost modal view controller. If modals are presented, // the root view controller's view might not be in the window hierarchy, and presenting from it will fail. while (presentingController.presentedViewController) { presentingController = presentingController.presentedViewController; } UIAlertController *alertController = [UIAlertController alertControllerWithTitle:title message:nil preferredStyle:UIAlertControllerStyleAlert]; if (allowsTextInput) { [alertController addTextFieldWithConfigurationHandler:^(UITextField *textField) { textField.text = message; }]; } else { alertController.message = message; } 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]; UIAlertActionStyle buttonStyle = [buttonKey isEqualToString:@"cancel"] ? UIAlertActionStyleCancel : UIAlertActionStyleDefault; UITextField *textField = allowsTextInput ? alertController.textFields.firstObject : nil; [alertController addAction:[UIAlertAction actionWithTitle:buttonTitle style:buttonStyle handler:^(__unused UIAlertAction *action) { if (callback) { if (allowsTextInput) { callback(@[buttonKey, textField.text]); } else { callback(@[buttonKey]); } } }]]; } if (!_alertControllers) { _alertControllers = [NSMutableArray new]; } [_alertControllers addObject:alertController]; [presentingController presentViewController:alertController animated:YES completion:nil]; } } #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]; if (alertView.alertViewStyle == UIAlertViewStylePlainTextInput) { callback(@[buttonKeys[buttonIndex], [alertView textFieldAtIndex:0].text]); } else { callback(@[buttonKeys[buttonIndex]]); } [_alerts removeObjectAtIndex:index]; [_alertCallbacks removeObjectAtIndex:index]; [_alertButtonKeys removeObjectAtIndex:index]; } @end