iOS本地通知和远程推送

iOS 本地通知和远程推送

推送通知的应用,可以推送最新的消息给用户,获得更多的关注。
推送分为本地推送和远程推送两种。并且在 iOS 8.0之后做了相关的改变。iOS 8.0之后不管是远程推送还是本地推送,都需要执行一个注册方法。

// 通知类型,显示未读信息标志, 显示提示通知,播放声音。
UIUserNotificationType notificationType =  UIUserNotificationTypeAlert | UIUserNotificationTypeBadge | UIUserNotificationTypeSound;

// 设置通知类型和执行动作
UIUserNotificationSettings *settings = [UIUserNotificationSettings settingsForTypes:type                                                             categories:nil];

// 注册通知
[[UIApplication sharedApplication] registerUserNotificationSettings:settings];

在调用此方法之后, 若是第一次调用注册,则会弹出是否允许此应用发送通知。并会调用 - (void)application:(UIApplication *)application didRegisterUserNotificationSettings:(UIUserNotificationSettings *)notificationSettings 方法。
在此方法中做操作。

本地通知

创建本地通知

- (void)sendLocalNotifi { 
// 添加本地通知
     UILocalNotification *localNoti = [[UILocalNotification alloc] init];
     NSDate *Date = [NSDate date];
     localNoti.fireDate = Date; // 发送通知时间 这里没做具体处理,若是要推送时间无误差,时间要精确到秒。
     localNoti.timeZone = [NSTimeZone localTimeZone];     // 设置时区 默认时区
     localNoti.repeatInterval = NSCalendarUnitSecond;     // 发送通知的间隔
     localNoti.alertTitle = @"通知";                      // 弹窗title
     localNoti.alertBody = @"起来搬砖了!!";               // 弹窗body显示内容

     localNoti.soundName = UILocalNotificationDefaultSoundName;
     localNoti.alertLaunchImage = @"1.jpg"; // 用于点击启动显示启动页,必须是 UILaunchImageFile
     localNoti.userInfo = [NSDictionary dictionaryWithObjectsAndKeys:@"value", @"key", nil];
     //    localNoti.soundName = @"1";                       // 响铃音乐文件名称, 放在main bundle 里边
      localNoti.applicationIconBadgeNumber = 2;         // app 的未读消息个数
          

     [[UIApplication sharedApplication] cancelAllLocalNotifications];
     [[UIApplication sharedApplication] scheduleLocalNotification:localNoti];    // 按照指定时间发送通知
     // [[UIApplication sharedApplication] presentLocalNotificationNow:localNoti];  // 立即发送通知

    
 }

在注册回调方法中

- (void)application:(UIApplication *)application didRegisterUserNotificationSettings:(UIUserNotificationSettings *)notificationSettings {
   // 判断若 types 不为 UIUserNotificationTypeNone 这可以添加本地通知
  if (notificationSettings.types != UIUserNotificationTypeNone) {
     [self sendLocalNotifi]; // 当然设置本地推送的方法不一定要要放在这里。 这仅仅是个例子。
   }
}

以上获得通知为最基本的通知,提示作用, 若是要有比如回复等操作, 则需要添加相应的 action
方法。

// 添加动作
UIMutableUserNotificationAction *action1 = [[UIMutableUserNotificationAction alloc] init];
action1.identifier = @"1";       // 标示用于区分点击的哪个按钮。
action1.title = @"OK";
action1.activationMode = UIUserNotificationActivationModeForeground; 
// UIUserNotificationActivationModeForeground   在前台处理事件,则在处理此事件时,系统会把 app 切换到前台。
// UIUserNotificationActivationModeBackground   在后台处理事件,点击事件在后台处理,不需要使app 切换到前台状态,
action1.destructive = NO;
action1.authenticationRequired = NO;
action1.behavior = UIUserNotificationActionBehaviorDefault; 
/*
UIUserNotificationActionBehaviorTextInput // 输入框样式
UIUserNotificationActionBehaviorDefault   // 默认样式
*/


UIMutableUserNotificationAction *action2 = [[UIMutableUserNotificationAction alloc] init];
    action2.identifier = @"2";
action2.title = @"cacel";
action2.activationMode = UIUserNotificationActivationModeForeground; 
action2.destructive = NO;       // 
action2.authenticationRequired = NO;

// 设置扩展 Categories 
    UIMutableUserNotificationCategory *catergory = [[UIMutableUserNotificationCategory alloc] init];

catergory.identifier = @"localCatergory"; // 设置识别的标示符

[catergory setActions:@[action1, action2] forContext:UIUserNotificationActionContextMinimal];
[catergory setActions:@[action2, action1] forContext:UIUserNotificationActionContextDefault];

// 别忘了给添加的本地通知添加扩展的识别符
   - (void)sendLocalNotifi { 
    ................
    localNoti.category = @"localCatergory";

   }
 
// 在第一步注册的时候别忘了 categories 
UIUserNotificationSettings *settings = [UIUserNotificationSettings settingsForTypes:type
                                               categories:[NSSet setWithObjects:catergory, nil]];
