iOS JPush初探

前言

最近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 

//此类别是为了拓展jpush sdk的api,封装出来使用,简化原本appDelegate的代码

@interface AppDelegate (Jpush)

//初始化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

你可能感兴趣的:(iOS JPush初探)