iOS的事件分发

        移动平台上的开发主要关注数据以及数据的处理,事件的处理以及UI。所以事件的分发处理是很重要的一个环节,对于一个平台的优劣来说也是一项重要的参数。如果事件的分发设计的不好,一些复杂的UI场景就会变得很难写甚至没法写。从小屏没有触摸的功能机开始到现在大屏多点触摸的智能机,对于事件的分发处理基本思路都是一样的——链(设计模式中有个模式就是职责链chain of responsibility),只是判定的复杂程度不同。

        iOS中的事件有3类,触摸事件(单点,多点,手势)、传感器事件(加速度传感器)和远程控制事件,这里我介绍的是第一种事件的分发处理。

        

        上面的这张图来自苹果的官方。描述了Responder的链,同时也是事件处理的顺序。通过这两张图,我们可以发现:

        1. 事件顺着responder chain传递,如果一环不处理,则传递到下一环,如果都没有处理,最后回到UIApplication,再不处理就会抛弃

        2. view的下一级是包含它的viewController,如果没有viewController则是它的superView

        3. viewController的下一级是它的view的superView

        4. view之后是window,最后传给application,这点iOS会比OS X简单(application就一个,window也一个)

         总结出来传递规则是这样的:

        

        这样事件就会从first responder逐级传递过来,直到被处理或者被抛弃。


        由于UI的复杂,这个responder chain是需要根据事件来计算的。比如,我现在在一个view内加入了2个Button,先点击了一个,则first responder肯定是这个点击过的button,但我下面可以去点击另一个button,所以显然,当触摸事件来时,这个chain是需要重新计算更新的,这个计算的顺序是事件分发的顺序,基本上是分发的反过来。

        

        无论是哪种事件,都是系统本身先获得,是iOS系统来传给UIApplication的,由Application再决定交给谁去处理,所以如果我们要拦截事件,可以在UIApplication层面或者UIWindow层面去拦截。

        

        

        UIView是如何判定这个事件是否是自己应该处理的呢?iOS系统检测到一个触摸操作时会打包一个UIEvent对象,并放入Application的队列,Application从队列中取出事件后交给UIWindow来处理,UIWindow会使用hitTest:withEvent:方法来递归的寻找操作初始点所在的view,这个过程成为hit-test view。

        hitTest:withEvent:方法的处理流程如下:调用当前view的pointInside:withEvent:方法来判定触摸点是否在当前view内部,如果返回NO,则hitTest:withEvent:返回nil;如果返回YES,则向当前view内的subViews发送hitTest:withEvent:消息,所有subView的遍历顺序是从数组的末尾向前遍历,直到有subView返回非空对象或遍历完成。如果有subView返回非空对象,hitTest方法会返回这个对象,如果每个subView返回都是nil,则返回自己。

        好了,我们还是看个例子:

        

        这里ViewA包含ViewB和ViewC,ViewC中继续包含ViewD和ViewE。假设我们点击了viewE区域,则hit-test View判定过程如下:

       1. 触摸在A内部,所以需要检查B和C

       2. 触摸不在B内部,在C内部,所以需要检查D和E

       3. 触摸不在D内部,但在E内部,由于E已经是叶子了,所以判定到此结束


        我们可以运行一段代码来验证,首先从UIView继承一个类myView,重写里面的

