iOS 為UILabel和UIImageView添加長按復(fù)制功能

在iOS中下面三個(gè)控件,自身就有復(fù)制-粘貼的功能:

1、UITextView

2、UITextField

3、UIWebView

UIKit framework提供了幾個(gè)類和協(xié)議方便我們在自己的應(yīng)用程序中實(shí)現(xiàn)剪貼板的功能。

1、UIPasteboard:我們可以向其中寫入數(shù)據(jù),也可以讀取數(shù)據(jù)

2、UIMenuController:顯示一個(gè)快捷菜單,用來展示復(fù)制、剪貼、粘貼等選擇的項(xiàng)。

3、UIResponder中的 canPerformAction:withSender:用于控制哪些命令顯示在快捷菜單中。

4、當(dāng)快捷菜單上的命令點(diǎn)擊的時(shí)候,UIResponderStandardEditActions將會被調(diào)用。

下面這些項(xiàng)能被放置到剪貼板中

1、UIPasteboardTypeListString —? 字符串?dāng)?shù)組, 包含kUTTypeUTF8PlainText

2、UIPasteboardTypeListURL —? URL數(shù)組,包含kUTTypeURL

3、UIPasteboardTypeListImage —? 圖形數(shù)組, 包含kUTTypePNG 和kUTTypeJPEG

4、UIPasteboardTypeListColor —? 顏色數(shù)組

剪貼板的類型分為兩種:

系統(tǒng)級:使用UIPasteboardNameGeneral和UIPasteboardNameFind,系統(tǒng)級應(yīng)用程序關(guān)閉,或者卸載的數(shù)據(jù)不會丟失。

應(yīng)用程序級:通過設(shè)置,可以讓數(shù)據(jù)在應(yīng)用程序關(guān)閉之后仍然保存在剪貼板中,但是應(yīng)用程序卸載之后數(shù)據(jù)就會失去。我們可用通過pasteboardWithName:create:來創(chuàng)建。

UILabel

例子如下:

有時(shí)候我們可能需要復(fù)制UILabel上的文本,或者UIImageView的圖片,而UILabel和UIImageView默認(rèn)是不響應(yīng)Touch事件的,也無法復(fù)制,那么我們就需要自己實(shí)現(xiàn)一個(gè)可復(fù)制的UILabel。

新添加一個(gè)類繼承自UILabel:

@interface UICopyLabel : UILabel

@end

#import "UICopyLabel.h"

@implementation UICopyLabel

@end

為了能接收到事件(能成為第一響應(yīng)者),我們需要覆蓋一個(gè)方法:

-(BOOL)canBecomeFirstResponder

{

return YES;

}

還需要針對復(fù)制的操作覆蓋兩個(gè)方法:

// 可以響應(yīng)的方法

-(BOOL)canPerformAction:(SEL)action withSender:(id)sender

{

return (action == @selector(copy:));

}

//針對于響應(yīng)方法的實(shí)現(xiàn)

-(void)copy:(id)sender

{

UIPasteboard *pboard = [UIPasteboard generalPasteboard];

pboard.string = self.text;

}

有了以上三個(gè)方法,我們就能處理copy了,當(dāng)然,在能接收到事件的情況下:

//UILabel默認(rèn)是不接收事件的,我們需要自己添加touch事件

-(void)attachTapHandler

{

self.userInteractionEnabled = YES;? //用戶交互的總開關(guān)

UITapGestureRecognizer *touch = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(handleTap:)];

touch.numberOfTapsRequired = 2;

[self addGestureRecognizer:touch];

[touch release];

}

//綁定事件

- (id)initWithFrame:(CGRect)frame

{

self = [super initWithFrame:frame];

if (self)

{

[self attachTapHandler];

}

return self;

}

//同上

-(void)awakeFromNib

{

[super awakeFromNib];

[self attachTapHandler];

}

我們已經(jīng)可以接收到事件了!由于我在上方將tap數(shù)設(shè)為2,所以需要雙擊才能捕獲,

接下來,我們需要處理這個(gè)tap,以便讓菜單欄彈出來:

-(void)handleTap:(UIGestureRecognizer*) recognizer

