随着极光版本得不断更迭,许多人看着极光demo看的一脸懵逼,这里更新最新的极光推送Xcode配置和相关代码。
点击后,会在xcode的文件栏生成一个 项目名.entitlements的文件
配置完以上之后,进入项目的appdelegate里面,首先导入头文件和遵循代理:
#import "AppDelegate.h"
// 引入 JPush 功能所需头文件
#import "JPUSHService.h"
// iOS10 注册 APNs 所需头文件
#ifdef NSFoundationVersionNumber_iOS_9_x_Max
#import <UserNotifications/UserNotifications.h>
#endif
@interface AppDelegate ()
在系统方法里面申请通知权限
注意:appkey一定不要忘了从极光复制过来
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
JPUSHRegisterEntity *entity = [[JPUSHRegisterEntity alloc]init];
if (@available(iOS 12.0, *)) {
entity.types = JPAuthorizationOptionAlert|JPAuthorizationOptionBadge|JPAuthorizationOptionSound|JPAuthorizationOptionProvidesAppNotificationSettings;
} else {
entity.types = JPAuthorizationOptionAlert|JPAuthorizationOptionBadge|JPAuthorizationOptionSound;
}
[JPUSHService setupWithOption:launchOptions appKey:appKey
channel:channel
apsForProduction:isProduction
advertisingIdentifier:nil]; // 这里是没有advertisingIdentifier的情况,有的话,大家在自行添加
//设置别名--后面方法有写
[AppDelegate setJPushAlias];
//极光透传消息--极光的服务器发送的message消息,非push
NSNotificationCenter *defaultCenter = [NSNotificationCenter defaultCenter];
[defaultCenter addObserver:self selector:@selector(networkDidReceiveMessage:) name:kJPFNetworkDidReceiveMessageNotification object:nil];
}
#pragma mark == 极光透传 极光message消息通知----用的极光服务器
- (void)networkDidReceiveMessage:(NSNotification *)notification {
KTPLog(@"---透传数据%@",notification);
NSDictionary * userInfo = [notification userInfo];
//写自己需要的逻辑处理。。。
}
获取deviceToken
// 获取deviceToken
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken
{
[JPUSHService registerDeviceToken:deviceToken];
}
// 注意 :此方法在ios 3 以后可以使用,但是在很多朋友可能会遇到不调用此方法的情况,也就无法获取deviceToke,这里提供一种解决方案
-(void)application:(UIApplication *)application didRegisterUserNotificationSettings:(UIUserNotificationSettings *)notificationSettings
{
[application registerForRemoteNotifications];
}
关于这个deviceToken,iOS10系统上,每次卸载重装将产生新的app,如果没有卸载重装,每次请求获得的device token是不变的。
//极光推送的iOS10代理方法
ios 10 support 处于前台时接收到通知
// ios 10 support 处于前台时接收到通知
- (void)jpushNotificationCenter:(UNUserNotificationCenter *)center willPresentNotification:(UNNotification *)notification withCompletionHandler:(void (^)(NSInteger))completionHandler
{
NSDictionary * userInfo = notification.request.content.userInfo;
if ([notification.request.trigger isKindOfClass:[UNPushNotificationTrigger class]]) {
[JPUSHService handleRemoteNotification:userInfo];
NSLog(@"iOS10处理远程推送-处于前台时接收到通知:%@", userInfo);
// 添加各种需求。。。。。
// 如果产品需要app处于前台是,通知栏也有alert弹窗,那么下面的代码需要打开
// UNNotificationRequest *request = notification.request; // 收到推送的请求
// UNNotificationContent *content = request.content; // 收到推送的消息内容
// NSNumber *badge = content.badge; // 推送消息的角标
// NSString *body = content.body; // 推送消息体
// UNNotificationSound *sound = content.sound; // 推送消息的声音
// NSString *subtitle = content.subtitle; // 推送消息的副标题
// NSString *title = content.title; // 推送消息的标题
// completionHandler(UNNotificationPresentationOptionBadge|UNNotificationPresentationOptionSound|UNNotificationPresentationOptionAlert); // 需要执行这个方法,选择是否提醒用户,有Badge、Sound、Alert三种类型可以设置
}
}
- 如果处于前台时需要自定义弹框或者弹出alert,可以看一下http://www.jianshu.com/p/d2a42072fad9 这篇文章
iOS 10 Support 点击处理事件
// iOS 10 Support 点击处理事件
- (void)jpushNotificationCenter:(UNUserNotificationCenter *)center didReceiveNotificationResponse:(UNNotificationResponse *)response withCompletionHandler:(void (^)())completionHandler {
NSDictionary * userInfo = response.notification.request.content.userInfo;
if([response.notification.request.trigger isKindOfClass:[UNPushNotificationTrigger class]]) {
[JPUSHService handleRemoteNotification:userInfo];
NSLog(@"iOS10 收到远程通知--点击事件:%@", userInfo);
//这里是点击app的消息通知栏后跳转进入的代理方法,如果用户需要进行相关操作可以在这里进行。
//比如点击通知栏的消息后,进入app要跳转到相应的页面,就可以在这里进行操作。注意不管是从后台进入还是杀死app进入,都会进入该代理方法
//这是我自己的跳转代码
UIViewController *currentController = [KTPNavigationController findCurrentViewController];
if (![currentController isKindOfClass:[UIAlertController class]] && ![currentController isKindOfClass:[KTPMyMessageVC class]]) {
//如果没弹窗,或者不是在message页面,那么就跳转
KTPMyMessageVC *messageVC = [[KTPMyMessageVC alloc]init];
[currentController.navigationController pushViewController:messageVC animated:YES];
}
}
completionHandler(); // 系统要求执行这个方法
}
//获取当前window下的currentContent方法我是写在了UINavigationController的扩展下了,方法也贴出来。
///获取当前正在显示的ViewController
+ (UIViewController *)findCurrentViewController
{
UIWindow *window = [[UIApplication sharedApplication].delegate window];
UIViewController *topViewController = [window rootViewController];
while (true) {
if (topViewController.presentedViewController) {
topViewController = topViewController.presentedViewController;
} else if ([topViewController isKindOfClass:[UINavigationController class]] && [(UINavigationController*)topViewController topViewController]) {
topViewController = [(UINavigationController *)topViewController topViewController];
} else if ([topViewController isKindOfClass:[UITabBarController class]]) {
UITabBarController *tab = (UITabBarController *)topViewController;
topViewController = tab.selectedViewController;
} else {
break;
}
}
return topViewController;
}
// iOS7 及以上接收到通知
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler {
// Required, iOS 7 Support
[JPUSHService handleRemoteNotification:userInfo];
completionHandler(UIBackgroundFetchResultNewData);
//这里在iOS7及以上系统的方法中,如果需要在前台和后台做不同的处理的时候,需要判断一下app 的状态,判断方式如下
if ([UIApplication sharedApplication].applicationState == UIApplicationStateActive) {
// 处于前台时 ,添加各种需求代码。。。。
}else if ([UIApplication sharedApplication].applicationState == UIApplicationStateBackground)
{
// app 处于后台 ,添加各种需求
}
}
iOS10前的系统,还需要考虑app被杀死的情况下点击通知栏的推送消息进入app的推送
前台后台收到推送的方法目前我们都有了。程序完全退出的时候,这种情况下,点击通知会走下面这个方法(即注册通知时的方法,程序的入口)
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions{
//在这个方法里,我们需要判断一下程序是否处于完全退出状态,判断方法如下
NSDictionary *remoteNotification = [launchOptions objectForKey: UIApplicationLaunchOptionsRemoteNotificationKey];
if (remoteNotification)
{
// 程序完全退出时,点击通知,添加需求。。。
}
}
需要注意的地方:
有可能有些小伙伴在集成过程中用ios10的手机测试的时候,
在推送通知的时候ios10和ios7的两个代理方法都走了,具体原因具体对待,如果实在找不到问题所在,这里提供两个方法,在ios10的两个代理方法的首尾加入以下语句:
#ifdef NSFoundationVersionNumber_iOS_9_x_Max
#endif // ios10中加入这两句话
在ios7的代理方法首尾加入以下语句
#if __IPHONE_OS_VERSION_MAX_ALLOWED > __IPHONE_7_1
#endif // ios7中加入这两句话
第二种方法是直接在ios7的代理方法中加入判断:
if ([[UIDevice currentDevice].systemVersion floatValue] < 10.0)
角标清除的也贴上
- (void)applicationDidEnterBackground:(UIApplication *)application {
[[UIApplication alloc] setApplicationIconBadgeNumber:0];
}
// 点击之后badge清零
- (void)applicationWillEnterForeground:(UIApplication *)application {
[application setApplicationIconBadgeNumber:0];
[[UNUserNotificationCenter alloc] removeAllPendingNotificationRequests];
}
//设置别名
//设置别名
+ (void)setJPushAlias {
NSString * userId = KTPUserId;//(用户的uuid)
[JPUSHService setAlias:userId completion:^(NSInteger iResCode, NSString *iAlias, NSInteger seq) {
NSLog(@"极光推送别名设置--------------->");
NSLog(@"rescode: %zd, \nalias: %@, \ntags: %zd\n",iResCode, alias, seq);
NSLog(@"极光推送别名设置 <---------------");
} seq:0];
}
}
//删除别名 - 退出账号时调用
+ (void)delJPushAlias {
[JPUSHService deleteAlias:^(NSInteger iResCode, NSString *iAlias, NSInteger seq) {
NSLog(@"删除别名返回--------------->");
NSLog(@"rescode: %zd, \nalias: %@, \ntags: %zd\n",iResCode, alias, seq);
} seq:0];
}
}
网友评论