iOS --苹果自带的UIMenuController功能扩展

一、UIMenuController认识

  • 1.默认情况下,UITextView / UITextFiled / UIWebView 都有苹果自带的有UIMenuController功能

  • 2.UITextFiled 的弹出菜单效果系统自带的


    Snip20151027_10.png
  • 3.当然,系统只给某些控件提供的该功能,但是我们自己给可以给指定控件添加该功能效果,至于如何添加,我们先简单学习一下,如何使用系统提供的UIMenuController功能

二、UIMenuController基本使用

  • 如何学UIMenuController使用,查看头文件是最直接的方式了。
Snip20151027_11.png
  • 可得:UIMenuController继承NSObject;我们可以自定义;通过sharedMenuController获取单例对象;必须手动设置弹窗菜单可见;指定弹窗相对哪个View的哪个位置显示;指定其显示方向(上下左右);指定Item多个(数组);而且可以指定menuFrame;不仅如此,系统也提供了可以监听menu的通知(即将显示/完全显示、即将隐藏/完全隐藏、menu的frame改变)

  • 1.获得菜单 menu

UIMenuController *menu = [UIMenuController sharedMenuController];
  • 2.设置菜单最终显示的位置
    // 菜单最终显示的位置
    CGRect rect = CGRectMake(100, 100, 100, 100);
    [menu setTargetRect:rect inView:self.label];
    /*
     targetRect:menuController指向的矩形框
     targetView:targetRect以targetView的左上角为坐标原点
     */
  • 3.手动设置需要,显示菜单
 // 显示菜单
[menu setMenuVisible:YES animated:YES];
  • 4.必须要得通过第一响应者,来告诉MenuController它内部应该显示什么内容

    • 4.1 那如何告诉显示内容呢?
      • 让第一响应者,实现下面方法,来告诉显示内容,监听哪些操作action

/**

  • 通过这个方法告诉UIMenuController它内部应该显示什么内容
  • 返回YES,就代表支持action这个操作
    */
  • (BOOL)canPerformAction:(SEL)action withSender:(id)sender
    {
    NSLog(@"%@", NSStringFromSelector(action));
    if (action == @selector(cut:)
    || action == @selector(copy:)
    || action == @selector(paste:)) {
    return YES; // YES -> 代表我们只监听 cut: / copy: / paste:方法
    }
    return NO; // 除了上面的操作,都不支持
    }
    // 打印如下:
    2015-7-28 10:06:25.578 UIMenuController[4735:388013] cut:
    2015-7-28 10:06:25.581 UIMenuController[4735:388013] copy:
    2015-7-28 10:06:25.581 UIMenuController[4735:388013] select:
    2015-7-28 10:06:25.582 UIMenuController[4735:388013] selectAll:
    2015-7-28 10:06:25.582 UIMenuController[4735:388013] paste:
    2015-7-28 10:06:25.582 UIMenuController[4735:388013] delete:
    2015-7-28 10:06:25.582 UIMenuController[4735:388013] _promptForReplace:
    2015-7-28 10:06:25.583 UIMenuController[4735:388013] _transliterateChinese:
    2015-7-28 10:06:25.583 UIMenuController[4735:388013] _showTextStyleOptions:
    2015-7-28 10:06:25.583 UIMenuController[4735:388013] _define:
    2015-7-28 10:06:25.583 UIMenuController[4735:388013] _addShortcut:
    2015-7-28 10:06:25.583 UIMenuController[4735:388013] _accessibilitySpeak:
    2015-7-28 10:06:25.583 UIMenuController[4735:388013] _accessibilitySpeakLanguageSelection:
    2015-7-28 10:06:25.583 UIMenuController[4735:388013] _accessibilityPauseSpeaking:
    2015-7-28 10:06:25.583 UIMenuController[4735:388013] _share:
    2015-7-28 10:06:25.584 UIMenuController[4735:388013] makeTextWritingDirectionRightToLeft:
    2015-7-28 10:06:25.584 UIMenuController[4735:388013] makeTextWritingDirectionLeftToRight:
        
   - 4.2 设置第一响应者
     + 前提是:必须要有第一响应者,让第一响应者,实现上面方法,告诉显示什么内容。实现下面方法,可以让某个视图或者控制器,成为第一响应者: canBecomeFirstResponder方法。

 ```objc
/**
 * 说明控制器可以成为第一响应者
 */
 - (BOOL)canBecomeFirstResponder
{
    return YES;
}
  • 5.实现监听菜单内容的对应的action方法
 - (void)cut:(UIMenuController *)menu
{
    NSLog(@"%s %@", __func__, menu);
}

- (void)copy:(UIMenuController *)menu
{
    NSLog(@"%s %@", __func__, menu);
}

- (void)paste:(UIMenuController *)menu
{
    NSLog(@"%s %@", __func__, menu);
}
  • 6.当然,我们还可以监听到menu菜单的显示与隐藏与frame改变的通知.
    • 如下:监听menu即将显示的通知
      • 1.注册通知监听
    // 注册监听 菜单即将显示 通知
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(show:) name:UIMenuControllerWillShowMenuNotification object:nil];
+ 2.实现监听到menu菜单显示调用方法

```objc
 - (void)dealloc{
   // 移除监听通知
   [[NSNotificationCenter defaultCenter] removeObserver:self];

}


 + 3.dealloc方法中,移除通知监听