{

[self becomeFirstResponder];

UIMenuItem *copyLink = [[[UIMenuItemalloc] initWithTitle:@"復(fù)制"

action:@selector(copy:)]autorelease];

[[UIMenuControllersharedMenuController] setMenuItems:[NSArrayarrayWithObjects:copyLink, nil]];

[[UIMenuControllersharedMenuController] setTargetRect:self.frameinView:self.superview];

[[UIMenuControllersharedMenuController] setMenuVisible:YESanimated: YES];

}

這樣一來,一個(gè)可復(fù)制的UILabel就誕生了!它能處理接收點(diǎn)擊、彈出菜單欄、處理copy,這是一個(gè)很普通的可復(fù)制控件。

UIImageView

接下來我們做一個(gè)可復(fù)制的UIImageView,創(chuàng)建一個(gè)新的viewController,放兩個(gè)imageView,默認(rèn)顯示不同的圖:

然后把上面的代碼直接拷過來,改三個(gè)地方:

-(BOOL)canPerformAction:(SEL)action withSender:(id)sender

{

return (action == @selector(copy:) || action == @selector(paste:));

}

-(void)copy:(id)sender

{

UIPasteboard *pboard = [UIPasteboard generalPasteboard];

pboard.image = self.image;

}

-(void)paste:(id)sender

{

UIPasteboard *pboard = [UIPasteboard generalPasteboard];

self.image = pboard.image;

}

UIPasteboard有系統(tǒng)級別和應(yīng)用級別兩種類型,所以不僅可以在應(yīng)用程序內(nèi)通信,還能在應(yīng)用程序間通信,比如我復(fù)制一個(gè)url,然后打開safari,粘貼到地址欄去,而我們可以在應(yīng)用程序間通信、共享數(shù)據(jù)。

在PasteBoardWrite里面點(diǎn)“寫入”后把textField中的文本寫入粘貼板,然后切換到PasteBoardRead的時(shí)候顯示出來。如果我們的粘貼板只想給“自己人”用的話,就不能用系統(tǒng)的通用粘貼板,需要我們自己創(chuàng)建一個(gè):

//需要提供一個(gè)唯一的名字,一般使用倒寫的域名:com.mycompany.myapp.pboard

//后面的參數(shù)表示,如果不存在,是否創(chuàng)建一個(gè)

UIPasteboard *pb = [UIPasteboard pasteboardWithName:@"testBoard" create:YES];

使用這個(gè)粘貼板,我們可以把文本存進(jìn)去,然后在另一個(gè)app里面讀出來,一些常用的類型已經(jīng)被設(shè)置為屬性了:

除此之外,如果是能夠轉(zhuǎn)換成plist的數(shù)據(jù)類型(NSString, NSArray, NSDictionary, NSDate, NSNumber 和 NSURL),我們可以調(diào)用setValue:forPasteboardType:方法去存儲數(shù)據(jù),其他類型只能調(diào)用setData:forPasteboardType:方法(plist數(shù)據(jù)類型也可使用),類似于這樣:

//存儲數(shù)據(jù)

NSDictionary *dict = [NSDictionary dictionaryWithObject:textField.text forKey:@"content"];

NSData *dictData = [NSKeyedArchiver archivedDataWithRootObject:dict];

[pb setData:dictData forPasteboardType:@"myType"];

//獲取就類似于這樣:

UIPasteboard *pb = [UIPasteboard pasteboardWithName:@"testBoard" create:YES];

NSDictionary *dict = [NSKeyedUnarchiver unarchiveObjectWithData:[pb dataForPasteboardType:@"myType"]];

caption.text = [dict objectForKey:@"content"];

上面提到了一個(gè)PasteboardType, 這是一個(gè)統(tǒng)一類型標(biāo)識符(Uniform Type Identifier? UTI),能幫助app獲取自己能處理的數(shù)據(jù)。比如你只能處理文本的粘貼,那給你一個(gè)UIImage顯然是無用的。你可以使用公用的UTI,也可以使用 任意字符,蘋果建議使用倒寫的域名加上類型名:com.myCompany.myApp.myType。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,117評論 6 537
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,860評論 3 423
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,128評論 0 381
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,291評論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,025評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,421評論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,477評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,642評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,177評論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,970評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,157評論 1 371
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,717評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,410評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,821評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,053評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,896評論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,157評論 2 375

推薦閱讀更多精彩內(nèi)容