玩轉 iOS 10 推送 —— UserNotifications Framework(中)

上一篇 我們聊了一些:json

  • iOS 10 before 推送的流程
  • iOS 10 beta 推送的基本使用方法
  • 以及跟 iOS 10 before 推送的區別

這一篇我們將繼續探討 iOS 10 推送,並向你們介紹一些進階的內容。數組

Notification Actions

在 iOS 10 中,能夠容許推送添加交互操做 action,這些 action 可使得 App 在前臺或後臺執行一些邏輯代碼。而且在鎖屏界面經過 3d-touch 觸發。如:推出鍵盤進行快捷回覆,該功能以往只在 iMessage 中可行。
(Notification Actions 在 iOS 8 引入,快捷回覆在 iOS 9 引入,在 iOS 10 中,這些 API 被統一。)
ui

在 iOS 10 中,這叫 category,是對推送功能的一個拓展,能夠經過 3d-touch 觸發。編碼

  1. 建立 action加密

    • 即一項交互操做spa

    • title 是交互按鈕的內容3d

    • options 可讓該 action 成爲一條可在前臺執行的 actioncode

    • 建立:orm

      UNNotificationAction *action = [UNNotificationAction actionWithIdentifier:@"reply" title:@"Reply" options:UNNotificationActionOptionNone];複製代碼
  2. 建立 categorycdn

    • 可添加多個 action 的數組,就像圖片中同樣,有多種操做

    • 其中的 id,須要填寫你想要添加到哪一個推送消息的 id

    • 建立:

      UNNotificationCategory *category = [UNNotificationCategory categoryWithIdentifier:@"message" actions:@[action] minimalActions:@[action] intentIdentifiers:@[] options:UNNotificationCategoryOptionNone];複製代碼
  3. category 添加到通知中心:

    [[UNUserNotificationCenter currentNotificationCenter] setNotificationCategories:[NSSet setWithArray:@[category]]];複製代碼
  4. 觸發方式:

    • Remote Notifications 配置 payload,指定其中 category 的值與第 2 步中 Identifier 一致:

      {
        aps : {
        alert : "Welcome to WWDC !",
        category : "message"
                }
      }複製代碼
    • Local Notifications 只須要在建立 contnet 的時候指定 Id 便可:(content 相關內容請參照 上一篇 中的 Content 部分)

      content。categoryIdentifier = @"message";複製代碼

Dismiss Actions

鎖屏及在通知中心收到推送,側滑,會展現 action。

只要點擊 Clear 就能夠將該條推送清除,而且重複的內容不會被髮送到你的其餘 iOS 設備上。


跟 Notification Actions 只有一點小區別,就是添加 action 到 category 的時候,增長一個 option 的值 UNNotificationCategoryOptionCustomDismissAction:

UNNotificationAction *clearAction = [UNNotificationAction actionWithIdentifier:@"clear" title:@"clear" options:UNNotificationActionOptionNone];

UNNotificationCategory *category = [UNNotificationCategory categoryWithIdentifier:@"clear" actions:@[clearAction] intentIdentifiers:@[] options:UNNotificationCategoryOptionCustomDismissAction];//這裏增長一個 dismiss 的值複製代碼

Response handling

用戶點擊這些 actions 之後,是啓動 App、觸發鍵盤、清除通知或是有其餘的響應,這些所有隻須要實現協議 UNUserNotificationCenterDelegate 中的一個方法就能夠控制:

@interface ClassName () <UNUserNotificationCenterDelegate>
-(void)userNotificationCenter:(UNUserNotificationCenter *)center didReceiveNotificationResponse:(UNNotificationResponse *)response withCompletionHandler:(void (^)())completionHandler{

}複製代碼

其中的 response 包含如下內容:

其中的 trigger 能夠用來判斷是遠程推送仍是本地推送。

處理 response 舉例:

