iOS10新特性-UserNotifications

iOS10增加了新的通知框架UserNotifications,整合了本地通知和APNS,新的API使用起来特别舒服。这篇文章就大概介绍一下新的框架。

新的特性:

  • 通知可以附加图片、音频、视频
  • 可以对通知更新、删除
  • 统一本地通知和APNS

权限申请

 @import UserNotifications;
 // 请求使用通知
 [[UNUserNotificationCenter currentNotificationCenter] requestAuthorizationWithOptions:UNAuthorizationOptionBadge | UNAuthorizationOptionSound | UNAuthorizationOptionAlert completionHandler:^(BOOL granted, NSError * _Nullable error) {
        if (granted) {
            //用户同意使用通知
        }
    }];

第一次调用这个方法时,会弹出一个系统弹窗。


UNNotification_author.png

要注意的是,一旦用户拒绝了这个请求,再次调用该方法也不会再进行弹窗,想要应用有机会接收到通知的话,用户必须自行前往系统的设置中为你的应用打开通知。

远程推送

用户同意了通知申请之后,就可以进行本地通知了。如果要进行远程推送,还需要获取token。然后服务器根据这个token,向Apple Push Notification Service服务器提交请求,然后APNS通过token识别用户,推送给用户。

//向APNS请求token
    [[UIApplication sharedApplication] registerForRemoteNotifications];
//请求token成功
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken {
}

通知权限
可以通知下面的这个方法,检查当前APP的通知权限。

[[UNUserNotificationCenter currentNotificationCenter] getNotificationSettingsWithCompletionHandler:^(UNNotificationSettings * _Nonnull settings) {
    }];

发送通知

// 创建通知
    UNMutableNotificationContent *content = [[UNMutableNotificationContent alloc] init];
    content.title = @"这是一条通知的标题";
    content.body = @"这是一条通知的内容";
    content.categoryIdentifier = @"ceshi";
    
    // 创建发送触发
    UNTimeIntervalNotificationTrigger *trigger = [UNTimeIntervalNotificationTrigger triggerWithTimeInterval:5 repeats:NO];
    
    // 发送请求标识符
    NSString *identifier = @"com.hanwei.firstNotificationIndentifier";
    
    // 创建一个通知请求
    UNNotificationRequest *request = [UNNotificationRequest requestWithIdentifier:identifier content:content trigger:trigger];
    
    
    // 将请求添加到通知中心
    [[UNUserNotificationCenter currentNotificationCenter] addNotificationRequest:request withCompletionHandler:^(NSError * _Nullable error) {
        if (!error) {
            NSLog(@"通知添加成功");
        }
    }];

发送通知成功之后,就可以回到首页,或者锁屏查看了。

UNNotification_home.png

取消和更新通知

远程推送只能更新,不能取消已经展示过的通知。
取消

    // 移除展示过的通知
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(6 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        [[UNUserNotificationCenter currentNotificationCenter] removeDeliveredNotificationsWithIdentifiers:@[identifier]];
        
    });
    
     //移除还未展示的通知
    [[UNUserNotificationCenter currentNotificationCenter] removePendingNotificationRequestsWithIdentifiers:@[identifier]];

更新

    // 不管通知是否已经展示过,都可以根据标示进行通知的更新
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(8 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        
//        NSLog(@"两秒钟后更新通知内容");
        
        // 创建发送触发
        UNTimeIntervalNotificationTrigger *trigger = [UNTimeIntervalNotificationTrigger triggerWithTimeInterval:2 repeats:NO];
        
        UNMutableNotificationContent *content = [[UNMutableNotificationContent alloc] init];
        content.title = @"这是一条通知的标题";
        content.body = @"这是新的内容??";
        content.categoryIdentifier = @"ceshi";
        
        // 创建一个通知请求
        // 同一个标示的话就会覆盖之前的通知
        UNNotificationRequest *request = [UNNotificationRequest requestWithIdentifier:identifier content:content trigger:trigger];
        
        [[UNUserNotificationCenter currentNotificationCenter] addNotificationRequest:request withCompletionHandler:^(NSError * _Nullable error) {
            if (!error) {
                NSLog(@"新的通知添加成功");
            }
        }];
    });

通知处理

通知处理就是,处理用户直接和通知的交互。比如微信发过来一条消息,你可以直接在通知上进行回复,而不用打开微信app。

应用内展示通知

我们知道当APP在前台使用的过程中,在默认情况下,通知是没有任何提示的。但是要想有展示的效果,可以通过下面的代码设置。

// The method will be called on the delegate only if the application is in the foreground. If the method is not implemented or the handler is not called in a timely manner then the notification will not be presented. The application can choose to have the notification presented as a sound, badge, alert and/or in the notification list. This decision should be based on whether the information in the notification is otherwise visible to the user.
- (void)userNotificationCenter:(UNUserNotificationCenter *)center willPresentNotification:(UNNotification *)notification withCompletionHandler:(void (^)(UNNotificationPresentationOptions options))completionHandler
{
    // 用户在前台使用APP的时候,收到通知,会调用此方法.
    // 写这个回调代表,会在APP打开的情况下,弹出通知。
    // 如果不复写次方法,或者`completionHandler();`,APP在前台收到通知将不会弹出提示。
    completionHandler(UNNotificationPresentationOptionBadge | UNNotificationPresentationOptionSound | UNNotificationPresentationOptionAlert);
}

