UIResponder学习 手势

UIResponder

介绍

UIResponder 这个类定义了很多用来处理响应和时间处理的类。他的子类有UIApplication,UIView以及UIWindow等。

IOS中分为两类事件:触摸事件,和移动事件。最原始的事件处理方是touchesBegan:withEvent:,touchesMoved:withEvent:touchesEnded:withEvent:, and touchesCancelled:withEvent:无论任何时候手指只要触摸屏幕或是在屏幕上移动拖拽甚至离开屏幕都会导致一个UIEvent对象产生。

Responder Chain

在UIResponder中有一个非常重要的概念叫做Responder Chain,个人的理解是这是按照一定规则组织的响应、处理事件的一条链表。在了解UIResponder之前还得在了解一个概念Hit-Testing。在IOS中通常使用hit-testing去找到那个被触摸的视图。这个视图叫hit-test view,当IOS找到hit-test view后就把touch event交个那个视图来处理。下面画个图来说明一下,当点击视图E时看一下hit-testing的工作过程。

1.确定改触摸事件发生在view A范围内,接下来测试view B以及view C。

2.检查发现事件不再view B范围内发生,接下来检查view C发现触摸事件发生在了view C中,所以检查 view D,view E。

3.最后发现事件发生在view E的范围内所以view E成为了hit-test view。

下面是关于调用hit-test的官方说明:

The hitTest:withEvent: method returns the hit test view for a given CGPoint and UIEvent. The hitTest:withEvent: method begins by calling thepointInside:withEvent: method on itself. If the point passed into hitTest:withEvent: is inside the bounds of the view, pointInside:withEvent: returns YES. Then, the method recursively calls hitTest:withEvent: on every subview that returns YES.

Responder Chain 是由responder对象组成的

responder chain是由一系列responder对象连接起来的,他从第一个responder对象开始一直到application对象结束。如果第一个responder不能够处理该事件则该事件会被发送到下一个在该responder chain中的responder来处理。

当自己定义的一个类想让他成为first responder时需要做两件事:

1.重写 canBecomeFirstResponder 方法让他返回YES

2.接受 becomeFirstResponder 消息,如果必要的话可让对象给自己发送该消息。

在这里有一个地方需要注意,当把一个对象变为first responder是要确保这个对象的图形界面已经建立起来,也就是说要在viewDidAppear中调用becomeFirstResponder方法。如果在veiwWillAppear方法中调用becomeFirstResponder将会得到一个NO。

Responder Chain 遵循一个特定的传播路径

如果hit-test view不能够处理该事件则UIKit会将事件传递给下一个Responder。下图则显示了事件在Responder Chain中传播的两种方式:

对于左边的app中事件传播路径如下:

1.初始的界面尝试去处理事件后者消息,打他处理不了则把事件交给它上一层视图处理,因为最开始的界面在他的view controller里的视图层次里不是最上层的。(这里的上下是按照树的结构而言的,下图解释:)

2.上层视图尝试处理事件,如果他不能处理则将事件交给他的上层视图处理,原因同上。

3.在view controller中最上层的视图尝试处理,他也不能处理则交给他的view controller来处理。

4.view controller也无法处理则交给window来处理。

5.window无法处理交给app object来处理

6.app object无法处理则将该事件丢弃掉。

右边的传播方式稍有不同:

1.一个视图在他的view controller 的视图层中向上传播一个事件直到它到达最顶层视图。

2.最顶层视图无法处理将event交给他的view controller来处理。

3.view controller 传递事件到他的最顶层视图的上一层视图,接下来重复1-3的步骤直到事件到达root view controller。

4.root view controller将事件传递到window object。

5.window 将事件传递给app object。

注意:事件,消息不要自己向上传送而要调用父类中的方法来处理,让UIKit来处理消息在responder chain中的传递。

Responder chain Demo:

[objc] view plain copy

