做iOS开发的同学想必都用过UIAlertVIew或者UIActionSheet。UIAlertVIew 可以弹出一个出现在屏幕中间的提示视图,给用户展示信息,并让用户自己选择操作,UIActionSheet可以弹出一个选择列表,让用户选择列表中的某一项操作。使用UIAlertVIew和UIActionSheet非常简单,以下是一个简单的示例代码:

//UIAlertView
- (void)someButtonClicked {//初始化AlertView
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"AlertViewTest"
message:@"message"
delegate:self
cancelButtonTitle:@"Cancel"
otherButtonTitles:@"OtherBtn",nil];
[alert show];
}
//按钮点击事件的代理
- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex{
NSLog(@"clickButtonAtIndex:%d",(int)buttonIndex);
//index为-1则是取消,
} //UIActionSheet
- (void)someButtonClicked {
UIActionSheet * sheet = [[UIActionSheet alloc] initWithTitle:nil delegate:self cancelButtonTitle:@"ddd" destructiveButtonTitle:@"aaa" otherButtonTitles:@"bbb", @"ccc", @"ddd", nil];
sheet.destructiveButtonIndex = 1;
[sheet showInView:self.view];
}
- (void)actionSheet:(UIActionSheet *)actionSheet clickedButtonAtIndex:(NSInteger)buttonIndex {
NSLog(@"result = %d", (int)buttonIndex);
}

虽然使用已经算是比较简单了,但我觉得还是比较麻烦他们都需要设置delegate来获得用户选择的结果。这么小的界面,把调用显示和回调方法分开写在2个方法中,使得原本简单的逻辑复杂了。虽然也不会复杂到哪儿去,但是每次调用他们就需要另外写一个delegate回调方法,觉得还是比较麻烦。
于是便产生了,分别给它们写一个category 用来对原生的ui再做一层简单地封装。(一些复杂的暂不考虑)然后用block来做回调,这样子,一块简单地东西就不用分开在两个地方了(才不要拆散它们)。
我们先简单地分析
UIAlertVIew 和
UIActionSheet,其实他们需要的东西,并不多,只是按钮的事件和对应的action,UIAlertVIew还多了一个Title和message,但这些在初始化里面就已经初始化好了。造成他们分离的主要原因还是action和init分离了。所以要完成这个不让他们分离的目的,实现这个category
首先我们需要一个 button的模型来封装 button的title 和 对应的事件,事件用block来代替代理(这样就可以让他们在一起了)。
于是我顺手在github上搜索了一下,发现了这个
https://github.com/jivadevoe/UIAlertView-Blocks
哈哈哈哈,全文终~~
好吧,不闹,既然那让我们来看一下它的实现,这个扩展很简单,就是六个文件,对应RIButtonItem.h、RIButtonItem.m
UIActionSheet+Blocks.h、UIActionSheet+Blocks.m
UIAlertView+Blocks.h、UIAlertView+Blocks.m

RIButtonItem.h

@interface RIButtonItem : NSObject
{
NSString *label;
void (^action)();
}
@property (strong, nonatomic) NSString *label;
@property (copy, nonatomic) void (^action)();
+(id)item;
+(id)itemWithLabel:(NSString *)inLabel;
@end

ok,我们很容易看到,这里面的内容和我们前面想的实现几乎一样,用一个label来存储标题,(^action)()来记录点击按钮的事件。
RIButtonItem.m代码很简单,这里不贴了。

剩下的两个UIAlertView+Blocks UIActionSheet+Blocks 因为实现类似我们来看一个的实现

UIAlertView+Blocks.h

#import <UIKit/UIKit.h>

#import "RIButtonItem.h"`

@interface UIAlertView (Blocks) <UIActionSheetDelegate>  

-(id)initWithTitle:(NSString *)inTitle message:(NSString *)inMessage cancelButtonItem:(RIButtonItem *)inCancelButtonItem otherButtonItems:(RIButtonItem *)inOtherButtonItems, ... NS_REQUIRES_NIL_TERMINATION;   //1

- (NSInteger)addButtonItem:(RIButtonItem *)item;   //2
@end

1.我们可以看到初始化方法几乎和UIAlertView 初始化一样,只是用RIButtonItem来代替原来的按钮标题,用RIButtonItem把action带上就省略了原来的delegate方法
2.这个方法则对应了 addButtonWithTitle 方法

