前段时间接手到一个关于本地推送的活,还可以设置重复本地推送,把此次记录拆分成两篇,第一篇主要是记录推送的设置,第二篇主要来讲一下如何实现重复推送。
初始化推送设置(项目支持最低iOS8.0)
头文件导入,需要增加版本的判断
#ifdef NSFoundationVersionNumber_iOS_9_x_Max
#import <UserNotifications/UserNotifications.h>
#endif
由于涉及到iOS版本的兼容,需要考虑iOS10以下及iOS8以上的设置
if (@available(iOS 10.0, *)) {
//iOS10特有
UNUserNotificationCenter *center = [UNUserNotificationCenter currentNotificationCenter];
// 必须写代理,不然无法监听通知的接收与点击
center.delegate = self;
[center requestAuthorizationWithOptions:(UNAuthorizationOptionAlert | UNAuthorizationOptionBadge | UNAuthorizationOptionSound) completionHandler:^(BOOL granted, NSError * _Nullable error) {
if (granted) {
// 点击允许
NSLog(@"注册成功");
[center getNotificationSettingsWithCompletionHandler:^(UNNotificationSettings * _Nonnull settings) {
NSLog(@"%@", settings);
}];
} else {
// 点击不允许
NSLog(@"注册失败");
}
}];
} else {
UIUserNotificationSettings *settings = [UIUserNotificationSettings settingsForTypes: (UIRemoteNotificationTypeBadge | UIRemoteNotificationTypeAlert | UIRemoteNotificationTypeSound) categories:nil];
[[UIApplication sharedApplication] registerUserNotificationSettings:settings];
}
推送的设置
if (@available(iOS 10.0, *)) {
// 使用 UNUserNotificationCenter 来管理通知
UNUserNotificationCenter* center = [UNUserNotificationCenter currentNotificationCenter];
//需创建一个包含待通知内容的 UNMutableNotificationContent 对象,注意不是 UNNotificationContent ,此对象为不可变对象。
UNMutableNotificationContent* content = [[UNMutableNotificationContent alloc] init];
content.title = title;
content.body = body;
UNNotificationSound *sound = [UNNotificationSound soundNamed:soundName];
content.sound = sound;
content.userInfo = param;
// 在 alertTime 后推送本地推送
UNTimeIntervalNotificationTrigger* trigger = [UNTimeIntervalNotificationTrigger
triggerWithTimeInterval:timeInterval repeats:NO];
UNNotificationRequest* request = [UNNotificationRequest requestWithIdentifier:notificationId
content:content trigger:trigger];
//添加推送成功后的处理!
[center addNotificationRequest:request withCompletionHandler:nil];
} else {
NSDate *fireDate = nil;
if (time > 0) {
// 定时时间存在,以定时时间为准
fireDate = [NSDate dateWithTimeIntervalSinceNow:time];
} else {
fireDate = [NSDate dateWithTimeIntervalSince1970:dateTime];
}
// 1.创建通知
UILocalNotification *localNotification = [[UILocalNotification alloc] init];
// 2.设置通知
// 设置通知标题
if (@available(iOS 8.2, *)) {
localNotification.alertTitle = title ? title : @"";
}
// 设置通知显示的内容
localNotification.alertBody = body ? body : @"";
// 设置通知的发送时间,单位秒
localNotification.fireDate = fireDate;
//收到通知时App icon的角标
localNotification.applicationIconBadgeNumber = 1;
//推送是带的声音提醒,设置默认的字段为UILocalNotificationDefaultSoundName
localNotification.soundName = soundName;
// 3.发送通知
// 方式一: 根据通知的发送时间(fireDate)发送通知
NSMutableDictionary *newParam = [NSMutableDictionary dictionaryWithDictionary:param];
newParam[@"identifier"] = notificationId;
localNotification.userInfo = newParam;
[[UIApplication sharedApplication] scheduleLocalNotification:localNotification];
}
推送的处理
在接收到推送的处理方式上,iOS8和iOS10上有所不同
iOS8上需要通过UIApplicationDelegate的代理方法如下:
- (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification {
NSLog(@"notification:%@", notification);
}
iOS10上需要通过UNUserNotificationCenterDelegate相关代理方法,iOS10上在应用内收到推送时,可以通过willPresentNotification这个代理方法里面通过completionHandler(0)关闭推送的展示
//在展示通知前进行处理,即有机会在展示通知前再修改通知内容。
-(void)userNotificationCenter:(UNUserNotificationCenter *)center willPresentNotification:(UNNotification *)notification withCompletionHandler:(void (^)(UNNotificationPresentationOptions))completionHandler API_AVAILABLE(ios(10.0)){
if (@available(iOS 10.0, *)) {
if([notification.request.trigger isKindOfClass:[UNPushNotificationTrigger class]]) {
NSLog(@"iOS10 收到远程通知:%@", notification);
}
else {
NSLog(@"iOS10 收到本地通知:%@", notification);
}
}
// 避免APP打开的状态下在iOS10+下出现本地推送
completionHandler(0);
}
- (void)userNotificationCenter:(UNUserNotificationCenter *)center didReceiveNotificationResponse:(UNNotificationResponse *)response withCompletionHandler:(void (^)(void))completionHandler API_AVAILABLE(ios(10.0)){
if (@available(iOS 10.0, *)) {
if([response.notification.request.trigger isKindOfClass:[UNPushNotificationTrigger class]]) {
NSLog(@"iOS10 收到远程通知:%@", response);
}
else {
NSLog(@"iOS10 收到本地通知:%@", response);
}
}
}
推送的更新
iOS8上通过注册时添加一个identifier,在更新时找到这个identifier的推送,并取消,然后重新注册推送即可实现更新。
iOS10上因为有identifier的存在,所以可以直接复用之前的identifier可以更新成最新的推送
if (@available(iOS 10.0, *)) {
// iOS10+ 只要重新设置一次 就能完成更新
} else {
// 获取所有本地通知数组
NSArray *localNotifications = [UIApplication sharedApplication].scheduledLocalNotifications;
for (UILocalNotification *notification in localNotifications)
{
NSDictionary *userInfo = notification.userInfo;
if (userInfo)
{
NSString *identifier = [self getNoticeIdentifierWithId:param[@"id"]];
// 如果找到需要取消的通知,则取消
if ([identifier isEqualToString:userInfo[@"identifier"]]) {
[[UIApplication sharedApplication] cancelLocalNotification:notification];
break;
}
}
}
}
推送的删除
iOS8和iOS10上思路都是通过一个identifier来找到之前的推送分别调用自己的取消方法即可,所以如果涉及到修改、删除推送的操作,identifier的设置很重要。
if (@available(iOS 10.0, *)) {
[[UNUserNotificationCenter currentNotificationCenter] removePendingNotificationRequestsWithIdentifiers:@[identifier]];
} else {
// 获取所有本地通知数组
NSArray *localNotifications = [UIApplication sharedApplication].scheduledLocalNotifications;
for (UILocalNotification *notification in localNotifications)
{
NSDictionary *userInfo = notification.userInfo;
if (userInfo)
{
// 如果找到需要取消的通知,则取消
if ([identifier isEqualToString:userInfo[@"identifier"]]) {
[[UIApplication sharedApplication] cancelLocalNotification:notification];
break;
}
}
}
}