print?

  1. /*PMBViewController*/
  2. #import "PMBViewController.h"
  3. #import "PMBViewOne.h"
  4. #import "PMBVIewTwo.h"
  5. #import "PMBVIewThree.h"
  6. @interface PMBViewController ()
  7. @property (weak, nonatomic) IBOutlet PMBVIewThree *viewThree;
  8. @property (weak, nonatomic) IBOutlet PMBVIewTwo *viewTwo;
  9. @property (weak, nonatomic) IBOutlet PMBViewOne *viewOne;
  10. @end
  11. @implementation PMBViewController
  12. - (void)viewDidLoad
  13. {
  14. [super viewDidLoad];
  15. // Do any additional setup after loading the view, typically from a nib.
  16. }
  17. - (void)didReceiveMemoryWarning
  18. {
  19. [super didReceiveMemoryWarning];
  20. // Dispose of any resources that can be recreated.
  21. }
  22. @end
  23. /*PMBViewOne*/
  24. #import "PMBViewOne.h"
  25. @implementation PMBViewOne
  26. - (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event{
  27. [super touchesBegan:touches withEvent:event];
  28. [[[touches anyObject] view] setBackgroundColor:[UIColor redColor]];
  29. }
  30. @end
  31. /*PMBVIewTwo*/
  32. #import "PMBVIewTwo.h"
  33. @implementation PMBVIewTwo
  34. @end
  35. /*PMBVIewThree*/
  36. #import "PMBVIewThree.h"
  37. @implementation PMBVIewThree
  38. - (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event{
  39. [super touchesBegan:touches withEvent:event];
  40. [[[touches anyObject] view] setBackgroundColor:[UIColor greenColor]];
  41. }
  42. @end
时间: 2024-08-05 06:37:27

UIResponder学习 手势的相关文章

UIResponder学习

http://blog.csdn.net/jimzhai/article/details/23283515 UIResponder 介绍 UIResponder 这个类定义了很多用来处理响应和时间处理的类.他的子类有UIApplication,UIView以及UIWindow等. iOS中分为两类事件:触摸事件,和移动事件.最原始的事件处理方是touchesBegan:withEvent:,touchesMoved:withEvent:, touchesEnded:withEvent:, and

iOS 初学-手势

一.学习手势,首先要先建立一个对象进行操作,我们以一个照片为例.首先创建一个imageView 1.声明一个属性: @property (nonatomic, retain)UIImageView *imageView; 2.开始真正创建对象 self.imageView = [[UIImageView alloc]initWithImage:[UIImage imageNamed:@"beautiful.jpg"]]; self.imageView.frame = CGRectMak

史上最详细的iOS之事件的传递和响应机制

前言: 按照时间顺序,事件的生命周期是这样的: 事件的产生和传递(事件如何从父控件传递到子控件并寻找到最合适的view.寻找最合适的view的底层实现.拦截事件的处理)->找到最合适的view后事件的处理(touches方法的重写,也就是事件的响应) 其中重点和难点是: 1.如何寻找最合适的view 2.寻找最合适的view的底层实现(hitTest:withEvent:底层实现) (一)iOS中的事件 iOS中的事件可以分为3大类型: 触摸事件 加速计事件 远程控制事件这里我们只讨论iOS中的

事件的传递和响应机制(全)

前言: 按照时间顺序,事件的生命周期是这样的: 事件的产生和传递(事件如何从父控件传递到子控件并寻找到最合适的view.寻找最合适的view的底层实现.拦截事件的处理)->找到最合适的view后事件的处理(touches方法的重写,也就是事件的响应) 其中重点和难点是: 1.如何寻找最合适的view 2.寻找最合适的view的底层实现(hitTest:withEvent:底层实现) (一)iOS中的事件 iOS中的事件可以分为3大类型: 触摸事件 加速计事件 远程控制事件 这里我们只讨论iOS中

二、Android应用的界面编程(七)ViewAnimator及其子类[ ViewSwitcher、ImageSwitcher、TextSwitcher、ViewFlipper ]

ViewAnimator是一个基类,它继承了FrameLayout.因此它表现出FrameLayout的特征,可以将多个View组“叠”在一起. ViewAnimator可以在View切换时表现出动画效果. ViewAnimator及其子类也是一组非常重要的UI组件,这种组件的主要功能是增加动画效果.从而使界面更加“炫”. [ViewAnimator及其子类的继承关系] 图2.56 [ViewAnimator支持的常见XML属性]android:animateFirstView 设置ViewAn

iOS-事件传递和响应机制

转自:http://www.jianshu.com/p/2e074db792ba 前言: 按照时间顺序,事件的生命周期是这样的: 事件的产生和传递(事件如何从父控件传递到子控件并寻找到最合适的view.寻找最合适的view的底层实现.拦截事件的处理)->找到最合适的view后事件的处理(touches方法的重写,也就是事件的响应) 其中重点和难点是: 1.如何寻找最合适的view 2.寻找最合适的view的底层实现(hitTest:withEvent:底层实现) (一)iOS中的事件 iOS中的

Android零基础入门第54节:视图切换组件ViewSwitcher

前面三期学习了ProgressBar系列组件,那本期开始一起来学习ViewAnimator组件. 一.ViewAnimator组件概述 ViewAnimator是一个基类,它继承了 FrameLayout,因此它表现出FrameLayout的特征,可以将多个View组件叠在一起. ViewAnimator额外增加的功能正如它的名字所暗示的一样,ViewAnimator可以在View切换时表现出动画效果. ViewAnimator及其子类的继承关系图如下图所示. ViewAnimator及其子类也

移动UI设计

移动设计 目录 概述... 4 让用户一见钟情的移动应用 什么是用户一见钟情的移动应用? 怎样做到让用户一见钟情? 移动应用的生命周期 转变交互设计思维 特征... 6 人-机-环协同的交互系统 应用的使用情景特征 光线 环境噪音 网络状态 用户的使用姿态 单双手 横竖屏 持续时间 干扰任务 移动网络的特性 设备的物理特性 小而繁多的屏幕尺寸 因平台而异的手机按键 丰富的传感器体验 触摸手势交互特性 用户的手指喜欢往哪点 手势交互不只是点击 为手势设计时避免犯的错 原则... 11 主流移动操作

ViewSwitcher的功能与用法

ViewSwitcher代表了视图切换组件,它本身继承了FramLayout,因此可以将多个View层叠在一起,每次只显示一个组件.当程序控制从一个View切换到另一个View时,ViewSwitcher支持指定动画效果. 为了给ViewSwitcher添加多个组件,一般通过调用ViewSwitcher的setFactory(ViewSwitcher.ViewFactory)方法为之设置ViewFactory,并由该ViewFactory为之创建View即可. 实例:仿Android系统的Lau