iOS中长按调出菜单组件UIMenuController的使⽤实例UIMenuController的使⽤
UIMenuController的展现需要基于⼀个View视图,其交互则需要基于其所在View视图的Responder。举例来说,如果⼀个UIMenuController展现在当前ViewController的View上,则此UIMenuController的交互逻辑交由当前的ViewController进⾏管理。
在界⾯展⽰出UIMenuController需要3个条件:
1.当前的Responder处于第⼀响应。
2.UIMenuController对象调⽤menuVisible⽅法。
3.当前的Responder实现了如下两个⽅法:
//是否可以成为第⼀相应
-(BOOL)canBecomeFirstResponder{
return YES;
}
//是否可以接收某些菜单的某些交互操作
-(BOOL)canPerformAction:(SEL)action withSender:(id)sender{
return YES;
}
实现了上⾯的两个⽅法,使⽤如下的代码可以唤出UIMenuController控件:
[self becomeFirstResponder];
//设置菜单显⽰的位置 frame设置其⽂职 inView设置其所在的视图
[[UIMenuController sharedMenuController] setTargetRect:frame inView:self.view];
//将菜单控件设置为可见
[UIMenuController sharedMenuController].menuVisible = YES;
在执⾏了上⾯的代码后,系统第⼀次调⽤canperformAction:withSender:⽅法会进⾏是否显⽰菜单栏的检
测,如果返回为NO,则不能显⽰菜单栏,如果返回为YES,之后系统会多次调⽤canPerformAction:withSender:⽅法,⽤于检测当前Responder对象是否实现了菜单栏上某个选项的触发⽅法,如果实现了,菜单栏上⾯的相应按钮会显⽰,否则不会显⽰。开发者可以在这个⽅法中通过判断action来确定菜单控件中显⽰的按钮种类。系统默认为开发者提供了⼀系列的菜单按钮,例如要显⽰剪切和赋值操作的菜单按钮,⽰例代码如下:
-(BOOL)canPerformAction:(SEL)action withSender:(id)sender{
if (action == @selector(cut:)||action == @selector(copy:)) {
return YES;
}
return NO;
}
效果如下图所⽰:
系统默认⽀持提供的按钮触发⽅法列举如下:
//剪切按钮的⽅法
- (void)cut:(nullable id)sender NS_AVAILABLE_IOS(3_0);
//复制按钮的⽅法
- (void)copy:(nullable id)sender NS_AVAILABLE_IOS(3_0);
//粘贴按钮的⽅法
- (void)paste:(nullable id)sender NS_AVAILABLE_IOS(3_0);
//选择按钮的⽅法
- (void)select:(nullable id)sender NS_AVAILABLE_IOS(3_0);
//全选按钮的⽅法
- (void)selectAll:(nullable id)sender NS_AVAILABLE_IOS(3_0);
/
/删除按钮的⽅法
- (void)delete:(nullable id)sender NS_AVAILABLE_IOS(3_2);
//改变书写模式为从左向右按钮触发的⽅法
- (void)makeTextWritingDirectionLeftToRight:(nullable id)sender NS_AVAILABLE_IOS(5_0);
//改变书写模式为从右向左按钮触发的⽅法
- (void)makeTextWritingDirectionRightToLeft:(nullable id)sender NS_AVAILABLE_IOS(5_0);
上⾯所列举的⽅法声明在UIResponder头⽂件中,实际上,除了上⾯的⽅法,关于UIMenuController上⾯的按钮,系统中还有许多私有⽅法,列举如下:
//替换按钮
- (void)_promptForReplace:(id)arg1{
NSLog(@"promptForReplace");
}
//简体繁体转换按钮
-(void)_transliterateChinese:(id)sender{
NSLog(@"transliterateChinese");
}
//⽂字风格按钮
-(void)_showTextStyleOptions:(id)sender{
NSLog(@"showTextStyleOptions");
}
//定义按钮
-(void)_define:(id)sender{
NSLog(@"define");
}
-(void)_addShortcut:(id)sender{
NSLog(@"addShortcut");
}
-(void)_accessibilitySpeak:(id)sender{
NSLog(@"accessibilitySpeak");
}
//语⾔选择按钮
-(void)_accessibilitySpeakLanguageSelection:(id)sender{
NSLog(@"accessibilitySpeakLanguageSelection");
}
/
/暂停发⾳按钮
-(void)_accessibilityPauseSpeaking:(id)sender{
NSLog(@"accessibilityPauseSpeaking");
}
//分享按钮
-(void)_share:(id)sender{
NSLog(@"share");
}
实例进阶
在实际开发中,开发这完全不需要使⽤这些私有的⽅法,UIMenuItem类提供给开发者进⾏⾃定义菜单按钮与触发⽅法,⽰例如下:
[self becomeFirstResponder];
UIMenuItem * item = [[UIMenuItem alloc]initWithTitle:@"⾃定义" action:@selector(newFunc)];
[[UIMenuController sharedMenuController] setTargetRect:[sender frame] inView:self.view];
[UIMenuController sharedMenuController].menuItems = @[item];
[UIMenuController sharedMenuController].menuVisible = YES;
-(BOOL)canBecomeFirstResponder{
return YES;
}
-(BOOL)canPerformAction:(SEL)action withSender:(id)sender{
if (action == @selector(newFunc)) {
return YES;
}
return NO;
}
-(void)newFunc{
NSLog(@"⾃定义⽅法");
}
效果如下图所⽰:
UIMenuController还有如下的属性⽤来设置其显⽰的位置:
//显⽰的位置
@property(nonatomic) UIMenuControllerArrowDirection arrowDirection;
//枚举如下:
/
*
typedef NS_ENUM(NSInteger, UIMenuControllerArrowDirection) {
//默认基于当前屏幕状态
UIMenuControllerArrowDefault, // up or down based on screen location
//箭头在上的显⽰模式
UIMenuControllerArrowUp NS_ENUM_AVAILABLE_IOS(3_2),
//箭头在下的显⽰模式
UIMenuControllerArrowDown NS_ENUM_AVAILABLE_IOS(3_2),
//箭头在左的显⽰模式
UIMenuControllerArrowLeft NS_ENUM_AVAILABLE_IOS(3_2),
//箭头在右的显⽰模式
UIMenuControllerArrowRight NS_ENUM_AVAILABLE_IOS(3_2),
};
*/
注意点总结
要正常显⽰菜单,必须做到以下⼏点:
1. -(BOOL)canBecomeFirstResponder 必须返回YES
2. -(BOOL)canPerformAction:(SEL)action withSender:(id)sender
该函数中,要显⽰的菜单项(包括系统的菜单项)的⽅法必须返回YES 3. 在显⽰菜单前,必须调⽤:
[self becomeFirstResponder]
成为第⼀个响应者
4. 为了马上可以正常显⽰第⼆个菜单,必须使⽤:
[menuController setMenuVisible:NO];
菜单栏包括
先关闭⼀下,不然就显⽰不出来!