美文网首页
事件处理之nextResponder

事件处理之nextResponder

作者: LX2014 | 来源:发表于2017-09-02 15:21 被阅读6次

一,通过分类向上传递信息

参考链接:http://www.jianshu.com/p/eb775f5d9032

这里通过给UIResponder添加分类方法:

- (void)routerInfo:(id)info;
- (void)routerInfo:(id)info sender:(UIResponder *)sender; //带有发送者的方法

省去了向上传递数据的代理方法和block调用。作者通过一个在cell上传递数据到控制器中的例子展示了nextResonder来处理事件信息处理的简便性。这里有个注意点需要提一下:如果tableView是自定义的并且实现了- (void)routerInfo:(id)info方法,接收了数据。那么控制器中就不能响应- (void)routerInfo:(id)info方法的事件了。如果控制器还需要处理,在tableView中继续调用该方法。

cell中的实现:

@implementation OKTestTableViewCell
- (void)awakeFromNib {
    [super awakeFromNib];
    // Initialization code
}
- (IBAction)btnClick:(id)sender {
    [self.nextResponder routerInfo:self.label.text];
}
@end

tableView中的实现:

@implementation OKTestTableView
//如果tableView中不实现,cell的事件就会向上抛给Controller处理。
- (void)routerInfo:(id)info {
    NSLog(@"tableView中的routerInfo");
    id tableViewData = nil;
    //如果需要向上抛,那么就继续调用
    [self.nextResponder routerInfo:tableViewData];
}
@end

如果tableView中没有实现- (void)routerInfo:(id)info 方法,而控制器中实现了,那么控制器会直接获取到cell中传递的信息。

控制器中:

- (void)routerInfo:(id)info {
    //如果tableView中没有实现,控制器就直接获取cell传递的info信息
    NSLog(@"控制器的routerInfo:%@",info);
}

二,长按显示UIMenuController时,防止键盘退出

参考:http://www.jianshu.com/p/b64a57538f74

实现长按弹出UIMenu的功能,需要在长按方法中设置控件(UILabel,UITextField)成为第一响应者,并且在canBecome:

- (void)longClick:(UILongPressGestureRecognizer *)longGesture {

    if (longGesture.state == UIGestureRecognizerStateBegan) {
        // 菜单已经打开不需重复操作
        UIMenuController *menu=[UIMenuController sharedMenuController];
        if (menu.isMenuVisible)return;
        //如果要不收起键盘还需要复写nextResponder方法
        //        if ([textField isFirstResponder]) {
        //            textField.inputNextResponder = self;//关键代码
        //        }else{
        //            [self becomeFirstResponder];
        //        }
          //注意要设置为第一响应者
          [self becomeFirstResponder];
        UIMenuItem *copy = [[UIMenuItem alloc] initWithTitle:@"复制"action:@selector(copyItem:)];
        UIMenuItem *paste = [[UIMenuItem alloc] initWithTitle:@"粘贴" action:@selector(pasteItem:)];
        [menu setMenuItems:@[copy,paste]];

        [menu setTargetRect:self.bounds inView:self];
        [menu setMenuVisible:YES animated:YES];
       //如果复写nextResponder方法,还要在menu隐藏时,移除inputNextResponder
       // [[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(menuWillHidden) name:UIMenuControllerWillHideMenuNotification object:nil];

    }
}

如果是可以输入的控件需要在键盘弹起时还能复制等长按操作,那么就要增加一个变量来保持自己的第一响应者身份:

MATextView.h
@interface MATextView : UITextView
@property (nonatomic, weak) UIResponder *inputNextResponder;
@end

MATextView.m

@implementation MATextView
- (UIResponder *)nextResponder {
 if (_inputNextResponder != nil)
   return _inputNextResponder;
  else
    return [super nextResponder];
}
 
//在设置外部的responder后,就不能执行方法
- (BOOL)canPerformAction:(SEL)action withSender:(id)sender {
if (self.inputNextResponder != nil) {
    return NO;
}
else {
    return [super canPerformAction:action withSender:sender];
}
}
@end

需要拷贝的自定义控件里,必须设置为YES才能添加UIMenu

- (BOOL)canBecomeFirstResponder {
    return YES;
}

如果没有实现本方法就只有自定义的item,如果实现了,就是实现中的item

- (BOOL)canPerformAction:(SEL)action withSender:(id)sender
{
    NSLog(@"%@", NSStringFromSelector(action));
    if (action == @selector(cut:)
        || action == @selector(copy:)
        || action == @selector(paste:) || action == @selector(copyItem:)|| action ==     @selector(pasteItem:)) {
        return YES; // YES ->  代表我们只监听 cut: / copy: / paste:方法
    }
    return NO; // 除了上面的操作,都不支持
}

如下是照搬代码供参考,如果添加了inputNextResponder还要将其置空:

- (void)copyBtnWillHidden{
    UIView *superView =  self.superview.superview.superview;
    KFChatToolView *toolView = [superView viewWithTag:kKF5ChatToolViewTag];
    if ([toolView isKindOfClass:[KFChatToolView class]]) {
            toolView.textView.inputNextResponder = nil;
    }
    self.chatMessageBackgroundView.highlighted = NO;
    [[NSNotificationCenter defaultCenter]removeObserver:self];
}

相关文章

网友评论

      本文标题:事件处理之nextResponder

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