一、事件分发处理【由外到内】
在ios中发生触摸后,事件会加到uiapplication事件队列,uiapplication会从事件队列取出最前面的事件进行分发处理,通常会先分发给主窗口,主窗口会调用hittest:withevent:方法,查找适合的事件触发视图,即 找到被触摸的视图对象
寻找流程如下:
- 在*视图(keywindow的视图)上调用pointinside:withevent:方法判断触摸点是否在当前视图内;
- 如果返回no,那么keywindow的hittest:withevent:返回nil;
- 如果返回yes,那么它会向当前视图的所有子视图发送hittest:withevent:消息,遍历所有子视图的顺序是从subviews数组的末尾向前遍历(从界面最上方开始向下遍历);
- 如果有subview的hittest:withevent:返回非空对象,则keywindow的hittest:withevent:返回此对象,处理结束;
- 如果所有subview遍历结束仍然没有返回非空对象,则keywindow的hittest:withevent:返回*视图;
二、响应者链条【由内到外】
找到被触摸的视图对象后,还需要判断该视图对象是否能处理该触摸事件,如果不能处理,又该让谁来处理,于是响应者链条出现,作用是 找到事件响应者
响应者链条原则:
- 触摸对象initalview无法响应事件时,传递给上级视图superview去响应
- 如果上级视图无法响应,继续往上传递
- 往上传递直到传递到视图控制器的根视图controllerview,如果根视图不响应,传递给视图控制器viewcontroller
- 视图控制器不响应,传递给父视图控制器的根视图supercontrollerview,如果根视图不响应,传递给父视图控制器superviewcontroller
- *视图控制器不能响应,传递给主窗口keywindow
- keywindow不能响应,传递给uiapplication处理
- uiapplication不能响应,该事件就会被 抛弃
三、继承uiresponder
以上 事件分发 和 响应者链条 ,都不需要我们关心,这些操作是自动执行的,不需要我们去操作,我们只需要了解它们的原理就行。
在ios中并不是所有的类都能处理并接受事件,只有继承uiresponder的对象才能处理事件(我们常用的uiview、uiviewcontroller、uiapplication都继承自uiresponder,它们都能接收并处理事件 ),但继承uiresponder又不意味着一定能处理事件
继承 uiresponder 的对象,不能处理事件的情况:
userinteractionenabled = no;
hidden = yes;
alpha = 0 ~ 0.01;
没有实现touchesbegan:withevent方法
重写uiresponder 触摸响应方法:
#pragma mark 触摸开始时会调用
- (void)touchesbegan:(nsset *)touches withevent:(uievent *)event;
#pragma mark 触摸移动时会频繁调用
- (void)touchesmoved:(nsset *)touches withevent:(uievent *)event;
#pragma mark 触摸结束离开屏幕时会调用
- (void)touchesended:(nsset *)touches withevent:(uievent *)event;
#pragma mark 触摸意外取消时会调用,比如触摸时电话打进来
- (void)touchescancelled:(nsset *)touches withevent:(uievent *)event;
四、uitouch对象
在 uiresponder 触摸响应方法中,需要获取 uitouch 对象:
uitouch *touch = [touches anyobject];
//取得在指定视图的触摸位置
cgpoint current = [touch locationinview:self.view];
//取得在指定视图的前一个触摸位置
cgpoint previous = [touch previouslocationinview:self.view];
其他常用属性:
window : 触摸所在窗口
view : 触摸所在视图
tapcount : 短时间点击次数
了解了这些,你就可以利用触摸事件做一些好玩的事情了,(^o^)/~,这里就不列具体代码了。