-(void)userNotificationCenter:(UNUserNotificationCenter *)center didReceiveNotificationResponse:(UNNotificationResponse *)response withCompletionHandler:(void (^)())completionHandler{

    NSString *categoryIdentifier = response.notification.request.content.categoryIdentifier;

    if ([categoryIdentifier isEqualToString:@"handle category"]) {//識別須要被處理的拓展

        if ([response.actionIdentifier isEqualToString:@"input text"]) {//識別用戶點擊的是哪一個 action

            //假設點擊了輸入內容的 UNTextInputNotificationAction 把 response 強轉類型
            UNTextInputNotificationResponse *textResponse = (UNTextInputNotificationResponse*)response;
              //獲取輸入內容
            NSString *userText = textResponse.userText;
            //發送 userText 給須要接收的方法
            [ClassName handleUserText: userText];
        }else{

        }

    }
    completionHandler();
}複製代碼

Service Extension


能夠在手機「接收到推送以後、展現推送以前」對推送進行處理,更改、替換原有的內容。

使用了這個玩意,大家公司原有發送推送的 payload 能夠徹底不變,而在客戶端對接收到的內容(只有一條字符串)進行加工,從而適配 iOS 10 的展現效果(標題+副標題+內容)。

「接收到推送以後、展現推送以前」:
  • 此時,你得到了一小段在後臺運行代碼的時間(也能夠用來幹別的壞事>。<,能夠偷偷的斷點下載大家 App 的更新包)
  • 而若是你更改推送內容出了錯誤,或者你調用什麼方法失敗了,那麼最終會正常的展現最初接收到的推送內容。

Potential uses

值得大家 App 充分發揮的是能夠作如下事情:

  • 端到端加密
  • 給推送展現內容添加附件(好比照片、背景音樂),使得內容更加豐富,就像從推送里拉出了一個網頁有木有!

不急,咱們先來介紹怎麼

添加 Service Extension

先在 Xcode 打開你的 App 工程,File - New - Target 而後添加這個:

而後會自動建立一個 UNNotificationServiceExtension 的子類 NotificationService,經過完善這個子類,來實現你的需求。

點開 NotificationService.m 會看到 2 個方法:

- (void)didReceiveNotificationRequest:(UNNotificationRequest *)request withContentHandler:(void (^)(UNNotificationContent * _Nonnull))contentHandler {
    self.contentHandler = contentHandler;
    self.bestAttemptContent = [request.content mutableCopy];

    self.bestAttemptContent.title = [NSString stringWithFormat:@"%@ [modified]", self.bestAttemptContent.title];

    self.contentHandler(self.bestAttemptContent);
}

- (void)serviceExtensionTimeWillExpire {
    self.contentHandler(self.bestAttemptContent);
}複製代碼
  • didReceiveNotificationRequest 讓你能夠在後臺處理接收到的推送,傳遞最終的內容給 contentHandler
  • serviceExtensionTimeWillExpire 在你得到的一小段運行代碼的時間即將結束的時候,若是仍然沒有成功的傳入內容,會走到這個方法,能夠在這裏傳確定不會出錯的內容,或者他會默認傳遞原始的推送內容
Example payload
{
  aps : {
    alert : "New Message",
    mutable-content : 1
  },
  encrypted-content : "#myencryptedcontent"
}複製代碼

首先須要添加 mutable-content : 1,這意味着此條推送能夠被 Service Extension 進行更改

同時能夠附加一條 encrypted-content,能夠提取該內容進行替換

- (void)didReceiveNotificationRequest:(UNNotificationRequest *)request withContentHandler:(void (^)(UNNotificationContent * _Nonnull))contentHandler {

    //用你的重編碼方法對該內容進行更改
    NSString *decryptedBody = [DecryptClass decrypt: request.content.userInfo[@"encrypted-content"]];

    //建立新的 content 並添加修改過的 body
    UNMutableNotificationContent *newContent = [UNMutableNotificationContent new];

    newContent.body = decryptedBody;

    //回調新的 content
    contentHandler(newContent);
}複製代碼
相關文章
相關標籤/搜索