一.基本概念

UIKit框架中,可以直接执行拷贝黏贴操作的有:UITextView、UITextField和UIWebView,其他控件需要实现相关方法。

关于UIPasteboard

·黏贴板是app内或者app之间,交换数据的标准机制

·有公开的也有私有的,公开的黏贴板为系统级别(system pasteboard),私有的黏贴板为应用程序级别(app pasteboard),系统级别的黏贴板可以分享数据给任意其他app,应用程序级别的黏贴板只能分享给应用本身或者有相同team ID的应用

·system pasteboard默认是持久化的,所谓持久化,就是设备重启或者app卸载依然有效,app pasteboard默认是非持久化的,可以通过persistent修改

@property(getter=isPersistent,nonatomic) BOOL persistent;

一般使用名为UIPasteboardNameGeneral的system pasteboard,这货是个单例对象,是整个手机所有app共享的黏贴板

UIPasteboard *generalPasteboard = [UIPasteboard generalPasteboard];

二.具体实现

以label为例,假设label轻触两下或者长按显示编辑菜单

屏幕快照 2016-04-28 下午8.55.08.png

过程将会涉及

·UIPasteboard:提供黏贴板分享数据,从黏贴板读-写数据

·UIMenuController:展示编辑菜单,用来执行拷贝-黏贴操作

·UIResponder类的canPerformAction:withSender:方法,决定是否显示菜单某些命令,例如剪切

·UIResponderStandardEditActions非正式协议中声明了copy、cut、paste等方法,需要实现

:展示编辑菜单之前,系统会发送canPerformAction:withSender:消息给第一响应者,所以控件需要成为第一响应者来接收消息,并且在这个方法中,根据上下文的环境评估命令是否可用,例如当黏贴板没有可处理的数据时,响应者应该返回NO,阻止黏贴命令

0.首先打开用户交互,这个不要忘记,添加手势

