本文会涉及iOS10中的UserNotifications框架,关于UN框架的介绍可以看这里iOS 7 8 9 10本地和推送通知踩坑之旅,适配iOS10之自定义推送通知
iOS10新加了一个Notification Service
,那么这个Notification Service
又是干什么的呢?
按照苹果的解释,Notification Service
是一个没有UI的Extension
,用于增加或者替换远程推送内容的。
反映到实际开发上:
-
Notification Service
可以解决推送敏感内容的端到端加密(End-to-end encryption
) - 也可以给远程推送添加本地的媒体文件
流程如下图所示:
新的key: mutable-content
如果需要更改某条推送的内容,则需要在payload中添加mutable-content
key,让系统知道这条推送内容可变。
{
"aps" : {
"alert" : "test message available",
"mutable-content" : 1
},
"encrypted-content" : "#myencryptedcontent"
}
创建Notification Service
创建好的
Notification Service
target中有一个继承自UNNotificationServiceExtension
的NotificationService
类,该类中包含两个方法:
- 方法一:
- (void)didReceiveNotificationRequest:(UNNotificationRequest *)request withContentHandler:(void (^)(UNNotificationContent *contentToDeliver))contentHandler{
self.contentHandler = contentHandler;
self.bestAttemptContent = [request.content mutableCopy];
NSString *newTitle = @"newTitle";
self.bestAttemptContent.title = [NSString stringWithFormat:@"%@ [modified]", newTitle];
self.contentHandler(self.bestAttemptContent);
}
当接收到包含mutable-content
key的payload之后会执行以上方法,同时会给extension一个很短的执行代码的时间,测试结果大概在30秒左右。
- 方法二:
- (void)serviceExtensionTimeWillExpire {
self.contentHandler(self.bestAttemptContent);
}
在Extension被系统杀死之前,会执行上面这个方法,这也是最后一次发出你修改后的通知的机会,如果不做任何操作,则原通知会正常发出。
Notification Service中的End-to-end encryption(端到端加密)
- (void)didReceiveNotificationRequest:(UNNotificationRequest *)request withContentHandler:(void (^)(UNNotificationContent *contentToDeliver))contentHandler{
self.contentHandler = contentHandler;
self.bestAttemptContent = [request.content mutableCopy];
NSDictionary *userInfo = request.content.userInfo;
NSString *encryptionContent = userInfo[@"encrypted-content"];
self.bestAttemptContent.title = [NSString stringWithFormat:@"%@ [modified]", encryptionContent];
}
当使用推送传送一些敏感信息时,可以在上述方法中拦截到推送来的加密数据,然后做解密操作,将推送内容随之做修改即可。
Notification Service中下载图片附件并添加显示
既然iOS10推出了媒体附件富推送,那么我们就可以使用Notification Service
扩展来下载媒体附件做一些推送显示的内容丰富,较好的示例如下:
上图展示了在Notification Service
中下载gif图并展示的效果,具体的代码实现在下面:
UNMutableNotificationContent *content = [self.bestAttemptContent mutableCopy];
NSURL *imageUrl = [NSURL URLWithString:@"http://upload-images.jianshu.io/upload_images/1159656-22419f571bb4bcd9.JPG?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240"];
NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]];
NSURLSessionDownloadTask *task = [session downloadTaskWithURL:imageUrl completionHandler:^(NSURL *location, NSURLResponse *response, NSError *error) {
NSLog(@"Downloading notification attachment completed with %@", error == nil ? @"success" : [NSString stringWithFormat:@"error: %@", error]);
NSError *fileError;
// create a local URL with extension
NSURL *urlWithExtension = [NSURL fileURLWithPath:[location.path stringByAppendingString:imageUrl.lastPathComponent]];
if (![[NSFileManager defaultManager] moveItemAtURL:location toURL:urlWithExtension error:&fileError]) {
NSLog(@"Could not append local attachment file name: %@", fileError);
contentHandler(content);
return;
}
UNNotificationAttachment *attachment = [UNNotificationAttachment attachmentWithIdentifier:imageUrl.absoluteString
URL:urlWithExtension options:nil
error:&fileError];
if (!attachment) {
NSLog(@"Could not create local attachment file: %@", fileError);
contentHandler(content);
return;
}
NSLog(@"Adding attachment: %@", attachment);
NSMutableArray *attachments = content.attachments ? [content.attachments mutableCopy] : [NSMutableArray array];
[attachments addObject:attachment];
content.attachments = attachments;
contentHandler(content);
}];
[task resume];
UNNotificationAttachment
的初始化需要传一个文件的url,而且该url必须是一个file URL
,那么我们就需要创建一个路径,然后把下载好的文件存下来。
参考相关问题写的oc的分类也在上面的链接里。