转载自 FlyOceanFish
序言
众所周知,iOS中消息推送扮演了不可或缺的位置。不管是本地通知还是远程通知无时不刻的在影响着我们的用户体验,以致于在iOS10的时候苹果对推送大规模重构,独立了已UserNotifications和UserNotificationsUI两个单独的framework,可见重要性一斑。针对于WWDC18苹果又给我们带来了什么惊喜呢?
新特性
-
Grouped notifications 推送分组
-
Notification content extensions 推送内容扩展中的可交互和动态更改Action
-
Notification management 推送消息的管理
-
Provisional authorization 临时授权
-
Critical alerts 警告性质的推送
推送分组
随着手机上应用的增多,尤其QQ和微信这两大聊天工具,当手机锁屏的时候,伴随着就是好满屏的推送消息。这一现象不知大家有没有觉着不高效和体验性比较差呢?苹果针对锁屏情况下,对消息进行了分组,从而有效的提高了用户的交互体验,分组形式如下:
分组形式:
-
苹果会自动帮我们以APP的为分类依据进行消息的分组;
-
如果我们设置了threadIdentifier属性则以此属性为依据,进行分组。
代码如下:
let content = UNMutableNotificationContent() content.title = "Notifications Team" content.body = "WWDC session after party" content.threadIdentifier = "notifications-team-chat"//通过这
摘要(Summary)格式定制
当苹果自动将推送消息的归拢到一起的时候,最下边会有一个消息摘要。默认格式是:n more notifications from xxx。不过此格式我们是可以定制的。
-
第一种
let summaryFormat = "%u 更多消息啦啦" return UNNotificationCategory(identifier: "category-identifier", actions: [], intentIdentifiers: [], hiddenPreviewsBodyPlaceholder: nil, categorySummaryFormat: summaryFormat, options: [])
-
第二种
let summaryFormat = “%u 更多消息啦啦!来自OceanFish” let content = UNMutableNotificationContent() content.body = "..." content.summaryArgument = "OceanFish"
同一个category的不同格式,苹果会将其合并在一起;并且不同的summaryArgument苹果也会将其默认合并到一起进行显示
也可以通过let summaryFormat = NSString.localizedUserNotificationString(forKey: "NOTIFICATION_SUMMARY", arguments: nil)来进行本地化服务
数字定制
有时会出现另一个场景:比如发送了2条推送消息,一条是“你有3个邀请函”,另一条是“你有5个邀请函”。那摘要则会显示你有2更多消息。这显然不是我们想要的!我们最好的期望肯定是"你有8个邀请函"。那这种效果怎么显示呢?
苹果给我们提供了另外一个属性,结合上边的摘要(Summary)格式定制我们可以实现以上效果。
let content = UNMutableNotificationContent() content.body = "..." content.threadIdentifier = "..." content.summaryArgument = "Song by Song" content.summaryArgumentCount = 3
当多个消息归拢到一起的时候,苹果会将summaryArgumentCount值加在一起,然后进行显示
推送内容扩展中的可交互和动态更改Action
之前消息是不支持交互的和动态更改Action的,比如界面有个空心喜欢按钮,用户点击则变成了实心喜欢按钮;有个Acction显示“喜欢”,用户点击之后变成"不喜欢"
推送界面可交互
如上图推送界面有个空心喜欢按钮
-
首先配置Notification Content Extention的UUNNotificationExtensionUserInteractionEnabled为YES
然后代码实现
import UserNotificationsUI class NotificationViewController: UIViewController, UNNotificationContentExtension { @IBOutlet var likeButton: UIButton? likeButton?.addTarget(self, action: #selector(likeButtonTapped), for: .touchUpInside) @objc func likeButtonTapped() { likeButton?.setTitle("♥", for: .normal) likedPhoto() } }
Action动态化
// Notification Content Extensions class NotificationViewController: UIViewController, UNNotificationContentExtension { func didReceive(_ response: UNNotificationResponse, completionHandler completion: (UNNotificationContentExtensionResponseOption) -> Void) { if response.actionIdentifier == "like-action" { // Update state... let unlikeAction = UNNotificationAction(identifier: "unlike-action", title: "Unlike", options: []) let currentActions = extensionContext?.notificationActions let commentAction = currentActions![1] let newActions = [ unlikeAction, commentAction ] extensionContext?.notificationActions = newActions } } }
performNotificationDefaultAction()用于点击推送的时候启动应用;dismissNotificationContentExtension()用于关闭锁屏页面的推送具体一条消息
推送消息的管理
这个主要是苹果针对消息增加了一个“管理”的按钮,消息左滑即可出现。
帮助我们快速的针对消息进行设置。
-
Deliver Quietly 则会不会播放声音。
-
turn off 则会关闭推送
-
Setttings 我们可以自己定制
import UIKit import UserNotifications class AppDelegate: UIApplicationDelegate, UNUserNotificationCenterDelegate { func userNotificationCenter(_ center: UNUserNotificationCenter, openSettingsFor notification: UNNotification? ) { } }
临时授权
临时授权主要体现就是推送消息过来会有两个按钮,会主动让用户自己选择
let notificationCenter = UNUserNotificationCenter.current() noficationCenter.requestAuthorization(options: [.badge,.alert,.sound,.provisional]) { (tag, error) in }
在申请权限的时候,加上provisional即可。
警告消息
比如家庭安全、健康、公共安全等因素的时候。此消息需要用户必须采取行动。最简单的一个场景是家里安装了一个摄像头,我们去上班了,此时如果家中有人,则摄像头会推送消息给我们。
-
证书申请
https://developer.apple.com/contact/request/notifications-critical-alerts-entitlement/
-
本地权限申请
let notificationCenter = UNUserNotificationCenter.current() noficationCenter.requestAuthorization(options: [.badge,.alert,.sound,.criticalAlert]) { (tag, error) in }
在申请权限的时候,加上criticalAlert。
播放声音
let content = UNMutableNotificationContent() content.title = "WARNING: LOW BLOOD SUGAR" content.body = "Glucose level at 57." content.categoryIdentifier = "low-glucose—alert" content.sound = UNNotificationSound.criticalSoundNamed(@"warning-sound" withAudioVolume: 1.00) // Critical alert push payload { // Critical alert push payload { "aps" : { "sound" : { "critical": 1, } } "name": "warning-sound.aiff", "volume": 1.0 } }