华山论剑之浅谈iOS剪切黏贴(UIPasteboard和UIMenuController)

每个程序猿都是一名强大的战士,名曰CV战士.
菜单功能


序言


如上图,我们在应用程序中时常需要有黏贴复制的功能,像UIWebView,UITextField,UITextView这些类都是自带黏贴复制功能的,但是我们先让某个控件具有黏贴复制功能,我们该怎么办呢?那就需要用到两个类,一个是UIPasteboard,另外一个就是UIMenuController,一个是剪切板,另外一个是菜单弹窗.


从UIMenuController到菜单弹窗的完美变身


UIMenuController的用法和UIAlertView的使用方法是类似的,如下图,我在storyboard的ViewController的控制器中,添加两个个UILabel对象,然后在ViewController其中添加手势.

把UILabel拖成属性,然后代码如下


#import "ViewController.h"

@interface ViewController ()

@property (strong, nonatomic) IBOutlet UILabel *testlLabel;//测试Label


@property (strong, nonatomic) IBOutlet UILabel *pasteLabel;//黏贴Label


@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
 
    [self loadingTapGR];
}

//添加点击手势
-(void)loadingTapGR{

    UILongPressGestureRecognizer *longPress = [[UILongPressGestureRecognizer alloc]initWithTarget:self action:@selector(showMenuVC:)];
    

    longPress.minimumPressDuration = 1;
    
    self.testlLabel.userInteractionEnabled = YES;   
    
    [self.testlLabel addGestureRecognizer:longPress];

    UILongPressGestureRecognizer *longPress2 = [[UILongPressGestureRecognizer alloc]initWithTarget:self action:@selector(showMenuVC:)];
    
    longPress2.minimumPressDuration = 1;
    
    self.pasteLabel.userInteractionEnabled = YES;
    
    [self.pasteLabel addGestureRecognizer:longPress2];

}

//显示菜单
-(void)showMenuVC:(UILongPressGestureRecognizer *)longPress{

    UIMenuController *menuController = [UIMenuController sharedMenuController];
    
    UIMenuItem * menuItem = [[UIMenuItem alloc] initWithTitle:@"复制" action:@selector(copyTitle)];
    
    UIMenuItem * pasteItem = [[UIMenuItem alloc] initWithTitle:@"黏贴" action:@selector(paste:)];
    
    NSLog(@"%@",menuController.menuItems);
    
    [menuController setMenuItems:@[menuItem,pasteItem]];
    
    CGPoint location = [longPress locationInView:[longPress view]];
    CGRect menuLocation = CGRectMake(location.x, location.y, 0, 0);
    [menuController setTargetRect:menuLocation inView:[longPress view]];
    
    [menuController setMenuVisible:YES animated:YES];



}

//复制文本
-(void)copyTitle{

 
    
}
//黏贴
-(void)paste:(id)sender
{
  

}



@end

代码是如上实现了,但是我们点击Label仍是没有菜单出现,这是为什么呢?因为我们还重新定义下面的这个方法.

//允许成为第一响应者
-(BOOL)canBecomeFirstResponder{
    return YES;
}

看一下效果图,好了菜单完成了,我们接下来要做复制黏贴的功能了.

效果图片


从UIPasteboard到复制黏贴功能的终极进化


在使用UIPasteboard之前,我们需要先了解一下剪贴板UIPasteboard的一些基本的知识.

剪贴板类型:

  • 系统级别:使用UIPasteboardNameGeneral和UIPasteboardNameFind,系统级应用程序关闭,或者卸载的数据不会丢失。
  • 应用程序级:通过设置,可以让数据在应用程序关闭之后仍然保存在剪贴板中,但是应用程序卸载之后数据就会失去。我们可用通过pasteboardWithName:create:来创建。

剪贴板可直接存放的类型

  • 1、UIPasteboardTypeListString — 字符串数组 包含kUTTypeUTF8PlainText
  • 2、UIPasteboardTypeListURL — URL数组,包含kUTTypeURL
  • 3、UIPasteboardTypeListImage — 图形数组, 包含kUTTypePNG 和kUTTypeJPEG
  • 4、UIPasteboardTypeListColor — 颜色数组

下面我们对上面代码中的 -(void)copyTitle 方法进行进一步的完善.使其能有简单的复制功能.

//复制文本
-(void)copyTitle{

    NSLog(@"复制功能的实现~");
    
    UIPasteboard *pasteboard = [UIPasteboard generalPasteboard];//普通的黏贴板

    pasteboard.string = self.testlLabel.text;
}

复制功能实现之后,我们需要实现黏贴功能.


-(void)paste:(id)sender
{
  
    UIPasteboard *pboard = [UIPasteboard generalPasteboard];
    
    //判断是否数据
    if (nil != pboard.string) {

            self.pasteLabel.text = pboard.string;
    }
}



到此,复制黏贴的功能就实现了,当然了,这是对一些简单对象(比如:string,URL,image,color)进行的复制黏贴,如果是复杂对象,改如何进行黏贴复制呢?这时候就需要使用到归档和反归档把数据变成NSData类型的对象,然后使用 setData:forPasteboardType:dataForPasteboardType: 两个方法进行数据的存储和获取,代码如下

//存储数据 
UIPasteboard *pb = [UIPasteboard pasteboardWithName:@"testBoard" create:YES];  
NSDictionary *dic = [NSDictionary dictionaryWithObject:self.testlLabel.text forKey:@"saoDong"];    
NSData *dictData = [NSKeyedArchiver archivedDataWithRootObject:dic];    
[pb setData:dictData forPasteboardType:@"myType"];    
  
//获取就类似于这样:   
UIPasteboard *pb = [UIPasteboard pasteboardWithName:@"testBoard" create:YES];    
NSDictionary *dic = [NSKeyedUnarchiver unarchiveObjectWithData:[pb dataForPasteboardType:@"myType"]];    
self.pasteLabel.text = [dict objectForKey:@"saoDong"];    


总结: UIPasteboard和UIMenuController两个类整体使用起来比较简单,难度一般,如果有什么疑问,请在下面评论,我会及时回复.谢谢您的查看,最后附上测试的Demo.

-------> UIPasteboard和UIMenuController使用Demo传送门

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,490评论 18 139
  • 用户可以在一个app中复制文本、图片、或者其他数据,并粘贴该数据到该app的其他位置,或不同的app中。例如,你可...
    raingu24阅读 4,102评论 0 1
  • 内容来自于 iOS文档中 About Text Handling in iOS 部分 ios平台提供了显示及编辑文...
    纵横而乐阅读 6,700评论 2 21
  • 留意到如何处理价值观不一致问题。首先尊重肯定对方的敢于表达自己的想法,这是我们逸才坦诚精神的体现。其次,表达对背后...
    sageness阅读 223评论 0 1
  • 我们无法预知未来的路, 只得集中精力走好脚下的每一步, 该发生的逃避不了, 生活没那么简单,你可以忧心,可以忧虑,...
    linguanjie关节阅读 503评论 0 4