IOS Today Extension 入门指南

简述:

Provide a quick update or enable a brief task in the Today view of Notification Center
即Widget窗口组件,类似于快捷入口or快捷窗口,展示在通知栏这

概览:

1.创建TodayExtension
2.运行
3.生命周期
4.界面绘制
5.跳转回主体App
6.数据共享
7.其他功能


一.创建TodayExtension

1.Flie -> New -> Target -> ios -> Application Extension -> Today Extension


二.运行TodayExtension

更改Target至TodayExtension
运行

运行后在手机通知栏查看如下:


111.gif


三.生命周期

5D96CD85-7199-4EAC-8C89-D94D688C6BE5.png

上图为通用的extension生命周期,today extension稍有区别
1.用户选择appExtension(用户切换至通知栏界面,并滑动显示出我们的extension界面时)
2.系统启动App Extension
3.App Extension 代码运行
4.运行完之后系统kill掉App Extension(离开了我们的extension界面一段时间后)


extension代码中的回调顺序

viewDidLoad (界面加载)
widgetActiveDisplayModeDidChange (控件显示模式发生改变代理,即紧凑和扩展)
widgetPerformUpdateWithCompletionHandler (官方建议在这个回调中进行数据获取)


四.界面绘制

widget有2种模式,compact(紧凑)及expended(扩展)
expended模式只有在IOS10+才有
选择expended模式时,系统会增加一个"展开"按钮,供用户自由切换compact和expended模式
compact mode:所有机型高度为110
expended mode:根据机型不同高度不同(也可以自由设置高度,小于最大值,大于110)
以expended模式为例:直接使用系统给的MainInterface.storyboard 进行绘制

1.stroyboard搭界面
前110height为compact时显示内容(绿色)
后面部分为展开时会显示的内容(蓝色)

直接更改MainInterface.storyboard的默认view.height至300(自定义expended的高度)并填充控件

E45131B4-5911-468B-93DF-6C0FA341DE28.png


2.代码实现

viewDidLoad中设置widget的displayMode为expended(largest)
widgetActiveDisplayModeDidChange中设置不同模式时的preferredContentSize
【widget的最终大小是由prefereedContentSize来决定,因此必须手动更改】
widgetPerformUpdateWithCompletionHandler中更新数据

- (void)viewDidLoad {
    [super viewDidLoad];
    //设置widget可扩展
    self.extensionContext.widgetLargestAvailableDisplayMode = YES;
}
//紧凑,扩展的显示模式发生改变时回调
-(void)widgetActiveDisplayModeDidChange:(NCWidgetDisplayMode)activeDisplayMode withMaximumSize:(CGSize)maxSize{
    NSLog(@"widgetActiveDisplayModeDidChange:%d,height = %.2f, width = %.2f",(int)activeDisplayMode,maxSize.height,maxSize.width);

    if (activeDisplayMode == NCWidgetDisplayModeCompact) {
        //紧凑时更改size至最大值
        self.preferredContentSize = CGSizeMake(maxSize.width, maxSize.height);
    }else{
        //扩展时,更改size.height至300
        self.preferredContentSize = CGSizeMake(maxSize.width, 300);
    }
}
//数据获取
//官方建议你通过实现它的回调来获取数据
- (void)widgetPerformUpdateWithCompletionHandler:(void (^)(NCUpdateResult))completionHandler {
      //官方建议在此获取数据
      completionHandler(NCUpdateResultNewData);
}


五.跳转回主体App(container)

点击widget的某个按钮,快捷进入到自己app的某个页面

1.主体app设置URL Schemes

A5A316F9-8332-44F0-9D57-C76E28BB9A31.png


2.extension的info.plist里设置白名单

    <key>CFBundleURLSchemes</key>
    <array>
        <string>AMNewInIOS8</string>
    </array>


3.触发跳转 @“Schemes://xxxxxx

- (IBAction)clickButton:(id)sender {
    NSURL *url = [NSURL URLWithString:@"AMNewInIOS8://"];
    [self.extensionContext openURL:url completionHandler:^(BOOL success) {
        NSLog(@"open status: %d",success);
    }];
}



4.主体App响应跳转至某个界面(注意ios9+及以前的区别)

//ios9实现此代理
-(BOOL)application:(UIApplication *)app openURL:(NSURL *)url options:(NSDictionary<UIApplicationOpenURLOptionsKey,id> *)options{
    UINavigationController* navi =  (UINavigationController*)self.window.rootViewController;
    UIStoryboard * sb = [UIStoryboard storyboardWithName:@"Main" bundle:nil];
    UIViewController * todayVC =[sb instantiateViewControllerWithIdentifier:@"NFTodayOpenURLVC"];
    [navi pushViewController:todayVC animated:YES];
    return YES;
}
//ios9以前实现此代理
- (BOOL)application:(UIApplication *)application handleOpenURL:(NSURL *)url{
    UINavigationController* navi =  (UINavigationController*)self.window.rootViewController;
    UIStoryboard * sb = [UIStoryboard storyboardWithName:@"Main" bundle:nil];
    UIViewController * todayVC =[sb instantiateViewControllerWithIdentifier:@"NFTodayOpenURLVC"];
    [navi pushViewController:todayVC animated:YES];
    return YES;
}


