ios Widget

转载:https://www.jianshu.com/p/012319813522

自iOS8之后,苹果支持了扩展(Extension)的开发,开发者可以通过系统提供给我们的扩展接入点 (Extension point) 来为系统特定的服务提供某些附加的功能。今年iOS10的推出,让Widget扩展应用渐渐的火了起来,地位得到重大的提升,从这也可以看出苹果对他的重视,今天我们就来一起学习下Widget,来实现一个简单的扩展程序。

iOS Widget扩展程序

程序效果

程序效果图

创建Widget程序

  • 创建工程,在工程中添加扩展程序
创建Widget程序
  • 创建成功后的目录
创建成功

顺便说一句,扩展程序虽然是程序的扩展,但是这两个应用其实是“独立”的。准确的来说,它们是两个独立的进程,默认情况下互相不应该知道对方的存在。扩展需要对宿主 app (host app,即调用该扩展的 app) 的请求做出响应,当然,通过进行配置和一些手段,我们可以在扩展中访问和共享一些容器 app 的资源,这个我们稍后再说。

Widget布局方式

  • 使用Interface Builder
    工程默认的方式就是使用Interface Builder,如果实现简单的布局的话可以考虑这种方式。
  • 使用代码进行布局
    当涉及到比较复杂的UI布局的时候,可以考虑使用这种布局方式,按大家平时的习惯来。这里需要注意一下,如果需要使用代码布局的话需要修改一下plist文件。
    首先将原有NSExtensionMainStoryboard字段删除,添加字段NSExtensionPrincipalClass,value是你所写的controller的名称,一般默认的都是TodayViewController
修改plist文件

实现相应的方法

1. 设置Widget的size
iOS10之后,Widget支持展开及折叠两种展现方式,通过设置widgetLargestAvailableDisplayMode属性可以让Widget程序实现展开布局。如下:

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view from its nib.
<span class="hljs-keyword">if</span> (isIOS10)
{
    <span class="hljs-keyword">self</span>.extensionContext.widgetLargestAvailableDisplayMode = NCWidgetDisplayModeExpanded;
}

<span class="hljs-keyword">self</span>.preferredContentSize = <span class="hljs-built_in">CGSizeMake</span>(kWidgetWidth, <span class="hljs-number">110</span>);

}

2. 重写切换展开及折叠布局时的方法(iOS10之后)

- (void)widgetActiveDisplayModeDidChange:(NCWidgetDisplayMode)activeDisplayMode withMaximumSize:(CGSize)maxSize
{
    NSLog(@"maxWidth %f maxHeight %f",maxSize.width,maxSize.height);
<span class="hljs-keyword">if</span> (activeDisplayMode == NCWidgetDisplayModeCompact)
{
    <span class="hljs-keyword">self</span>.preferredContentSize = <span class="hljs-built_in">CGSizeMake</span>(maxSize.width, <span class="hljs-number">110</span>);
}
<span class="hljs-keyword">else</span>
{
    <span class="hljs-keyword">self</span>.preferredContentSize = <span class="hljs-built_in">CGSizeMake</span>(maxSize.width, <span class="hljs-number">200</span>);
}

}

3. iOS10之前,视图原点默认存在一个间距,可以实现以下方法来调整视图间距
注:该方法在iOS10之后被遗弃,iOS10默认不存在间距。

- (UIEdgeInsets)widgetMarginInsetsForProposedMarginInsets:(UIEdgeInsets)defaultMarginInsets
{
    return UIEdgeInsetsMake(0, 10, 0, 10);
}

应用唤醒

本来想叫应用间跳转的,想想还是这个名字比较高大上些?
如下,配置url scheme,这个定义的时候尽量不要和其他用用冲突,笔者定义的为WidgetDemo。这样,通过访问WidgetDemo://就可以实现应用唤醒了。代码如下:

配置url scheme
- (void)redButtonPressed:(UIButton *)button
{
    NSLog(@"%s",__func__);
<span class="hljs-built_in">NSURL</span> *url = [<span class="hljs-built_in">NSURL</span> URLWithString:<span class="hljs-string">@"WidgetDemo://red"</span>];

[<span class="hljs-keyword">self</span>.extensionContext openURL:url completionHandler:^(<span class="hljs-built_in">BOOL</span> success) {
    
    <span class="hljs-built_in">NSLog</span>(<span class="hljs-string">@"isSuccessed %d"</span>,success);
}];

}

