1

我正在将一些自 iOS 9.0 以来已弃用的 UIAlertViews 更新为 UIAlertViewControllers。

使用 UIAlertView,可以从任何正在执行的代码中抛出警报——即使是在实用程序类或共享实例中——只需一行:

[alertView show];

因此,如果我调用共享实例,例如

- (void)didTapDeleteButton:(id)sender {
    NSArray *itemsToDelete = [self.selectedIndexPathToContact allValues];

    [[IDModel sharedInstance] deleteItems:itemsToDelete];

//其中包含代码:

UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"Keep Archive Copy?"
                                                    message:nil
                                                   delegate:self
                                          cancelButtonTitle:@"No"
                                          otherButtonTitles:@"OK",nil];
alertInvite.alertViewStyle = UIAlertViewStyleDefault;
alertInvite.tag=100;
[alertView show];

一切正常。

但是,对于 UIAlertController,这是不允许的。如果您将以下代码放在可通过共享实例访问的类的方法中,当您到达 presentViewController 时,它会引发错误:

UIAlertController *alertView = [UIAlertController alertControllerWithTitle:@"Delete Item?" message:nil preferredStyle:UIAlertControllerStyleAlert];

UIAlertAction* yesButton = [UIAlertAction actionWithTitle:@"OK" style:UIAlertActionStyleDefault handler:^(UIAlertAction * action) {
    [alertView dismissViewControllerAnimated:YES completion:nil];
}];

UIAlertAction* noButton = [UIAlertAction actionWithTitle:@"Not Now" style:UIAlertActionStyleDefault handler:^(UIAlertAction * action) {
    [alertView dismissViewControllerAnimated:YES completion:nil];
}];

[alertView addAction:noButton];
[alertView addAction:yesButton];
if ([alertView respondsToSelector:@selector(setPreferredAction:)]) {
    [alertView setPreferredAction:yesButton];
}
//FOLLOWING THROWS ERROR
[self presentViewController:alertView animated:YES completion:nil];

在最后一行,该类(通过共享实例访问)没有此方法。看来您必须使用更复杂的方式来引发警报。我见过一些复杂的方法,例如:

id rootViewController = [UIApplication sharedApplication].delegate.window.rootViewController;
if([rootViewController isKindOfClass:[UINavigationController class]])
{
    rootViewController = ((UINavigationController *)rootViewController).viewControllers.firstObject;
}
if([rootViewController isKindOfClass:[UITabBarController class]])
{
    rootViewController = ((UITabBarController *)rootViewController).selectedViewController;
}
[rootViewController presentViewController:alertInvite animated:YES completion:nil];

但是,这对我不起作用,因为我认为我的共享实例没有 rootviewcontroller。任何人都可以提出一个简单、直接的方法来做到这一点吗?

4

3 回答 3

1

从我能想到的任何代码中显示警报:

UIAlertController *alert = [UIAlertController alertControllerWithTitle:@"Alert" message:@"message" preferredStyle:UIAlertControllerStyleAlert];
[alert addAction: [UIAlertAction actionWithTitle:@"Yes" style:UIAlertActionStyleDefault handler:nil]];
alert.TitleColor = [UIColor whiteColor];

id<UIApplicationDelegate> delegate = [UIApplication sharedApplication].delegate;
UIViewController *vc = delegate.window.rootViewController;
[vc presentViewController:alert animated:YES completion:nil];

笔记:

请注意,在大多数情况下,我不会这样做。
非ui代码不应该做ui!这可能也是苹果做出改变的部分原因:它鼓励适当的模型||视图分离

于 2018-08-22T16:03:34.870 回答
1

我在 UIViewController 上创建了一个扩展,它允许我创建一个新窗口并从那里呈现一个视图控制器。这使我可以从任何类进行演示,而不仅仅是视图控制器。此外,它还可以防止您尝试从已经呈现视图控制器的视图控制器显示警报视图的问题。

extension UIViewController {
    func presentFromNewWindow(animated: Bool = true, completion: (() -> Void)? = nil) {
        let window = newWindow()

        if let rootViewController = window.rootViewController {
            window.makeKeyAndVisible()
            rootViewController.present(self, animated: animated, completion: completion)
        }
    }

    private func newWindow() -> UIWindow {
        let window = UIWindow(frame: UIScreen.main.bounds)
        let rootViewController = UIViewController()
        rootViewController.view.backgroundColor = .clear
        window.backgroundColor = .clear
        window.rootViewController = rootViewController
        window.windowLevel = UIWindowLevelAlert

        return window
    }
}

然后,您可以使用此方法来呈现您的警报控制器(或任何 UIViewController):

alertViewController.presentFromNewWindow()

当您关闭警报视图控制器时,它会从视图控制器中移除,并且窗口也会从层次结构中移除。

于 2018-08-22T16:46:56.283 回答
0

我相信你的问题很模糊。但我认为你一直在寻找的是一个展示UIAlertController. 那是对的吗?如果是这样,请继续阅读...

例子:

UIAlertController *alert = [UIAlertController alertControllerWithTitle:@"Alert" message:@"message" preferredStyle:UIAlertControllerStyleAlert];
[alert addAction: [UIAlertAction actionWithTitle:@"Yes" style:UIAlertActionStyleDefault handler:nil]];
alert.TitleColor = [UIColor whiteColor];
[self presentViewController:alert animated:YES completion:nil];

文档:https ://developer.apple.com/documentation/uikit/uialertcontroller

于 2018-08-22T15:54:38.150 回答