这是通知中的代理方法,需要先设置代理、签协议。<UNUserNotificationCenterDelegate>

To guarantee that your app is able to respond to actionable notifications, you must set the value of this property before your app finishes launching. For example, this means assigning a delegate object to this property in an iOS app’s application:willFinishLaunchingWithOptions:or application:didFinishLaunchingWithOptions: method.

需要注意的是,这个代理必须要在程序启动之前设置好。一般在 application:didFinishLaunchingWithOptions:这个方法设置就可以。

    // 设置代理
    [[UNUserNotificationCenter currentNotificationCenter] setDelegate:self];

处理用户和通知的交互

/**
     设置通知上面的交互按钮
     UNNotificationActionOptionAuthenticationRequired //操作这个按钮会先检查iPhone是否解锁
     UNNotificationActionOptionDestructive //按钮会被高亮标记(红色)
     UNNotificationActionOptionForeground // 点击按钮会将APP唤起
     */
    UNNotificationAction *unlocking = [UNNotificationAction actionWithIdentifier:@"unlocking"
                                                                           title:@"unlocking"
                                                                         options:UNNotificationActionOptionAuthenticationRequired];
    UNNotificationAction *destructive = [UNNotificationAction actionWithIdentifier:@"destructive"
                                                                             title:@"destructive"
                                                                           options:UNNotificationActionOptionDestructive];
    UNNotificationAction *foreground = [UNNotificationAction actionWithIdentifier:@"foreground"
                                                                            title:@"foreground"
                                                                          options:UNNotificationActionOptionForeground];
    UNTextInputNotificationAction *input = [UNTextInputNotificationAction actionWithIdentifier:@"text" title:@"text" options:UNNotificationActionOptionAuthenticationRequired textInputButtonTitle:@"text_btn" textInputPlaceholder:@"placeholder"];
    // 注意:这里的category的标示如果和发送通知时,写的category标示不一样的时候,发过来的通知不会显示action!??!发送通知的时候可以不写标示。
    // 我这里是,两边都写得 “ceshi”
    UNNotificationCategory *category = [UNNotificationCategory categoryWithIdentifier:@"ceshi"
                                                                              actions:@[unlocking, destructive, foreground, input] intentIdentifiers:@[@""]
                                                                              options:UNNotificationCategoryOptionNone];
    
    [[UNUserNotificationCenter currentNotificationCenter] setNotificationCategories:[NSSet setWithObject:category]];
UNNotification_text_01.png
UNNotification_text_02.png
// The method will be called on the delegate when the user responded to the notification by opening the application, dismissing the notification or choosing a UNNotificationAction. The delegate must be set before the application returns from applicationDidFinishLaunching:.
- (void)userNotificationCenter:(UNUserNotificationCenter *)center didReceiveNotificationResponse:(UNNotificationResponse *)response withCompletionHandler:(void(^)())completionHandler
{
    //当用户是通过和通知的交互进入app的时候,拿到通知对象。
    UNNotificationContent *content = response.notification.request.content;
    
    if ([content.categoryIdentifier isEqualToString:@"ceshi"]) {
        
        if ([response.actionIdentifier isEqualToString:@"text"]) {
            UNTextInputNotificationResponse *textResp = (UNTextInputNotificationResponse *)response;
            NSLog(@"输入文字的内容是%@", textResp.userText);
        }
        else if ([response.actionIdentifier isEqualToString:@"unlocking"]) {
            NSLog(@"unlocking");
        }
        else if ([response.actionIdentifier isEqualToString:@"destructive"]) {
            NSLog(@"destructive");
        }
        else if ([response.actionIdentifier isEqualToString:@"foreground"]) {
            NSLog(@"foreground");
        }
    }
    
    completionHandler();
}

刚才我看评论,有一人问我是不是自己实践的??,我都是自己尝试过,才发上去的,多谢大家支持。这里放上我的Demo

关于通知发送富文本部分,正在整理,等整理好了,我会继续补充!

2016-10-09补充:
多媒体通知已经更新:iOS10新特性-UserNotifications(二)

参考:
https://onevcat.com/2016/08/notification/

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,644评论 18 139
  • 我的Blog链接 上篇文章主要介绍了新的通知框架的基本使用,这篇文章主要说一下多媒体通知和扩展。 扩展(Notif...
    mieGod阅读 660评论 3 1
  • PushNotification发展历史 iOS 10 中,把以前杂乱的和通知相关的 API 都统一了,现在开发者...
    Arackboss阅读 1,088评论 0 0
  • 今天学完车,回家手痒,准备使用一下python elasticsearch。安装完后发现 原来是Redis没弄好!...
    朱小虎XiaohuZhu阅读 268评论 0 1
  • 一位视养花如孩子的老花匠在今天的立春这天,发生了一意外。 2017-02-04节气是立春,本是万物复苏,春暖花开时...
    革斤123阅读 262评论 0 0