stack overflow上的有人提出responder chain问题UITableViewCell skipped in responder chain,并附上其git demo验证responder chain;我在阅读代码后,在此写下几点体会
demo分析
代码构造的view层次结构如下:
cell中button点击操作代码:
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
TableCell *cell = [[TableCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:@"Cell"];
ContentView *view = [[ContentView alloc] initWithFrame:cell.bounds];
cell.selectionStyle = UITableViewCellSelectionStyleBlue;
cell.backgroundColor = [UIColor greenColor];
view.backgroundColor = [UIColor clearColor];
UIButton *button = [[UIButton alloc] initWithFrame:CGRectMake(0, 0, 100, 44)];
[button setTitle:@"Button" forState:UIControlStateNormal];
button.backgroundColor = [UIColor blueColor];
[button addTarget:nil action:@selector(customEventFired:) forControlEvents:(1 << 24)];
[button addTarget:self action:@selector(sendStuff:) forControlEvents:UIControlEventTouchUpInside];
[view addSubview:button];
[cell.contentView addSubview:view];
return cell;
}
- (void)sendStuff:(id)sender {
UIButton *btn = (UIButton *)sender;
[btn sendActionsForControlEvents:(1 << 24)];
}
点击按钮时,会通过hitTest来查找touch事件是在哪个view上,之后触发类中的方法sendStuff:
,该方法中又会触发controlEvent(1<<24)
,因为[button addTarget:nil action:@selector(customEventFired:) forControlEvents:(1 << 24)];
,target是nil,会根据responder chain来查找可以处理事件的responder;接下来先介绍hitTest过程。
ios事件分成三类:
- 触摸事件(Touch event):触摸事件会被分发给触摸产生的view,查找这个view的过程就被成为 hitTest
- 运动事件(Motion event):会被分发给first responder处理
- 远程事件 (remote control event):同上
hitTest:withEvent:
touch event产生后,会被加入到app的事件队列,按照先进先出的原则,依次取出事件,系统先发给主窗口,主窗口按照view层次结构,去查找最小的发生触摸事件的view;主要过程如下:
- 判断view是否接收touch event,以下三种情况不接收:
- userInteractionEnabled为no
- hidden为YES
- alpha为0
- 触摸点是否在view范围中
- (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event
{
CGPoint btnPoint = [self convertPoint:point toView:_btn];
if ([_btn pointInsider:btnPoint withEvent:event])
{
return YES;
}
else
{
return [super pointInside:point withEvent:event];
}
}
- 依照上述过程遍历子控件
- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event
{
CGPoint btnPoint = [_btn convertPoint:point fromView:self];
if ([_btn pointInside:btnPoint withEvent:event])
{
return _btn;
}
return [super hitTest:point withEvent:event];
}
- 如果没有符合的子控件,则自身就是hitTest-view
在项目中,运行结果调用,会发现
屏幕快照 2016-07-31 下午4.27.41.png上述结果中没有展示button的hitTest:withEvent:结果,其中{{0,0},{100,44}}是button的frame
注意clipsToBounds设置为NO,因为subview可以超出superView,所以这个时候要重写
hitTest:withEvent:
或者pointInside:withEvent:
responder chain
hitTest view会处理touch event,但是如果其不能处理;则会根据响应链从firstResponder开始往上传递,寻找可以处理的responder为止。
根据responder chain确认event handler
responder chain.jpgfirst responder
被指派第一个接收事件,可以通过重写
- (BOOL)canBecomeFirstResponder {
NSLog(@"canBecomeFirstResponder");
return YES;
}
再调用[cell becomeFirstResponder]
,可以成为first responder;例如tableCell中重写canBecomeFirstResponder
,并调用becomeFirstResponder
,则在点击button时,响应链并没有调用contentView中的customEventFired:
方法,而是调用tableCell中的customEventFired:
方法。
参考
responder object
Event Delivery:The Responder Chain
iOS事件分发机制(一) hit-Testing
iOS事件分发机制(二)The Responder Chain
网友评论