```objc
  - (void)dealloc{
     // 移除监听通知
    [[NSNotificationCenter defaultCenter] removeObserver:self];
}
 ```

+ 7.当然系统提供的menuItem的标题,默认是英文的,我们可以设置menu支持中文,显示中文 ->修改软件应用支持中文

![Snip20151028_12.png](http://upload-images.jianshu.io/upload_images/831339-2cdc40153334b8da.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

+ 8.那么如何自定义meun显示自己定义的文字
- 解决方案:查看UIMenuController的头文件,我们发现有个属性menuItems数组,通过它我们可以增加额外的菜单项。

##### 注意:
+ 1.创建只能通过[UIMenuController sharedMenuController];单例方式获取,不能通过init方式创建,否则报如下错误

```objc

// 获得菜单 -> 回报如下错误
UIMenuController *menu = [[UIMenuController alloc] init]; 

�Terminating app due to uncaught exception 'NSInternalInconsistencyException',
reason: 'There can only be one UIMenuController instance.'

三、应用

1. 如何给Label添加UIMenuController功能

  • 1.设置UILabel允许交互
  • 2.给UILabel添加手势,
  • 3.在UILabel手势监听方法中,创建UIMenuController-》menu
  • 4.设置menu位置,利用UIMenuController的对象方法setTargetRect: inView:方法来设置menu显示在在那个控件的那个位置
  • 6.显示menu, -》 menu setMenuVisible: animation:
  • 7.设置menu显示内容
    • 注意:得通过第一响应者,来告诉menu它内部显示什么内容。如果显示中文标题,就需要手动设置APP支持中文
    • 实现:
      • 7.1让label成为第一响应者(注意:不一定第一响应者必须是控制器)
      • 7.2设置menu显示menuItem,告诉menu可以显示什么内容。
#import "JPLabel.h"

@implementation JPLabel

- (void)awakeFromNib
{
   // 给Label添加手机 
   [self addGestureRecognizer:[[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(labelClick)]];
}

-  (void)initWithFrame:(CGRect)rect{
   if(self = [super initWithFrame:rect]){
      // 给Label添加手机 
      [self addGestureRecognizer:[[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(labelClick)]];
   }
}
- (void)labelClick
{
    // 让label成为第一响应者
    [self becomeFirstResponder];
    
    // 获得菜单
    UIMenuController *menu = [UIMenuController sharedMenuController];
    
    // 设置菜单内容,显示中文,所以要手动设置app支持中文
    menu.menuItems = @[
                       [[UIMenuItem alloc] initWithTitle:@"顶" action:@selector(ding:)],
                       [[UIMenuItem alloc] initWithTitle:@"回复" action:@selector(reply:)],
                       [[UIMenuItem alloc] initWithTitle:@"举报" action:@selector(warn:)]
                       ];
    
    // 菜单最终显示的位置
    [menu setTargetRect:self.bounds inView:self];
    
    // 显示菜单
    [menu setMenuVisible:YES animated:YES];
}

#pragma mark - UIMenuController相关
/**
 * 让Label具备成为第一响应者的资格
 */
- (BOOL)canBecomeFirstResponder
{
    return YES;
}

/**
 * 通过第一响应者的这个方法告诉UIMenuController可以显示什么内容
 */
- (BOOL)canPerformAction:(SEL)action withSender:(id)sender
{
    if ( (action == @selector(copy:) && self.text) // 需要有文字才能支持复制
        || (action == @selector(cut:) && self.text) // 需要有文字才能支持剪切
        || action == @selector(paste:)
        || action == @selector(ding:)
        || action == @selector(reply:)
        || action == @selector(warn:)) return YES;
    
    return NO;
}

#pragma mark - 监听MenuItem的点击事件
- (void)cut:(UIMenuController *)menu
{
    // 将label的文字存储到粘贴板
    [UIPasteboard generalPasteboard].string = self.text;
    // 清空文字
    self.text = nil;
}

- (void)copy:(UIMenuController *)menu
{
    // 将label的文字存储到粘贴板
    [UIPasteboard generalPasteboard].string = self.text;
}

- (void)paste:(UIMenuController *)menu
{
    // 将粘贴板的文字赋值给label
    self.text = [UIPasteboard generalPasteboard].string;
}

- (void)ding:(UIMenuController *)menu
{
    NSLog(@"%s %@", __func__, menu);
}

- (void)reply:(UIMenuController *)menu
{
    NSLog(@"%s %@", __func__, menu);
}

- (void)warn:(UIMenuController *)menu
{
    NSLog(@"%s %@", __func__, menu);
}
@end

2.如何在cell中间实现,弹出菜单

  • 实现:见文章(tableView的Cell功能实现)中所示
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 148,637评论 1 318
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 63,443评论 1 266
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 99,164评论 0 218
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 42,075评论 0 188
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 50,080评论 1 266
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 39,365评论 1 184
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 30,901评论 2 283
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 29,649评论 0 176
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 33,122评论 0 223
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 29,734评论 2 225
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,093评论 1 236
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 27,548评论 2 222
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,028评论 3 216
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 25,765评论 0 9
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,291评论 0 178
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 34,162评论 2 239
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 34,293评论 2 242

推荐阅读更多精彩内容