案例
此时有一个按钮,就比如说是保存按钮,暂叫保存Button。
它的上面有一个子视图,就比如说是保存进度圈,暂叫进度圈View。
那么,现在有一个需求,点击保存按钮,保存按钮置灰不可点击。并且显示保存进度圈,并支持点击手势,比如点击之后出来pop一个面板。
# 抽象一下,就是:
父Button.enable = NO,
子View能够响应点击事件。正常来说点击是没有反应的,如何实现?
方案
重写保存Button的-hitTest:withEvent:方法,强行让进度圈View能够响应点击事件:
-(UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event {
if (CGRectContainsPoint(progressView.frame, point)) {
return progressView;
}
return [super hitTest:point withEvent:event];
}
可以试一下,结论是:即使保存Button设为不可点击,子视图进度圈View依然能够响应点击事件。
为什么可以这样?这就涉及到事件传递响应链的知识了。下面开始带着浅入了解
事件传递响应链
这里整个流程,分为传递链+响应链,一个个看:
#1 - 事件传递
第一步,当我们点击一台iOS设备,首先得知道我们的点击,具体是落在哪一个视图上面。
也就是说,这一步,把我们的点击事件,传递给具体某个视图。
以点击微信聊天界面-语音按钮为例:

点击语音按钮,此时系统生成一个事件,以上面的图层树型结构,
1、根结点调用它的-hitTest:withEvent:方法
2、然后自上至下,若有一个结点有子结点,对所有子结点,都调用子结点的-hitTest:withEvent:
这里的-hitTest:withEvent:,作用是找到最顶层的视图,将事件传递给它处理。
#2 - 事件响应
由上一步,已找到了事件由哪个视图来处理。
第二步,我们让这个视图,调用它的-pointInside:withEvent:,来决定如何响应事件。
-(BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event {
return YES; // 这个方法默认返回YES
}
比如,想要一个View,点击它圆里的范围,才响应事件。点击圆外的范围,则透过去。

代码可以是这样:
-(BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event {
if ([self isPointInRound:point]) {
return YES;
} else {
return NO;
}
}
另外,都知道UIView有nextResponder这个属性,当return NO,这个事件,将由nextResponder来响应。比如是它的父View。
总结
总结一下,就是主要通过以下两个方法,实现事件的传递与响应:
#1 先传递事件: -hitTest:withEvent:
#2 再响应事件: -pointInside:withEvent:
