Best practice to clear application badges

一般情况下,我们都会用此方法来消除角标:

[UIApplication sharedApplication].applicationIconBadgeNumber = 0;

然而这种方法会使得系统通知栏的历史推送消息全部被清除。

试了下即使用 iOS 10 的消息推送框架 UserNotification 也存在这种情况。

会造成相同效果的还有

[UIApplication sharedApplication].scheduledLocalNotifications = nil;

解决方案:

- (void)applicationDidEnterBackground:(UIApplication *)application {
UILocalNotification *clearEpisodeNotification = [[UILocalNotification alloc] init];
clearEpisodeNotification.fireDate = [NSDate dateWithTimeIntervalSinceNow:1];
clearEpisodeNotification.timeZone = [NSTimeZone systemTimeZone];
clearEpisodeNotification.applicationIconBadgeNumber = -1;
[[UIApplication sharedApplication] scheduleLocalNotification: clearEpisodeNotification];
}

在程序退到后台后,添加该通知。
关键的地方在于 applicationIconBadgeNumber 是 -1。
接着你可以看到红点带着动画优雅地消失了,
同时,历史推送消息被保留了下来。

不过,上述这个方式到 iOS 11 就失效了。
本来以为没辙了,直到发现某个新闻客户端仍然可以实现。

于是,我反编译了该新闻客户端。
applicationDidEnterBackground 入手,很快就找到了相关代码。

image
image

那么答案就出来了,直接使用 UserNotifications 即可,猜测 badge 应该还是 -1。

- (void)applicationDidEnterBackground:(UIApplication *)application {
UNMutableNotificationContent *content = [[UNMutableNotificationContent alloc] init];
content.badge = @(-1);
UNNotificationRequest *request = [UNNotificationRequest requestWithIdentifier:@"clearBadge" content:content trigger:nil];
[[UNUserNotificationCenter currentNotificationCenter] addNotificationRequest:request withCompletionHandler:^(NSError * _Nullable error) {
}];
}

最后记得不要设置 title,否则会收到空白通知。

参考资料:
https://stackoverflow.com/questions/5375355/clear-app-badge-with-local-notifications