在很多种语言中,alert都表示弹窗操作,弹窗功能非常有用,不仅可以用于正式的app功能中,也可以在调试中使用。在OC中,UIAlertController类用来控制弹窗操作。在IOS 8.0之前, 其使用UIAlertView类来控制弹窗,如果我们想弹出一个带有输入框的窗口,可以使用如下的代码:
// Prior to iOS 8.0 UIAlertView *alert = [[UIAlertView init] initWithTitle:@"Alert Title" message:@"Alert Message" delegate:self cancelButtonTitle:@"Cancel" otherButtonTitles:@"OK", nil]; alert.alertViewStyle = UIAlertViewStylePlainTextInput; [alert textFieldAtIndex:0].text = @"Show Text Here"; [alert show]; - (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex { if (buttonIndex == 0) return; // Do something after clicking OK button }
我们可以看出如果要实现弹窗中的按钮功能,需要使用代理模式,功能代码需要写在clickedButtonAtIndex()函数中,这种写法固然可以实现功能,但是在某些情况下并不是很方便。比如说我们有一个UITableView,我们点击任意某一行进行弹窗操作,那么上面声明弹窗的代码需要写在didSelectRowAtIndexPath()函数中,但是由于弹窗按钮的操作在另一个代理函数中,我们不能直接访问当前行数indexPath.row,只能将其存入一个全局变量中,或者我们在声明完弹窗后,将当前行数绑定到alert的tag上,这样在代理函数中才可以直接使用。而且我们如果要用reloadRowsAtIndexPaths函数来更新当前行的时候,还需要自己创建一个包含indexPath类对象的数组,略显麻烦。
而在IOS 8.0之后,UIAlertView被抛弃了,取而代之的是UIAlertController类,我们想实现相同功能的弹窗需要使用如下的代码:
// After iOS 8.0 UIAlertController *alert = [UIAlertController alertControllerWithTitle:@"Alert Title" message:@"Alert Message" preferredStyle:UIAlertControllerStyleAlert]; [alert addTextFieldWithConfigurationHandler:^(UITextField *textField) { textField.placeholder = NSLocalizedString(@"TextPlaceholder", @"Text"); }]; UIAlertAction *okButton = [UIAlertAction actionWithTitle:@"OK" style:UIAlertActionStyleDefault handler:^(UIAlertAction *action) { // Do something after clicking OK button UITextField *textField = alert.textFields.firstObject; }]; UIAlertAction *cancelButton = [UIAlertAction actionWithTitle:@"Cancel" style:UIAlertActionStyleCancel handler:^(UIAlertAction *action) { // Do something after clicking Cancel button }]; [alert addAction:okButton]; [alert addAction:cancelButton]; [self presentViewController:alert animated:YES completion:nil];
我们发现貌似写法变得更复杂了,虽然代码量多了点,但是却完全摒弃了代理模式,而且添加按钮和输入框的方式变的更加*了,按钮功能代码也不必写在其他的代理函数中,整个结构一目了然,面向对象的感觉扑面而来,感觉很叼。
参考资料: