• iOS开发事件分发机制—响应链—手势影响


    1、提纲

    • 什么是iOS的事件分发机制 ?
    • 一个事件UIEvent又是如何响应的?
    • 手势对于响应链有何影响?

    2、事件分发机制

      2.1、来源

        以直接触摸事件为例:

      1. 当用户一个手指触摸屏幕是会生成一个UITouch对象,多个手指就是多个对象,手指移动系统会更新对象的相应信息。
      2. 系统会将UITouch对象封装生成一个事件UIEvent对象,将这个事件交给最佳具有响应能力的视图处理。
      3. 系统有个UIResponder类,只有继承UIResponder的类才具有响应事件能力,所以UIKit系统控件大多是继承此类。

        如何找到处理这个事件的视图的过程——事件分发机制

      2.2、具体步骤

        2.2.1、事件Event的产生

          点击一下iOS设备的屏幕,UIKit就会生成一个事件对象UIEvent,然后会把这个Event分发给当前活动的app; 

          当前活动的app得知有事件,UIApplication 单例就会从事件队列中去取最新的事件,然后分发给能够处理该事件的对象(具体视图)。

        2.2.2、运用到的两个UIView中的方法

    - (nullable UIView *)hitTest:(CGPoint)point withEvent:(nullable UIEvent *)event;   //找到并返回最合适的视图来
    - (BOOL)pointInside:(CGPoint)point withEvent:(nullable UIEvent *)event;   //判断点是否在视图内

          当一个视图View收到hitTest消息时,会调用自己的poinInside方法;

          如果返回YES,View会遍历自己的子视图(遍历顺序先addSubView先遍历),子视图就会调用自己的hitTest方法;

          如果返回NO,View就不会遍历自己子视图(很节约);

          直到找到最小的能够处理事件的view,如果整了一圈没找到能够处理的view,则返回自身。

        2.2.3、举例说明

          白色:ViewController , 绿色:一个View视图  , 蓝色:一个Button按钮

          现象:点击绿色视图内的按钮区域,正常;点击绿色视图外的按钮区域,按钮的selector方法不会调用?

          重新绿色view的hitTest方法:

          点击有效区域时:返回视图坐标{{88, -26}, {47, 52}},此坐标是相对父视图的;

          点击区域外的部分时:返回视图坐标{{0, 0}, {0, 0}},所以按钮不会响应;

          重新hitTest方法,修改返回view,当点击区域外按钮部分时也返回有效视图{{88, -26}, {47, 52}},如图所示;

          另外一种解决方案:更暴力 

    - (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event{
        return YES;
    }

    号外:

      如果 View 的 userInteractionEnabled = NO,enabled = NO( UIControl ),或者 alpha <= 0.01, hidden = YES 等情况的时候,直接返回 nil。
      如果触摸点不在 view 中,直接返回 nil (例子就是此现象)。


    3、响应链
      响应链:在事件分发机制寻找view时会生成响应链;
      第一响应者:上例中Button就是第一响应者;
      下一个响应者:UIResponder有个属性只读的为nextResponder,当第一响应者不处理事件时就在吧事件抛给下一个响应者;
      响应链的一些规则:
    1. UIView 的 nextResponder 是直接管理它的 UIViewController (也就是 VC.view.nextResponder = VC ),如果当前 View 不是 ViewController 直接管理的 View,则 nextResponder 是它的 superView( view.nextResponder = view.superView )。
    2. UIViewController 的 nextResponder 是它直接管理的 View 的 superView( VC.nextResponder = VC.view.superView )。
    如果viewcontroller的view是window的根view,那么下一个响应者是window;
    如果viewcontroller是另一个viewcontroller模态推出的,那么下一个响应者是另一个viewcontroller;
    如果viewcontroller的view被add到另一个viewcontroller的根view上,那么下一个响应者是另一个viewcontroller的根view
    3. UIWindow 的 nextResponder 是 UIApplication 。
    4. UIApplication 的 nextResponder 是 AppDelegate。
      上例中的响应链:UIButton.nextResponder = 绿色view.nextResponder = VC.view.nextResponder = VC.nextResponder = VC.view.superViewe(本例中为UIWindow).nextResponder = UIApplication.nextResponder = AppDelegate ;

       一般来说,某个 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);

      如果想自己处理后,继续让事件响应下去:

    - (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];
    }

    4、手势对响应链有何影响

      手势识别器并不是响应者链中的一员,但是手势识别器会观察touch事件,并延迟事件向所绑定的视图传递;

      在上例中给Button和view添加点击手势:

      按钮会优先响应手势,而不是自身的selector;

      设置手势代理:让按钮响应自身的selector
    - (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldReceiveTouch:(UITouch *)touch{
        if ([touch.view isKindOfClass:[UIButton class]]) {
            return NO;
        }else{
            return YES;
        }
    }

      

      添加手势又屏蔽:有毛病?

      在其他UICollectionView和UIScrollView、UITableView等,如果需要在视图上添加手势,cell上也会响应手势,此方法可以解决手势和自身方法的冲突!



     
  • 相关阅读:
    Azure SQL Storage
    T-SQL quries
    映射盘符
    繁体及其输入法、乱码问题
    匈牙利命名法
    C++四种转换总结
    windows系统下进程间通信
    Qt 中文字符串问题
    PDB文件详解
    DbgView 无法开启Capture Kernel问题
  • 原文地址:https://www.cnblogs.com/xianfeng-zhang/p/9473063.html
Copyright © 2020-2023  润新知