查找響應(yīng)者
- (nullable UIView *)hitTest:(CGPoint)point withEvent:(nullable UIEvent *)event;
- (BOOL)pointInside:(CGPoint)point withEvent:(nullable UIEvent *)event;
hitTexst
它首先會通過調(diào)用自身的 pointInside 方法判斷用戶觸摸的點是否在當前對象的響應(yīng)范圍內(nèi),如果 pointInside 方法返回 NO hitTest方法直接返回 nil
如果 pointInside 方法返回 YES hitTest方法接著會判斷自身是否有子視圖.如果有則調(diào)用頂層子視圖的 hitTest 方法 直到有子視圖返回 View
如果所有子視圖都返回 nil hitTest 方法返回自身.
事件傳遞
- (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);
1、找到第一響應(yīng)者 application 便會根據(jù) event 調(diào)用第一響應(yīng)者響
2朱监、第一響應(yīng)者在這幾個方法中處理響應(yīng)的事件,處理完成后根據(jù)需要調(diào)用 nextResponder 的 touch 方法,通常 nextResponder 就是第一響應(yīng)者的 superView 文章的第一張圖倒著看就是nextResponder 的順序
事件處理流程
1 當用戶點擊屏幕時焰手,會產(chǎn)生一個觸摸事件,系統(tǒng)會將該事件加入到一個由UIApplication管理的事件隊列中
2 UIApplication會從事件隊列中取出最前面的事件進行分發(fā)以便處理粟按,通常,先發(fā)送事件給應(yīng)用程序的主窗口(UIWindow)
3 主窗口會調(diào)用hitTest:withEvent:方法在視圖(UIView)層次結(jié)構(gòu)中找到一個最合適的UIView來處理觸摸事件
(hitTest:withEvent:其實是UIView的一個方法,UIWindow繼承自UIView布蔗,因此主窗口UIWindow也是屬于視圖的一種)
通常第一響應(yīng)者都是響應(yīng)鏈中最末端的響應(yīng)者,事件攔截就是在響應(yīng)鏈中截獲事件,停止下發(fā).將事件交由中間的某個響應(yīng)者執(zhí)行.比如這樣:
應(yīng)用
1、擴大按鈕的點擊區(qū)域(上下左右各增加20)
重寫按鈕的 pointInside 方法
- (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event
{
if (CGRectContainsPoint(CGRectInset(self.bounds, -20, -20), point)) {
return YES;
}
return NO;
}
2、子view超出了父view的bounds響應(yīng)事件
正常情況下田藐,子View超出父View的bounds的那一部分是不會響應(yīng)事件的右遭。一般解決方案:修改父view的大小
解決方法:重寫父View
的pointInside方法,使事件Point 在超出父view的部分返回 true
- (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event
{
BOOL flag = NO;
for (UIView *view in self.subviews) {
if (CGRectContainsPoint(view.frame, point)){
flag = YES;
break;
}
}
return flag;
}
3滥搭、如果一個Button被一個View蓋住了,在觸摸View時,希望該Button能夠響應(yīng)事件
解決方法1:點擊View及View的非交互子View(例如UIImageView)帚桩,則該Button可以響應(yīng)事件
- (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event
{
BOOL next = YES;
for (UIView *view in self.subviews) {
if ([view isKindOfClass:[UIControl class]]) {
if (CGRectContainsPoint(view.frame, point)){
next = NO;
break;
}
}
}
return !next;
}
解決方法2:點擊View本身Button會響應(yīng)該事件,點擊View的任何一個子View嘹黔,Button不會響應(yīng)事件
- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event
{
UIView *view = [super hitTest:point withEvent:event];
if (view == self) {
return nil;
}
return view;
}
4 特殊的UIScrollView
該ScrollView可以顯示上一頁和下一頁的部分界面账嚎,紅色框是ScrollView的frame,綠色框部分是設(shè)置了clipsToBounds = NO的結(jié)果儡蔓,但是正如情況2提到的郭蕉,超出部分是不響應(yīng)事件的。
CGSize size = [UIScreen mainScreen].bounds.size;
CGFloat width = size.width - 80;
JCScrollView *scrollView = [[JCScrollView alloc] initWithFrame:CGRectMake(40, size.height - 150 - 30, width, 150)];
scrollView.pagingEnabled = YES;
scrollView.clipsToBounds = NO;
scrollView.showsVerticalScrollIndicator = NO;
scrollView.showsHorizontalScrollIndicator = NO;
[scrollView setContentSize:CGSizeMake(width * 5, 150)];
for (NSInteger i = 0; i < 5; i++) {
UIView *view = [[UIView alloc] initWithFrame:CGRectMake(5 + width * i, 0, width - 10, 150)];
view.backgroundColor = [UIColor colorWithRed:arc4random_uniform(255) / 255.0f green:arc4random_uniform(255) / 255.0f blue:arc4random_uniform(255) / 255.0f alpha:1];
[scrollView addSubview:view];
}
[self.view addSubview:scrollView];
如果需要綠色框響應(yīng)ScrollView的滾動事件喂江,則原理和情況1一樣
- (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event
{
if (CGRectContainsPoint(CGRectInset(self.bounds, -40, 0), point)) {
return YES;
}
return NO;
}