在ios中,事件UIEvent类来表示,当一个事件发生时,系统会搜集的相关事件信息,创建一个UIEvent对象,最后将该事件转发给应用程序对象(UIApplication)。日常生活中,主要有三种类型的事件:触摸事件,加速计事件以及远程遥控事件。下面是官方的一张图片:

当用户通过以上方式触发一个事件时,会将相应的事件对象添加到UIApplication的事件队列中。UIApplication会循环的从队列中拿出第一个事件来处理。首先将该事件分发给UIApplication 的主窗口对象(KeyWindow),然后由主窗口决定如何将事件交给最合适的响应者(UIResponder)来处理取决于事件的类型。这里主要分两种情况:

  1、触摸事件:UIApplication通过一个触摸检测来决定最合适来处理该事件的响应者,一般情况下,这个响应者是UIView对象。

  2、加速计事件或远程遥控事件:UIApplication寻找UIWindow中的第一响应者。找到第一响应者(The First Responder)后,会将该事件对象派发给该响应者以便处理。

下面分别讨论上述两种情况。

一、触摸事件中的触摸检测

  首先我们需要明确一个UIView对象能够接收触摸事件至少要保证以下三个条件:

  1、userInteractionEnabled属性为YES,该属性表示允许控件同用户交互。

  2、Hidden属性为NO。控件都看不见,还触摸啥?

  3、opacity属性值0 ~0.01,不能透明过分了吧?

  接下来的我们仅仅认为该三个基本属性都满足要求,方便描述,当然对于不满足要求的自然是不能接收触摸说事件的。

  当用户手指触摸到屏幕中的某一块区域时,UIWindow查找其子控件,然后通过调用所有自控件的方法:

- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event

  来通过指定的触摸点获取最合适的UIView来处理该触摸事件。如何通过触摸点获取UIView原理其实非常简单,只需要检查该触摸点是否在该控件所在的矩形区域内就可以了,其实hitTest:withEvent方法内部也是调用方法:

- (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event

  如果检测到传入的控件包含该触摸点就返回YES。

  当通过hitTest方法检测获取到UIView后,会继续对该UIView对象做一次检测操作,也就是查找subViews的subViews做触摸检测。最终该方法会返回一个最合适的控件来响应该事件。再次申明,如果之前的三个条件不满足,那么该UIView以及其subViews都不可以响应该触摸事件。

  找到响应者后,响应者可以重写以下方法来对触摸事件做响应:

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
[super touchesBegan:touches withEvent:event];//让下一个响应者可以有机会继续处理
}
- (void)touchesMoved:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
[super touchesBegan:touches withEvent:event];
}
- (void)touchesEnded:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
[super touchesBegan:touches withEvent:event];
}
- (void)touchesCancelled:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
[super touchesBegan:touches withEvent:event];
}

  在响应方法内部,我们也可以将这个触摸事件继续传递给父控件的对应方法处理。然后父控件还可以将该事件继续向上传递,直到传递给UIApplication对象。这一系列的响应者对象就构成了一个响应者链条。

二、第一响应者 (The First Responder)

  什么是第一响应者?简单的讲,第一响应者是一个UIWindow对象接收到一个事件后,第一个来响应的该事件的对象。注意:这个第一响应者与之前讨论的触摸检测到的第一个响应的UIView并不是一个概念。第一响应者一般情况下用于处理非触摸事件(手机摇晃、耳机线控的远程空间)或非本窗口的触摸事件(键盘触摸事件),通俗点讲其实就是管别人闲事的响应者。在IOS中,当然管闲事并不是所有控件都愿意的,这么说好像并不是很好理解,或着是站在编程人员的角度来看待这个问题,程序员负责告诉系统哪个对象可以成为第一响应者(canBecomeFirstResponder),如果方法canBecomeFirstResponder返回YES,这个响应者对象才有资格称为第一响应者。有资格并不代表一定可以成为第一响应者,就好像符合要求并不一定能够应聘成功一样,所以还差一个聘用环节,那就是becomeFirstResponder正式成为第一响应者。

  请原谅我的这些可能不太正常的想法,个人感觉上面的过程又有点像招聘流程,简历筛选就是canBecomeFirstResponder,becomeFirstResponder就是正式成为公司的员工。那么既然公司由聘用,那么对应的就有辞退咯!对应的方法就是canResignFirstResponder,这个表示第一响应者是否可以被辞退,有些牛逼到逆天的员工并不是说辞退就辞退的,争取有一天可以成为这个逆天员工,好吧,我又扯远了。还有一个方法就是resignFirstResponder,正式辞退该员工。

  值得注意的是,一个UIWindow对象在某一时刻只能有一个响应者对象可以成为第一响应者。我们可以通过isFirstResponder来判断某一个对象是否为第一响应者。

  大家先看下面的一个手机界面:

  

  界面中包含两个输入框,一个切换第一响应者的按钮。我为两个输入框绑定了开始编辑事件,然后在事件中打印第一响应者相关的信息,代码如下:

