zoukankan      html  css  js  c++  java
  • iOS事件的响应和传递机制

    跟二狗子哥哥交流的时候,他总说我,说的过程太业余。故 好好学习整理一下。努力不那么业余。

    一、事件的产生、传递、响应:

    1、事件从父控件依次传递到子控件,寻找最合适的子控件View。

    2、寻找最合适的View的底层实现,拦截事件的处理。

    3、找到最合适的view之后的事件处理,也就是事件响应,重写touch方法等。

    传递过程中比较重要的两点:

    1.如何寻找最合适的view
    2.寻找最合适的view的底层实现(hitTest:withEvent:底层实现)

    触摸事件 加速事件 远程控制事件 我们现在 讨论触摸事件

    二、响应者对象(UIResponder)

    只有继承了UIResponder的对象,才可以接受并处理事件,我们称之为响应者对象

    响应者对象有:UIApplication UIViewController UIView

    UIResponder提供了方法以下方法处理触摸事件:

    - (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;

    - (void)touchesMoved:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;

    - (void)touchesEnded:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;

    - (void)touchesCancelled:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;

    - (void)touchesEstimatedPropertiesUpdated:(NSSet<UITouch *> *)touches NS_AVAILABLE_IOS(9_1);

    Presses事件

    - (void)pressesBegan:(NSSet<UIPress *> *)presses withEvent:(nullable UIPressesEvent *)event NS_AVAILABLE_IOS(9_0);

    - (void)pressesChanged:(NSSet<UIPress *> *)presses withEvent:(nullable UIPressesEvent *)event NS_AVAILABLE_IOS(9_0);

    - (void)pressesEnded:(NSSet<UIPress *> *)presses withEvent:(nullable UIPressesEvent *)event NS_AVAILABLE_IOS(9_0);

    - (void)pressesCancelled:(NSSet<UIPress *> *)presses withEvent:(nullable UIPressesEvent *)event NS_AVAILABLE_IOS(9_0);

    UIResponder提供了方法以下方法处理加速事件:

    - (void)motionBegan:(UIEventSubtype)motion withEvent:(nullable UIEvent *)event NS_AVAILABLE_IOS(3_0);

    - (void)motionEnded:(UIEventSubtype)motion withEvent:(nullable UIEvent *)event NS_AVAILABLE_IOS(3_0);

    - (void)motionCancelled:(UIEventSubtype)motion withEvent:(nullable UIEvent *)event NS_AVAILABLE_IOS(3_0);

    UIResponder提供了方法以下方法处理远程事件:

    - (void)remoteControlReceivedWithEvent:(nullable UIEvent *)event NS_AVAILABLE_IOS(4_0);

    另外还有:

    - (BOOL)canPerformAction:(SEL)action withSender:(nullable id)sender NS_AVAILABLE_IOS(3_0);

    允许一个事件处理是可转发给其他对象, 默认情况是 调用方法 -canPerformAction:withSender: 来返回自己或者提交返回给响应链去判断处理。

    // Allows an action to be forwarded to another target. By default checks -canPerformAction:withSender: to either return self, or go up the responder chain.

    - (nullable id)targetForAction:(SEL)action withSender:(nullable id)sender NS_AVAILABLE_IOS(7_0)

    三、事件传递:用户的触摸事件,首先被封装成一个UIEvent事件,然后UIEvent事件传递给UIResponder的事件,进行判断,寻找最合适的View。

    UIEvent事件中会有UITouch对象集,保存着UITouch对象

    @property(nonatomic, readonly, nullable) NSSet <UITouch *> *allTouches;

    一根手指对应一个UITouch对象,

    如果两根手指同事触摸,会调用一次- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;,其中的UIEvent事件中有两个UITouch对象

    如果两根手指先后触摸,会调用两次- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(nullable UIEvent *)event;方法,每个方法的UIEvent事件中各有一个UITouch对象

    UITouch中保存着手指相关的信息,触摸的位置,时间,阶段。

    当手指移动时,系统会更新同一个UITouch对象,使之能够一直保存该手指在的触摸位置

    当手指离开屏幕时,系统会销毁相应的UITouch对象

    事件产生之后,系统会将事件添加到UIApplication管理的事件队列中,UIApplication会从事件队列中取出最前面的事件,并将事件分发下去处理,先将事件发送给应用程序的主窗口 keyWindow。

    keyWindow会在视图层次结构中找到一个最合适的视图来处理触摸事件,这也是整个事件处理过程的第一步。

    事件产生之后,就是事件的传递过程:

    从父控件到子控件寻找合适的视图控件:

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

              recursively calls -pointInside:withEvent:. point is in the receiver's coordinate system

             递归调用-pointInside:withEvent:方法,判断point是否在其范围内

    • 1.首先判断主窗口(keyWindow)自己是否能接受触摸事件
    • 2.判断触摸点是否在自己身上(pointInside方法)

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

             // default returns YES if point is in bounds  如果点在bouns范围内,默认返回yes

    • 3.子控件数组中从后往前遍历子控件,重复前面的两个步骤(所谓从后往前遍历子控件,就是首先查找子控件数组中最后一个元素,然后执行1、2步骤)
    • 4.view,比如叫做fitView,那么会把这个事件交给这个fitView,再遍历这个fitView的子控件,直至没有更合适的view为止。
    • 5.如果没有符合条件的子控件,那么就认为自己最合适处理这个事件,也就是自己是最合适的view。

    期间用到的重要的两个方法:

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

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

    UIView不能接收触摸事件的三种情况:

    • 不允许交互:userInteractionEnabled = NO
    • 隐藏:如果把父控件隐藏,那么子控件也会隐藏,隐藏的控件不能接受事件
    • 透明度:如果设置一个控件的透明度<0.01,会直接影响子控件的透明度。alpha:0.0~0.01为透明。

    主要方法解析:

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

    调用时机:

    只要事件一传给某个控件,控件就会调用自己的该方法,寻找合适的子控件返回。

    作用:

    寻找合适的子控件返回

    注 意:不管这个控件能不能处理事件,也不管触摸点在不在这个控件上,事件都会先传递给这个控件,随后再调用hitTest:withEvent:方法

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

    拦截事件的处理:

    • 正因为hitTest:withEvent:方法可以返回最合适的view,所以可以通过重写hitTest:withEvent:方法,返回指定的view作为最合适的view。
    • 不管点击哪里,最合适的view都是hitTest:withEvent:方法中返回的那个view。
    • 通过重写hitTest:withEvent:,就可以拦截事件的传递过程,想让谁处理事件谁就处理事件。

    如果hitTest:withEvent:方法中返回nil,那么调用该方法的控件本身和其子控件都不是最合适的view,也就是在自己身上没有找到更合适的view。那么最合适的view就是该控件的父控件。

    事件传递顺序:

    产生触摸事件->UIApplication事件队列->[UIWindow hitTest:withEvent:]->返回更合适的view->[子控件 hitTest:withEvent:]->返回最合适的view

    重写的技巧:在父控件的hitTest:withEvent:中返回子控件作为最合适的view!

    hit:withEvent:方法底层会调用pointInside:withEvent:方法判断点在不在方法调用者的坐标系上

    pointInside:withEvent:方法

    判断点在不在当前view上(方法调用者的坐标系上)如果返回YES,代表点在方法调用者的坐标系上;返回NO代表点不在方法调用者的坐标系上,那么方法调用者也就不能处理事件。

    四、事件的响应:

    响应者链条

    如何判断上一个响应者

    • 1> 如果当前这个view是控制器的view,那么控制器就是上一个响应者
    • 2> 如果当前这个view不是控制器的view,那么父控件就是上一个响应者

    响应者链的事件传递过程:

    • 1>如果当前view是控制器的view,那么控制器就是上一个响应者,事件就传递给控制器;如果当前view不是控制器的view,那么父视图就是当前view的上一个响应者,事件就传递给它的父视图
    • 2>在视图层次结构的最顶级视图,如果也不能处理收到的事件或消息,则其将事件或消息传递给window对象进行处理
    • 3>如果window对象也不处理,则其将事件或消息传递给UIApplication对象
    • 4>如果UIApplication也不能处理该事件或消息,则将其丢弃

    五、事件处理的整个流程总结:
      1.触摸屏幕产生触摸事件后,触摸事件会被添加到由UIApplication管理的事件队列中(即,首先接收到事件的是UIApplication)。
      2.UIApplication会从事件队列中取出最前面的事件,把事件传递给应用程序的主窗口(keyWindow)。
      3.主窗口会在视图层次结构中找到一个最合适的视图来处理触摸事件。(至此,第一步已完成)
      4.最合适的view会调用自己的touches方法处理事件
      5.touches默认做法是把事件顺着响应者链条向上抛。

    参考:http://www.cnblogs.com/machao/p/5471094.html

    十分感谢前辈的分享

  • 相关阅读:
    python基础学习(九)
    python基础学习(八)
    python基础学习(七)
    python基础学习(六)
    python基础学习(五)
    python基础学习(四)
    python基础学习(三)
    mysql-binlog server的实现
    percona-toolkit常用工具
    Linux下如何快速定位系统瓶颈在哪里
  • 原文地址:https://www.cnblogs.com/Jordandan/p/6483835.html
Copyright © 2011-2022 走看看