一,通过分类向上传递信息
这里通过给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时,防止键盘退出
实现长按弹出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];
}
网友评论