responder chain

stack overflow上的有人提出responder chain问题UITableViewCell skipped in responder chain,并附上其git demo验证responder chain;我在阅读代码后,在此写下几点体会

demo分析

代码构造的view层次结构如下:

responder chain_第1张图片
view tree.jpg

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事件分成三类:

  1. 触摸事件(Touch event):触摸事件会被分发给触摸产生的view,查找这个view的过程就被成为 hitTest
  2. 运动事件(Motion event):会被分发给first responder处理
  3. 远程事件 (remote control event):同上

hitTest:withEvent:

touch event产生后,会被加入到app的事件队列,按照先进先出的原则,依次取出事件,系统先发给主窗口,主窗口按照view层次结构,去查找最小的发生触摸事件的view;主要过程如下:

  1. 判断view是否接收touch event,以下三种情况不接收:
  • userInteractionEnabled为no
  • hidden为YES
  • alpha为0
  1. 触摸点是否在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];
  }
}
  1. 依照上述过程遍历子控件
- (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];
}
  1. 如果没有符合的子控件,则自身就是hitTest-view

在项目中,运行结果调用,会发现

responder chain_第2张图片
屏幕快照 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_第3张图片
responder chain.jpg

first 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

你可能感兴趣的:(responder chain)