web-dev-qa-db-fra.com

Comment utiliser UIAlertController pour remplacer UIActionSheet?

Je maintiens un ancien projet iOS basé sur SDK 6.0.

Une méthode sur ce projet appelée

-(void) showComboBox:(UIView*)view:withOptions:(NSDictionary*)options

est utilisé pour afficher une liste déroulante. Pour atteindre cet objectif, il a utilisé UIActionSheet, qui est obsolète sur iOS8.

Ma solution est la suivante:

        if (floor(NSFoundationVersionNumber) > NSFoundationVersionNumber10_8) {
        UIAlertController* alertController = [UIAlertController 
           alertControllerWithTitle:@"title" 
           message:@"message" 
           preferredStyle:UIAlertControllerStyleActionSheet];

        UIAlertAction* item = [UIAlertAction actionWithTitle:@"item" 
           style:UIAlertActionStyleDefault 
           handler:^(UIAlertAction *action) {
            //do something here 
            //inform the selection to the WebView 
            ...
            [alertController dismissViewControllerAnimated:YES completion:nil];
        }];

        UIAlertAction* cancelAction = [UIAlertAction actionWithTitle:@"Cancel" style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) {
            [alertController dismissViewControllerAnimated:YES completion:nil];
        }];

        [alertController addAction:item];
        [alertController addAction:cancelAction];
        //I am not sure whether it's the right way
        if ([view.nextResponder isKindOfClass:UIViewController.class]) {
            UIViewController* vc = (UIViewController*)view.nextResponder;
            [vc presentViewController:alertController animated:YES completion:nil];
        }

Est-ce une solution appropriée?

C’est ce que j’ai principalement préoccupation concernant: UIAlertController doit être ajouté à un UIViewController mais je ne peux obtenir que le pointeur de UIView. J’ai donc utilisé view.nextResponder pour obtenir ce que je veux, mais c’est un bon moyen?

48
Don_Chen

J'ai utilisé le code suivant pour afficher la feuille d'action à l'aide de UIAlertViewController et cela fonctionne parfaitement.

- (IBAction)buttonClicked:(id)sender {

    UIAlertController *actionSheet = [UIAlertController alertControllerWithTitle:@"Action Sheet" message:@"Using the alert controller" preferredStyle:UIAlertControllerStyleActionSheet];

    [actionSheet addAction:[UIAlertAction actionWithTitle:@"Cancel" style:UIAlertActionStyleCancel handler:^(UIAlertAction *action) {

        // Cancel button tappped.
        [self dismissViewControllerAnimated:YES completion:^{
        }];
    }]];

    [actionSheet addAction:[UIAlertAction actionWithTitle:@"Delete" style:UIAlertActionStyleDestructive handler:^(UIAlertAction *action) {

        // Distructive button tapped.
        [self dismissViewControllerAnimated:YES completion:^{
        }];
    }]];

    [actionSheet addAction:[UIAlertAction actionWithTitle:@"OK" style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) {

        // OK button tapped.

        [self dismissViewControllerAnimated:YES completion:^{
        }];
    }]];

    // Present action sheet.
    [self presentViewController:actionSheet animated:YES completion:nil];
}

Modifier:

Vous devez obtenir l'objet UIViewController ici. Vous pouvez définir une variable globale ou appeler une méthode de délégation, ou vous pouvez utiliser notification pour obtenir un objet contrôleur de vue dans ce code.

et la dernière ligne dans le code ci-dessus sera comme.

[self.viewController presentViewController:actionSheet animated:YES completion:nil];

self.viewController est une variable globale qui sera définie avant que vous obteniez réellement cette vue. 

Parce que l'approche que vous suivez maintenant utilise view.nextResponder. J'ai peur que cela ne fonctionne pas.

65
Kampai

J'ai utilisé la feuille d'action pour changer de photo de profil. Je suivais l'approche de Kampai, je venais de retirer l'appel de licenciementContrôleur puisqu'il me renvoyait d'une vue lorsque j'appuyais sur Annuler ou sur la vue de sélection de photo

UIAlertController *actionSheet = [UIAlertController alertControllerWithTitle:nil message:nil preferredStyle:UIAlertControllerStyleActionSheet];

[actionSheet addAction:[UIAlertAction actionWithTitle:@"Cancel" style:UIAlertActionStyleCancel handler:^(UIAlertAction *action) {

    // Cancel button tappped do nothing.

}]];

[actionSheet addAction:[UIAlertAction actionWithTitle:@"Take photo" style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) {

    // take photo button tapped.
    [self takePhoto];

}]];

[actionSheet addAction:[UIAlertAction actionWithTitle:@"Choose photo" style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) {

    // choose photo button tapped.
    [self choosePhoto];

}]];

[actionSheet addAction:[UIAlertAction actionWithTitle:@"Delete Photo" style:UIAlertActionStyleDestructive handler:^(UIAlertAction *action) {

    // Distructive button tapped.
    [self deletePhoto];

}]];
28
vladicabg