六.数据共享

Extension与主体app处于不同的沙盒中,数据共享及交互需要使用group来进行共享
同一账号下的所有app及extension可以通过group来进行数据共享

1.证书建立

创建App Group (ID必须以group.com开头)
创建主体App以及Extension的AppId并勾选App Groups功能
点击App Groups边的Edit进行编辑,勾选之前创建的Group,成为组员

[图片上传中...(9D8DF0E6-C5F3-42EA-8266-DB5891622E0E.png-793d71-1524216372694-0)]

9D8DF0E6-C5F3-42EA-8266-DB5891622E0E.png


2.项目配置

主体App及Extension的Capabilities中打开App Groups选项
勾选Group

0AC8ED3A-97A7-4038-BD4C-EDFCA85A139D.png


3.通过UserDefaults共享基本数据

//通过UserDefaults存储
- (void)saveDataByNSUserDefaults
{
    //填写正确的groupID
    NSUserDefaults *shared = [[NSUserDefaults alloc] initWithSuiteName:@"group.com.aimi.extension"];
    [shared setObject:@"Hello,UserDefaults" forKey:@"todayExTextStr"];
    [shared synchronize];
}
//通过UerDefaults 读取数据
- (NSString *)readDataFromNSUserDefaults
{
    //填写正确的GroupId
    NSUserDefaults *shared = [[NSUserDefaults alloc] initWithSuiteName:@"group.com.aimi.extension"];
    NSString *value = [shared valueForKey:@"todayExTextStr"];
    return value;
}


4.通过NSFileManager共享文件

//通过NSFileManager 存储
- (BOOL)saveDataByNSFileManager
{
    NSError *error = nil;
    //填写正确的GroupID
    NSURL *containerURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.com.aimi.extension"];
    containerURL = [containerURL URLByAppendingPathComponent:@"Library/Caches/todayExtension.txt"];
    
    NSString *value = @"Hello,FileManager";
    BOOL result = [value writeToURL:containerURL atomically:YES encoding:NSUTF8StringEncoding error:&error];
    if (!result) {
        NSLog(@"%@",error);
    } else {
        NSLog(@"save value:%@ success.",value);
    }
    
    return result;
}
//通过NSFileManager 读取数据
- (NSString *)readDataByNSFileManager
{
    NSError *error = nil;
    //填写正确的GroupId
    NSURL *containerURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.com.aimi.extension"];
    containerURL = [containerURL URLByAppendingPathComponent:@"Library/Caches/todayExtension.txt"];
    NSString *value = [NSString stringWithContentsOfURL:containerURL encoding:NSUTF8StringEncoding error:&error];
    
    return value;
}


七.其他

在Today Extension中同样可以进行网络请求,设置图片
并且Extension界面可以设置动画以及界面切换甚至可以展现一个小游戏(只是没有必要)


extension和主体App处于不同target,不能直接使用主体App中的框架及图片
若需要使用网络框架或图片可以另外导入到extension中
或者在加入图片或文件时,勾选上extension的target即可只用

推荐做成framework框架,勾选上extension的target (所有extension和主题共用)

96C47F7C-4DE6-443F-8466-BF331D88E363.png


其他API:

//使用这个UIVibrancyEffect效果,让widget里的自定义文字看起来更有原生的那种模糊半透明效果
//即滑动时文字会随着背景的颜色而变化
//primary和sencondary暂时测试下来没什么区别
+ (UIVibrancyEffect *)widgetPrimaryVibrancyEffect NS_AVAILABLE_IOS(10_0); 
+ (UIVibrancyEffect *)widgetSecondaryVibrancyEffect NS_AVAILABLE_IOS(10_0);
//如何使用UIVibracyEffect
    UIVibrancyEffect* primaryEffect = [UIVibrancyEffect widgetPrimaryVibrancyEffect];
    UIVisualEffectView* primaryView = [[UIVisualEffectView alloc]initWithEffect:primaryEffect];
    primaryView.frame = CGRectMake(100, 40, 100, 40);
    
    UILabel* l1 = [[UILabel alloc]initWithFrame:primaryView.bounds];
    l1.tintColor = [UIColor redColor];
    l1.font = [UIFont systemFontOfSize:16];
    l1.textAlignment = NSTextAlignmentLeft;
    l1.text = @"Primary";

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