2015-03-23 20:28:42 +00:00
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*/
|
2015-02-20 04:10:52 +00:00
|
|
|
|
|
|
|
#import "RCTAlertManager.h"
|
|
|
|
|
2015-04-07 14:36:26 +00:00
|
|
|
#import "RCTAssert.h"
|
2015-11-03 22:45:46 +00:00
|
|
|
#import "RCTConvert.h"
|
2015-02-20 04:10:52 +00:00
|
|
|
#import "RCTLog.h"
|
2015-09-22 17:43:56 +00:00
|
|
|
#import "RCTUtils.h"
|
2015-02-20 04:10:52 +00:00
|
|
|
|
2015-12-01 02:44:06 +00:00
|
|
|
@implementation RCTConvert (UIAlertViewStyle)
|
|
|
|
|
|
|
|
RCT_ENUM_CONVERTER(UIAlertViewStyle, (@{
|
|
|
|
@"default": @(UIAlertViewStyleDefault),
|
|
|
|
@"secure-text": @(UIAlertViewStyleSecureTextInput),
|
|
|
|
@"plain-text": @(UIAlertViewStylePlainTextInput),
|
|
|
|
@"login-password": @(UIAlertViewStyleLoginAndPasswordInput),
|
|
|
|
}), UIAlertViewStyleDefault, integerValue)
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
2015-02-20 04:10:52 +00:00
|
|
|
@interface RCTAlertManager() <UIAlertViewDelegate>
|
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation RCTAlertManager
|
|
|
|
{
|
2015-11-03 22:45:46 +00:00
|
|
|
NSMutableArray<UIAlertView *> *_alerts;
|
|
|
|
NSMutableArray<UIAlertController *> *_alertControllers;
|
|
|
|
NSMutableArray<RCTResponseSenderBlock> *_alertCallbacks;
|
|
|
|
NSMutableArray<NSArray<NSString *> *> *_alertButtonKeys;
|
2015-02-20 04:10:52 +00:00
|
|
|
}
|
|
|
|
|
2015-04-08 12:42:43 +00:00
|
|
|
RCT_EXPORT_MODULE()
|
|
|
|
|
2015-04-20 19:06:02 +00:00
|
|
|
- (dispatch_queue_t)methodQueue
|
|
|
|
{
|
|
|
|
return dispatch_get_main_queue();
|
|
|
|
}
|
|
|
|
|
2015-09-18 15:40:49 +00:00
|
|
|
- (void)invalidate
|
|
|
|
{
|
2015-10-30 18:23:47 +00:00
|
|
|
for (UIAlertView *alert in _alerts) {
|
|
|
|
[alert dismissWithClickedButtonIndex:0 animated:YES];
|
|
|
|
}
|
|
|
|
for (UIAlertController *alertController in _alertControllers) {
|
|
|
|
[alertController.presentingViewController dismissViewControllerAnimated:YES completion:nil];
|
|
|
|
}
|
2015-09-18 15:40:49 +00:00
|
|
|
}
|
|
|
|
|
2015-02-20 04:10:52 +00:00
|
|
|
/**
|
|
|
|
* @param {NSDictionary} args Dictionary of the form
|
|
|
|
*
|
|
|
|
* @{
|
|
|
|
* @"message": @"<Alert message>",
|
|
|
|
* @"buttons": @[
|
|
|
|
* @{@"<key1>": @"<title1>"},
|
2015-12-01 02:44:06 +00:00
|
|
|
* @{@"<key2>": @"<title2>"},
|
|
|
|
* ],
|
|
|
|
* @"cancelButtonKey": @"<key2>",
|
2015-02-20 04:10:52 +00:00
|
|
|
* }
|
|
|
|
* The key from the `buttons` dictionary is passed back in the callback on click.
|
2015-12-01 02:44:06 +00:00
|
|
|
* Buttons are displayed in the order they are specified.
|
2015-02-20 04:10:52 +00:00
|
|
|
*/
|
2015-04-08 15:52:48 +00:00
|
|
|
RCT_EXPORT_METHOD(alertWithArgs:(NSDictionary *)args
|
|
|
|
callback:(RCTResponseSenderBlock)callback)
|
2015-02-20 04:10:52 +00:00
|
|
|
{
|
2015-11-03 22:45:46 +00:00
|
|
|
NSString *title = [RCTConvert NSString:args[@"title"]];
|
|
|
|
NSString *message = [RCTConvert NSString:args[@"message"]];
|
2015-12-01 02:44:06 +00:00
|
|
|
UIAlertViewStyle type = [RCTConvert UIAlertViewStyle:args[@"type"]];
|
|
|
|
NSArray<NSDictionary *> *buttons = [RCTConvert NSDictionaryArray:args[@"buttons"]];
|
2016-03-15 11:25:43 +00:00
|
|
|
NSString *defaultValue = [RCTConvert NSString:args[@"defaultValue"]];
|
2015-12-01 02:44:06 +00:00
|
|
|
NSString *cancelButtonKey = [RCTConvert NSString:args[@"cancelButtonKey"]];
|
|
|
|
NSString *destructiveButtonKey = [RCTConvert NSString:args[@"destructiveButtonKey"]];
|
2015-02-20 04:10:52 +00:00
|
|
|
|
|
|
|
if (!title && !message) {
|
2015-03-01 23:33:55 +00:00
|
|
|
RCTLogError(@"Must specify either an alert title, or message, or both");
|
2015-02-20 04:10:52 +00:00
|
|
|
return;
|
2015-12-01 02:44:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (buttons.count == 0) {
|
|
|
|
if (type == UIAlertViewStyleDefault) {
|
|
|
|
buttons = @[@{@"0": RCTUIKitLocalizedString(@"OK")}];
|
|
|
|
cancelButtonKey = @"0";
|
|
|
|
} else {
|
|
|
|
buttons = @[
|
|
|
|
@{@"0": RCTUIKitLocalizedString(@"OK")},
|
|
|
|
@{@"1": RCTUIKitLocalizedString(@"Cancel")},
|
|
|
|
];
|
|
|
|
cancelButtonKey = @"1";
|
|
|
|
}
|
2015-02-20 04:10:52 +00:00
|
|
|
}
|
2015-10-30 18:23:47 +00:00
|
|
|
|
2016-09-02 02:37:54 +00:00
|
|
|
UIViewController *presentingController = RCTPresentedViewController();
|
|
|
|
if (presentingController == nil) {
|
|
|
|
RCTLogError(@"Tried to display alert view but there is no application window. args: %@", args);
|
|
|
|
return;
|
|
|
|
}
|
2015-10-30 18:23:47 +00:00
|
|
|
|
2016-09-02 02:37:54 +00:00
|
|
|
UIAlertController *alertController = [UIAlertController
|
|
|
|
alertControllerWithTitle:title
|
|
|
|
message:nil
|
|
|
|
preferredStyle:UIAlertControllerStyleAlert];
|
|
|
|
switch (type) {
|
|
|
|
case UIAlertViewStylePlainTextInput: {
|
|
|
|
[alertController addTextFieldWithConfigurationHandler:^(UITextField *textField) {
|
|
|
|
textField.secureTextEntry = NO;
|
|
|
|
textField.text = defaultValue;
|
|
|
|
}];
|
|
|
|
break;
|
Simplified AlertIOS
Summary:
Ok, so this started as fixing #5273 but ended up getting a little more complicated. :smile:
Currently, AlertIOS has the following API:
* `alert(title, message, buttons, type)`
* `prompt(title, defaultValue, buttons, callback)`
I've changed the API to look like the following:
* `alert(title, message, callbackOrButtons)`
* `prompt(title, message, callbackOrButtons, type, defaultValue)`
I know that breaking changes are a big deal, but I find the current alert API to be fairly inconsistent and unnecessarily confusing. I'll try to justify my changes one by one:
1. Currently `type` is an optional parameter of `alert`. However, the only reason to change the alert type from the default is in order to create one of the input dialogs (text, password or username/password). So we're in a weird state where if you want a normal text input, you use `prompt`, but if you want a password input you use `alert` with the 'secure-text' type. I've moved `type` to `prompt` so all text input is now done with `pro
Closes https://github.com/facebook/react-native/pull/5286
Reviewed By: svcscm
Differential Revision: D2850400
Pulled By: androidtrunkagent
fb-gh-sync-id: 2986cfa2266225df7e4dcd703fce1e322c12b816
2016-01-21 18:48:58 +00:00
|
|
|
}
|
2016-09-02 02:37:54 +00:00
|
|
|
case UIAlertViewStyleSecureTextInput: {
|
|
|
|
[alertController addTextFieldWithConfigurationHandler:^(UITextField *textField) {
|
|
|
|
textField.placeholder = RCTUIKitLocalizedString(@"Password");
|
|
|
|
textField.secureTextEntry = YES;
|
|
|
|
textField.text = defaultValue;
|
|
|
|
}];
|
|
|
|
break;
|
2015-10-30 18:23:47 +00:00
|
|
|
}
|
2016-09-02 02:37:54 +00:00
|
|
|
case UIAlertViewStyleLoginAndPasswordInput: {
|
|
|
|
[alertController addTextFieldWithConfigurationHandler:^(UITextField *textField) {
|
|
|
|
textField.placeholder = RCTUIKitLocalizedString(@"Login");
|
|
|
|
textField.text = defaultValue;
|
|
|
|
}];
|
|
|
|
[alertController addTextFieldWithConfigurationHandler:^(UITextField *textField) {
|
|
|
|
textField.placeholder = RCTUIKitLocalizedString(@"Password");
|
|
|
|
textField.secureTextEntry = YES;
|
|
|
|
}];
|
|
|
|
break;
|
2015-11-25 11:09:00 +00:00
|
|
|
}
|
2016-09-02 02:37:54 +00:00
|
|
|
case UIAlertViewStyleDefault:
|
|
|
|
break;
|
|
|
|
}
|
2015-11-03 22:45:46 +00:00
|
|
|
|
2016-09-02 02:37:54 +00:00
|
|
|
alertController.message = message;
|
2015-11-03 22:45:46 +00:00
|
|
|
|
2016-09-02 02:37:54 +00:00
|
|
|
for (NSDictionary<NSString *, id> *button in buttons) {
|
|
|
|
if (button.count != 1) {
|
|
|
|
RCTLogError(@"Button definitions should have exactly one key.");
|
2015-11-16 18:20:34 +00:00
|
|
|
}
|
2016-09-02 02:37:54 +00:00
|
|
|
NSString *buttonKey = button.allKeys.firstObject;
|
|
|
|
NSString *buttonTitle = [RCTConvert NSString:button[buttonKey]];
|
|
|
|
UIAlertActionStyle buttonStyle = UIAlertActionStyleDefault;
|
|
|
|
if ([buttonKey isEqualToString:cancelButtonKey]) {
|
|
|
|
buttonStyle = UIAlertActionStyleCancel;
|
|
|
|
} else if ([buttonKey isEqualToString:destructiveButtonKey]) {
|
|
|
|
buttonStyle = UIAlertActionStyleDestructive;
|
2015-04-18 17:43:20 +00:00
|
|
|
}
|
2016-09-02 02:37:54 +00:00
|
|
|
[alertController addAction:[UIAlertAction actionWithTitle:buttonTitle
|
|
|
|
style:buttonStyle
|
|
|
|
handler:^(__unused UIAlertAction *action) {
|
|
|
|
switch (type) {
|
|
|
|
case UIAlertViewStylePlainTextInput:
|
|
|
|
case UIAlertViewStyleSecureTextInput:
|
|
|
|
callback(@[buttonKey, [alertController.textFields.firstObject text]]);
|
|
|
|
break;
|
|
|
|
case UIAlertViewStyleLoginAndPasswordInput: {
|
|
|
|
NSDictionary<NSString *, NSString *> *loginCredentials = @{
|
|
|
|
@"login": [alertController.textFields.firstObject text],
|
|
|
|
@"password": [alertController.textFields.lastObject text]
|
|
|
|
};
|
|
|
|
callback(@[buttonKey, loginCredentials]);
|
|
|
|
break;
|
2015-10-30 18:23:47 +00:00
|
|
|
}
|
2016-09-02 02:37:54 +00:00
|
|
|
case UIAlertViewStyleDefault:
|
|
|
|
callback(@[buttonKey]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}]];
|
|
|
|
}
|
2015-11-25 11:09:00 +00:00
|
|
|
|
2016-09-02 02:37:54 +00:00
|
|
|
if (!_alertControllers) {
|
|
|
|
_alertControllers = [NSMutableArray new];
|
2015-10-30 18:23:47 +00:00
|
|
|
}
|
2016-09-02 02:37:54 +00:00
|
|
|
[_alertControllers addObject:alertController];
|
|
|
|
|
|
|
|
[presentingController presentViewController:alertController animated:YES completion:nil];
|
2015-02-20 04:10:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#pragma mark - UIAlertViewDelegate
|
|
|
|
|
|
|
|
- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex
|
|
|
|
{
|
|
|
|
NSUInteger index = [_alerts indexOfObject:alertView];
|
|
|
|
RCTAssert(index != NSNotFound, @"Dismissed alert was not recognised");
|
2015-03-23 20:28:42 +00:00
|
|
|
|
2015-02-20 04:10:52 +00:00
|
|
|
RCTResponseSenderBlock callback = _alertCallbacks[index];
|
2015-11-03 22:45:46 +00:00
|
|
|
NSArray<NSString *> *buttonKeys = _alertButtonKeys[index];
|
2015-05-11 23:40:04 +00:00
|
|
|
|
2015-12-01 02:44:06 +00:00
|
|
|
switch (alertView.alertViewStyle) {
|
|
|
|
case UIAlertViewStylePlainTextInput:
|
|
|
|
case UIAlertViewStyleSecureTextInput:
|
|
|
|
callback(@[buttonKeys[buttonIndex], [alertView textFieldAtIndex:0].text]);
|
|
|
|
break;
|
|
|
|
case UIAlertViewStyleLoginAndPasswordInput: {
|
|
|
|
NSDictionary<NSString *, NSString *> *loginCredentials = @{
|
|
|
|
@"login": [alertView textFieldAtIndex:0].text,
|
|
|
|
@"password": [alertView textFieldAtIndex:1].text,
|
|
|
|
};
|
|
|
|
callback(@[buttonKeys[buttonIndex], loginCredentials]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case UIAlertViewStyleDefault:
|
|
|
|
callback(@[buttonKeys[buttonIndex]]);
|
|
|
|
break;
|
2015-05-11 23:40:04 +00:00
|
|
|
}
|
|
|
|
|
2015-02-20 04:10:52 +00:00
|
|
|
[_alerts removeObjectAtIndex:index];
|
|
|
|
[_alertCallbacks removeObjectAtIndex:index];
|
|
|
|
[_alertButtonKeys removeObjectAtIndex:index];
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|