2020阿里云推送iOS SDK配置

2020-07-08  本文已影响0人  的的可可

阿里云推送
总的来说官方文档已经写得很清楚了,我这里做一点补充,避免以后踩没必要的坑。
1:iOS应用推送需配置开发环境/生产环境推送证书,具体可参见制作推送证书里面在推送证书哪里有详细的说明。
2:sdk下载位置:阿里云主页->控制台主页->搜索移动推送(要先登录阿里云账号)

样子已经不是官方文档的样子了
3:下载SDK统一配置文件
在上面图片遮盖部分点击你已经添加的app
截屏2020-07-08下午4.37.22.png

这这里获取你所注册app的AppKey,AppSecret。
然后点击箭头处应用配置


这里下载配置文件

然后拖入自己的Xcode工程中就可以了。

然后下面是SDK初始化的代码

AppDelegate.m中

@interface AppDelegate ()<UIApplicationDelegate,UNUserNotificationCenterDelegate>

@end

@implementation AppDelegate{
// iOS 10 通知中心
UNUserNotificationCenter * _notificationCenter;
    
}
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
// 初始化阿里云推送SDK
[self initCloudPush];
// APNs注册,获取deviceToken并上报
[self registerAPNS:application];
// 监听推送消息到达
[self registerMessageReceive];
// 点击通知将 App 从关闭状态启动时,将通知打开回执上报 // 计算点击 OpenCount
// [CloudPushSDK handleLaunching:launchOptions];(Deprecated from v1.8.1)
[CloudPushSDK sendNotificationAck:launchOptions];
// 监听推送通道打开动作
[self listenerOnChannelOpened];
// 主动获取设备通知是否授权 (iOS 10+)
[self getNotificationSettingStatus];

 /*
     
     当程序处于关闭状态 (APP未运行) 收到推送通知时,
     点击 图标 会调用该方法 didFinishLaunchingWithOptions:
     那么通知可以通过 launchOptions 这个参数获取到。
     在该方法中 didFinishLaunchingWithOptions: 这个函数 在正常启动下 launchOptions 是空,
     如果你是从点击推送通知过来的,那么 laungchOptions 里面会包含你的推送的内容。
     在这里就可以进行相应的处理,你就可以发一个通知,可以在 rootViewController 中接收执行相应的操作
    */
    
    // 当APP为关闭状态 收到推送通知,点击 图标 会调用该方法 didFinishLaunchingWithOptions:
    if (launchOptions) {
        NSLog(@"\n ================app关闭时通知包的内容 launchOptions: %@",launchOptions);
        NSDictionary *pushNotificationKey = [launchOptions objectForKey:UIApplicationLaunchOptionsRemoteNotificationKey];
        if (pushNotificationKey) {
            // 这里定义自己的处理方式
            // 如需要代码控制 BadgeNum (icon右上角的数字)
            //[UIApplication sharedApplication].applicationIconBadgeNumber = 10;
        }
    }
    
    /* 当APP为关闭状态,收到推送通知, 点击 通知栏 跳转到指定的页面 */
    [self jumpViewController:launchOptions];

