美文网首页
iOS事件传递与响应

iOS事件传递与响应

作者: 哈哈哈我的简书账号 | 来源:发表于2017-05-17 21:05 被阅读117次

触摸事件发生时,会递归调用hitTest:withEvent获得响应事件的试图,然后将触摸事件包装成UITouch,传递给[UIWindow hitTest:withEvent]调用返回的试图处理
• hitTest:withEvent:方法大致处理流程是这样的:
首先调用当前视图的pointInside:withEvent:方法判断触摸点是否在当前视图内:
▶ 若pointInside:withEvent:方法返回NO,说明触摸点不在当前视图内,则当前视图的hitTest:withEvent:返回nil
▶ 若pointInside:withEvent:方法返回YES,说明触摸点在当前视图内,则遍历当前视图的所有子视图(subviews),调用子视图的hitTest:withEvent:方法重复前面的步骤,子视图的遍历顺序是从top到bottom,即从subviews数组的末尾向前遍历,直到有子视图的hitTest:withEvent:方法返回非空对象或者全部子视图遍历完毕:
▷ 若第一次有子视图的hitTest:withEvent:方法返回非空对象,则当前视图的hitTest:withEvent:方法就返回此对象,处理结束
▷ 若所有子视图的hitTest:withEvent:方法都返回nil,则当前视图的hitTest:withEvent:方法返回当前视图自身(self)
• 最终,这个触摸事件交给主窗口的hitTest:withEvent:方法返回的视图对象去处理
我大致画了个iOS触摸事件分发的原理图:



• hitTest:withEvent:方法会忽略以下视图:
1> 隐藏(hidden=YES)的视图

2> 禁止用户操作(userInteractionEnabled=NO)的视图

3> alpha<0.01的视图

4> 如果一个子视图的区域超过父视图的区域(如果父视图的clipsToBounds属性为NO,超过父视图区域的子视图内容也会显示),那么正常情况下在父 视图区域外的触摸操作不会被识别,因为父视图的pointInside:withEvent:方法会返回NO,这样就不会继续向下遍历子视图了。当然,也 可以重写pointInside:withEvent:方法来处理这种

综上所述可得:如果父视图的userInteractionEnabled=NO,触摸事件不会继续往下传递给子视图,所以子视图永远无法处理触摸事件。而UIImageView在默认情况下的userInteractionEnabled就是NO。

事件的反向传递:

重写视图的touchesBegan方法,打印nextResponder,查看输出

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
    NSMutableString *str = [[NSMutableString alloc] init];
    printf("%s%s\n",str.UTF8String,NSStringFromClass([self class]).UTF8String);
    UIResponder *nextResponder = self.nextResponder;
    while (nextResponder) {
        printf("%s%s\n",str.UTF8String,NSStringFromClass([nextResponder class]).UTF8String);
        nextResponder = nextResponder.nextResponder;
        [str appendString:@"--"];
    }
}
6EC0EA82-7ADA-4797-BD87-DF2C35CD3339.png

EventView的控制器是EnventChuandiViewController,父视图是NextResponderView,NextResponderView的控制器是MethodSwizzlingViewController

hintTest是UIView的方法,所以产生event后寻找的入口是UIWindow。
但是当事件反向传递的时候,会传到UIApplication和AppDelegte.

那么问题来啦
1,responder是什么时候被赋值的?
2,hitTest的实现是什么?
3,UIScoroll如何响应事件?

responder是什么时候被赋值的?

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view.
    self.view.backgroundColor = [UIColor yellowColor];
    EnventChuandiViewController *eventViewVC = [[EnventChuandiViewController alloc] init];
//(lldb) po eventViewVC.view.nextResponder
//<EnventChuandiViewController: 0x7fc55282e210>
//(lldb) po eventViewVC.nextResponder
// nil
    [self.view addSubview:eventViewVC.view];
//(lldb) po eventViewVC.nextResponder
//<NextResponderView: 0x7fc552805a30; frame = (0 0; 0 0); layer = <CALayer: 0x608000034c80>>
    [self addChildViewController:eventViewVC];
    [eventViewVC didMoveToParentViewController:self];
    eventViewVC.view.frame = CGRectMake(100, 100, 100, 100);
    UIImageView *view = [[UIImageView alloc] init];
(lldb) po view.nextResponder
// nil
    [self.view addSubview:view];
