responder chain

作者: linatan | 来源:发表于2016-07-31 17:09 被阅读93次

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

    demo分析

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

    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

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

    屏幕快照 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.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

        本文链接:https://www.haomeiwen.com/subject/idoajttx.html