NSString * NSStringFromBoolValue(BOOL boolValue){
return boolValue ? @"YES" : @"NO";
} - (IBAction)editingBegin:(id)sender {
NSLog(@"top : 是否可以成为第一响应者=>%@,是否第一响应者=>%@",NSStringFromBoolValue(self.topInputView.canBecomeFirstResponder),NSStringFromBoolValue(self.topInputView.isFirstResponder));
NSLog(@"down : 是否可以成为第一响应者=>%@,是否第一响应者=>%@",NSStringFromBoolValue(self.downInputView.canBecomeFirstResponder),NSStringFromBoolValue(self.downInputView.isFirstResponder));
}

  当点击第一个输入框时,打印如下:

  

  我们可以看到两个输入框都可以成为第一响应者。但是只有第一个输入框才是第一响应者。

  当点击第二个输入框时,打印如下:

  

  我们可以看到两个输入框都是第一响应者,但是只有下面那个输入框才是第一响应者。

  我们注意到,两个输入框的下方有一个按钮用于切换第一响应者。按钮的响应事件方法为:

  

- (IBAction)switch:(id)sender {
/**
* 1、如果顶部输入框是第一响应者就将第一响应者切换为下方的输入框
2、如果顶部输入框不是第一响应者,就将其设置为第一响应者。
*/
if(self.topInputView.isFirstResponder){
[self.downInputView becomeFirstResponder];
}else{
[self.topInputView becomeFirstResponder];
} NSLog(@"父控件中的第一响应者:%@",[self.uiview findFirstResponder]); NSLog(@"top : 是否可以成为第一响应者=>%@,是否第一响应者=>%@",NSStringFromBoolValue(self.topInputView.canBecomeFirstResponder),
NSStringFromBoolValue(self.topInputView.isFirstResponder));
NSLog(@"down : 是否可以成为第一响应者=>%@,是否第一响应者=>%@",NSStringFromBoolValue(self.downInputView.canBecomeFirstResponder),
NSStringFromBoolValue(self.downInputView.isFirstResponder)); }

  在点击了第一个输入框后,我们点击切换第一响应者,屏幕打印如下:

  我们可以看到,这个时候下方的输入框成为第一响应者,并且触发了开始编辑事件,所以有两次打印。切换后,焦点也切换到第二个输入框中,我们通过键盘输入时,内容会在第二个输入框中出现。

  时常有人碰到希望点击空白区域,隐藏键盘的问题。例如输入框输入一半,觉得不想再编辑了,可以点击空白区域来隐藏键盘,这个时候其实只需要告诉系统,这个第一响应者的位置我不想要了,我想辞职,这就够了!下面是点击第一个输入框后,然后点击空白区域,触发touchesBegin事件,最后topInputView辞职的过程,通过这种方式就可以隐藏键盘了。代码如下:

-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
[self.topInputView resignFirstResponder];
}

  

  今天就先写到这里,后续会补充一些细节性的东西。

  

 

