NSExtension

而在引入扩展之后,其他app可以与扩展进行数据交换。基于安全和性能的考虑,每一个扩展运行在一个单独的进程中,它拥有自己的bundle, bundle后缀名是.appex。扩展bundle必须包含在一个普通应用的bundle的内部。


图片2.png

iOS 8系统有6个支持扩展的系统区域,分别是Today、Share、Action、Photo Editing、Storage Provider、Custom keyboard。支持扩展的系统区域也被称为扩展点。

Today Widget

对于赛事比分,股票、天气、快递这类需要实时获取的信息,可以在通知中心的Today视图中创建一个Today扩展实现。Today扩展又称为Widget。


图片3.png

Share

在iOS 8之前,用户只有Facebook,Twitter等有限的几个分享选项可以选择。如果希望将内容分享到Pinterest,开发者则需要一些额外的努力。在iOS 8中,开发者可以创建自定义的分享选项。


图片4.png

Action

action在所有支持的扩展点中扩展性最强的一个。它可以实现转换另一个app上下文中的内容。苹果在WWDC大会上演示了一个Bing翻译动作扩展,它可以将在Safari中选中的文本翻译成不同的语言。

图片6.png

Photo Editing

在iOS 8之前,如果你想为你的照片添加一个特殊的滤镜,你需要进入第三方app中,这个过程是相当繁琐的。在iOS 8中,你可以直接在Photos中使用第三方app,如Instagram,VSCO cam、Aviary提供的Photo Editing扩展完成对图片的编辑,而无需离开当前的app。


图片7.png

Storage Provider

Storage Provider让跨多个文件存储服务之间的管理变得更简单。类似Dropbox、Google Drive等存储提供商通过在iOS 8中提供一个Storage Provider扩展,app直接可以使用这些扩展检索和存储文件而不再需要创建不必要的拷贝。

图片8.png

Custom Keyboard

苹果公司在2007年率先推出了触摸屏键盘,但一直没多大改进。在这一方面,Android则将键盘权限开放给了第三方开发者,所以出现了许多像Swype,SwiftKey等优秀的键盘输入法。在iOS 8中,苹果终于将键盘权限开发给了第三方开发者,自定义键盘输入法可以让用户在整个系统范围内使用。

二、创建扩展与发布扩展

在创建扩展之前,你需要创建一个用来包含扩展的常规app项目。该包含扩展的app被称为containing app。在创建好containing app之后,选择File->New->Target菜单,从弹出的对话框中选择一个适当的扩展目标模版。每一个扩展目标模版都包含了与扩展点相关的文件和设置。一个containing app可以包含多个不同类型的扩展。

图片9.png

每一个扩展目标模版包含一个头文件和实现文件,一个info.plist文件,以及一个storyboard文件。info.plist文件包含了对扩展的配置信息,其中最重要的键是NSExtension。下面列出了一个NSExtension可能包含的常用键值对。

<key>NSExtension</key> 
 
 <dict> 
 
    <key>NSExtensionAttributes</key> 
 
    <dict> 
 
            <key>NSExtensionActivationRule</key> <!--1--> 
 
            <dict> 
 
                <key>NSExtensionActivationSupportsImageWithMaxCount</key> 
 
                <integer>10</integer> 
 
                <key>NSExtensionActivationSupportsMovieWithMaxCount</key> 
 
                <integer>1</integer> 
 
                 </dict> 
 
            <key>NSExtensionJavaScriptPreprocessingFile</key> <!--2--> 
 
            <string>MyJavaScriptFile</string> 
 
           <key>NSExtensionPointVersion</key> 
 
           <string>1.0</string> 
 
       </dict> 
 
     <key>NSExtensionMainStoryboard</key>  <!--3--> 
 
     <string>MainInterface</string> 
 
       <key>NSExtensionPointIdentifier</key>  <!--4--> 
 
       <string>com.apple.ui-services</string> 
 
     <key>NSExtensionPrincipalClass</key>  <!--5--> 
 
       <string>ActionViewController</string> 
 
</dict>
  1. NSExtensionActivationRule定义了当前的扩展支持的数据类型及数据项个数,例如当前的设置只支持图片格式和视频格式的数据,并且最多不超过10张图片和1个视频。

  2. NSExtensionJavaScriptPreprocessingFile用于配置与脚本交互的JS脚本文件的名字。

  3. NSExtensionMainStoryboard配置扩展的Storyboard文件名。

  4. NSExtensionPointIdentifier用于表示扩展点,每一个扩展点拥有一个唯一的名字。

  5. NSExtensionPrincipalClass配置当扩展启动时,扩展点首先要实例化的类

为了将扩展提交苹果商店,你需要提交你的containg app。并且需要注意,除了扩展必须包含功能以外,同时containg app还需要提供一些功能,而针对OS X平台的扩展则无此限制。当用户安装了你的containg app,containg app中包含的扩展也会一同被安装。

三、理解扩展如何运作

