Application Extension(二):Today Extension

简介

在通知中心的Today的视图中显示的 extension 叫做 widget ,widget 可以方便用户快速的得到想要的信息,不用再通过复杂的步骤打开app才能找到自己想要的东西,只要下拉就可方便的看到,iOS设备中即使在锁屏的状态下也能查看。例如,用户下拉滑到Today查看现在的股票信息、天气、日程等等,如今越来越多的用户会频繁的通过Today来快速的得到信息。

Today Widget

在创建widget的时候应该注意:
1、确保每次查看时都要更新,显示最新的内容
2、交互流程合理
3、注意性能问题,确保使用流畅

由于用户在使用Today widget是快速短暂的,所以在设计widget的时候应注意UI简洁,限制显示的内容,只把用户最感兴趣的内容显示出来。

在不同的平台上的widget有所不同:
iOS:widget不支持键盘输入,用户需要在 containing app 中去设置对应的 widget 要显示的内容。例如,天气的 widget 要是用户想添加要显示的城市天气,那就需要打开天气应用,在城市列表中添加要显示的城市。

iOS天气widget

macOS:当 Today widget 运行时,用户可以去修改配置改变要显示的内容。例如天气widget可以添加想要显示的城市天气。


macOS天气widget
macOS天气widget编辑

当用户安装了带有Today widget 的应用,可以在通知中心Today的视图内添加该应用的widget,可以在编辑widget的页面添加、删除、修改widget。

创建Today Extension

这里在iOS上创建个Today Extension的小例子用来显示主应用里面的内容。
注意本例子是在 iOS10以上的系统上创建,之前的系统要另行适配。
新建一个项目,在项目里添加target,选择Today Extension:

  • Xcode的Today模板会生成默认的实现文件,在Today的Info.plist文件中默认的NSExtension值如下:
<key>NSExtension</key>
    <dict>
        <key>NSExtensionMainStoryboard</key>
        <string>MainInterface</string>
        <key>NSExtensionPointIdentifier</key>
        <string>com.apple.widget-extension</string>
    </dict>

如不想使用系统默认的 storyboard 可以删除NSExtensionMainStoryboard 键值对,添加自己的实现文件 ,NSExtensionPrincipalClass 对应的是自定义的view controller名。
NSExtensionPointIdentifier对应的值是Extension的反向DNS名称。

创建好之后运行项目,在通知中心中编辑widget添加到刚创建的widget就可看到,如下图:


图标就是你的应用图标,标题默认的创建的Today名称,可以在Info.plist中修改CFBundleDisplayName显示的标题的内容。
然后就可以在TodayViewController中添加想要显示的内容了。常见的widget,在右上角带有展开/折叠按钮,这个在TodayViewController中设置好属性就可以实现

//支持展开、折叠
self.extensionContext.widgetLargestAvailableDisplayMode = NCWidgetDisplayModeExpanded;

Apple 为扩展提供了一个 NSExtensionContext 类来与host app应用进行交互。用户在host app中启动扩展后,host app提供一个上下文给扩展,里面最主要的是包含了 inputItems 这样的待处理的数据。

当点击展开或者折叠时会调用下面的方法,从而去改变widget的高度,默认的为展开的widget的高度是固定的110。
Xcode的TodayExtension的模板中生成的TodayViewController中默认的遵循NCWidgetProviding协议,协议里有下面几个方法:

  • widgetPerformUpdateWithCompletionHandler 在TodayViewController中实现这个方法后,当用户操作显示通知中心Today视图,就会调用这个方法只要以显示就会调用,从而可以及时的去更新状态。即使你的扩展现在不可见 (也就是用户没有拉开通知中心),系统也会时不时地调用实现了的这个方法,来要求扩展刷新界面。在这个方法中我们一般可以做一些像 API 请求之类的事情,在获取到了数据并更新了界面,或者是失败后都使用提供的 completionHandler 来向系统进行报告。
  • widgetActiveDisplayModeDidChange 当widget的展开或折叠的状态改变时就会调用该方法,在这个方法里可以通过改变self.preferredContentSize的值改变widget的高度,从而适应不同的状态。展开折叠是在iOS10以后才加入的功能,所以这个方法只有在iOS10之后才有。

Extension和Containing App间共享代码

在App里创建的某个类,要想在Extension中也能使用,在Target Membership中勾选就可以了。同样的要想在Extension中使用App Assets.xcassets里面的图片,勾选就可以了。


Extension和Containing App间共享数据

沙盒限制了我们在设备上随意读取和写入。但是从iOS8开始有了新的功能,App Groups 为同一个 vender 的应用或者扩展定义了一组域,在这个域中同一个 group 可以共享一些资源。对于我们的例子来说,我们只需要使用同一个 group 下的 NSUserDefaults 就能在主体应用不活跃时向其中存储数据,然后在扩展初始化时从同一处进行读取就行了。

对于应用和其对应的扩展来说,可以使用 App Groups 来进行数据共享。

首先我们需要开启 App Groups。选择containing app 的 target,打开它的 Capabilities ,找到 App Groups 并打开开关,然后添加一个 group 名字。接下来你还需要为 TodayExtension 这个 target 进行同样的配置,只不过不再需要新建 group,而是勾选刚才创建的 group 就行。

然后在App中保存要在widget中显示的数据:

NSUserDefaults * groupDefault = [[NSUserDefaults alloc] initWithSuiteName:@"group.myzTodayGroup"];
[groupDefault setObject:dataArray forKey:@"myzTodayDataArray"];

在TodayExtension中就可以获取到数据,从而显示在widget中:

NSUserDefaults * groupDefault = [[NSUserDefaults alloc] initWithSuiteName:@"group.myzTodayGroup"];
NSArray * dataArray = [groupDefault objectForKey:@"myzTodayDataArray"];

从TodayExtension启动应用

接着来处理在widget点击内容后,然后跳转到 containing app 查看详细信息。也就是从 widget 启动 containing app ,还要向 app 传递数据。可以使用系统提供的 NSExtensionContext 类,通过NSExtensionContext 来调用 openURL(URL:completionHandler:)启动 containing app :

[self.extensionContext openURL:[NSURL URLWithString:[NSString stringWithFormat:@"myzwidget://open]] completionHandler:nil];

然后选择containing app 的 target,设置对应的 URL Scheme:


然后在 AppDelegate 中获取打开的事件,做出相应的反应:

- (BOOL)application:(UIApplication *)application openURL:(nonnull NSURL *)url options:(nonnull NSDictionary<UIApplicationOpenURLOptionsKey,id> *)options {
    
    if ([[url scheme] isEqualToString:@"myzwidget"]) {
        //...
        return YES;
    }
    return NO;
}

小例子实现了在widget中显示应用中前三行的内容,效果如下:

Widget.gif


Demo地址:https://github.com/MA806P/MYZAppExtension


Reference

App Extension Programming Guide - Today
https://onevcat.com/2014/08/notification-today-widget/

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

推荐阅读更多精彩内容