最近因为公司的业务需求,对推送做了相应的修改。因为之前用的是极光的推送,对于推送也没有太多的要求,来消息就在通知栏展示,有角标展示即可。可是因为最近需求变更,对于推送需求也有了详细的要求。这也让我从新来review遍关于推送方面的知识。尤其是iOS10之后,推送的细分化,有了更多的认识。特在此做一个总结。
这里不再对原理进行阐述。
一、远程推送
iphone在接受到远端推送的通知,打开App后,程序会运行appDelegate的代理,这里分三种状态
1.APP在未运行,已杀死进程的状态,远程推送过来消息
2.APP在已运行,但APP未在前台,就是在后台,APP展示的不是此时的这个程序,这个时候,远程推送过来消息
3.APP已运行,此时程序在前台,就是我们正在玩我们的APP时,远程推送过来消息
此时我们点击通知栏消息后,在AppDelegate.m文件中相应的代理(代码中对应的是个推的一些特殊跳转处理):
1.1下面是对应1这种情况,app首次打开会进入这个方法:
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
//获取 APNs(通知) 推送内容(app未启动时接受推送消息)
//推送的信息会含在launchOption的字典内,取出后,分析其对应的key值
NSDictionary *remoteDic = [launchOptions objectForKey: UIApplicationLaunchOptionsRemoteNotificationKey];
if (remoteDic)
{
//这里因为个推平台有一个所谓的透传消息,从平台推送过来和我们自己服务器搭建后台推送的时候有所不同,所以做了处理,以防止后面做跳转崩溃
NSString *url = nil;
if ([remoteDic objectForKey:@"payload"])
{
NSString *payloadStr = [remoteDic objectForKey:@"payload"];
NSDictionary *payload = [NSString zhGetJSONSerializationObjectFormString:payloadStr];
url = [payload objectForKey:@"url"];
}
else
{
url = [remoteDic objectForKey:@"url"];
}
//这里写自己的跳转
[WYUserDefaultManager setDidFinishLaunchRemoteNoti:url];
}
}
1.2这里对应2的情况,因为iOS10的原因所以,需要在两个方法里都写
//在iOS 10 以前,为处理 APNs 通知点击事件,统计有效用户点击数,需在AppDelegate.m里的didReceiveRemoteNotification回调方法中调用个推SDK统计接口
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler {
// 将收到的APNs信息传给个推统计
[GeTuiSdk handleRemoteNotification:userInfo];
completionHandler(UIBackgroundFetchResultNewData);
//推送后台跳转
if (application.applicationState == UIApplicationStateActive) {
NSLog(@"active");
//程序当前正处于前台
}
else if(application.applicationState == UIApplicationStateInactive)
{
[GeTuiSdk setBadge:0];
NSLog(@"inactive");
//程序处于后台
[self jumpPage:userInfo];
}
}
// iOS 10: 点击通知进入App时触发,在该方法内统计有效用户点击数
- (void)userNotificationCenter:(UNUserNotificationCenter *)center didReceiveNotificationResponse:(UNNotificationResponse *)response withCompletionHandler:(void (^)())completionHandler {
NSLog(@"didReceiveNotification:%@", response.notification.request.content.userInfo);
// [ GTSdk ]:将收到的APNs信息传给个推统计
[GeTuiSdk handleRemoteNotification:response.notification.request.content.userInfo];
completionHandler();
[GeTuiSdk setBadge:0];
//iOS10以上,点击通知栏推送跳转
NSDictionary *dic = response.notification.request.content.userInfo;
[self jumpPage:dic];
}
1.3第三种情况,这里要说下关于个推的透传功能,个推的透传功能说是为了补充APN的不稳定的。但是在使用中,个推其实推送的到达率也并不是100%,而因为早期对个推透传功能的理解不太深入,尤其是在线和离线的消息推送,走了不少坑。现在先说下在app开启运行的时候,程序会运行下面这个代理
- (void)GeTuiSdkDidReceivePayloadData:(NSData *)payloadData andTaskId:(NSString *)taskId andMsgId:(NSString *)msgId andOffLine:(BOOL)offLine fromGtAppId:(NSString *)appId {
//收到个推消息
NSString *payloadMsg = nil;
if (payloadData) {
payloadMsg = [[NSString alloc] initWithBytes:payloadData.bytes length:payloadData.length encoding:NSUTF8StringEncoding];
}
NSString *msg = [NSString stringWithFormat:@"taskId=%@,messageId:%@,payloadMsg:%@%@",taskId,msgId, payloadMsg,offLine ? @"<离线消息>" : @""];
NSLog(@"\n>>>[GexinSdk ReceivePayload]:%@\n\n", msg);
NSData *jsonData = [payloadMsg dataUsingEncoding:NSUTF8StringEncoding];
NSDictionary *dic = [NSJSONSerialization JSONObjectWithData:jsonData options:NSJSONReadingMutableContainers error:nil];
NSString *type = [[dic objectForKey:@"extra"] objectForKey:@"type"];
//不是离线消息处理
if (!offLine)
{
}
}
上面是当app不在前台的时候个推处于离线状态,app走APNS过来的。对于个推而言,当点击推送的通知栏,打开app都会走上面的代码,但由于app开启后,个推平台有可能未能连接,会出现推送不出来,或者有时候一下好多条。所以,我对个推这个离线的队列感到不可信。所以,在离线的时候走APNS,因此这样写的推送相关的代码。
二、红点,角标,通知栏
对于个推,角标是由+ (void)setBadge:(NSUInteger)value这个方法来设置的,而这个角标就是在个推平台计数的角标,所以当使用+1的时候,如果不把这个角标设置的话,角标都会+1的,所以当再次来推送的时候,会在原有的基础上+1,不会从0开始计数,所以,要在每个点击通知栏后的代理方法里对个推的角标设置
[GeTuiSdk setBadge:0];
而如果想把app图标角标清除,需要用
[[UIApplication sharedApplication] setApplicationIconBadgeNumber:0];
但是使用这个的话,通知栏也会完全清除掉。当然如果你需要清除所有通知栏,可以使用这个方法,但是我们要求的是要保留通知栏,可点击app不是点击通知栏的时候,角标清除。查看了许多网上的大家的处理方式,最后使用本地推送的折中方式来解决。
在app进入从后台变为前台的时候启动一条本地推送,并设置角标为-1,紧接着马上取消本地推送。
- (void)applicationDidBecomeActive:(UIApplication *)application {
//设置推送红点取消
//1.设置本地推送
UILocalNotification *localNote = [[UILocalNotification alloc] init];
localNote.applicationIconBadgeNumber = -1;
[[UIApplication sharedApplication] scheduleLocalNotification:localNote];
[[UIApplication sharedApplication] cancelLocalNotification:localNote];
}
网友评论