标签:
UIAlertView是iOS开发过程中最常用的控件之一,是提醒用户做出选择最主要的工具.在iOS8及后来的系统中,苹果更推荐使用UIAlertController来代替UIAlertView.所以本文也并不提倡开发者再使用UIAlertView,
本文的目的是探讨如何将原来的给变量赋值和通过Delete来回调的方式变成链式编程风格和通过Block来回调.通过学习对UIAlertView的改造让各位iOS开发者能够学会这种更加便捷的开发方式
对于有一定开发经验的开发者来说,链式编程并不陌生.有很多知名的开源库使用了这种编程风格,比如大名鼎鼎的JQuery,而iOS库中,Masory和Snapkit也是典型的使用链式编程的例子.
大体来说,链式编程就是将多个操作(多行代码)通过某种操作符(通常是点号.)链接成一句的代码.便代码更加紧凑,可读性也更好,降低了代码的重复度.比如以下代码:
1 //使用普通的赋值模式 2 txtUserName.placeHolder = "请输入用户名" 3 txtUserName.font = UIFont.systemFont(15) 4 txtUserName.textColor = UIColor.GrayColor() 5 //使用链式编程的语句 6 txtUserName.setPlaceHolder("请输入用户名").setFont(15).setTextColor(UIColor.GrayColor())
通过这个例子读者应该可以更清楚的了解什么是链式编程风格.简单来说,链式编程风格要有以下特点
使用链式编程最主要的好处是可以使代码更简洁,写起来一种"爽快"感.设计优秀的链式编程可以大大降低重复的代码,增强逻辑感.不足之处就是对开
发者的业务逻辑能力要求较高,同时因为链式编程都是调用函数,所以有可能会造成过深的函数调用栈.稍微影响性能.
iOS开发中有很多回调都是使用Delegate完成的,相信各位读者已经写过很多次了.相对来说,使用Delegate比较繁琐,有时还需要在Delegate里
判断是哪个对象的Delegate,优点是运行效率较高.而Block则相反,写起来更直观,开发效率更高.苹果也是逐步使用Block来代替Delegate,iOS
8最新的UIViewController里的Action已经全部使用Block来实现.而且Swift里的闭包,匿名函数更上比比皆是.所以大胆地使用Block来代替
Delegate和Target-Action吧,苹果会帮我们处理好各种性能问题的.但是需要注意的是retian-circle问题,初识Block很容易出现这种问题.
目前有两种方式可以实现将UIAlertView的Delegate改成Block的回调,一是使用运行时给UIAlertView加上Block属性.二是再写一个新的类继承
UIAlertView,本文使用的是第二种方式,写起来更简单一点.
1 class BlockAlert:UIAlertView,UIAlertViewDelegate{ 2 var completion:((buttonIndex:Int,alert:UIAlertView)->Void)? //定义各个Block用来回调,其参数名和Delegate回调的函数参数名一至 3 var willDismissBlock:((buttonIndex:Int,alert:UIAlertView)->Void)? 4 var didDismissBlock:((buttonIndex:Int,alert:UIAlertView)->Void)? 5 var didPresentBlock:((alert:UIAlertView)->Void)? 6 var willPresentBlock:((alert:UIAlertView)->Void)? 7 var alertWithCancelBlock:((alert:UIAlertView)->Void)? 8 override init(frame: CGRect) { 9 super.init(frame: frame) 10 self.delegate = self //将delegate设为自己 11 } 12 func alertView(alertView: UIAlertView, clickedButtonAtIndex buttonIndex: Int) { 13 if let block = completion{ //最主要的Block,当用户点了按钮时回调,先要判断这个Block存在不? 14 block(buttonIndex: buttonIndex, alert: alertView) 15 } 16 } 17 func alertView(alertView: UIAlertView, didDismissWithButtonIndex buttonIndex: Int) { 18 if let block = didDismissBlock{//其他根据相应的Delegate回调函数依次调用各个Block 19 block(buttonIndex: buttonIndex, alert: alertView) 20 } 21 } 22 func alertView(alertView: UIAlertView, willDismissWithButtonIndex buttonIndex: Int) { 23 if let block = willDismissBlock{ 24 block(buttonIndex: buttonIndex, alert: alertView) 25 } 26 } 27 // 其他几个委托方法也省略了,原理都是一样的 28 required init?(coder aDecoder: NSCoder) { 29 fatalError("init(coder:) has not been implemented") 30 } 31 }
参考上面的代码,总体思路是在构造器里面将委托设为自己,再将Delegate里面所有的函数用Block来实现,Block里面的参数和委托回调的函数参数一至.当有委托函数回调时,先判断这个
Block是不是nil,如果不是,则执行该Block
接下来的操作就很容易了
1 extension UIAlertView { 2 static func setMessage(msg:String)->UIAlertView{//在这里设定这个为静态函数,在里面实例化一个BlockAlert对象,再返回该对象 3 let alert = BlockAlert() 4 alert.message = msg 5 return alert 6 } 7 8 func addAlertStyle(style:UIAlertViewStyle)->UIAlertView{ //直接在各个函数中封装一些操作.再返回自身 9 self.alertViewStyle = style 10 return self 11 } 12 13 func addTitle(title:String)->UIAlertView{ 14 self.title = title 15 return self 16 } 17 18 func addFirstButton(btnTitle:String)->UIAlertView{ 19 self.addButtonWithTitle(btnTitle) 20 return self 21 } 22 23 func addSecondButton(btnTitle:String)->UIAlertView{ 24 self.addButtonWithTitle(btnTitle) 25 return self 26 } 27 28 func addButtons(btnTitles:[String])->UIAlertView{ 29 for title in btnTitles{ 30 self.addButtonWithTitle(title) 31 } 32 return self 33 } 34 35 func addButtonClickEvent(clickButton:((buttonIndex:Int,alert:UIAlertView)->Void)?)->UIAlertView{ 36 if let alert = self as? BlockAlert{//对于block,先要判断它是不是BlockAlert,如果是的话才添加该block,后面都是这种操作 37 alert.completion = clickButton 38 } 39 return self 40 } 41 42 //这里也是省略了部分函数,因为原理都是一样的 43 func addAlertCancelEvent(event:((alert:UIAlertView)->Void)?)->UIAlertView{ 44 if let alert = self as? BlockAlert{ 45 alert.alertWithCancelBlock = event 46 } 47 return self 48 } 49 50 51 func alertWithButtonClick(clickButton:((buttonIndex:Int,alert:UIAlertView)->Void)?){ 52 if let alert = self as? BlockAlert{ //这个为终结函数,没有返回值,在里面直接调用了show()方法. 53 alert.completion = clickButton 54 alert.show() 55 } 56 } 57 }
在这里面我选择了用UIAlverView来扩展,而不是BlockAlert,这样可能会导致开发过程中踩进陷阱.因为在首个静态函数返回的对象是一个BlockAction对象,而不是UIAlertView对象
在后面添加Block时先要判断本身是不是BlockAction对象,如果是的话再将设定Block,所以在开发过程中要小心这个陷阱.当然读者也可以直接扩展BlockAlert,这样会更安全一些.
UIAlertView.setMessage("这里要设置信息").addTitle("我是标题").addFirstButton("取消").addSecondButton("确认").alertWithButtonClick { (buttonIndex, alert) -> Void in print("你按下了第\(buttonIndex)个按键") //你按下了第1个按键 //你按下了第0个按键 }
如果要使用其他的委托方法也很简单
1 UIAlertView.setMessage("这里要设置信息").addTitle("我是标题").addFirstButton("取消").addSecondButton("确认").addWillPresentEvent { (alert) in 2 print("the alertView will present") 3 }.addDidPresentEvent { (alert) in 4 print("the alertView did present") 5 }.alertWithButtonClick { (buttonIndex, alert) in 6 print("你按下了第\(buttonIndex)个按键") 7 }
当然其他的一些自定义样式,或者是带用户名输入和密码输入框的也很简单
1 UIAlertView.setMessage("这里要设置信息").addTitle("我是标题").addFirstButton("取消").addSecondButton("确认").addAlertStyle(.LoginAndPasswordInput).addDidPresentEvent { (alert) in 2 print("the alertView did present") 3 }.alertWithButtonClick { (buttonIndex, alert) in 4 let txt1 = alert.textFieldAtIndex(0)?.text 5 let txt2 = alert.textFieldAtIndex(1)?.text 6 print("userName:\(txt1) password:\(txt2)") 7 //打印出the alertView did present 8 //userName:Optional("111") password:Optional("222") 9 }
可见,使用链式编程加Block的方式基本只需要短短几行代码就可以满足大部分使用UIAlertView的使用场景,写起来很方便快捷.对于UIActionSheet也一样,可以将其完全改造成使用链式编程加Block的形式来使用.这个便留给读者
来完成了.相信看完上面的代码并实践过的读者很快就可以写出来.以上所有代码都可以在我的Github看到DuckDeck-GrandCue
但是目前Apple还是更推荐开发者使用最新的UIAlertController,它的API设计和UIAlertView是完全不同的.改成了基于Block的回调方式,使用起来更方便了.在这里这篇文章给大家详细地介绍了
UIAlertController详解,有兴趣的读者可以去看看
本文用一个足够简单的例子和大家探讨了怎么将原先使用UIAlertView编程方式改造成基于链式编程和block的编程方式.但是在实际的开发过程中,情况要比这个复杂得多,首先要考虑该业务和逻辑适不适合使用链式编程,其次设计出良好的
链式编程API也是需要花费很大精力的.函数返回值的设计也是很需要功力的,因为它直接影响了下一个调用函数.
iOS开发技巧系列---使用链式编程和Block来实现UIAlertView
标签:
原文地址:http://www.cnblogs.com/fengmin/p/5403170.html