return YES;
}
//初始化阿里云推送SDK
- (void)initCloudPush {
//    [CloudPushSDK turnOnDebug];//打开调试日志,正式上线建议关闭
    // SDK初始化
    [CloudPushSDK asyncInit:你注册的AppKey appSecret:你注册的AppSecret callback:^(CloudPushCallbackResult *res) {
        if (res.success) {
            NSLog(@"\n =================================== 阿里云推送SDK初始化成功Push SDK init success, deviceId: %@.", [CloudPushSDK getDeviceId]);
            
        } else {
            NSLog(@"\n ==================================== 阿里云推送SDK初始化失败Push SDK init failed, error: %@", res.error);
        }
    }];
}
// APNs注册,获取deviceToken并上报
- (void)registerAPNS:(UIApplication *)application {
    
    float systemVersionNum = [[[UIDevice currentDevice] systemVersion] floatValue];
    if (systemVersionNum >= 10.0) {
        
        // iOS 10 notifications
        _notificationCenter = [UNUserNotificationCenter currentNotificationCenter];
        
        // 创建 category,并注册到通知中心
        [self createCustomNotificationCategory];
        
        // 遵循协议
        _notificationCenter.delegate = self;
        
        // 请求客户推送通知权限,以及推送的类型
        [_notificationCenter requestAuthorizationWithOptions:UNAuthorizationOptionAlert | UNAuthorizationOptionBadge | UNAuthorizationOptionSound completionHandler:^(BOOL granted, NSError * _Nullable error) {
            
            if (granted) {
                // granted
                NSLog(@"\n ====== 请求客户推送通知权限,以及推送的类型User authored notification.");
        
                // 向APNs注册,获取deviceToken  // 要求在主线程中
                dispatch_async(dispatch_get_main_queue(), ^{
                    [application registerForRemoteNotifications];
                });
                
            } else {
                // not granted
                NSLog(@"\n ====== User denied notification.");
                
                // 即使客户不允许通知也想让它通知 // 待测试
                // 向APNs注册,获取deviceToken  // 要求在主线程中
                dispatch_async(dispatch_get_main_queue(), ^{
                    [application registerForRemoteNotifications];
                });
                
            }}];
        
        /**
         *  主动获取设备通知是否授权 (iOS 10+)
         */
        [_notificationCenter getNotificationSettingsWithCompletionHandler:^(UNNotificationSettings * _Nonnull settings) {
            
            // 进行判断做出相应的处理
            if (settings.authorizationStatus == UNAuthorizationStatusNotDetermined) {
                NSLog(@"\n ====== 未选择是否允许通知");
            } else if (settings.authorizationStatus == UNAuthorizationStatusDenied) {
                NSLog(@"\n ====== 未授权允许通知");
            } else if (settings.authorizationStatus == UNAuthorizationStatusAuthorized){
                NSLog(@"\n ====== 已授权允许通知");
            }
        }];
        
        
    } else if (systemVersionNum >= 8.0) { // 适配 iOS_8, iOS_10.0
        
        // iOS 8 Notifications
        // 不会有黄色叹号
#pragma clang diagnostic push
#pragma clang diagnostic ignored"-Wdeprecated-declarations"
        [application registerUserNotificationSettings:
         [UIUserNotificationSettings settingsForTypes:
          (UIUserNotificationTypeSound | UIUserNotificationTypeAlert | UIUserNotificationTypeBadge) categories:nil]];
        [application registerForRemoteNotifications];
        
        /*
         // 提出弹窗,授权是否允许通知
         UIUserNotificationSettings * settings = [UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeBadge|UIUserNotificationTypeSound|UIUserNotificationTypeAlert categories:nil];
         [[UIApplication sharedApplication] registerUserNotificationSettings:settings];
         
         // 注册远程通知 (iOS_8+)
         [[UIApplication sharedApplication] registerForRemoteNotifications];
         
         if ([[UIApplication sharedApplication] currentUserNotificationSettings].types  == UIUserNotificationTypeNone) { //判断用户是否打开通知开关
             NSLog(@"没有打开");
           } else {
             NSLog(@"已经打开");
           }
         */
        
#pragma clang diagnostic pop
    } else {
        
        // iOS < 8 Notifications
#pragma clang diagnostic push
#pragma clang diagnostic ignored"-Wdeprecated-declarations"
        [[UIApplication sharedApplication] registerForRemoteNotificationTypes:
         (UIRemoteNotificationTypeAlert | UIRemoteNotificationTypeBadge | UIRemoteNotificationTypeSound)];
        /*
         UIRemoteNotificationType types = UIRemoteNotificationTypeBadge|UIRemoteNotificationTypeSound|UIRemoteNotificationTypeAlert;
         [[UIApplication sharedApplication] registerForRemoteNotificationTypes:types];
         
         if ([[UIApplication sharedApplication] enabledRemoteNotificationTypes]  == UIRemoteNotificationTypeNone) { //判断用户是否打开通知开关
         }
         */
        
#pragma clang diagnostic pop
    }

}