[objc] view plaincopy在CODE上查看代码片派生到我的代码片
  1. - (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event  
  2. {  
  3.     UIView *retView = nil;  
  4.     NSLog(@"hitTest %@ Entry! event=%@"self.name, event);  
  5.       
  6.     retView = [super hitTest:point withEvent:event];  
  7.     NSLog(@"hitTest %@ Exit! view = %@"self.name, retView);  
  8.      
  9.     return retView;  
  10. }  

[objc] view plaincopy在CODE上查看代码片派生到我的代码片
  1. - (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event  
  2. {  
  3.     BOOL ret = [super pointInside:point withEvent:event];  
  4. //    if ([self.name isEqualToString:@"viewD"]) {  
  5. //        ret = YES;  
  6. //    }  
  7.     if (ret) {  
  8.         NSLog(@"pointInside %@ = YES"self.name);  
  9.     } else {  
  10.         NSLog(@"pointInside %@ = NO"self.name);  
  11.     }  
  12.       
  13.     return ret;  
  14. }  
        在viewDidLoad方法中手动加入5个view,都是myView的实例。

[objc] view plaincopy在CODE上查看代码片派生到我的代码片
  1. - (void)viewDidLoad  
  2. {  
  3.     [super viewDidLoad];  
  4.       
  5.     _viewA = [[myView alloc] initWithFrame:CGRectMake(1010300200) Color:[UIColor blackColor] andName:@"viewA"];  
  6.     [self.view addSubview:_viewA];  
  7.     [_viewA release];  
  8.       
  9.     _viewB = [[myView alloc] initWithFrame:CGRectMake(10240300200) Color:[UIColor blackColor] andName:@"viewB"];  
  10.     [self.view addSubview:_viewB];  
  11.     [_viewB release];  
  12.       
  13.     _viewC = [[myView alloc] initWithFrame:CGRectMake(1010120180) Color:[UIColor blueColor] andName:@"viewC"];  
  14.     [_viewB addSubview:_viewC];  
  15.     [_viewC release];  
  16.       
  17.     _viewD = [[myView alloc] initWithFrame:CGRectMake(17010120180) Color:[UIColor blueColor] andName:@"viewD"];  
  18.     [_viewB addSubview:_viewD];  
  19.     [_viewD release];  
  20.       
  21.     _viewE = [[myView alloc] initWithFrame:CGRectMake(304060100) Color:[UIColor redColor] andName:@"viewE"];  
  22.     [_viewD addSubview:_viewE];  
  23.     [_viewE release];  
  24.   
  25. }  
        这个样式如下:


        当我点击viewE的时候,打印信息如下:

2014-01-25 18:32:46.538 eventDemo[1091:c07] hitTest viewB Entry! event=<UITouchesEvent: 0x8d0cae0> timestamp: 6671.26 touches: {(

)}

2014-01-25 18:32:46.538 eventDemo[1091:c07] pointInside viewB = YES

2014-01-25 18:32:46.539 eventDemo[1091:c07] hitTest viewD Entry! event=<UITouchesEvent: 0x8d0cae0> timestamp: 6671.26 touches: {(

)}

2014-01-25 18:32:46.539 eventDemo[1091:c07] pointInside viewD = YES

2014-01-25 18:32:46.539 eventDemo[1091:c07] hitTest viewE Entry! event=<UITouchesEvent: 0x8d0cae0> timestamp: 6671.26 touches: {(

)}

2014-01-25 18:32:46.540 eventDemo[1091:c07] pointInside viewE = YES

2014-01-25 18:32:46.540 eventDemo[1091:c07] hitTest viewE Exit! view = <myView: 0x8c409f0; frame = (30 40; 60 100); layer = <CALayer: 0x8c40a90>>

2014-01-25 18:32:46.540 eventDemo[1091:c07] hitTest viewD Exit! view = <myView: 0x8c409f0; frame = (30 40; 60 100); layer = <CALayer: 0x8c40a90>>

2014-01-25 18:32:46.541 eventDemo[1091:c07] hitTest viewB Exit! view = <myView: 0x8c409f0; frame = (30 40; 60 100); layer = <CALayer: 0x8c40a90>>

2014-01-25 18:32:46.541 eventDemo[1091:c07] touchesBegan viewE

2014-01-25 18:32:46.624 eventDemo[1091:c07] touchesEnded viewE

        从打印信息可以看到,先判断了viewB,然后是viewD,最后是viewE,但事件就是直接传给了viewE。


转载自:http://blog.csdn.net/pi9nc/article/details/18779781

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/500076.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

iOS开发ARC入门和使用

本文部分实例取自iOS 5 Toturail一书中关于ARC的教程和公开内容&#xff0c;仅用于技术交流和讨论。请不要将本文的部分或全部内容用于商用&#xff0c;谢谢合作。 欢迎转载本文&#xff0c;但是转载请注明本文出处&#xff1a;http://www.onevcat.com/2012/06/arc-hand-by-han…

UIBezierPath 的使用介绍

使用UIBezierPath类可以创建基于矢量的路径。此类是Core Graphics框架关于path的一个封装。使用此类可以定义简单的形状&#xff0c;如椭圆或者矩形&#xff0c;或者有多个直线和曲线段组成的形状。1.Bezier Path 基础UIBezierPath对象是CGPathRef数据类型的封装。path如果是基…

iOS中的XML解析

解析方式分类: 解析 XML 通常有两种方式&#xff0c;DOM 和 SAX&#xff1a;DOM解析XML时&#xff0c;读入整个XML文档并构建一个驻留内存的树结构&#xff08;节点树&#xff09;&#xff0c;通过遍历树结构可以检索任意XML节点&#xff0c;读取它的属性和值。 iOS中XML解析是…

iOS中的JSON解析

一、在iOS中&#xff0c;JSON的常见解析方案有4种&#xff08;1&#xff09;第三方框架&#xff1a;JSONKit、SBJson、TouchJSON&#xff08;性能从左到右&#xff0c;越差&#xff09; &#xff08;2&#xff09;苹果原生&#xff08;自带&#xff09;&#xff1a;NSJSONSeria…

iOS中NSLog的优化使用

一、概述&#xff1a;Debug模式下输出日志&#xff0c;Release模式下自动屏蔽日志输出。通过宏定义实现。二、代码如下&#xff1a;#ifndef __OPTIMIZE__ #define DLog(fmt, ...) NSLog(("%s [Line %d] " fmt), __PRETTY_FUNCTION__, __LINE__, ##__VA_ARGS__) #else…

iOS中bundle的使用

一、概述1.在我们使用第三方框架时&#xff0c;常常看到XXX.bundle的文件。我们找到该文件&#xff0c;显示包内容&#xff0c;大致看到很多资源文件&#xff1a;图片、配置文本、XIB文件……2.什么是Bundle文件&#xff1a;简单理解&#xff0c;就是资源文件包。我们将许多图片…

NSRunloop小总结

一、概述1.Run loops是线程的基础架构部分。一个run loop就是一个事件处理循环&#xff0c;用来不停的调配工作以及处理输入事件。使用run loop的目的是使你的线程在有工作的时候工作&#xff0c;没有的时候休眠。2.Run loop的管理并不完全是自动的。你仍必须设计你的线程代码以…

NSURLProtocol概述

一、概念 NSURLProtocol也是苹果众多黑魔法中的一种&#xff0c;使用它可以轻松地重定义整个URL Loading System。当你注册自定义NSURLProtocol后&#xff0c;就有机会对所有的请求进行统一的处理&#xff0c;基于这一点它可以让你&#xff1a; 1.自定义请求和响应 2.提供自定义…

使用NSURLProtocol实现离线缓存

一、说明&#xff1a;NSURLProtocol可以拦截任何网络请求&#xff0c;包含UIWebView中发出的所有请求。但是在WKWebView中&#xff0c;只能拦截到最初始的请求&#xff0c;内嵌的资源下载拦截不到。比如通过WKWebView加载"http://www.baidu.com",则只能拦截到"h…

WKWebView概述

一、概述1.iOS 8 SDK中发布了新的WebView框架----WebKit.framework。2.WebKit使用WKWebView来代替IOS的UIWebView和OSX的NSWebView&#xff0c;并且使用Nitro JavaScript引擎&#xff0c;这意味着所有第三方浏览器运行JavaScript将会跟safari一样快。3.内存问题&#xff1a;(1)…

CoreData 自定义数据类型

在CoreData中&#xff0c;Entity中Attribute的类型只有固定的几种可选。如下图&#xff1a; 如果我们要想直接存放UIImage到数据库&#xff0c;如何做&#xff1f; 1.在coredata中新建的attribute中类形选择Transformable. 意思表示这个字段是自定义的类型。 2.在生成的NSMana…

XMLDictionary iOS的XML处理包

1.概述&#xff1a;XMLDictionary 提供一种简单的方法实现 iOS 和 Mac OS X 下解析和生成 XML 的方法。XMLDictionary 将 XML 转成 NSDictionary ,也可以将 NSDictionary 装成 XML 结构的字符串。2.实现原理&#xff1a;XMLDictionary 使用 NSXMLParser 类解析XML&#xff0c;使…

CoreData并发操作模式简介

iOS5.0中&#xff0c;苹果为CoreData的并发处理添加了两个内容。一、首先介绍第一个内容&#xff1a;CoreData框架中的NSManagedObjectContext类增加新的初始化方法&#xff1a;initWithConcurrencyType:(NSManagedObjectContextConcurrencyType)ct;1.参数方法介绍&#xff1a;…

java图片的导出,并压缩

java图片的导出,并压缩 java 压缩包jar包使用的是commons-compress-1.6.jar /*** 导出图片* param request* param response*/RequestMapping("/exportPicture")public void exportPicture(HttpServletRequest request,HttpServletResponse response) throws Exce…

【Linux系统基础】(3)在Linux上部署运维监控Zabbix和Grafana

目录 运维监控Zabbix部署简介安装安装前准备 - Mysql安装Zabbix Server 和 Zabbix Agenta. 安装Zabbix yum库b. 安装Zabbix Server、前端、Agentc. 初始化Mysql数据库d. 为Zabbix Server配置数据库e. 配置Zabbix的PHP前端 配置zabbix 前端&#xff08;WEB UI&#xff09; 运维监…

Pods简介

每种语言发展到一个阶段&#xff0c;就会出现相应的依赖管理工具&#xff0c;例如 Java 语言的 Maven&#xff0c;nodejs 的 npm。随着 iOS 开发者的增多&#xff0c;业界也出现了为 iOS 程序提供依赖管理的工具&#xff0c;它的名字叫做&#xff1a;CocoaPods。 CocoaPods项目…

HashMap源码解释

HashMap 前言: 本文的hashMap是基于jdk1.7的hashMap. 关于jdk1.8的hashMap在另一篇中,那里将会介绍与1.7的差异与优势 首先基础知识介绍: 1.HashMap的成员变量   int DEFAULT_INITIAL_CAPACITY 16&#xff1a;默认的初始容量为2 ^ 4   int MAXIMUM_CAPACITY 1 <<…

MagicRecord For IOS 简介

一、概述 MagicalRecord 灵感来自于简洁的Ruby语言中 Rails Active Record 查询方式. MagicalRecord 这个开源库的核心思想是: 1.清除 Core Data 相关的代码2.简洁的清除,简单的一行搜索记录的功能3.当然允许使用NSFetchRequest,当存在着复杂的搜索条件时 二、使用 1. 导入框架…

对象引用 String引用 基本类型引用 差别

最近遇到一个线上问题,原因是忽略的引用的一些语法,导致出错,现在记录一下: Testpublic void testList(){List<String> list new ArrayList<String>();list.add("1");list.add("2");list.add("3");List<String> list2 new …

Mantle For iOS

Mantle可以很方便的去书写一个模型层的代码。 使用它可以很方便的去反序列化JSON或者序列化为JSON(需要在MTLModel子类中实现<MTLJSONSerializing>协议) 使用一个解释器MTLJSONAdapter去转换模型对象。 NSError *error nil; MyObject *myObject [MTLJSONAdapter modelO…