Mise à jour rapide -

    let actionSheet = UIAlertController.init(title: "Please choose a source type", message: nil, preferredStyle: .actionSheet)
    actionSheet.addAction(UIAlertAction.init(title: "Take Photo", style: UIAlertActionStyle.default, handler: { (action) in
        self.openCamera()
    }))
    actionSheet.addAction(UIAlertAction.init(title: "Choose Photo", style: UIAlertActionStyle.default, handler: { (action) in
        self.showPhotoLibrary()
    }))
    actionSheet.addAction(UIAlertAction.init(title: "Cancel", style: UIAlertActionStyle.cancel, handler: { (action) in
        // self.dismissViewControllerAnimated(true, completion: nil) is not needed, this is handled automatically,
         //Plus whatever method you define here, gets called,
        //If you tap outside the UIAlertController action buttons area, then also this handler gets called.
    }))
    //Present the controller
    self.present(actionSheet, animated: true, completion: nil)
14
Saurabh Yadav

Swift 4

        let alert = UIAlertController(title: "Select One", message: nil, preferredStyle: UIAlertControllerStyle.actionSheet)
        alert.addAction(UIAlertAction(title: "Cancel", style: UIAlertActionStyle.cancel, handler: nil))
        alert.addAction(UIAlertAction(title: "Export", style: UIAlertActionStyle.default, handler: { (action) in

            // TODO: Export wordlist

        }))
        alert.addAction(UIAlertAction(title: "Import", style: UIAlertActionStyle.default, handler: { (action) in

            // TODO: Import wordlist
        }))

        self.present(alert, animated: true, completion: nil)
4

Bien que cela puisse sembler très simple, l'utilisation de UIAlertController pose un vilain problème. C'est une fuite de mémoire sujette. Pour vérifier si vous avez le problème, il suffit de placer un point d'arrêt dans la méthode dealloc de votre contrôleur de vue et de voir s'il est désalloué correctement. 

Je cherchais une solution depuis un certain temps et voici comment utiliser un contrôleur d'alerte dans mon application.

+ (void)alertWithPresenting:(UIViewController *)presenting title:(NSString *)title
                       text:(NSString *)text buttons:(NSArray *)buttons
                    handler:(void (^)(UIAlertAction *action, NSUInteger index))handler
{
        UIAlertController *alert = [UIAlertController
                                    alertControllerWithTitle:title message:text
                                    preferredStyle:UIAlertControllerStyleAlert];
        __weak __typeof(alert) weakAlert = alert;
        for (NSString *title in buttons) {
                UIAlertActionStyle style = UIAlertActionStyleDefault;
                if ([title isEqualToString:[L10n cancelButton]])
                        style = UIAlertActionStyleCancel;
                else if ([title isEqualToString:[L10n deleteButton]])
                        style = UIAlertActionStyleDestructive;
                else if ([title isEqualToString:[L10n archiveButton]])
                        style = UIAlertActionStyleDestructive;

                UIAlertAction *action = [UIAlertAction actionWithTitle:title style:style handler:^(UIAlertAction *action) {
                        if (handler != nil)
                                handler(action, [buttons indexOfObject:action.title]);
                        [weakAlert dismissViewControllerAnimated:YES completion:nil];
                }];
                [alert addAction:action];
        }
        [presenting presentViewController:alert animated:YES completion:nil];
}

Ce n'est pas tout. Voici un exemple d'utilisation de celui-ci dans votre contrôleur de vue. Dans mon cas, c'est tableview avec recherche, donc le contrôleur de présentation peut être différent.

- (void) deleteCases:(NSArray *)selectedRows
{
        NSString *text = NSLocalizedStringWithDefaultValue(@"cases.delete.alert.text",
                                                           @"Localizable",  [NSBundle mainBundle],
                                                           @"Deleted cases cannot be restored. Continue with delete?",
                                                           @"Delete alert text");
        NSString *title = NSLocalizedStringWithDefaultValue(@"cases.delete.alert.title",
                                                            @"Localizable",  [NSBundle mainBundle],
                                                            @"Delete cases", @"Detete alert title");
        UIViewController *presenting = self.searchController.active ? self.searchController : self;
        __weak __typeof(presenting) weakPresenting = presenting;
        __weak __typeof(self) weakSelf = self;
        [YourClassName alertWithPresenting:weakPresenting title:title text:text
                                   buttons:@[[L10n deleteButton], [L10n cancelButton]]
                                   handler:^(UIAlertAction *action, NSUInteger index)
        {
                if (action.style == UIAlertActionStyleDestructive) {
                        __typeof(weakSelf) strongSelf = weakSelf;
                        // Perform your actions using @strongSelf
                }
         }];
}
0
Cynichniy Bandera

Vous pouvez utiliser view.window.rootViewController à la place. Si vous ne vous souciez pas du présentateur, c'est bien. 

0
Roman Temchenko