iOS事件传递与响应

触摸事件发生时,会递归调用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继续响应滑动事件,事件不会传递给子视图

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容