/**
 *  创建并注册通知category (iOS 10+)
 */
- (void)createCustomNotificationCategory {
    // 自定义 action1 和 action2
    UNNotificationAction *action1 = [UNNotificationAction actionWithIdentifier:@"action1" title:@"test1" options: UNNotificationActionOptionNone];
    UNNotificationAction *action2 = [UNNotificationAction actionWithIdentifier:@"action2" title:@"test2" options: UNNotificationActionOptionNone];
    // 创建id为`test_category`的category,并注册两个action到category
    // UNNotificationCategoryOptionCustomDismissAction 表明可以触发通知的 dismiss 回调
    UNNotificationCategory *category = [UNNotificationCategory categoryWithIdentifier:@"test_category" actions:@[action1, action2] intentIdentifiers:@[] options: UNNotificationCategoryOptionCustomDismissAction];
    // 注册category到通知中心
    [_notificationCenter setNotificationCategories:[NSSet setWithObjects:category, nil]];
}

/*
 *  APNs注册成功回调,将返回的deviceToken上传到CloudPush服务器
 */
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken NS_AVAILABLE_IOS(3_0) {
    NSLog(@"Upload deviceToken to CloudPush server");
    [CloudPushSDK registerDevice:deviceToken withCallback:^(CloudPushCallbackResult *res) {
        if (res.success) {
            NSLog(@"\n 回调的device Token--------------------- APNs注册成功回调,将返回的deviceToken上传到CloudPush服务器: %@", [CloudPushSDK getApnsDeviceToken]);
        } else {
            NSLog(@"\n ====== APNs注册出问题了,将返回的deviceToken上传到CloudPush服务器: %@", res.error);
        } }];
}
 
/*
 *  APNs注册失败回调
 */