重点来了,来看UIAlertView+Blocks.m的实现

#import "UIAlertView+Blocks.h"
#import <objc/runtime.h> static NSString *RI_BUTTON_ASS_KEY = @"com.random-ideas.BUTTONS"; @implementation UIAlertView (Blocks) -(id)initWithTitle:(NSString *)inTitle message:(NSString *)inMessage cancelButtonItem:(RIButtonItem *)inCancelButtonItem otherButtonItems:(RIButtonItem *)inOtherButtonItems, ...
{
if((self = [self initWithTitle:inTitle message:inMessage delegate:self cancelButtonTitle:inCancelButtonItem.label otherButtonTitles:nil]))
{
NSMutableArray *buttonsArray = [NSMutableArray array]; RIButtonItem *eachItem;
va_list argumentList; //1
if (inOtherButtonItems)
{
[buttonsArray addObject: inOtherButtonItems];
va_start(argumentList, inOtherButtonItems); //2
while((eachItem = va_arg(argumentList, RIButtonItem *))) //3
{
[buttonsArray addObject: eachItem];
}
va_end(argumentList); //4
} for(RIButtonItem *item in buttonsArray)
{
[self addButtonWithTitle:item.label];
} if(inCancelButtonItem)
[buttonsArray insertObject:inCancelButtonItem atIndex:0]; objc_setAssociatedObject(self, (const void *)RI_BUTTON_ASS_KEY, buttonsArray, OBJC_ASSOCIATION_RETAIN_NONATOMIC); //5 [self setDelegate:self];
}
return self;
} - (NSInteger)addButtonItem:(RIButtonItem *)item
{
NSMutableArray *buttonsArray = objc_getAssociatedObject(self, (const void *)RI_BUTTON_ASS_KEY); NSInteger buttonIndex = [self addButtonWithTitle:item.label];
[buttonsArray addObject:item]; return buttonIndex;
} - (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex
{
// If the button index is -1 it means we were dismissed with no selection
if (buttonIndex >= 0)
{
NSArray *buttonsArray = objc_getAssociatedObject(self, (const void *)RI_BUTTON_ASS_KEY); //6
RIButtonItem *item = [buttonsArray objectAtIndex:buttonIndex];
if(item.action)
item.action();
} objc_setAssociatedObject(self, (const void *)RI_BUTTON_ASS_KEY, nil, OBJC_ASSOCIATION_RETAIN_NONATOMIC); //7
} @end
  1. va_list 是ios实现传递不定长的多个参数的方法时所使用的
  2. 然后用va_start初始化刚定义的va_list变量
  3. 然后用va_arg返回可变的参数,va_arg的第二个参数是你要返回的参数的类型.如果函数有多个可变参数的,依次调用va_arg获取各个参数;
    下面就是如果还有其他的buttonItem就把它加到buttonsArray
  4. 最后用va_end宏结束可变参数的获取。
  5. 因为在category里面不能添加成员变量,所以用objc_setAssociatedObject和objc_getAssociatedObject来变相的添加buttonArray方便下面delegate的时候取出buttonArray里的action
  6. 取出buttonArray,使用block来实现回调方法。
  7. set nil;

代码分析差不多,还学习了va_list 以后也能方便的使用这两个弹出框了。

上一个使用例子

code.png

敲下一块代码就出现了下面的弹框,很方便是不是。

alertView.png

iOS:简单使用UIAlertVIew和UIActionSheet的更多相关文章

  1. iOS 8 中 UIAlertView 和 UIActionSheet 河里去了?

    iOS 8 中 UIAlertView 和 UIActionSheet 河里去了? 太阳火神的漂亮人生 (http://blog.csdn.net/opengl_es) 本文遵循"署名-非商 ...

  2. iOS开发——UI篇Swift篇&UIAlertView/UIActionSheet

    UIAlertView/UIActionSheet UIAlertView //一个按钮的提醒 @IBAction func oneButtonAler() { //创建单一按钮提醒视图 let on ...

  3. iOS8以后UIAlertView和UIActionSheet两种alert页面都将通过UIAlertController来创建

    1. Important: UIAlertView is deprecated in iOS 8. (Note that UIAlertViewDelegate is also deprecated. ...

  4. iOS开发之UIAlertView与UIAlertController的详尽用法说明

    本文将从四个方面对IOS开发中UIAlertView与UIAlertController的用法进行讲解: 一.UIAlertView与UIAlertController是什么东东? 二.我们为什么要用 ...

  5. UIAlertView、 UIActionSheet

    一.UIAlertView. UIActionSheet都是ios系统自带的弹出式对话框,当UIAlertView或UIActionSheet弹出来时用户无法与应用界面中的其它控件交互,UIAlert ...

  6. UIAlertView、UIActionSheet兼容iOS8

    链接地址:http://blog.csdn.net/nextstudio/article/details/39959895?utm_source=tuicool 1.前言 iOS8新增了UIAlert ...

  7. UIAlertView及UIActionSheet 在ios8极其以下版本的兼容问题解决方案

    本文转载至 http://www.aichengxu.com/view/35326 UIAlertView及UIActionSheet在ios8中被放弃,其功能将完全由UIAlertControlle ...

  8. iOS 简单工厂模式

    iOS 简单工厂模式 什么是简单工厂模式? 简单工厂模式中定义一个抽象类,抽象类中声明公共的特征及属性,抽象子类继承自抽象类,去实现具体的操作.工厂类根据外界需求,在工厂类中创建对应的抽象子类实例并传 ...

  9. 用block将UIAlertView与UIActionSheet统一起来

    用block将UIAlertView与UIActionSheet统一起来 效果 1. 将代理方法的实例对象方法转换成了类方法使用 2. 要注意单例block不要长期持有,用完就释放掉 源码 https ...

随机推荐

  1. 朴素的UNIX之-调度器细节

    0.多进程调度的本质 我们都知道UNIX上有一个著名的nice调用.何谓nice,当然是"好"了.常规的想法是nice值越大越好,实际上,nice值越好,自己的优先级越低.那么为何 ...

  2. C语言的长处和缺点

     C语言的长处和缺点 C语言的长处: 1.面向过程的语言C语言是面向过程的语言,在这里用户创建过程或函数来运行他们的任务. 面向过程的语言是非常easy学.因为它遵循的算法来运行你的语句.要使用面 ...

  3. 超高性能管线式HTTP请求(实践·原理·实现)

    超高性能管线式HTTP请求(实践·原理·实现) 一.总结 一句话总结:实际pipe早就被http1.1所支持,并且大部分nginx服务器也支持并开启了这一功能. pipe之所以能比常规请求方式性能高出 ...

  4. 5.listview(QStringList QStringListModel)

    UI mainwindow.h #ifndef MAINWINDOW_H #define MAINWINDOW_H #include <QMainWindow> #include < ...

  5. [转]Linux+XAMPP+eolinker开源版v3.2.4

    eolinker是一个由国人开源的接口管理系统(AMS),特性及介绍详见开源中国-eolinker首页. 搭建步骤参考:eolinker开源指南 系统环境:CentOS Linux release 7 ...

  6. Core Bluetooth的基本常识

    每个蓝牙4.0设备都是通过服务(Service)和特征(Characteristic)来展示自己的 一个设备必然包含一个或多个服务,每个服务下面又包含若干个特征 特征是与外界交互的最小单位 比如说,一 ...

  7. MS SQL 获取数据字典的经典sql语句

    select [表名]=c.Name, [表说明]=isnull(f.[value],''), [列名]=a.Name, [列序号]=a.Column_id, [标识]=case when is_id ...

  8. SVG实现波浪效果

    SVG实现波浪效果 svg path:C 贝塞尔曲线绘制波浪形状   A 绘制圆弧形 svg animate:制作波浪动画,为了波浪动画效果自然,设置values关键点  attributeName: ...

  9. VS运行程序发生栈溢出的分析

    这两天时间都浪费在解决各种栈溢出了,没想到最后各个部件合在一起的时候会出现这么多的问题,这其实是我第一次解决栈溢出问题,很认真的查了查资料,找了几种解决方法,但是作用都不大,拆了东墙补西墙,虽然知道及 ...

  10. TensorFlow初学

    TensorFlow初学 基本概念 1.激活函数和成本函数 激活函数(activation function):一般是非线性函数,就是每个神经元通过这个函数将原有的来自其他神经的输入做一个非线性变化, ...