相应的,在AppDelegate中实现以下方法,这里可以处理传过来的action,对于传过来不同的值可以进行不同的操作,这里我们打印了请求url的内容。

- (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation
{
    if ([[url absoluteString] hasPrefix:@"WidgetDemo"])
    {
        UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"提示" message:[NSString stringWithFormat:@"你点击了%@按钮",[url host]] delegate:nil cancelButtonTitle:@"好的?" otherButtonTitles:nil, nil];
        [alert show];
    }
    return  YES;
}
  • 简易的应用快速启动器
    既然说到了应用唤醒,这里再稍稍拓展以下,想必大家都有用过类似launcher这种的应用快速启动器。其实就是运用了应用间跳转的原理,每款应用都有自定义的url scheme,我们只要知道他们的url scheme就可以跳转至改款应用,例如进行微信的跳转:
- (void)wechatLoginButtonPressed
{
    NSLog(@"%s",__func__);
<span class="hljs-built_in">NSURL</span> *url = [<span class="hljs-built_in">NSURL</span> URLWithString:<span class="hljs-string">@"wechat://"</span>];

[<span class="hljs-keyword">self</span>.extensionContext openURL:url completionHandler:^(<span class="hljs-built_in">BOOL</span> success) {
    
    <span class="hljs-built_in">NSLog</span>(<span class="hljs-string">@"isSuccessed %d"</span>,success);
}];

}

以下是我们比较常用的软件的url scheme,有兴趣的同学们可以试一试:
QQ mqq:// 微信 weixin:// 淘宝taobao:// 微博 sinaweibo:// 支付宝alipay://

数据共享

扩展程序一般都不是脱离宿主程序单独运行的,难免需要和宿主程序进行数据交互。而相对于一般的APP,数据可以用单例,NSUserDefault等等。但由于拓展与宿主应用是两个完全独立的App,并且iOS应用基于沙盒的形式限制,所以一般的共享数据方法都是实现不了数据共享,这里就需要使用App Groups。

  • 在宿主程序和扩展程序中分别设置打开App Group,设置一个group的名称,这里要保证宿主APP和扩展APP的groupName要是相同的。
设置App Group

两种数据存储方式

  • 使用NSUserDefault
    这里不能使用[NSUserDefaults standardUserDefaults];方法来初始化NSUserDefault对象,正像之前所说,由于沙盒机制,拓展应用是不允许访问宿主应用的沙盒路径的,因此上述用法是不对的,需要搭配app group完成实例化UserDefaults。正确的使用方式如下:
    写入数据
    NSUserDefaults *userDefaults = [[NSUserDefaults alloc] initWithSuiteName:@"group.com.japho.widgetDemo"];
    [userDefaults setObject:self.textField.text forKey:@"widget"];
    [userDefaults synchronize];

读取数据

    NSUserDefaults *userDefaults = [[NSUserDefaults alloc] initWithSuiteName:@"group.com.japho.widgetDemo"];
    self.contentStr = [userDefaults objectForKey:@"widget"];
  • 通过NSFileManager共享数据
    写入数据
-(BOOL)saveDataByNSFileManager
{
    NSError *err = nil;
    NSURL *containerURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.com.xxx"];
    containerURL = [containerURL URLByAppendingPathComponent:@"Library/Caches/ widget"];
    NSString *value = @"asdfasdfasf";
    BOOL result = [value writeToURL:containerURL atomically:YES encoding:NSUTF8StringEncoding error:&err];
    if (!result)
    {
        NSLog(@"%@",err);
    }
    else
    {
        NSLog(@"save value:%@ success.",value);
    }
    return result;
}

读取数据

-(NSString *)readDataByNSFileManager
{
    NSError *err = nil;
    NSURL *containerURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.com.xxx"];
    containerURL = [containerURL URLByAppendingPathComponent:@"Library/Caches/ widget"];
    NSString *value = [NSString stringWithContentsOfURL:containerURL encoding: NSUTF8StringEncoding error:&err];
    return value;
}

其他

补充:widget的上线也是需要单独申请APP ID的 需要配置证书和Provisioning Profiles文件
没有配置相关证书时:



配置证书及描述文件:(列举一些)




证书与描述文件配置好之后:
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值