[[UIApplication sharedApplication] registerUserNotificationSettings:settings];   

注册完成,回调方法。当app处于活跃状态的时候, 会调用以下代理方法- (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification ,开着的状态可以调用此方法进行处理 。

    - (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification
 {
   NSDictionary *infoDic = notification.userInfo;
   NSLog(@"%@", infoDic);
 }

若是在后台或者关闭app的状态,如下图所示,这不同的点击方法则会走不同的方法。

  • 若点击对应的动作这回调用 - (void)application:(UIApplication *)application handleActionWithIdentifier:(nullable NSString *)identifier forLocalNotification:(UILocalNotification *)notification completionHandler:(void(^)())completionHandler 方法对不同的按钮分别进行判别,以进行不同的处理。
    此方法为iOS8.0 方法, 若是iOS 9.0 之后,可以执行- (void)application:(UIApplication *)application handleActionWithIdentifier:(nullable NSString *)identifier forLocalNotification:(UILocalNotification *)notification withResponseInfo:(NSDictionary *)responseInfo completionHandler:(void(^)())completionHandler方法进行判别处理相应的操作。但别忘了在方法最后执行 completionHandler(); 以告诉系统已经处理完毕。
    - (void)application:(UIApplication *)application handleActionWithIdentifier:(NSString *)identifier forLocalNotification:(UILocalNotification *)notification completionHandler:(void (^)())completionHandler {
        if ([identifier isEqualToString:@"accept_action"]) {
    NSLog(@"-----accept action");
         } else if ([identifier isEqualToString:@"cancel_action"]) {
         NSLog(@"-----cancel action");
         }

        if (completionHandler) {
        completionHandler();
        } 
     }
  • 若点击显示信息的区域,未点击 action 按钮, 若app只是在后台,系统则会把app 转化为前台状态,并在- (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification 方法中启用操作, 但若是app是关闭状态, 这系统 - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions 初始化方法中,在 launchOptions 中通过键值对获取到对应的消息。

      - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions{
          
          UILocalNotification * localNotify = [launchOptions objectForKey:UIApplicationLaunchOptionsLocalNotificationKey];
       return YES;
      }
    
  • 取消本地通知

     // 取消所有本地通知
     [[UIApplication sharedApplication] cancelAllLocalNotifications];
     [[UIApplication sharedApplication] cancelLocalNotification:localNoti];
    
远程推送

远程推送的 Action 和 Categories 以及 Settings 与上边的设置一样。
不一样的地方在于接受的方法。

远程推送,所有消息大小不超过2KB,我们获取远程推送的json格式的消息,解析这个消息就是我们的远程推送了;

在注册的回调方法中,

  - (void)application:(UIApplication *)application didRegisterUserNotificationSettings:(UIUserNotificationSettings *)notificationSettings {
   if (notificationSettings.types != UIUserNotificationTypeNone) {
      [application registerForRemoteNotifications];
      }  
  }

在获取远程推送成功之后,APNs 服务器会返回一个 deviceToken,如果是自己家的服务器,则需要获取到相应的 deviceToken 返回到自身的服务器,以方便后台的服务器做推送使用。若是用第三方的,则需要包 deviceToken 发送给第三方。

- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken {
// 注册远程通知成功之后,会接收到 APNs 服务器的 deviceToken
}

获取 deviceToken 失败调用, 可以重新发送注册远程推送请求。

- (void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error{
// 获取 device token 失败, 则会回调此方法,并报出错误
  if(error){
   [application registerForRemoteNotifications];
  }
}

以下几个方法和本地通知方法类似

- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo {
// 接收到远程通知信息
}

-(void)application:(UIApplication *)application handleActionWithIdentifier:(NSString *)identifier forRemoteNotification:(NSDictionary *)userInfo completionHandler:(void (^)())completionHandler
{
//在没有启动本App时,收到服务器推送消息,下拉消息会有快捷回复的按钮,点击按钮后调用的方法,根据identifier来判断点击的哪个按钮
    
    completionHandler();
 }
  
// iOS 9.0 之后的方法, 和上边的方法类似,但多个参数更方便操作
- (void)application:(UIApplication *)application handleActionWithIdentifier:(nullable NSString *)identifier forRemoteNotification:(NSDictionary *)userInfo withResponseInfo:(NSDictionary *)responseInfo completionHandler:(void(^)())completionHandler {

completionHandler();
}

// 若是点击重新启动app
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions{
         
           NSDictionary   * userInfo = [launchOptions objectForKey:UIApplicationLaunchOptionsRemoteNotificationKey];
      return YES;
      
     }
  • 如何实现远程推送的时候,设置Categories,以便系统找到对应的Categories来实现 action 操作。这需要在推送的消息格式中添加相应的字段来识别。当 推送消息到达用户时, iOS能识别这个类别的键值对,并查找到与之对应的注册的通知,并显示相应的操作, 添加的字段如下:

      {
      "aps" :  {
          "alert" : "You’re invited!",
          "category" : "INVITE_CATEGORY"
           }
      }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容