華山論劍之淺談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閱讀 229,001評論 6 537
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,786評論 3 423
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,986評論 0 381
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,204評論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,964評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,354評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,410評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,554評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,106評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,918評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,093評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,648評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,342評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,755評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,009評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,839評論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,107評論 2 375

推薦閱讀更多精彩內容

  • Spring Cloud為開發人員提供了快速構建分布式系統中一些常見模式的工具(例如配置管理,服務發現,斷路器,智...
    卡卡羅2017閱讀 134,803評論 18 139
  • 用戶可以在一個app中復制文本、圖片、或者其他數據,并粘貼該數據到該app的其他位置,或不同的app中。例如,你可...
    raingu24閱讀 4,194評論 0 1
  • 內容來自于 iOS文檔中 About Text Handling in iOS 部分 ios平臺提供了顯示及編輯文...
    縱橫而樂閱讀 6,819評論 2 21
  • 留意到如何處理價值觀不一致問題。首先尊重肯定對方的敢于表達自己的想法,這是我們逸才坦誠精神的體現。其次,表達對背后...
    sageness閱讀 237評論 0 1
  • 我們無法預知未來的路, 只得集中精力走好腳下的每一步, 該發生的逃避不了, 生活沒那么簡單,你可以憂心,可以憂慮,...
    linguanjie關節閱讀 507評論 0 4