- (void)awakeFromNib {    
   [super awakeFromNib];    
   UILongPressGestureRecognizer *longPress = [[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(showMenu:)];    
   longPress.minimumPressDuration = 0.5;    
   [self addGestureRecognizer:longPress];
}  
- (void)touchesEnded:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {    
   UITouch *touch = [touches anyObject];    
   if (touch.tapCount == 2) {          
       [self showMenu:nil];      
   }
}

1.是否可以成为第一响应者,默认为NO

- (BOOL)canBecomeFirstResponder {    
   return YES;
}

2.成为第一响应者

[self becomeFirstResponder];

3.创建编辑菜单

//获得单例对象UIMenuController *menu = [UIMenuController sharedMenuController];//view是参照物,指向以view为参照物的rect[menu setTargetRect:self.bounds inView:self];//显示[menu setMenuVisible:YES animated:YES];

4.实现canPerformAction:withSender:方法,参数action具体会有哪些方法,基本都在UIResponderStandardEditActions声明了

- (BOOL)canPerformAction:(SEL)action withSender:(id)sender {    
   NSLog(@"%@",NSStringFromSelector(action));    
   NSLog(@"%@",sender);    
   if (action == @selector(copy:) && self.text) {        //显示条件是文本不为空        
       return YES;      
   }    else if (action ==@selector(cut:) && self.text) {        //显示条件是文本不为空          
        return YES;      
   }    else if (action == @selector(paste:) && [UIPasteboard generalPasteboard].string) {        //显示条件是黏贴板不为空        
       return YES;      
   }    else {        
       return NO;      
   }
}

5.实现copy、cut、paste等方法

//实现剪切
- (void)cut:(id)sender {    
   [self copy:sender];    
   self.text = nil;
}
//实现拷贝
- (void)copy:(id)sender {    
   [UIPasteboard generalPasteboard].string = self.text;
}
//实现黏贴
- (void)paste:(id)sender {    
   UIPasteboard *generalPasteboard = [UIPasteboard generalPasteboard];    NSMutableArray *types = [[NSMutableArray alloc] init];    
   [types addObjectsFromArray:UIPasteboardTypeListString];    
   if ([generalPasteboard containsPasteboardTypes:types]) {        
       self.text = generalPasteboard.string;      
   }
}

6.如果想要添加自定义命令,可以创建UIMenuItem

//添加自定义菜单项
UIMenuItem *item = [[UIMenuItem alloc] initWithTitle:@"Change Color" action:@selector(changeColor:)];
menu.menuItems = @[item];

再以imageView为例,展示图片拷贝黏贴操作

#import "CustomImageView.h"@implementation CustomImageView- (void)awakeFromNib {    
   [super awakeFromNib];    
   UILongPressGestureRecognizer *longPress = [[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(showMenu:)];    
   longPress.minimumPressDuration = 0.5;    
   [self addGestureRecognizer:longPress];
}  
- (void)touchesEnded:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {    
   UITouch *touch = [touches anyObject];    
   if (touch.tapCount == 2) {          
       [self showMenu:nil];      
   }
}
/**  *  4.响应者是否可以执行该方法  *  *  @param action 方法  *  @param sender 请求对象  */
- (BOOL)canPerformAction:(SEL)action withSender:(id)sender {    
   if (action == @selector(copy:) && self.image) {        
       return YES;      
   }    else if (action == @selector(cut:) && self.image) {        
       return YES;      
   }    else if (action == @selector(paste:) && [UIPasteboard generalPasteboard].image) {        
       return YES;      
   }    else if (action == @selector(changeColor:)) {        
       return YES;      
   }    else {        
       return NO;      
   }
}
/**  *  1.是否可以成为第一响应者,默认为NO  *
/- (BOOL)canBecomeFirstResponder {    
return YES;
}  
- (void)showMenu:(UILongPressGestureRecognizer *)longPress {    
   if (longPress.state == UIGestureRecognizerStateEnded || longPress == nil) {        //2.成为第一响应者        
       [self becomeFirstResponder];        //添加自定义菜单项        
       UIMenuItem *item = [[UIMenuItem alloc] initWithTitle:@"Change Color" action:@selector(changeColor:)];        //3.创建编辑菜单        
       UIMenuController *menu = [UIMenuController sharedMenuController];        
       [menu setTargetRect:self.bounds inView:self];          
       menu.menuItems = @[item];          
       [menu setMenuVisible:YES animated:YES];      
   }
}  
- (void)cut:(id)sender {    
   [self copy:sender];    
   self.image = nil; }  
- (void)copy:(id)sender {    
   [UIPasteboard generalPasteboard].image = self.image;
}  
   - (void)paste:(id)sender {    
   UIPasteboard *generalPasteboard = [UIPasteboard generalPasteboard];    
   NSMutableArray *types = [[NSMutableArray alloc] init];    
   [types addObjectsFromArray:UIPasteboardTypeListImage];    
   if ([generalPasteboard containsPasteboardTypes:types]) {        
       self.image = generalPasteboard.image;      
   }
}  
- (void)changeColor:(id)sender {    
   self.backgroundColor = [UIColor colorWithRed:arc4random_uniform(256)/255.0 green:arc4random_uniform(256)/255.0  blue:arc4random_uniform(256)/255.0  alpha:1.0];
}

aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAIAAACQd1PeAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAyBpVFh0WE1MOmNvbS5hZG9iZS54bXAAAAAAADw/eHBhY2tldCBiZWdpbj0i77u/IiBpZD0iVzVNME1wQ2VoaUh6cmVTek5UY3prYzlkIj8+IDx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6bWV0YS8iIHg6eG1wdGs9IkFkb2JlIFhNUCBDb3JlIDUuMC1jMDYwIDYxLjEzNDc3NywgMjAxMC8wMi8xMi0xNzozMjowMCAgICAgICAgIj4gPHJkZjpSREYgeG1sbnM6cmRmPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5LzAyLzIyLXJkZi1zeW50YXgtbnMjIj4gPHJkZjpEZXNjcmlwdGlvbiByZGY6YWJvdXQ9IiIgeG1sbnM6eG1wPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvIiB4bWxuczp4bXBNTT0iaHR0cDovL25zLmFkb2JlLmNvbS94YXAvMS4wL21tLyIgeG1sbnM6c3RSZWY9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9zVHlwZS9SZXNvdXJjZVJlZiMiIHhtcDpDcmVhdG9yVG9vbD0iQWRvYmUgUGhvdG9zaG9wIENTNSBXaW5kb3dzIiB4bXBNTTpJbnN0YW5jZUlEPSJ4bXAuaWlkOkJDQzA1MTVGNkE2MjExRTRBRjEzODVCM0Q0NEVFMjFBIiB4bXBNTTpEb2N1bWVudElEPSJ4bXAuZGlkOkJDQzA1MTYwNkE2MjExRTRBRjEzODVCM0Q0NEVFMjFBIj4gPHhtcE1NOkRlcml2ZWRGcm9tIHN0UmVmOmluc3RhbmNlSUQ9InhtcC5paWQ6QkNDMDUxNUQ2QTYyMTFFNEFGMTM4NUIzRDQ0RUUyMUEiIHN0UmVmOmRvY3VtZW50SUQ9InhtcC5kaWQ6QkNDMDUxNUU2QTYyMTFFNEFGMTM4NUIzRDQ0RUUyMUEiLz4gPC9yZGY6RGVzY3JpcHRpb24+IDwvcmRmOlJERj4gPC94OnhtcG1ldGE+IDw/eHBhY2tldCBlbmQ9InIiPz6p+a6fAAAAD0lEQVR42mJ89/Y1QIABAAWXAsgVS/hWAAAAAElFTkSuQmCC" alt="" data-ratio="1.6079136690647482" data-src="http://mmbiz.qpic.cn/mmbiz/g4uoJOMA38LLFWhB4dvm9mtEdAwLxmoiaytJLSjtVbuSAKPeicp3CpAXZBLib0COTrgwZxl1iaMWIChoXBic9yk2kRQ/0?wx_fmt=png" data-type="png" data-w="" />

屏幕快照 2016-04-28 下午9.28.28.png

屏幕快照 2016-04-28 下午9.30.34.png

感谢分享

UIMenuController搭配UIPasteboard,执行拷贝-黏贴操作-b的更多相关文章

  1. Service 是否在 main thread 中执行, service 里面是否能执行耗时的操作?

    默认情况,如果没有显示的指 service 所运行的进程, Service 和 activity 是运行在当前 app 所在进程的 main thread(UI 主线程)里面.service 里面不能 ...

  2. 闯祸了,生成环境执行了DDL操作《死磕MySQL系列 十四》

    由于业务随着时间不停的改变,起初的表结构设计已经满足不了如今的需求,这时你是不是想那就加字段呗!加字段也是个艺术活,接下来由本文的主人咔咔给你吹. 试想一下这个场景 事务A在执行一个非常大的查询 事务 ...

  3. Android AlarmManager(全局定时器/闹钟)指定时长或以周期形式执行某项操作

    AlarmManager的使用机制有的称呼为全局定时器,有的称呼为闹钟.通过对它的使用,个人觉得叫全局定时器比较合适,其实它的作用和Timer有点相似.都有两种相似的用法:(1)在指定时长后执行某项操 ...

  4. svn执行clean up 操作时报错 "Previous operation has not finished; run 'cleanup' if it was interrupted"解决如下!

    今天在项目中更新的时候,突然间爆了一个svn的这个错误,当时提示我去clean up操作,结果我执行clean up操作时候,还是报错,后来坚持出来,是因为ios项目中的一个图标出了问题,使svn进入 ...

  5. Android之AlarmManager(全局定时器/闹钟)指定时长或以周期形式执行某项操作

    1.AlarmManager,顾名思义,就是“提醒”,是Android中常用的一种系统级别的提示服务,可以实现从指定时间开始,以一个固定的间隔时间执行某项操作,所以常常与广播(Broadcast)连用 ...

  6. 不借助工具在浏览器中通过Web API执行Dynamics 365操作(Action)实例

    摘要: 本人微信和易信公众号: 微软动态CRM专家罗勇 ,回复262或者20170727可方便获取本文,同时可以在第一间得到我发布的最新的博文信息,follow me!我的网站是 www.luoyon ...

  7. java for循环里面执行sql语句操作,有效结果只有一次,只执行了一次sql mybatis 循环执行update生效一次 实际只执行一次

    java后台controller中,for循环执行数据库操作,但是发现实际仅仅执行了一次,或者说提交成功了一次,并没有实际的个数循环 有可能是同一个对象导致的 可以仔细看一下下面两段代码有什么区别 p ...

  8. Python 标准类库-数据类型之copy-深拷贝浅拷贝操作

    标准类库-数据类型之copy-深拷贝浅拷贝操作   by:授客 QQ:1033553122 Python中赋值并不会拷贝对象,只是创建目标和对象的绑定关系. copy.copy(x) 返回x的浅拷贝 ...

  9. Shell脚本中执行sql语句操作mysql的5种方法【转】

    对于自动化运维,诸如备份恢复之类的,DBA经常需要将SQL语句封装到shell脚本.本文描述了在Linux环境下mysql数据库中,shell脚本下调用sql语句的几种方法,供大家参考.对于脚本输出的 ...

随机推荐

  1. 安装MySQL在最后的start service停住了解决方法

    今天自己安装mysql在start service卡住了,原来是以前安装过,但是没有删干净.通过下面的方法解决了,特分享下 由于我的MySQL不知道什么原因突然打不开了并报了个10061的错,查了下原 ...

  2. inheritance,菱形继承, 虚继承,virtual

    //菱形继承   |||||||   虚继承 #include <iostream> using namespace std; class R {     int r; public:   ...

  3. Hibernate学习笔记--------2.一多|多多的CRUD

    一.一多关系 例如用户(Tb_User)和订单(Tb_Order)之间,一个用户对应了多个订单,多个订单对应一个用户. 除了基本的配置外,需要在用户类(单方)中添加订单的集合同样需要get/set方法 ...

  4. Redis主备自动切换

    Sentinel(哨兵)是用于监控redis集群中Master状态的工具. 一.Sentinel作用  1.Master状态检测   2.如果Master异常,则会进行Master-Slave切换,将 ...

  5. 3.x vector的用法

    #include<vector> //struct struct GOLD_STRUCT {     Sprite  * goldspSprite;     int goldValue; ...

  6. oracle交集,并集,差集

    引自:http://www.2cto.com/database/201308/238777.html [sql] create table test1 ( name ), NN ) ); insert ...

  7. ASP.Net Core 运行在Linux(Ubuntu)

    这段时间一直在研究asp.net core部署到linux,今天终于成功了,这里分享一下我的部署过程. Linux Disibutaion:Ubuntu 14.04 Web Server:nginx. ...

  8. Android studio 删除Module、project

    很简单: 1 选中项目右键:Open Module Setting 2 选择要删除的项目,点击“-”即可

  9. Jquery on 事件

    $(document).on("click", 'a.AAA', function(){ var flag=$(this).attr('flag'); alert(flag); } ...

  10. ###《Effective STL》--Chapter1

    点击查看Evernote原文. #@author: gr #@date: 2014-09-12 #@email: forgerui@gmail.com Chapter1 容器 Topic 4: 调用e ...