(lldb) po view.nextResponder
//<NextResponderView: 0x7fc552805a30; frame = (0 0; 0 0); layer = <CALayer: 0x608000034c80>>
}

可以看到在创建视图和addSubView的时候fuzhi

hintTest实现如下:

- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event {
    if (!self.userInteractionEnabled || self.alpha < 0.01 || ![self pointInside:point withEvent:event] || self.isHiden) {
        return nil;
    }
    for (UIView *view in [self.subviews reverseObjectEnumerator]) {
       UIView *viewTmp = [view hitTest:[view convertPoint:point fromView:self] withEvent:event];
        if (viewTmp) {
            return viewTmp;
        }
    }
    return self;
}

为证明,给UIView写一个分类替换掉系统实现:

#import <UIKit/UIKit.h>

@interface UIView (hint)

@end
#import "UIView+hint.h"
#import "objc/runtime.h"
@implementation UIView (hint)
+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        SEL originalSelector = @selector(hitTest:withEvent:);
        SEL swizzledSelector = @selector(myhitTest:withEvent:);
        Class class = [self class];
        Method originalMethod = class_getInstanceMethod(class, originalSelector);
        Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector);
        BOOL didAddMethod =
        class_addMethod(class,
                        originalSelector,
                        method_getImplementation(swizzledMethod),
                        method_getTypeEncoding(swizzledMethod));
        if (didAddMethod) {
            class_replaceMethod(class,
                                swizzledSelector,
                                method_getImplementation(originalMethod),
                                method_getTypeEncoding(originalMethod));
        } else {
            method_exchangeImplementations(originalMethod, swizzledMethod);
        }

    });
}
- (UIView *)myhitTest:(CGPoint)point withEvent:(UIEvent *)event {
    if (!self.userInteractionEnabled || self.alpha < 0.01 || ![self pointInside:point withEvent:event] || self.isHidden) {
        return nil;
    }
 
    for (UIView *view in [self.subviews reverseObjectEnumerator]) {
        UIView *viewTmp = [view hitTest:[view convertPoint:point fromView:self] withEvent:event];
        if (viewTmp) {
            return viewTmp;
        }
    }
    return self;
}

@end

系统运行正常
3,UIScoroll如何响应事件?
视图上默认添加的手势识别器会拦截触摸事件
(1),触摸时间小于阈值
如果滑动了手指,UIScrollView响应滑动事件
如果没滑动过,不作处理
(2),触摸时间大于阈值
如果没有滑动过手指,那么会把触摸事件交给子视图处理(如果之后手指滑动,会取消发送给子视图的事件,UIScrollView响应滑动事件)
如果滑动过手指,UIScrollView继续响应滑动事件,事件不会传递给子视图

相关文章

  • iOS 响应链

    iOS开发 - 事件传递响应链iOS 响应者链,事件的传递事件传递之响应链Cocoa Touch事件处理流程--响...

  • 深入浅出iOS事件机制

    深入浅出iOS事件机制事件传递:响应链事件传递响应链

  • iOS 触摸事件与响应理解

    参考文章: iOS触摸事件的流动 iOS触摸事件的传递与响应 UIViewController UIAppli...

  • iOS响应者链

    参考好文 iOS开发-事件传递响应链,用运行时分析 iOS事件传递:响应者链[译] http://www.jian...

  • 初识iOS事情处理机制

    参考:史上最详细的iOS之事件的传递和响应机制-原理篇iOS触摸事件全家桶史上最详细的iOS之事件的传递和响应机制...

  • 二、事件传递链和响应者链

    iOS触摸事件详解iOS开发-事件传递响应链 响应者链 UIResponser包括了各种Touch message...

  • iOS之事件的传递和响应机制

    iOS之事件的传递和响应机制

  • iOS事件传递与响应

    触摸事件发生时,会递归调用hitTest:withEvent获得响应事件的试图,然后将触摸事件包装成UITouch...

  • iOS事件传递与响应

    在 UIKit 中我们使用响应者对象(Responder)接收和处理事件。一个响应者对象一般是 UIRespond...

  • iOS 中事件的响应链和传递链

    iOS 事件的主要由:响应连 和 传递链 构成。一般事件先通过传递链,传递下去。响应链,如果上层不能响应,那么一层...

网友评论

      本文标题:iOS事件传递与响应

      本文链接:https://www.haomeiwen.com/subject/cokdxxtx.html