雖然通知經(jīng)常被過(guò)度使用章钾,但是通知確實(shí)是一種獲得用戶關(guān)注和通知他們需要更新或行動(dòng)的有效方式。iOS 10有了新的通知艺演,如新消息却紧、商業(yè)信息和時(shí)間表的變化。在本教程中,我將向你展示如何使用通知在你的iOS應(yīng)用程序胎撤,并且顯示iOS 10引入了新特性晓殊。開(kāi)發(fā)iOS 10推送通知你需要最新版本的Xcode,Xcode 8測(cè)試版,這些目前都是可下載的,在下載頁(yè)面伤提。
你可以去Github下載本教程的整個(gè)工程巫俺。
開(kāi)始
在Xcode中啟用推送通知是很容易的,但你需要幾個(gè)步驟。
創(chuàng)建一個(gè)新的工程肿男,給它起一個(gè)唯一的Bundle Identifier.
當(dāng)您已經(jīng)創(chuàng)建了project,去Project Settings頁(yè)選擇Capabilities欄介汹。打開(kāi)推送通知,如下所示。
注意: 如果你是蘋(píng)果的付費(fèi)開(kāi)發(fā)者成員舶沛,你就能看到推送通知功能這一欄嘹承。
去Developer Account這一欄,從左側(cè)的菜單欄中選擇證書(shū)如庭,IDs叹卷,和描述文件,然后選擇App IDs在Identifiers欄中柱彻。找到已經(jīng)創(chuàng)建的App的名稱豪娜,在服務(wù)列表中選中。注意哟楷,有兩個(gè)可配置狀態(tài)的推送通知瘤载。
不要關(guān)閉這個(gè)網(wǎng)頁(yè),你很快就會(huì)回來(lái)的。
發(fā)送通知
在本文中,我將使用Pusher發(fā)送推送通知卖擅。您還可以使用其他的解決方案如Houston鸣奔。無(wú)論哪種方式,發(fā)送一個(gè)通知,你都需要一個(gè)證書(shū)墨技。
去創(chuàng)建一個(gè)證書(shū),打開(kāi)Keychain Access挎狸,從證書(shū)認(rèn)證菜單中選擇Keychain Access -> Certificate Assistant -> Request a Certificate扣汪。
填寫(xiě)表單并單擊Continue。確保你選擇保存到了磁盤(pán)锨匆。
返回到開(kāi)發(fā)者賬戶的網(wǎng)頁(yè)崭别。你可以為你的App IDs生成開(kāi)發(fā)(調(diào)試)證書(shū)或發(fā)布證書(shū)。
之后在選擇右側(cè)的申請(qǐng)恐锣,在底部茅主,單擊編輯。在推送通知部分,單擊創(chuàng)建開(kāi)發(fā)(調(diào)試)證書(shū)土榴。
在需要時(shí),從Keychain,繼續(xù)上傳生成證書(shū)請(qǐng)求诀姚。
現(xiàn)在你已經(jīng)創(chuàng)建了證書(shū),可以下載它。打開(kāi)下載的文件安裝它玷禽。
下載并運(yùn)行Pusher赫段。這個(gè)程序的頂部需要填入一個(gè)推送的證書(shū)。為它位于你的鑰匙鏈,OS X將詢問(wèn)是否允許Pusher訪問(wèn)證書(shū)矢赁。
第二個(gè)字段需要device token,你會(huì)在下一步中得打它糯笙。
收到通知
是時(shí)候敲代碼了。收到通知的設(shè)備必須注冊(cè)到蘋(píng)果推送通知服務(wù)(APNS)坯台。在應(yīng)用啟動(dòng)的時(shí)候你要發(fā)送一個(gè)唯一的token炬丸。
打開(kāi)AppDelegate.swift然后添加如下方法。
注意:該代碼是基于Swift3.0蜒蕾。語(yǔ)法可能看起來(lái)不同于你之前使用過(guò)的。
func registerPushNotifications() {
DispatchQueue.main.async {
let settings = UIUserNotificationSettings(types:[.badge, .sound, .alert], categories: nil)
UIApplication.shared().registerUserNotificationSettings(settings)
}
}
我之后會(huì)解釋焕阿,在這個(gè)設(shè)置中你會(huì)收到指定的通知類型咪啡。調(diào)用這個(gè)方法在應(yīng)用程序啟動(dòng)的的文件里。
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [NSObject: AnyObject]?) -> Bool {
registerPushNotifications()
return true
}
此時(shí),應(yīng)用程序?qū)⒆詣?dòng)彈出一個(gè)Alert暮屡,詢問(wèn)用戶是否要收到該通知撤摸。
通知必須被注冊(cè),才能發(fā)送褒纲,而是否接受通知?jiǎng)t需要用戶批準(zhǔn)准夷。UIApplicationDelegate方法處理響應(yīng)。
func application(_ application: UIApplication, didRegister notificationSettings: UIUserNotificationSettings) {
if notificationSettings.types != UIUserNotificationType() {
application.registerForRemoteNotifications()
}
}
首先檢查用戶授予權(quán)限,然后調(diào)用該方法注冊(cè)遠(yuǎn)程通知莺掠。當(dāng)請(qǐng)求完成后者將調(diào)用另一個(gè)代理方法衫嵌。這個(gè)方法響應(yīng)包含一個(gè)device token,你可以打印進(jìn)行調(diào)試。在發(fā)送推送通知來(lái)識(shí)別設(shè)備需要這個(gè)device token彻秆。
如果出現(xiàn)錯(cuò)誤,調(diào)用下面的方法楔绞。
func application(_ application: UIApplication, didFailToRegisterForRemoteNotificationsWithError error: NSError) {
print("Registration failed!")
}
注意:重要的是在應(yīng)用程序啟動(dòng)時(shí)要調(diào)用registerUserNotificationSettings,因?yàn)橛脩艨梢愿淖儥?quán)限的設(shè)置结闸。同樣registerForRemoteNotifications也是很重要的,因?yàn)橛行﹫?chǎng)景device token可以改變那么通知將不再發(fā)送。
到目前為止,這足以讓你收到一個(gè)簡(jiǎn)單的通知酒朵。
通知內(nèi)容
通過(guò)不同的通知內(nèi)容桦锄,有不同的方式來(lái)使一個(gè)App來(lái)收到不同類型的通知,這些通知內(nèi)容包括應(yīng)用程序通知用戶的信息蔫耽,或者用戶自定義的信息结耀。
給用戶發(fā)送通知,使用JSON格式匙铡,這個(gè)格式本身包含一個(gè)字典,對(duì)應(yīng)aps的key图甜。在這第二個(gè)字典你指定載內(nèi)容和key。
最常見(jiàn)的是:
向用戶顯示的通知消息慰枕。這是一個(gè)簡(jiǎn)單的字符串,或一個(gè)字典key和標(biāo)題一樣,正文等等具则。
接收到通知的聲音。它可以是一個(gè)定制的聲音,或一個(gè)系統(tǒng)的聲音具帮。
應(yīng)用圖標(biāo)右上角的角標(biāo)個(gè)數(shù)博肋。將其設(shè)置為0,消除角標(biāo)。
有效的內(nèi)容蜂厅。使用值1發(fā)送一個(gè)無(wú)聲的通知給用戶匪凡。它不會(huì)播放任何聲音,或任何角標(biāo)設(shè)置,但是當(dāng)通知被喚醒,應(yīng)用將與服務(wù)器進(jìn)行溝通掘猿。
本教程的一個(gè)簡(jiǎn)單的通知內(nèi)容:
{
"aps": {
"alert": {
"title":"Hello! :)",
"body":"App closed..."
},
"badge":1,
"sound":"default"
}
}
應(yīng)用程序的生命周期
拷貝device token粘貼在Pusher的token部分病游,拷貝這個(gè)JSON對(duì)象在Pusherd的payload部分。
試著發(fā)送第一個(gè)通知稠通。如果設(shè)備的屏幕被鎖定衬衬,它將看起來(lái)如下,但什么都不會(huì)發(fā)生,當(dāng)用戶點(diǎn)擊了這個(gè)通知視圖。
接受通知改橘,你需要添加新的方法:
private func getAlert(notification: [NSObject:AnyObject]) -> (String, String) {
let aps = notification["aps"] as? [String:AnyObject]
let alert = aps?["alert"] as? [String:AnyObject]
let title = alert?["title"] as? String
let body = alert?["body"] as? String
return (title ?? "-", body ?? "-")
}
這將返回收到的通知標(biāo)題和正文,如果結(jié)構(gòu)是相同的滋尉。
func notificationReceived(notification: [NSObject:AnyObject]) {
let viewController = window?.rootViewController
let view = viewController as? ViewController
view?.addNotification(
title: getAlert(notification: notification).0,
body: getAlert(notification: notification).1)
}
這個(gè)方法將在應(yīng)用程序主要視圖UITableView內(nèi)添加一行(參見(jiàn)ViewController的完整項(xiàng)目代碼)。
我測(cè)試了三個(gè)案例的推送通知:
- 當(dāng)應(yīng)用關(guān)閉時(shí)
如果用戶打開(kāi)應(yīng)用程序的通知,調(diào)用didFinishLaunchingWithOptions方法更新,如下:
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [NSObject: AnyObject]?) -> Bool {
// Override point for customization after application launch.
application.applicationIconBadgeNumber = 0; // Clear badge when app launches
// Check if launched from notification
if let notification = launchOptions?[UIApplicationLaunchOptionsRemoteNotificationKey] as? [String: AnyObject] {
window?.rootViewController?.present(ViewController(), animated: true, completion: nil)
notificationReceived(notification: notification)
} else {
registerPushNotifications()
}
return true
}
假設(shè)用戶已經(jīng)看過(guò)了這個(gè)通知飞主,那么角標(biāo)就被清除了狮惜。然后,檢查應(yīng)用程序是從圖標(biāo)打開(kāi)還是通過(guò)通知打開(kāi)的。在第一種情況下,調(diào)用registerPushNotifications()方法然后繼續(xù)之前的流程碌识。如果應(yīng)用是通過(guò)打開(kāi)通知的方式運(yùn)行,則調(diào)用自定義notificationReceived方法來(lái)添加行碾篡。
- 當(dāng)應(yīng)用運(yùn)行在前臺(tái)時(shí)
如果用戶正在使用應(yīng)用程序,這意味著應(yīng)用程序在前臺(tái),接受通知的方法如下。在這個(gè)通知的方法中加入對(duì)tableView的處理:
func application(_ application: UIApplication, didReceiveRemoteNotification userInfo: [NSObject : AnyObject]) {
notificationReceived(notification: userInfo)
}
注意:在這種情況下,通知將不會(huì)發(fā)出聲音筏餐。
- 當(dāng)應(yīng)用運(yùn)行在后臺(tái)時(shí)
在這種情況下,我添加了一個(gè)方法來(lái)清除角標(biāo)數(shù)目开泽。通知的處理和應(yīng)用程序在前臺(tái)的處理是一樣的。
func applicationWillEnterForeground(_ application: UIApplication) {
application.applicationIconBadgeNumber = 0; // Clear badge when app is or resumed
}
最后胖烛,這個(gè)列表中有三行來(lái)自通知的內(nèi)容眼姐。
最后
隨著iOS 10的通知诅迷,開(kāi)發(fā)者有了更多比之前有趣的機(jī)會(huì)和不曾有的交互權(quán)限。我希望本教程中關(guān)于如何使用通知能幫助你更好的理解通知是如何工作的众旗。