前言
最近iOS10的更新,客户反映收不到推送消息,经测试发现是在iOS10的上,JPush不回调之前设置好的响应方法。
(ps:之前是同事负责这个模块的,所以更新JPush SDK、适配iOS10的重任落在我身上叻 lol )
初步了解
粗略地浏览了一遍JPush SDK的官方文档,发现极光官方文档也是很详细的。
1、JPush推送机制:
推送大致分两种
一种是封装苹果原生的APNs的推送(以代理形式,简化APNs推送的配置)
另一种是应用内部推送(通过SDK的TCP长连接进行消息推送)。
2、JPush SDK的更新配置:
这里就不作细述叻。iOS JPush SDK配置官方文档
3、消息推送功能:
3.1)远程通知推送,APNs推送通知
3.2)自定义消息推送,应用内部消息推送,只有app处于前台时才会收到的推送,当然极光方面也有离线消息功能。
3.3)富媒体推送,官方说明此功能仅支持安卓端=。=,个人觉得iOS10的Service Extension也可以实现此功能(自定义消息推送+iOS内部处理)。
3.4)本地通知推送,应用内部生产一个定时或定点推送,到达某个时刻、或者地点,就会触发iOS的原生本地通知
ps:(如果应用打开的时候,不会弹出下拉通知框。如果应用在后台的时候,就会弹出通知框,这个跟Lifeline游戏app推送有点像)。
4、推送证书配置:
如果进行推送调试,苹果开发者证书配置是必不可少的。极光官方证书配置教程
这里吐槽一下我绕过的坑,推送环境证书分两种:
- 开发环境证书(Development开发证书)
- 生产环境证书(Production发布证书)
二者都要相互配置好并上传到极光开发者服务后台,后台认证后方可收到推送。
那么问题就来了,如果我们测试的时候使用企业发布证书(客户测试人员比较多,所以使用企业发布),appStore上面版本使用的是正式发布证书,那岂不是会出现冲突?
(ps:极光开发者后台只能配置一个发布证书)
经过iOS小组内部讨论,提出一个可行的方案:
同时申请2个极光开发者账号,分别配置企业发布证书和正式发布证书,安卓、后台也需要同步切换账号所使用的AppKey和Master Secret。
这样就可以解决这个比较尴尬的问题。
思路如下:
(1)iOS、安卓、后台在【测试】的接口环境使用,企业发布证书的账号的AppKey
(2)iOS、安卓、后台在【正式】的接口环境使用,正式发布证书的账号的AppKey
(3)三端判断接口环境,同步切换AppKey,这样就可以同时推送【app的企业版】和【app store版】
读者思考:假如我想使用【正式】的接口环境打包企业版,那岂不是会出现问题?
(ps:实际上不会出现这种情况,具体原因留作读者琢磨:p)
自定义消息推送
自定义消息推送功能与客户需求比较吻合,所以下面展开自定义消息推送的探讨。
- 推送里面分4种:广播推送、标签推送、别名推送、 用户分群推送。
我们的应用使用的比较简单的广播推送(这种推送是所有用户都会收到此推送)。经iOS、安卓、后台三个端同事的讨论,总结出一套比较可行的公司内部推送机制、推送接口。推送业务流程如下:
1、客户端或者后台发起推送,按照约定好的规则,调用推送接口
2、客户端收到广播推送,按照规则响应方法处理,作出相应的操作、提示
如果是购物发货流程业务场景,推送例子如下(推送发起者、响应者都在客户端):
1、商家进行发货操作,客户端回调推送接口(发货通知推送规则)
2、用户接受到发货通知,客户端进行判断,如果收货人是当前用户,则响应推送,弹窗提醒。其它用户均不提醒
具体优化方案也是在做拓展(如标签推送、别名推送、 用户分群推送)。
talk is cheap,show me the code
当我看到AppDelegate.m文件的那一刻,整个人都不好了,推送、支付、分享、其它第三方sdk的应用操作,都堆在AppDelegate里面叻 LOL。
因此,我萌生写一个jpush工具类的想法。go!
继承封装:生成一个类,继承于AppDelegate,重写父类的方法。(ps:每次导入这个工具类都需要更改AppDelegate的父类,略麻烦,所以pass掉叻)
类别封装:生成一个类别作用于(AppDelegate),头文件开放初始化、响应操作的api(这种可能比较友好一点,下面就贴类别的code)
AppDelegate+Jpush.h
#import "AppDelegate.h"
#import "JPUSHService.h"
#import <UserNotifications/UserNotifications.h>
//此类别是为了拓展jpush sdk的api,封装出来使用,简化原本appDelegate的代码
@interface AppDelegate (Jpush)<JPUSHRegisterDelegate>
//初始化jpush
-(void)setupJpush:(nullable NSDictionary *)launchOptions;
@end
AppDelegate+Jpush.m
#import"AppDelegate+Jpush.h"
#define jpush_appKey_inHouse @"xxxxxxxxxx"
#define jpush_channel_inHouse @"Enterprise"
#define jpush_appKey @"xxxxxxxxxx"
#define jpush_channel_AppStore @"App Store"
#define hostUrl @"http://www.test.com/"
@implementationAppDelegate (Jpush)
@implementation AppDelegate (Jpush)
-(void)setupJpush:(nullable NSDictionary *)launchOptions
{
NSNotificationCenter *defaultCenter = [NSNotificationCenter defaultCenter];
[defaultCenter addObserver:self selector:@selector(networkDidReceiveMessage:) name:kJPFNetworkDidReceiveMessageNotification object:nil];
// NSString *advertisingId = [[[ASIdentifierManager sharedManager] advertisingIdentifier] UUIDString];
//Required
if ([[UIDevice currentDevice].systemVersion floatValue] >= 10.0) {
JPUSHRegisterEntity * entity = [[JPUSHRegisterEntity alloc] init];
entity.types = UNAuthorizationOptionAlert|UNAuthorizationOptionBadge|UNAuthorizationOptionSound;
[JPUSHService registerForRemoteNotificationConfig:entity delegate:self];
}
else if ([[UIDevice currentDevice].systemVersion floatValue] >= 8.0) {
//可以添加自定义categories
[JPUSHService registerForRemoteNotificationTypes:(UIUserNotificationTypeBadge |
UIUserNotificationTypeSound |
UIUserNotificationTypeAlert)
categories:nil];
}
else {
//categories 必须为nil
[JPUSHService registerForRemoteNotificationTypes:(UIRemoteNotificationTypeBadge |
UIRemoteNotificationTypeSound |
UIRemoteNotificationTypeAlert)
categories:nil];
}
//Required
// init Push(2.1.5版本的SDK新增的注册方法,改成可上报IDFA,如果没有使用IDFA直接传nil )
// 如需继续使用pushConfig.plist文件声明appKey等配置内容,请依旧使用[JPUSHService setupWithOption:launchOptions]方式初始化。
BOOL isTestHost=![hostUrl isEqualToString:@"http://www.test.com/"];
[JPUSHService setupWithOption:launchOptions appKey:isTestHost?jpush_appKey_inHouse:jpush_appKey
channel:nil
apsForProduction:YES
advertisingIdentifier:nil];
}
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken
{
/// Required - 注册 DeviceToken
[JPUSHService registerDeviceToken:deviceToken];
}
-(void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(nonnull NSError *)error
{
NSLog(@"did Fail To Register For Remote Notifications With Error: %@", error);
}
#pragma mark- JPUSHRegisterDelegate
#pragma mark- JPUSHRegisterDelegate
// iOS 10 Support
- (void)jpushNotificationCenter:(UNUserNotificationCenter *)center willPresentNotification:(UNNotification *)notification withCompletionHandler:(void (^)(NSInteger))completionHandler {
// Required
NSDictionary * userInfo = notification.request.content.userInfo;
if([notification.request.trigger isKindOfClass:[UNPushNotificationTrigger class]]) {
[JPUSHService handleRemoteNotification:userInfo];
}
completionHandler(UNNotificationPresentationOptionAlert); // 需要执行这个方法,选择是否提醒用户,有Badge、Sound、Alert三种类型可以选择设置
}
// iOS 10 Support
- (void)jpushNotificationCenter:(UNUserNotificationCenter *)center didReceiveNotificationResponse:(UNNotificationResponse *)response withCompletionHandler:(void (^)())completionHandler {
// Required
NSDictionary * userInfo = response.notification.request.content.userInfo;
if([response.notification.request.trigger isKindOfClass:[UNPushNotificationTrigger class]]) {
[JPUSHService handleRemoteNotification:userInfo];
}
completionHandler(); // 系统要求执行这个方法
}
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler {
// Required, iOS 7 Support
[JPUSHService handleRemoteNotification:userInfo];
completionHandler(UIBackgroundFetchResultNewData);
}
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo {
// Required,For systems with less than or equal to iOS6
[JPUSHService handleRemoteNotification:userInfo];
}
-(void)application:(UIApplication *)application didRegisterUserNotificationSettings:(nonnull UIUserNotificationSettings *)notificationSettings
{
[application registerForRemoteNotifications];
}
- (void)networkDidReceiveMessage:(NSNotification *)notification {
//jpush回调响应方法
NSDictionary * userInfo = [notification userInfo];
NSString *content = userInfo[@"content"];
NSLog(@"%@",content);
UIAlertView*alterView=[[UIAlertView alloc] initWithTitle:@"" message:content delegate:self cancelButtonTitle:@"取消" otherButtonTitles:@"确定", nil];
[alterView show];
}
-(void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex{
}
@end
调用方法:
在AppDelegate导入AppDelegate+Jpush.h
#import "AppDelegate.h"
#import "AppDelegate+Jpush.h"
@implementation AppDelegate
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
//初始化jpush工具类
[self setupJpush:launchOptions];
return YES;
}
jpush iOS10不回调响应方法的原因是:
需要实现jpushiOS10的代理方法,iOS先回调
-(void)jpushNotificationCenter:(UNUserNotificationCenter *)center willPresentNotification:(UNNotification *)notification withCompletionHandler:(void(^)(NSInteger options))completionHandler;
在此方法中可拓展iOS10新特性Service Extension,如果不特殊处理,则在此方法中调用iOS9响应的jpush方法
完结,撒花:D