Extension Point
系統中支持extension的區域,extension的類別也是據此區分的,iOS上共有Today、Share、Action、Photo Editing、Storage Provider、Custom keyboard幾種,其中Today中的extension又被稱為widget。每種extension point的使用方式和適合干的活都不一樣,因此不存在通用的extension。
App Extension
即為本文所說的extension。extension并不是一個獨立的app,它有一個包含在app bundle中的獨立bundle,extension的bundle后綴名是.appex。其生命周期也和普通app不同。extension不能單獨存在,必須有一個包含它的containing app。另外,extension需要用戶手動激活,不同的extension激活方式也不同,比如:比如Today中的widget需要在Today中激活和關閉;Custom keyboard需要在設置中進行相關設置;Photo Editing需要在使用照片時在照片管理器中激活或關閉;Storage Provider可以在選擇文件時出現;Share和Action可以在任何應用里被激活,但前提是開發者需要設置Activation Rules,以確定extension需要在合適出現。
Containing App
盡管蘋果開放了extension,但是在iOS中extension并不能單獨存在,要想提交到AppStore,必須將extension包含在一個app中提交,并且app的實現部分不能為空,這個包含extension的app就叫containing app。extension會隨著containing app的安裝而安裝,同時隨著containing app的卸載而卸載。
Host App
能夠調起extension的app被稱為host app,比如widget的host app就是Today。
1 extension和host app之間可以通過extensionContext屬性直接通信,該屬性是新增加的UIViewController類別。
實際上extension和host app之間是通過IPC(interprocess communication)實現的,只是蘋果把調用接口高度抽象了,我們并不需要關注那么底層的東西。
2 containing app和host app
他們之間沒有任何直接關系,也從來不需要通信。
3 extension和containing app
這二者之間的關系最復雜,糾糾纏纏扯不清關系。
首先,盡管extension的bundle是放在containing app的bundle中,但是他們是兩個完全獨立的進程,之間不能直接通信。不過extension可以通過openURL的方式啟動containing app(當然也能啟動其它app),不過必須通過extensionContext借助host app來實現。
可以共享Shared resources
extension和containing app可以共同讀寫一個被稱為Shared resources的存儲區域,這是通過App Groups實現。
使用說明:
1 Extension?info.plist中字段NSExtension
1)???NSExtensionActivationRule定義了當前的擴展支持的數據類型及數據項個數,例如當前的設置只支持圖片格式和視頻格式的數據,并且最多不超過10張圖片和1個視頻。
2)???NSExtensionJavaScriptPreprocessingFile用于配置與腳本交互的JS腳本文件的名字。
3)???NSExtensionMainStoryboard配置擴展的Storyboard文件名。
4)???NSExtensionPointIdentifier用于表示擴展點,每一個擴展點擁有一個唯一的名字。
5)???NSExtensionPrincipalClass配置當擴展啟動時,擴展點首先要實例化的類
如果不使用自動生成的storyBoard,純代碼實現的話就在plist文件刪除第二個字段,添加第五個值為你的VC名字。
2 Extension VC的屬性NSExtensionContext
1.通過URL Schemes打開contain app
- (void)openURL:(NSURL*)URL completionHandler:(void(^)(BOOLsuccess))completionHandler;
2.對于其他的耗時任務似乎并沒有更好的方式,由于返回host app后其生命周期會馬上結束,因此,對于其它類型的耗時任務,只能在任務完成后,再調用
- (void)completeRequestReturningItems:(NSArray?*)items completionHandler:(void(^)(BOOL?expired))completionHandler;
3 與host app的協議NCWidgetProviding更新視圖
- (void)widgetPerformUpdateWithCompletionHandler:(void(^)(NCUpdateResultresult))completionHandler;
每次顯示這個頁面都會調用這個函數去更新界面
- (UIEdgeInsets)widgetMarginInsetsForProposedMarginInsets:(UIEdgeInsets)defaultMarginInsets;
布局的邊界
4 NCWidgetController控制顯示與隱藏
[[NCWidgetController widgetController] setHasContent:YESforWidgetWithBundleIdentifier:@"com.qunar.iphoneclientdev.cyjToday"];
5 Host App在調用Extension的時候會首先
- (void)beginRequestWithExtensionContext:(NSExtensionContext?*)context
如果實現了的話,會先執行里面的函數再viewDidLoad
Demo