iOS的Extension Widget开发:Today

[TOC]

1、Tips

  • 苹果官方文档

  • 扩展App想要使用主App中的代码,需要在如下位置引入


    01.png
  • 调试主App则运行主App;调试扩展则运行扩展 (解决扩展不走断点的问题)

  • 扩展App想要使用的图片资源等,需要引入到扩展文件夹下

2、纯代码需要配置info.plist的俩项参数

移除NSExtensionMainStoryboard键,并添加NSExtensionPrincipalClass键,使用view controller的名字作为值。

3、UI样式

  • 背景:尽量不要使用背景,默认的毛玻璃效果很好,主要文字颜色最好是白色,次要文字的颜色最好是 lightTextColor
  • 不要在今日面板里使用可以滚动的 Scroll View,而是要完全展开
  • 缩进:尽量保持默认的缩进,即左边会空几个像素。如果想改变默认缩进,在TodayViewController里面实现以下方法
- (UIEdgeInsets)widgetMarginInsetsForProposedMarginInsets: (UIEdgeInsets)defaultMarginInsets {      
    return UIEdgeInsetsZero;
}

4、与主App进行交互

  • Today跳转App(唤起App,调用App某项功能)
    [UIApplication sharedApplication]在扩展App中是无法访问的,需要通过NSExtensionContext来调用主App的openURL方法
// 
[self.extensionContext openURL:[NSURL URLWithString:@"跳转链接"] completionHandler:^(BOOL success) {
   NSLog(@"open url result:%d",success);
}];

02.png

跳转链接示例:iMyApp://为跳转页面做标识

  • App 处理URL Schemes
// iOS 9+
- (BOOL)application:(UIApplication *)app openURL:(NSURL *)url options:(NSDictionary<NSString*, id> *)options {
   // [url.absoluteString hasPrefix:@"iMyApp://"]
   if ([url.host isEqualToString:@"iMyApp"]) {
       // 操作
      return YES;
   }
   return YES;
}
// iOS 7、iOS 8
- (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation {
      // [url.absoluteString hasPrefix:@"iMyApp://"]
      if ([url.host isEqualToString:@"iMyApp"]) {
        // 操作
       return YES;
      }
   return YES;
}

5、与主App共享数据

  • 利用group,将主App和扩展App做一个数据共享空间(NSUserDefault),先在主App的Targets中创建并设置,再在扩展App的Targets设置(如图)
03.png
  • 主App存扩展App所需要数据
NSUserDefaults* userDefault = [[NSUserDefaults alloc] initWithSuiteName:@"group.com.iMyApp"];
[userDefault setBool:YES forKey:@"islogin"];
  • 扩展App取所需要数据
NSUserDefaults *myDefaults = [[NSUserDefaults alloc] initWithSuiteName:@"group.com.iMyApp"];
BOOL isLogin = [myDefaults objectForKey:@"islogin"];

6、 - (void)widgetPerformUpdateWithCompletionHandler:(void (^)(NCUpdateResult result))completionHandler 方法说明

对于扩展App,即使扩展App现在不可见 (即用户没有拉开通知中心),系统也会时不时地调用实现了 NCWidgetProviding 的扩展的这个方法,来要求扩展刷新界面。
这个机制和 iOS 7 引入的后台机制是很相似的。在这个方法中我们一般可以做一些像 API 请求之类的事情,在获取到了数据并更新了界面,或者是失败后都使用提供的 completionHandler 来向系统进行报告

7、定时更新机制(通过增加定时更新的NSTimer)

把NSTimer fire触发代码调用放到viewWillAppear方法中来(viewDidLoad方法并不是每次都执行).同理当Widget关闭后在viewDidDisappear方法取消NSTimer invalidate定时更新即可。

8、关闭today widget中扩展App的显示

有时候在没有数据的时候需要隐藏扩展,可以使用以下方法:

NCWidgetController *widgetController = [NCWidgetController widgetController];
[widgetController setHasContent:NO forWidgetWithBundleIdentifier:@"扩展的id"];

9、iOS10 的适配- 展开、折叠按钮

在NSExtensionContext中,新添了widgetLargestAvailableDisplayMode属性,来确认当前widget是展开还是折叠状态。所以,先在viewWillAppear中设置widget的mode为展开。

self.extensionContext.widgetLargestAvailableDisplayMode = NCWidgetDisplayModeExpanded;

展开和折叠状态变化时的处理

-(void)widgetActiveDisplayModeDidChange:(NCWidgetDisplayMode)activeDisplayMode withMaximumSize:(CGSize)maxSize {    
  if (activeDisplayMode == NCWidgetDisplayModeCompact) {
    self.preferredContentSize = CGSizeMake([UIScreen mainScreen].bounds.size.width, 110);    
    // 处理~~    
  } else {        
    self.preferredContentSize = CGSizeMake([UIScreen mainScreen].bounds.size.width, 300);     
    // 处理~~    
  }
}
  • 如何用XCode 7.3打出能够适配iOS 10的widget呢?

Xcode 7没有iOS 10,可以通过KVC来解决这个问题

[self.extensionContext setValue:@"1" forKey:@"widgetLargestAvailableDisplayMode"];
  if (activeDisplayMode == 0) {        
    self.preferredContentSize = CGSizeMake([UIScreen mainScreen].bounds.size.width, 110);    
  } else {        
    self.preferredContentSize = CGSizeMake([UIScreen mainScreen].bounds.size.width, 300);    
  }
} 
  • 切记:UI的更新要在主线程操作
//通知主线程刷新
dispatch_async(dispatch_get_main_queue(), ^{    
    //...........;
});
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 221,695评论 6 515
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,569评论 3 399
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 168,130评论 0 360
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,648评论 1 297
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,655评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,268评论 1 309
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,835评论 3 421
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,740评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,286评论 1 318
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,375评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,505评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,185评论 5 350
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,873评论 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,357评论 0 24
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,466评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,921评论 3 376
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,515评论 2 359

推荐阅读更多精彩内容