一、原生开发推送
今天看到一篇很详细的关于推送的相关文章,一系列的本地推送和远程推送都有,收藏下来,以备后用!
第一、远程推送相关
一、前往开发者中心配置 推送调试证书(用于未上线前推送调试)和推送发布证书(用于上线后推送)
注册证书链接
二、将证书的.cer和.p12文件提供给后台合成.pem文件
三、代码相关
0.点击工程 - TARGETS - Capabilities - Push Notifications 打开,会在项目中生成一个 工程名.entitlements 文件
1.在 AppDelegate 中导入 头文件和协议
// 推送
#ifdef NSFoundationVersionNumber_iOS_9_x_Max
#import <UserNotifications>
#endif
<UNUserNotificationCenterDelegate>
2.在 - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions 方法里实现 请求权限和注册通知代码
// 推送相关
if ([[UIDevice currentDevice].systemVersion floatValue] >= 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 if ([[UIDevice currentDevice].systemVersion floatValue] >= 8.0){
//iOS8 - iOS10
// 申请用户权限
[application registerUserNotificationSettings:[UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeAlert | UIUserNotificationTypeSound | UIUserNotificationTypeBadge categories:nil]];
}
//此方法不写 无法调用接收推送回调方法 didRegisterForRemoteNotificationsWithDeviceToken
//注册远程通知
[[UIApplication sharedApplication] registerForRemoteNotifications];
3.将获取到的 deviceToken值传给后台服务器,
#pragma mark - 推送相关
/* 获取苹果服务器返回的 Device Token */
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken{
NSString *tokenString = [[[[deviceToken description] stringByReplacingOccurrencesOfString:@">" withString:@""] stringByReplacingOccurrencesOfString:@"<" withString:@""] stringByReplacingOccurrencesOfString:@" " withString:@""];
NSLog(@"deviceTokenString : %@", tokenString);
// 返给后台 deviceToken 必须返回
// NSString *urlHttp = @"v.showbuy100.com/index.php?r=order/apns";
// NSString *urlString = [urlHttp stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet characterSetWithCharactersInString:@"`#%^{}\"[]|\\<> "].invertedSet];
// [[MSUAFNRequest sharedInstance] postRequestWithURL:urlString parameters:tokenString withBlock:^(id obj, NSError *error) {
// if (!error) {
// NSLog(@"------------obj--------------%@",obj);
// }else{
// NSLog(@"-----------error---------------%@",error);
//
// }
// }];
}
/* 获得Device Token失败 */
- (void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error{
NSLog(@"did Fail To Register For Remote Notifications With Error: %@", error);
}
4.调用接收通知的代理方法 (注:前台模式下暂无法接受推送消息,需要退到后台才会显示消息)
/* 接收到推送消息的代理方法 */
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler{
NSLog(@"-----------接收到了远程通知--------------%@",userInfo);
if (application.applicationState == UIApplicationStateInactive) {
NSLog(@"Inactive");
completionHandler(UIBackgroundFetchResultNewData);
} else if (application.applicationState == UIApplicationStateActive){
NSLog(@"Active");
completionHandler(UIBackgroundFetchResultNewData);
} else if (application.applicationState == UIApplicationStateBackground){
NSLog(@"Backround");
completionHandler(UIBackgroundFetchResultNewData);
}
}
5.iOS10 新增方法
/* iOS10 以后版本新增代理 关闭或后台模式 点击通知栏会调用这个方法 */
- (void)userNotificationCenter:(UNUserNotificationCenter *)center didReceiveNotificationResponse:(UNNotificationResponse *)response withCompletionHandler:(void (^)())completionHandler{
NSLog(@"iOS10 后新增代理 点击通知栏会调用这个方法");
NSDictionary *dict = response.notification.request.content.userInfo;
NSLog(@"推送--- %@",dict);
NSDictionary *dataDic = dict[@"aps"];
// NSString *str = dict[@"aps"][@"alert"][@"title"];
if ([dataDic[@"type"] isEqualToString:@"100"] || [dataDic[@"type"] isEqualToString:@"99"] || [dataDic[@"type"] isEqualToString:@"97"] || [dataDic[@"type"] isEqualToString:@"95"]) {// 100-发货提醒:买家提醒买家发货 | 99-您出售的商品 XX已完成交易,小贝已将XX元存入您的钱包 | 97-XX 买家购买的XX 商品,已发起退货/退款申请,点击查看详情>> | 95-XX买家购买的XX 商品已退货,请注意处理
[self pushOrderDetailCenterWithPushCode:200];
} else if ([dataDic[@"type"] isEqualToString:@"98"] || [dataDic[@"type"] isEqualToString:@"94"] || [dataDic[@"type"] isEqualToString:@"93"] ){//98-您购买的xx卖家已发货 | 94-您购买的XX商品,卖家已拒绝您的退货/退款申请,点击查看详情>> | 93-您购买的XX商品,卖家已同意您的退货/退款申请,点击查看详情>>
[self pushOrderDetailCenterWithPushCode:201];
} else if ([dataDic[@"type"] isEqualToString:@"96"] || [dataDic[@"type"] isEqualToString:@"92"] || [dataDic[@"type"] isEqualToString:@"91"] ){ // 96-您出售的商品 XX已完成交易,小贝已将XX元存入您的钱包 | 92-您申请退货的XX商品卖家已签收,货款XX元已退回至您的付款账户
[self pushMoneyMineWithPushCode:0];
} else if ([dataDic[@"alert"][@"body"] isEqualToString:@"您有一条新消息"]){
[self pushMessageWithPushCode:0];
}
NSString *categoryIdentifier = response.notification.request.content.categoryIdentifier;
NSLog(@"推送2--- %@",categoryIdentifier);
// if ([categoryIdentifier isEqualToString:@"message"]) {//识别需要被处理的拓展
//
// if ([response.actionIdentifier isEqualToString:@"回复"]) {//识别用户点击的是哪个 action
//
// //假设点击了输入内容的 UNTextInputNotificationAction 把 response 强转类型
// UNTextInputNotificationResponse *textResponse = (UNTextInputNotificationResponse*)response;
// //获取输入内容
// NSString *userText = textResponse.userText;
// //发送 userText 给需要接收的方法
//
// } else if ([response.actionIdentifier isEqualToString:@"清除"]){
//
// }
// }
}
/* iOS10 以后版本新增代理 前台模式 可显示通知提示内容 */
- (void)userNotificationCenter:(UNUserNotificationCenter *)center willPresentNotification:(UNNotification *)notification withCompletionHandler:(void (^)(UNNotificationPresentationOptions))completionHandler{
NSLog(@"iOS10 后新增代理 前台可显示通知提示内容");
// 显示通知
completionHandler(UNNotificationPresentationOptionAlert | UNNotificationPresentationOptionBadge | UNNotificationPresentationOptionSound);
}
四、注意点
1.iOS7以后配置文件 Xcode会自动生成,所以配置文件不需要!(前往 开发者中心 - Account - Certificates, Identifiers & Profiles - Provisioning Profiles - All 查看是否有多余的配置文件)
2.推送出现问题原因有三种可能 ,
1)证书不对 检验证书可以通过后台是否可以跟苹果服务器对接,若能对接说明证书正确,若不能对接证书很大几率出现问题
2)deviceToken不对 检验deviceToken,可以打印出来,如果在出现不卸载不重装系统情况下,值不发生变化说明 deviceToken值正确!(注:APP应用卸载会造成 deviceToken值发生变化 ,不会发生变化的是UUID)
3)后台错误 (接口不对应,服务器地址问题,.pem文件配置错误)
3.远程推送 在前台模式接收不到消息,只有退到后台模式才会接受到消息,- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler 方法才会调用!
4.申请用户权限后,切记需要注册远程通知 ! 如果不注册远程通知,获取deviceToken的方法 didRegisterForRemoteNotificationsWithDeviceToken 不会调用,注册代码如下:
[[UIApplication sharedApplication] registerForRemoteNotifications];
二、个推 - 第三方
1.集成和Appdelegate代码参考个推文档中心说明!
2.注意点
1)手动集成导入包的时候,注意文字 :“**将GtSdkLib目录拷贝到项目工程目录下,导入GtSdkLib文件夹:**” !跟百度API一样,需要先把SDK拖到项目本地工程中,在 addfile到项目中!
1)注意 TARGET 里面的 Linked Frameworks and libraries 里面的库!切记不能出现 GTExtensionSDK.framework ,一旦出现就会出现77个重复报错问题!严格遵守下列图片中的各类库添加!
EB7D0F34-9751-4E51-9AD7-388A361850B0.png