在安装扩展之后,扩展并不会自动运行,用户必须执行特定的操作来启用扩展。如果是Today扩展,用户可以在通知中心的Today视图中编辑启用扩展。如果是自定义键盘扩展,用户需要在系统设置的通用选项下的键盘选项中启用自定义键盘扩展。而如果是Share扩展,用户只需点击系统提供的分享按钮


图片10.png

,即可在分享列表中找到分享扩展。

一个扩展并不是一个app,它的生命周期和运行环境不同于普通app。在生命周期方面,扩展的生命周期从用户在另一个app中选择了扩展开始,一直到扩展完成了用户的请求生命周期结束。在运行环境方面,扩展的限制要比普通app更严格,扩展的可用内存上限以及可用的API都比普通app要少。严格限制扩展的内存是因为在同一时间可能会有多个扩展同时运行,如Widget扩展。如果API声明包含NS_EXTENSION_UNAVAILABLE宏,则此API在扩展中将不可用,常见的API如:

+ (UIApplication *)sharedApplication NS_EXTENSION_UNAVAILABLE_IOS("Use view controller based solutions where appropriate instead."); 

调用扩展的应用称为host app,对于Widget扩展,host app就是Today。host app会在扩展的有效生命周期内定义一个扩展上下文。通过扩展上下文,host app可以和扩展互传数据。注意,扩展只和host app直接通信,扩展与containg app以及containing app与host app之间不存在通信关系,如果扩展需要打开containg app,则通过自定义URL scheme方式实现,而不是直接向containg app发送消息。三者的关系见下图:

图片11.png

扩展是一个单独的个体。扩展拥有独立的target,独立的bundle文件,独立的运行进程,独立的地址空间。这意味着即使你的containing app不在运行,系统也可以启动扩展。或者你的containing app处于挂起状态,同样不会影响扩展的运行。所以系统可以单独对扩展执行优化。扩展与containg app的关系:

图片12.png

四、设计扩展过程中常见的几个问题

1. containg app与扩展如何通过扩展上下文互传数据

在iOS 8中,UIViewController新增了一个扩展上下文属性extensionContext。来处理containing app与扩展之间的通信,上下文的类型是NSExtensionContext。假设你现在需要在host app中将一张图片传递给扩展做滤镜处理,host app中的代码如下:

UIActivityViewController *activityViewController = [[UIActivityViewController alloc] initWithActivityItems:@[[self.imageView image]] applicationActivities:nil]; 
 
[self presentViewController:activityViewController animated:YES completion:nil]; 

当用户在弹出的Action列表中选择了扩展,扩展将被启动,然后在扩展的viewDidLoad方法中,通过extensionContext检索host app传回的数据项。扩展中的代码如下:

- (void)viewDidLoad { 
 
    [super viewDidLoad]; 
 
    NSExtensionItem *imageItem = [self.extensionContext.inputItems firstObject]; //extensionContext表示一个扩展到host app的连接,通过extionContext,你可以访问一个NSExtensionItem的数组,每一个NSExtensionItem项表示从host app传回的一个逻辑单元。
 
    if(!imageItem){ 
 
        return; 
 
    } 
 
    NSItemProvider *imageItemProvider = [[imageItem attachments] firstObject]; //可以从NSExtensionItem项中的attachments属性中获得附件数据,如音频、视频、图片等,NSItemProvide就是实例的表示
 
    if(!imageItemProvider){ 
 
        return; 
 
    } 
 
   // 检查是否包含文本 
 
    if([imageItemProvider hasItemConformingToTypeIdentifier:(NSString *)kUTTypeImage]){ 
 
        [imageItemProvider loadItemForTypeIdentifier:(NSString *)kUTTypeImage options:nil completionHandler:^(UIImage *image, NSError *error) { //
 
            if(image){ 
 
                dispatch_async(dispatch_get_main_queue(), ^{ 
 
                    self.imageView.image = image; 
 
                }); 
 
            } 
 
        }]; 
 
        
 
    } 
 
} 

当扩展处理完host app传来的图片数据后,它需要将处理好的的数据在传给host app,在扩展中的代码如下:

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

推荐阅读更多精彩内容

  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,066评论 4 62
  • 1. Maven跑在JDK上,eclipse中配置JDK 代码如下: -Dmaven.multiModulePro...
    Mark_Van阅读 640评论 0 0
  • 重看我是歌手4月1日突围赛,彻底喜欢上赵传这样的大哥和信这样的音乐小顽童。 还记得第一次看赵传大哥出场的时候,给我...
    蟹萌萌阅读 389评论 0 0
  • 那天娃问了我一个问题:“妈妈,我们今天的作文题目是“快乐”,我想问问你,你觉得快乐是什么?"我竟然一时语塞,半...
    素笺_如昔阅读 438评论 0 0
  • 跟母亲通电话,母亲问我在这边的情况怎么样?冷不冷啊?笑笑闹不闹人呀?最后总要说一句,你俩别“磨牙”,磨牙在我们家乡...
    利萍阅读 212评论 0 0