ios中的事件处理、响应者链条以及第一响应者的更多相关文章

  1. iOS中的事件处理

    前言:iOS中事件处理,是一个非常重要也非常难得地方.涉及到响应者链的地方的面试题,非常多工作两三年的老鸟也未必能回答的非常专业.这里具体介绍一下iOS中的事件处理,以及响应者链. 1. 三大事件 触 ...

  2. OS开发中的事件处理(二)-事件传递,响应者链条

    事件处理的事件传递 简介: 发生触摸事件后,系统会将该事件加入到一个由UIApplication管理的事件 队列中,UIApplication会从事件队列中取出最前面的事件,并将事件分发下去以便处理, ...

  3. 响应者链条,iOS中touchs事件的处理流程。

    用户在使用app的时候,会产生各样的事件.在iOS中的事件可以分为三种 触摸事件(Touch Event) 加速计事件(Accelerometer Event) 远程控制事件(Remote Contr ...

  4. IOS - 响应者链条

    简单来说就是:一级一级的找到响应的视图,如果没有就传给UIWindow实例和UIApplication实例,要是他们也处理不了,就丢弃这次事件... 对于IOS设备用户来说,他们操作设备的方式主要有三 ...

  5. IOS 响应者链条 and UIGestureRecognizer 手势识别器)

    一次完整的触摸事件的传递响应的过程 UIAppliction --> UIWiondw -->递归找到最适合处理事件的控件 控件调用touches方法-->判断是否实现touches ...

  6. IOS中键盘隐藏几种方式

    在ios开发中,经常需要输入信息.输入信息有两种方式: UITextField和UITextView.信息输入完成后,需要隐藏键盘,下面为大家介绍几种隐藏键盘的方式. <一> 点击键盘上的 ...

  7. Quartz 2D在ios中的使用简述二:创建画布

    在iOS中使用Quartz画图时,第一步就是要获取画布(图形上下文),然后再画布上做各种操作.先看下CoreGraphics.h这个头文件,就可以知道能够创建多少种上下文类型. #include &l ...

  8. UI开发--响应者链条

    一.触摸事件处理的详细过程 用户点击屏幕后产生的一个触摸事件,经过一些列的传递过程后,会找到最合适的视图控件来处理这个事件 找到最合适的视图控件后,就会调用控件的touches方法来作具体的事件处理 ...

  9. iOS开发——UI进阶篇(十二)事件处理,触摸事件,UITouch,UIEvent,响应者链条,手势识别

    触摸事件 在用户使用app过程中,会产生各种各样的事件 一.iOS中的事件可以分为3大类型 触摸事件加速计事件远程控制事件 响应者对象在iOS中不是任何对象都能处理事件,只有继承了UIResponde ...

随机推荐

  1. 使用sqlite的命令操作

    一:  首先进入到D:\java\android\android-sdk\platform-tools文件夹里面 二:使用adb  shell进入shell命令方式行(注意要想进入shell里面的操作 ...

  2. 字符集乱码问题:ISO-8859-1和GBK

    问题,引用百度知道的问题吧: http://zhidao.baidu.com/question/51342167.html?qbl=relate_question_0&word=%C3%84% ...

  3. Chrome 开发者工具的Timeline和Profiles提高Web应用程序的性能

    Chrome 开发者工具的Timeline和Profiles提高Web应用程序的性能 二.减少 HTTP 的请求数    当用户浏览页面时,如果我们在用户第一次访问时将一些信息一次性加载到客户端缓存, ...

  4. [Angular 2] Set Properties on Dynamically Created Angular 2 Components

    When you generate Angular 2 components, you’re still able to access the component instance to set pr ...

  5. 使用javaScript解决asp.net中mvc使用ajax提交数组参数的匹配问题

    想到在asp.net的mvc中如果使用ajax向服务端传递参数时如果参数是一个类或者是个数组(或List集合)以及更复杂的对象时,服务端总是会发生取不到值的情况,当然网上也有很多解决的例子,但都是在服 ...

  6. cocos2d-x在android下的编译

    $(call import-add-path,E:/cocos2d-2.0-x-2.0.3) include $(BUILD_SHARED_LIBRARY) http://www.cnblogs.co ...

  7. [原创]SSIS-执行包任务调用子包且子包读取父包变量

    背景:       有时候需要将一个个开发好的独立的ETL包串接起来形成一个独立而庞大的包,如:每家分公司都开发不同的ETL包,最后使用执行包任务来将这些分公司的包给串联起来形成一个独立而完整运行的E ...

  8. Orm图解教程

    entity framework框架生成摘要文档为空(没有元数据文档可用)的bug解决方案 西安.王磊 2012-10-25 10:47 阅读:1234 评论:2   ORM for Net主流框架汇 ...

  9. Golang学习 - errors 包

    ------------------------------------------------------------ Go 语言使用 error 类型来返回函数执行过程中遇到的错误,如果返回的 e ...

  10. uboot 网络不通问题解决一例1

    平台:Hi3531 PHY:RTL8211 现象:在uboot中执行ping命令的时候,总是超时. 过程: 使用uboot自带的phy操作命令mii读出的数据全是0xff.这里要介绍一下uboot中的 ...