- (void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error {
    NSLog(@"\n ====== APNs注册失败回调 %@", error);
}


/**
 *  注册推送消息 到来 监听
 */
- (void)registerMessageReceive {
    [[NSNotificationCenter defaultCenter] addObserver:self
                                             selector:@selector(onMessageReceived:)
                                                 name:@"CCPDidReceiveMessageNotification"
                                               object:nil];
}
 
/**
 *    处理到来推送消息
 *
 */
- (void)onMessageReceived:(NSNotification *)notification {
    NSLog(@"\n ====== 收到一条推送消息 !!!!!!!");
    CCPSysMessage *message = [notification object];
    NSString *title = [[NSString alloc] initWithData:message.title encoding:NSUTF8StringEncoding];
    NSString *body = [[NSString alloc] initWithData:message.body encoding:NSUTF8StringEncoding];
    NSLog(@"\n ====== 处理到来推送消息Receive message title: %@, content: %@.", title, body);
    NSLog(@"\n ====== 当前线程 %@",[NSThread currentThread]);
    
}
/*
*  App处于启动状态时,通知打开回调
*/
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo {
    // 通过远程通知进入应用时候
    
//    application.applicationIconBadgeNumber = 0;// iOS badge 清0
    /*
     1、  在 iOS7 上时,只要收到通知,就会调起 didReceiveRemoteNotification
     1.2、当程序处于前台工作时,这时候若收到消息推送,会调用该这个方法
     1.3、当程序处于后台运行时,这时候若收到消息推送,如果点击消息或者点击消息图标时,也会调用该这个方法
     2、  但在 iOS_8 上时,只有app前台运行或充电时,才会调起didReceiveRemoteNotification
     2.1、并且 iOS_8 得开启后台模式下接收远程通知。工程配置 TARGATES --> Capabilities BackgroundModes -> ON 选择 RemoteNotification
     */
    
    // 2.2、处理方法:
    if (application.applicationState == UIApplicationStateActive) {
        //前台时候
        if ([[userInfo objectForKey:@"aps"] objectForKey:@"alert"]!= NULL) {
            //处理情况
            NSLog(@"Receive one notification.");
            // 取得APNS通知内容
            NSDictionary *aps = [userInfo valueForKey:@"aps"];
            // 内容
            NSString *content = [aps valueForKey:@"alert"];
            // badge数量
            NSInteger badge = [[aps valueForKey:@"badge"] integerValue];
            // 播放声音
            NSString *sound = [aps valueForKey:@"sound"];
            // 取得Extras字段内容
            NSString *Extras = [userInfo valueForKey:@"Extras"]; //服务端中Extras字段,key是自己定义的
            NSLog(@"内容content = [%@], badge数量 = [%ld], sound播放声音 = [%@], 取得Extras字段内容 = [%@]", content, (long)badge, sound, Extras);
            // iOS badge 清0
            application.applicationIconBadgeNumber = 0;
            // 通知打开回执上报
            // [CloudPushSDK handleReceiveRemoteNotification:userInfo];(Deprecated from v1.8.1)
            [CloudPushSDK sendNotificationAck:userInfo];
        }
    } else {
        //后台时候
        //这里定义自己的处理方式
        PHlog(@"在后台的时候可以不做任何处理???");
    }
}
/**
 *  注册推送通道 打开 监听
 */
- (void)listenerOnChannelOpened {
    [[NSNotificationCenter defaultCenter] addObserver:self
                                             selector:@selector(onChannelOpened:)
                                                 name:@"CCPDidChannelConnectedSuccess"
                                               object:nil];
    
}
/**
 *   推送通道打开回调
 */
- (void)onChannelOpened:(NSNotification *)notification {
    NSLog(@"\n ====== 温馨提示,消息通道建立成功,该通道创建成功表示‘在线’,可以接收到推送的消息");
}
/**
 *  主动获取设备通知是否授权 (iOS 10+) 可以单独调用
 */
- (void)getNotificationSettingStatus {
    [_notificationCenter getNotificationSettingsWithCompletionHandler:^(UNNotificationSettings * _Nonnull settings) {
        if (settings.authorizationStatus == UNAuthorizationStatusAuthorized) {
            NSLog(@"\n 主动获取设备通知是否授权====== User authed.");
        } else {
            NSLog(@"\n 主动获取设备通知是否授权======  User denied.");
        }}];
}
/* 当APP为关闭状态,收到推送通知, 点击 通知栏 跳转到指定的页面 */
- (void)jumpViewController:(NSDictionary *)tfdic {
    
    NSDictionary *remoteNotification = [tfdic objectForKey:UIApplicationLaunchOptionsRemoteNotificationKey];
    for (NSString *tfStr in remoteNotification) {
        
        // 通知里面的内容需要包含 " "
        if ([tfStr isEqualToString:@"careline"]) {
        //这里做跳转操作
        }
    }
}
- (void)applicationDidBecomeActive:(UIApplication *)application {
    //阿里云推送
    NSLog(@" \n ------ App 在后台运行 点击 icon 激活的APP ------");
        
        if (application.applicationIconBadgeNumber != 0) {
            
            // 发起请求获取未读消息的内容
            
            // 点击 icon 从后台进入应用时, 对 badge 的处理
            application.applicationIconBadgeNumber = 0;
            
            // 清除导航栏未读的通知
            [_notificationCenter removeAllDeliveredNotifications];
            
        }
        
        /* 备注:
         APP在后台:  如果不是点击通知栏进入APP, 是通过点击icon 进入程序是拿不到推送消息的。
         原因是这样的:如果堆积了多条应用,回调将会变得复杂,而且没用。
         正确的做法是,服务器要缓存好当前的未读消息,进入应用的时候去获取未读消息。
         */
    
}
/**
 *  iOS 10 + 实现两个代理方法之一。
 *  当APP处于后台  点击通知栏通知
 *  触发通知动作时回调,比如点击、删除通知和点击自定义 action(iOS 10+)
 */
- (void)userNotificationCenter:(UNUserNotificationCenter *)center didReceiveNotificationResponse:(UNNotificationResponse *)response withCompletionHandler:(void (^)(void))completionHandler {
    
    NSString *userAction = response.actionIdentifier;
    
    // 点击通知打开
    if ([userAction isEqualToString:UNNotificationDefaultActionIdentifier]) {
        NSLog(@"\n ====== 点击通知打开User opened the notification.");
        // 处理iOS 10通知,并上报通知打开回执
        [self handleiOS10Notification:response.notification];
    }
    // 通知dismiss,category 创建时传入 UNNotificationCategoryOptionCustomDismissAction 才可以触发
    if ([userAction isEqualToString:UNNotificationDismissActionIdentifier]) {
        NSLog(@"\n ====== 用户关闭了通知User dismissed the notification.");
    }
    NSString *customAction1 = @"action1";
    NSString *customAction2 = @"action2";
    
    // 点击用户自定义Action1
    if ([userAction isEqualToString:customAction1]) {
        NSLog(@"点击用户自定义Action1 User custom action1.");
    }
    
    // 点击用户自定义Action2
    if ([userAction isEqualToString:customAction2]) {
        NSLog(@"点击用户自定义Action2 User custom action2.");
    }
    completionHandler();
}
 
 
/**
 *  App 处于前台时收到通知 (iOS 10+ )
 *  iOS 10 + 实现两个代理方法之一。
 *  只有当应用程序位于前台时,该方法才会在委托上调用。如果方法未被执行或处理程序没有及时调用,则通知将不会被提交。
 *  应用程序可以选择将通知呈现为声音、徽章、警报和/或通知列表中。此决定应基于通知中的信息是否对用户可见。
 */
- (void)userNotificationCenter:(UNUserNotificationCenter *)center willPresentNotification:(UNNotification *)notification withCompletionHandler:(void (^)(UNNotificationPresentationOptions))completionHandler {
    NSLog(@"\n ====== App 处于前台时收到通知 (iOS 10+ ) Receive a notification in foregound.");
    
    //音效文件路径
    NSString *path = [[NSBundle mainBundle] pathForResource:@"PushSound" ofType:@"caf"];
    // 这里是指你的音乐名字和文件类型
    NSLog(@"path-----------------------这里是指你的音乐名字和文件类型 %@",path);
    
    //组装并播放音效
//    SystemSoundID soundID;
//    NSURL *filePath = [NSURL fileURLWithPath:path isDirectory:NO];
//    AudioServicesCreateSystemSoundID((__bridge CFURLRef)filePath, &soundID);
//    AudioServicesPlaySystemSound(soundID);
    
    // 处理iOS 10通知,并上报通知打开回执
    [self handleiOS10Notification:notification];
    
    /*
     处理完成后调用 completionHandler ,用于指示在前台显示通知的形式
     completionHandler() 功能:可设置是否在应用内弹出通知
     在 iOS 10 中 通知在前台的显示设置:
     */
    
    // 1、通知在前台不显示
    // 如果调用下面代码: 通知不在前台弹出也不在通知栏显示
    // completionHandler(UNNotificationPresentationOptionNone);
    
    // 2、通知在前台显示
    // 如果调用下面代码: 通知在前台弹出也在通知栏显示
    // completionHandler(UNNotificationPresentationOptionAlert);
    
    
    // 3、通知在前台显示 并带有声音
    // 如果调用下面代码:通知弹出,且带有声音、内容和角标
    completionHandler(UNNotificationPresentationOptionSound | UNNotificationPresentationOptionAlert | UNNotificationPresentationOptionBadge);
}
 
/**
 *  处理App 处于前台时收到通知 (iOS 10+ )
 */
- (void)handleiOS10Notification:(UNNotification *)notification {
    
    UNNotificationRequest *request = notification.request;
    UNNotificationContent *content = request.content;
    
    NSDictionary *userInfo = content.userInfo;
    // 通知时间
    NSDate *noticeDate = notification.date;
    // 标题
    NSString *title = content.title;
    // 副标题
    NSString *subtitle = content.subtitle;
    // 内容
    NSString *body = content.body;
    // 角标
    int badge = [content.badge intValue];
    // 取得通知自定义字段内容,例:获取key为"Extras"的内容
    NSString *extras = [userInfo valueForKey:@"Extras"];
    // 通知角标数清0
    [UIApplication sharedApplication].applicationIconBadgeNumber = 0;
    
    // 同步角标数到服务端 SDK1.9.5 以后才支持
     [self syncBadgeNum:0];
    
    // 通知打开回执上报
    [CloudPushSDK sendNotificationAck:userInfo];
    
    NSLog(@"\n ====== App 处于前台时收到通知 (iOS 10+ ) Notification, == date: %@, == title: %@, == subtitle: %@, == body: %@, == badge: %d, == extras: %@.", noticeDate, title, subtitle, body, badge, extras);
}
 
/* 同步通知角标数到服务端 */
- (void)syncBadgeNum:(NSUInteger)badgeNum {
    [CloudPushSDK syncBadgeNum:badgeNum withCallback:^(CloudPushCallbackResult *res) {
        if (res.success) {
            NSLog(@"\n ====== 同步通知角标数到服务端成功Sync badge num: [%lu] success.", (unsigned long)badgeNum);
        } else {
            NSLog(@"\n ====== 同步通知角标数到服务端失败Sync badge num: [%lu] failed, error: %@", (unsigned long)badgeNum, res.error);
        }
    }];
}
// iOS(8_0, 10_0) 当应用程序被用户从远程通知中选择操作时激活,调用该方法处理程序(
- (void)application:(UIApplication *)application handleActionWithIdentifier:(nullable NSString *)identifier forRemoteNotification:(NSDictionary *)userInfo completionHandler:(nonnull void (^)(void))completionHandler {}
 
// iOS 7+ 不论是前台还是后台只要有远程推送都会调用(如果这条和658行的代理都实现了,系统只会执行这条)
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult result))completionHandler  NS_AVAILABLE_IOS(7_0) {
    
    NSLog(@"\n ====== iOS 7+ 前台后台都会调用");
    
    /*
     建议使用该方法,还有一个作用。根据苹果给出的文档,系统给出30s的时间对推送的消息进行处理,此后就会运行CompletionHandler 程序块。
     在处理这类推送消息(即程序被启动后接收到推送消息)的时候,通常会遇到这样的问题 :
     就是当前的推送消息是当前程序正在前台运行时接收到的,还是说是程序在后台运行,用户点击系统消息通知栏对应项进入程序时而接收到的?这个其实很简单,用下面的代码就可以解决:
     */
    
    // 做相应的判断是前台还是后台
    if (application.applicationState == UIApplicationStateActive) {
        
        // 程序当前正处于前台 如果不做处理里面不用写 就可以了
        
        /*
         关于userInfo的结构,参照苹果的官方结构:
         {
         "aps" : {
         "alert" : "You got your emails.",
         "badge" : 9,
         "sound" : "bingbong.aiff"
         "acme1" : "bar",
         "acme2" : 42
         }
         
         即key aps 对应了有一个字典,里面是该次推送消息的具体信息。具体跟我们注册的推送类型有关。另外剩下的一些key就是用户自定义的了。
         */
        
    } else if (application.applicationState == UIApplicationStateInactive) {
        // 程序处于后台 做相应的处理
        for (NSString *tfStr in userInfo) {
            if ([tfStr isEqualToString:@"careline"]) {
#warning message - 跳转界面写在这里
                
            }
        }
    }
}
上一篇下一篇

猜你喜欢

热点阅读