iOS10系統(tǒng)里面的的通知與推送

介紹一下iOS10的通知新功能文黎,用戶體驗(yàn)的提升和開發(fā)者能夠發(fā)揮的地方非常多膏斤,使得iOS更具有競(jìng)爭(zhēng)力。

1.iOS 10通知系統(tǒng)支持Images, GIFs, Audio and Video類型

2.iOS 10推出Notification Service Extension與Notification Content Extension逮京,可以實(shí)現(xiàn)推送數(shù)據(jù)在展示前進(jìn)行下載更新、定制通知UI

3.iOS 10統(tǒng)一了通知類型盅安,具有時(shí)間間隔通知、地理位置通知和日歷通知


iOS里的通知擴(kuò)展

User Notifications Framework 介紹:

關(guān)系圖:


User Notifications Framework類關(guān)系圖

重點(diǎn)介紹:

UNUserNotificationCenter通知中心,用以管理通知的注冊(cè)、權(quán)限獲取和管理值戳、通知的刪除與更新本鸣,通過代理分發(fā)事件等筷黔。

UNNotification 通知實(shí)體,在UNUserNotificationCenter的代理回調(diào)事件中,告知App接收到一條通知,包含一個(gè)發(fā)起通知的請(qǐng)求UNNotificationRequest

UNNotificationRequest包含通知內(nèi)容UNNotificationContent和觸發(fā)器UNNotificationTrigger

UNNotificationContent 通知內(nèi)容墙歪,通知的title,sound贝奇,badge以及相關(guān)的圖像虹菲、聲音、視頻附件UNNotificationAttachment掉瞳,觸發(fā)打開App時(shí)候指定的LacnchImage等

UNNotificationResponse毕源,用戶在觸發(fā)了按鈕或者文本提交的UNNotificationAction的時(shí)候,會(huì)形成一個(gè)response陕习,通過通知中心的代理方法回調(diào)給App進(jìn)行處理或者是交給擴(kuò)展處理霎褐。

UNNotificationServiceExtension,是一個(gè)在接收到APNs服務(wù)器推送過來的數(shù)據(jù)進(jìn)行處理的服務(wù)擴(kuò)展该镣,如果App提供了服務(wù)擴(kuò)展瘩欺,那么APNs下發(fā)推送后在通知顯示觸發(fā)之前,會(huì)在UNNotificationServiceExtension內(nèi)接收到拌牲,此處有大約30秒的處理時(shí)間,開發(fā)者可以進(jìn)行一些數(shù)據(jù)下載歌粥、數(shù)據(jù)解密塌忽、更新等操作,然后交由而后的內(nèi)容擴(kuò)展(UNNotificationContentExtension)或者是App進(jìn)行觸發(fā)顯示

UNNotificationCategory,用以定義一組樣式類型失驶,該分類包含了某一個(gè)通知包含的交互動(dòng)作的組合土居,比如說UNNotificationRequest內(nèi)包含了一個(gè)Category標(biāo)示,那該通知就會(huì)以預(yù)定義好的交互按鈕或者文本框添加到通知實(shí)體上嬉探。

UNNotificationAttachment擦耀,通知內(nèi)容UNNotificationContent包含的附件,一般為圖片涩堤、視頻和音頻眷蜓,雖然iOS10的通知數(shù)據(jù)容量為4k,但依舊很少胎围,在添加了UNNotificationServiceExtension擴(kuò)展的情況下吁系,可以在服務(wù)里下載圖片德召,生成圖片、視頻等的本地緩存汽纤,UNNotificationAttachment根據(jù)緩存數(shù)據(jù)生成并添加到UNNotificationContent中上岗,交由UI顯示

UNNotificationAction,是通知中添加的action蕴坪,展示在通知欄的下方肴掷。默認(rèn)以的button樣式展示。有一個(gè)文本輸入的子類UNTextInputNotificationAction背传〈粽埃可以在點(diǎn)擊button之后彈出一個(gè)鍵盤,輸入信息续室。用戶點(diǎn)擊信息和輸入的信息可以在UNNotificationResponse中獲取

User Notifications UI Framework介紹:

關(guān)系圖:

User Notifications UI Framework類關(guān)系圖


10.?UNNotificationContentExtension<協(xié)議>栋烤,NotificationViewController實(shí)現(xiàn)該協(xié)議,可以獲得iOS展示自定義UI時(shí)候分發(fā)的UNNotification對(duì)象和用戶交互的Response

11.NotificationViewController挺狰,App添加Notification Content Extension擴(kuò)展的時(shí)候明郭,自動(dòng)生成的Controller,可以定義通知UI的主題部分丰泊,由StoryBoard指定設(shè)計(jì)

iOS10注冊(cè)通知薯定、回調(diào)處理

只簡(jiǎn)單介紹一下iOS10的通知注冊(cè)

UNUserNotificationCenter *center = [UNUserNotificationCenter currentNotificationCenter];

//?必須寫代理,不然無法監(jiān)聽通知的接收與點(diǎn)擊

center.delegate?=?self;

//設(shè)置預(yù)設(shè)好的交互類型瞳购,NSSet里面是設(shè)置好的UNNotificationCategory

[center?setNotificationCategories:[self?createNotificationCategoryActions]];

[center?getNotificationSettingsWithCompletionHandler:^(UNNotificationSettings?*?_Nonnull?settings)?{

if(settings.authorizationStatus==UNAuthorizationStatusNotDetermined)?{

[center?requestAuthorizationWithOptions:(UNAuthorizationOptionAlert?|?UNAuthorizationOptionBadge?|?UNAuthorizationOptionSound)?completionHandler:^(BOOL?granted,?NSError?*?_Nullable?error)?{

if(granted)?{

}else{

}

}];

}

else{

//do?other?things

}

}];

-?(void)application:(UIApplication?*)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData?*)deviceToken?{

//上傳token

}

-?(void)application:(UIApplication?*)application

didFailToRegisterForRemoteNotificationsWithError:(NSError?*)error?{

//獲取token失敗话侄,開發(fā)調(diào)試的時(shí)候需要關(guān)注,必要的情況下將其上傳到異常統(tǒng)計(jì)

}

//代理回調(diào)方法学赛,通知即將展示的時(shí)候

-?(void)userNotificationCenter:(UNUserNotificationCenter?*)center?willPresentNotification:(UNNotification?*)notification?withCompletionHandler:(void?(^)(UNNotificationPresentationOptions?options))completionHandler{

UNNotificationRequest?*request?=?notification.request;//?原始請(qǐng)求

NSDictionary?*?userInfo?=?notification.request.content.userInfo;//userInfo數(shù)據(jù)

UNNotificationContent?*content?=?request.content;//?原始內(nèi)容

NSString?*title?=?content.title;//?標(biāo)題

NSString?*subtitle?=?content.subtitle;//?副標(biāo)題

NSNumber?*badge?=?content.badge;//?角標(biāo)

NSString?*body?=?content.body;//?推送消息體

UNNotificationSound?*sound?=?content.sound;//?指定的聲音

//建議將根據(jù)Notification進(jìn)行處理的邏輯統(tǒng)一封裝年堆,后期可在Extension中復(fù)用~

completionHandler(UNNotificationPresentationOptionBadge|UNNotificationPresentationOptionSound|UNNotificationPresentationOptionAlert);//?回調(diào)block,將設(shè)置傳入

}

//用戶與通知進(jìn)行交互后的response盏浇,比如說用戶直接點(diǎn)開通知打開App变丧、用戶點(diǎn)擊通知的按鈕或者進(jìn)行輸入文本框的文本

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

UNNotificationRequest?*request?=?response.notification.request;//?原始請(qǐng)求

NSDictionary?*?userInfo?=?notification.request.content.userInfo;//userInfo數(shù)據(jù)

UNNotificationContent?*content?=?request.content;//?原始內(nèi)容

NSString?*title?=?content.title;//?標(biāo)題

NSString?*subtitle?=?content.subtitle;//?副標(biāo)題

NSNumber?*badge?=?content.badge;//?角標(biāo)

NSString?*body?=?content.body;//?推送消息體

UNNotificationSound?*sound?=?content.sound;

//在此,可判斷response的種類和request的觸發(fā)器是什么绢掰,可根據(jù)遠(yuǎn)程通知和本地通知分別處理痒蓬,再根據(jù)action進(jìn)行后續(xù)回調(diào)

}

以上就是iOS10的通知中心注冊(cè)和設(shè)置管理的過程,一下還有一些比較有用API:

//獲取在Pending狀態(tài)下待觸發(fā)的通知

-?(void)getPendingNotificationRequestsWithCompletionHandler:(void(^)(NSArray?*requests))completionHandler;

//移除未觸發(fā)的通知

-?(void)removePendingNotificationRequestsWithIdentifiers:(NSArray?*)identifiers;

-?(void)removeAllPendingNotificationRequests;

//?通知已經(jīng)觸發(fā)滴劲,但是還在操作系統(tǒng)的通知中心上攻晒,可以進(jìn)行查詢和刪除

-?(void)getDeliveredNotificationsWithCompletionHandler:(void(^)(NSArray?*notifications))completionHandler?__TVOS_PROHIBITED;

-?(void)removeDeliveredNotificationsWithIdentifiers:(NSArray?*)identifiers?__TVOS_PROHIBITED;

-?(void)removeAllDeliveredNotifications?__TVOS_PROHIBITED;


iOS10本地通知

UNMutableNotificationContent *content = [[UNMutableNotificationContent alloc] init];

content.title?=?@"\"Fly?to?the?moon\"";

content.subtitle?=?@"by?Neo";

content.body?=?@"the?wonderful?song?with?you~";

content.badge?=?@0;

NSString?*path?=?[[NSBundle?mainBundle]?pathForResource:@"image1"ofType:@"png"];

NSError?*error?=?nil;

//將本地圖片的路徑形成一個(gè)圖片附件,加入到content中

UNNotificationAttachment?*img_attachment?=?[UNNotificationAttachment?attachmentWithIdentifier:@"att1"URL:[NSURL?fileURLWithPath:path]?options:nil?error:&error];

if(error)?{

NSLog(@"%@",?error);

}

content.attachments?=?@[img_attachment];

//設(shè)置為@""以后班挖,進(jìn)入app將沒有啟動(dòng)頁(yè)

content.launchImageName?=?@"";

UNNotificationSound?*sound?=?[UNNotificationSound?defaultSound];

content.sound?=?sound;

//設(shè)置時(shí)間間隔的觸發(fā)器

UNTimeIntervalNotificationTrigger?*time_trigger?=?[UNTimeIntervalNotificationTrigger?triggerWithTimeInterval:10?repeats:NO];

NSString?*requestIdentifer?=?@"time?interval?request";

content.categoryIdentifier?=?@"";

UNNotificationRequest?*request?=?[UNNotificationRequest?requestWithIdentifier:requestIdentifer?content:content?trigger:time_trigger];

[[UNUserNotificationCenter?currentNotificationCenter]?addNotificationRequest:request?withCompletionHandler:^(NSError?*?_Nullable?error)?{

NSLog(@"%@",error);

}];


這里面的圖片附件后面再述鲁捏,過程上能感受到,通知數(shù)據(jù)部分在UNMutableNotificationContent中設(shè)置萧芙,附件UNNotificationAttachment也是在其中包含碴萧,categoryIdentifier為指定該通知對(duì)應(yīng)的交互樣式乙嘀,也就是前面設(shè)置的UNNotificationCategory的對(duì)象,后面再述破喻。然后創(chuàng)建觸發(fā)器虎谢,UNTimeIntervalNotificationTrigger,觸發(fā)器有很多種曹质,UNNotificationTrigger有四個(gè)子類:

1.UNPushNotificationTrigger婴噩,遠(yuǎn)程推送觸發(fā)器,一般是遠(yuǎn)程推送推過來的通知帶有這類觸發(fā)器

2.UNTimeIntervalNotificationTrigger羽德,時(shí)間間隔觸發(fā)器几莽,定時(shí)或者是重復(fù),在本地推送設(shè)置中有用

3.UNCalendarNotificationTrigger宅静,日歷觸發(fā)器章蚣,指定日期進(jìn)行通知

4.UNLocationNotificationTrigger,地理位置觸發(fā)器姨夹,指定觸發(fā)通知的條件是地理位置CLRegion這個(gè)類型纤垂。

觸發(fā)器和內(nèi)容最后形成UNNotificationRequest,一個(gè)通知請(qǐng)求磷账,本地通知的請(qǐng)求峭沦,直接交給通知中心進(jìn)行發(fā)送,發(fā)送成功后逃糟,該通知會(huì)按照觸發(fā)器的觸發(fā)條件進(jìn)行觸發(fā)吼鱼,并且會(huì)顯示到通知中心上,用戶可與指定的category交互方式與通知進(jìn)行交互

如下圖:


iOS10遠(yuǎn)程通知

遠(yuǎn)程通知與本地通知的流程一樣绰咽,只不過觸發(fā)器是UNPushNotificationTrigger菇肃,并且不需要形成request,又Provider Service發(fā)送給APNs到iOS以后生成取募,在代理回調(diào)的函數(shù)中獲取request

通知的代理回調(diào)

上面代碼有些代理回調(diào)函數(shù)琐谤,可以在這兩個(gè)代理回調(diào)函數(shù)里做一些事情

- (void)userNotificationCenter:(UNUserNotificationCenter *)center willPresentNotification:(UNNotification *)notification withCompletionHandler:(void (^)(UNNotificationPresentationOptions options))completionHandler{

//該回調(diào)函數(shù)是在通知條即將顯示之前調(diào)用的

if([request.trigger?isKindOfClass:[UNPushNotificationTrigger?class]])?{

//遠(yuǎn)程通知處理

}

if([request.trigger?isKindOfClass:[UNTimeIntervalNotificationTrigger?class]])?{

//時(shí)間間隔通知處理

}

if()?{

//加解密,數(shù)據(jù)下載,完成后調(diào)用completionHandler

}

else{

completionHandler(UNNotificationPresentationOptionBadge|UNNotificationPresentationOptionSound|UNNotificationPresentationOptionAlert);

}

}

如果你的App在前臺(tái)矛辕,一般在這個(gè)回調(diào)函數(shù)里做一些數(shù)據(jù)加解密、數(shù)據(jù)下載付魔,然后將下載的數(shù)據(jù)組裝成UNNotificationAttachment或者是根據(jù)通知里面的content里面的userinfo里與后端服務(wù)約定好的修改通知對(duì)應(yīng)的categoryId聊品,調(diào)用相應(yīng)的交互組件到通知上,completionHandler在你想要做的邏輯完成以后調(diào)用几苍。

如果App在前臺(tái)翻屈,你接收到通知,不想顯示系統(tǒng)提示框妻坝,想使用App 自定義的通知消息彈窗伸眶,可以在completionHandler回調(diào)的時(shí)候傳入的opinion不要帶上UNAuthorizationOptionAlert惊窖,然后直接彈自定義的彈窗就Ok。

注意:改回調(diào)函數(shù)僅僅用來處理數(shù)據(jù)和重新選擇交互方式厘贼,其他遠(yuǎn)程推送到達(dá)設(shè)備要做的業(yè)務(wù)邏輯界酒,最好不要在此回調(diào)函數(shù)觸發(fā),保持職責(zé)單一


//用戶與通知進(jìn)行交互后的response嘴秸,比如說用戶直接點(diǎn)開通知打開App毁欣、用戶點(diǎn)擊通知的按鈕或者進(jìn)行輸入文本框的文本

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

//在此,可判斷response的種類和request的觸發(fā)器是什么岳掐,可根據(jù)遠(yuǎn)程通知和本地通知分別處理凭疮,再根據(jù)action進(jìn)行后續(xù)回調(diào)

if([response.actionIdentifier?isEqualToString:@""])?{

}

//也可根據(jù)response?判斷是否是text文本輸入

if([response?isKindOfClass:[UNTextInputNotificationResponse?class]])?{

//該函數(shù)是在用戶點(diǎn)擊通知或者是與通知上面指定好的action進(jìn)行了交互回調(diào)的函數(shù),用戶觸發(fā)通知的業(yè)務(wù)邏輯最好放在此處

}

}

iOS10通知交互串述,UNNotificationAction與UNNotificationCategory

組合按鈕


組合文本框

首先說明的是执解,iOS10通知上的交互只有兩種,一種是Button一種是text纲酗,就算使用了iOS10 Notification Content Extension也不能添加自定義的按鈕或者其他交互組件衰腌,因?yàn)椴粫?huì)響應(yīng)。

-(NSSet *)createNotificationCategoryActions{

//定義按鈕的交互button?action

UNNotificationAction?*?likeButton?=?[UNNotificationAction?actionWithIdentifier:@"see1"title:@"I?love?it~"options:UNNotificationActionOptionAuthenticationRequired|UNNotificationActionOptionDestructive|UNNotificationActionOptionForeground];

UNNotificationAction?*?dislikeButton?=?[UNNotificationAction?actionWithIdentifier:@"see2"title:@"I?don't?care~"options:UNNotificationActionOptionAuthenticationRequired|UNNotificationActionOptionDestructive|UNNotificationActionOptionForeground];

//定義文本框的action

UNTextInputNotificationAction?*?text?=?[UNTextInputNotificationAction?actionWithIdentifier:@"text"title:@"How?about?it~?"options:UNNotificationActionOptionAuthenticationRequired|UNNotificationActionOptionDestructive|UNNotificationActionOptionForeground];

//將這些action帶入category

UNNotificationCategory?*?choseCategory?=?[UNNotificationCategory?categoryWithIdentifier:@"seeCategory"actions:@[likeButton,dislikeButton]?intentIdentifiers:@[@"see1",@"see2"]?options:UNNotificationCategoryOptionNone];

UNNotificationCategory?*?comment?=?[UNNotificationCategory?categoryWithIdentifier:@"seeCategory1"actions:@[text]?intentIdentifiers:@[@"text"]?options:UNNotificationCategoryOptionNone];

return[NSSet?setWithObjects:choseCategory,comment,nil];

}


在上面封裝了上面兩張圖中所示的兩個(gè)category組合耕姊,每一個(gè)category攜帶的action如圖所示桶唐,在通知中心初始化的時(shí)候設(shè)置app要支持的category。

UNUserNotificationCenter *center = [UNUserNotificationCenter currentNotificationCenter];

[center?setNotificationCategories:[self?createNotificationCategoryActions]];

UNNotificationAction 在初始化的時(shí)候需要定義UNNotificationActionOptions茉兰,這個(gè)UNNotificationActionOptions的意思是:

typedef NS_OPTIONS(NSUInteger, UNNotificationActionOptions) {

//?Whether?this?action?should?require?unlocking?before?being?performed.

//指定該動(dòng)作是否需要用戶解鎖驗(yàn)證身份

UNNotificationActionOptionAuthenticationRequired?=?(1?<<?0),

//?Whether?this?action?should?be?indicated?as?destructive.

//指定用戶執(zhí)行該動(dòng)作是否要將通知從iOS的通知中心移除尤泽,以防止處理過該通知以后重復(fù)處理

UNNotificationActionOptionDestructive?=?(1?<<?1),

//?Whether?this?action?should?cause?the?application?to?launch?in?the?foreground.

//指定通知action點(diǎn)擊后是否要進(jìn)入app到前臺(tái),如果到前臺(tái)规脸,這個(gè)對(duì)Notification?Content?Extension的自定義的通知UI有意義坯约,

//可以在Extension中處理用戶的點(diǎn)擊或者提交文字,那么就可以指定該action不需要進(jìn)入app莫鸭,

//UNNotificationActionOptionAuthenticationRequired這個(gè)就不要加入

UNNotificationActionOptionForeground?=?(1?<<?2),

}?__IOS_AVAILABLE(10.0)?__WATCHOS_AVAILABLE(3.0)?__TVOS_PROHIBITED;


測(cè)試我們預(yù)設(shè)好的category

-(void)timeLoacl{

UNMutableNotificationContent?*content?=?[[UNMutableNotificationContent?alloc]?init];

content.title?=?@"\"Fly?to?the?moon\"";

content.subtitle?=?@"by?Neo";

content.body?=?@"the?wonderful?song?with?you~";

content.badge?=?@0;

NSString?*path?=?[[NSBundle?mainBundle]?pathForResource:@"image1"ofType:@"png"];

NSError?*error?=?nil;

UNNotificationAttachment?*img_attachment?=?[UNNotificationAttachment?attachmentWithIdentifier:@"att1"URL:[NSURL?fileURLWithPath:path]?options:nil?error:&error];

if(error)?{

NSLog(@"%@",?error);

}

content.attachments?=?@[img_attachment];

//設(shè)置為@""以后闹丐,進(jìn)入app將沒有啟動(dòng)頁(yè)

content.launchImageName?=?@"";

UNNotificationSound?*sound?=?[UNNotificationSound?defaultSound];

content.sound?=?sound;

UNTimeIntervalNotificationTrigger?*time_trigger?=?[UNTimeIntervalNotificationTrigger?triggerWithTimeInterval:5?repeats:NO];

NSString?*requestIdentifer?=?@"time?interval?request";

//在此指定通知內(nèi)容的categoryIdentifier,就是上面我們預(yù)設(shè)好的category被因,一個(gè)category代表一種交互組合類型

content.categoryIdentifier?=?@"seeCategory1";

//????content.categoryIdentifier?=?@"seeCategory";

UNNotificationRequest?*request?=?[UNNotificationRequest?requestWithIdentifier:requestIdentifer?content:content?trigger:time_trigger];

[[UNUserNotificationCenter?currentNotificationCenter]?addNotificationRequest:request?withCompletionHandler:^(NSError?*?_Nullable?error)?{

NSLog(@"%@",error);

}];

}


在點(diǎn)擊某個(gè)按鈕或者是輸入了文本后卿拴,會(huì)在通知中心的代理回調(diào)函數(shù)里處理交互的response

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

//在此,可判斷response的種類和request的觸發(fā)器是什么梨与,可根據(jù)遠(yuǎn)程通知和本地通知分別處理堕花,再根據(jù)action進(jìn)行后續(xù)回調(diào)

if([response?isKindOfClass:[UNTextInputNotificationResponse?class]])?{

UNTextInputNotificationResponse?*?textResponse?=?(UNTextInputNotificationResponse*)response;

NSString?*?text?=?textResponse.userText;

//do?something

}

else{

if([response.actionIdentifier?isEqualToString:@"see1"])?{

//I?love?it~的處理

}

if([response.actionIdentifier?isEqualToString:@"see2"])?{

//I?don't?care~

[[UNUserNotificationCenter?currentNotificationCenter]?removeDeliveredNotificationsWithIdentifiers:@[response.notification.request.identifier]];

}

}

completionHandler();

}


這里需要根據(jù)response的類型或者根據(jù)actionIdentifier來區(qū)分用戶的交互結(jié)果來處理邏輯

iOS10通知附件UNNotificationAttachment,展示圖片粥鞋、Gif缘挽、Audio和Video

gif通知

-(void)timeLoaclWithImage{

UNMutableNotificationContent?*content?=?[[UNMutableNotificationContent?alloc]?init];

content.title?=?@"\"Fly?to?the?moon\"";

content.subtitle?=?@"by?Neo";

content.body?=?@"the?wonderful?song?with?you~";

content.badge?=?@0;

NSString?*path?=?[[NSBundle?mainBundle]?pathForResource:@"image1"ofType:@"png"];

NSError?*error?=?nil;

UNNotificationAttachment?*img_attachment?=?[UNNotificationAttachment?attachmentWithIdentifier:@"att1"URL:[NSURL?fileURLWithPath:path]?options:nil?error:&error];

if(error)?{

NSLog(@"%@",?error);

}

content.attachments?=?@[img_attachment];

//設(shè)置為@""以后,進(jìn)入app將沒有啟動(dòng)頁(yè)

content.launchImageName?=?@"";

UNNotificationSound?*sound?=?[UNNotificationSound?defaultSound];

content.sound?=?sound;

UNTimeIntervalNotificationTrigger?*time_trigger?=?[UNTimeIntervalNotificationTrigger?triggerWithTimeInterval:5?repeats:NO];

NSString?*requestIdentifer?=?@"time?interval?request";

content.categoryIdentifier?=?@"seeCategory1";

UNNotificationRequest?*request?=?[UNNotificationRequest?requestWithIdentifier:requestIdentifer?content:content?trigger:time_trigger];

[[UNUserNotificationCenter?currentNotificationCenter]?addNotificationRequest:request?withCompletionHandler:^(NSError?*?_Nullable?error)?{

NSLog(@"%@",error);

}];

}

UNNotificationAttachment需要指定image、gif壕曼、audio與video的文件路徑苏研,

+ (nullable instancetype)attachmentWithIdentifier:(NSString *)identifier URL:(NSURL *)URL options:(nullable NSDictionary *)options error:(NSError *__nullable *__nullable)error;

此處有一個(gè)options的字典,傳入的key有一下幾點(diǎn):

// Key to manually provide a type hint for the attachment. If not set the type hint will be guessed from the attachment's file extension. Value must be an NSString.

extern?NSString?*?const

//指定文件類型腮郊,查看文檔可以發(fā)現(xiàn)支持哪些文件

UNNotificationAttachmentOptionsTypeHintKey?__IOS_AVAILABLE(10.0)?__WATCHOS_AVAILABLE(3.0);

//?Key?to?specify?if?the?thumbnail?for?this?attachment?is?hidden.?Defaults?to?NO.?Value?must?be?a?boolean?NSNumber.

extern?NSString?*?const

//指定通知上是否顯示文件的縮略圖

UNNotificationAttachmentOptionsThumbnailHiddenKey?__IOS_AVAILABLE(10.0)?__WATCHOS_AVAILABLE(3.0);

//?Key?to?specify?a?normalized?clipping?rectangle?to?use?for?the?attachment?thumbnail.?Value?must?be?a?CGRect?encoded?using?CGRectCreateDictionaryRepresentation.

//指定縮略圖的切割比例

extern?NSString?*?const?UNNotificationAttachmentOptionsThumbnailClippingRectKey?__IOS_AVAILABLE(10.0)?__WATCHOS_AVAILABLE(3.0);

//?Key?to?specify?the?animated?image?frame?number?or?the?movie?time?to?use?as?the?thumbnail.

//?An?animated?image?frame?number?must?be?an?NSNumber.?A?movie?time?must?either?be?an?NSNumber?with?the?time?in?seconds?or?a?CMTime?encoded?using?CMTimeCopyAsDictionary.

extern?NSString?*?const

//影片切割時(shí)間

UNNotificationAttachmentOptionsThumbnailTimeKey?__IOS_AVAILABLE(10.0)?__WATCHOS_AVAILABLE(3.0);


從網(wǎng)上獲取gif下載后展示

-(void)timeLoaclWithGif{

NSURLSessionConfiguration?*config?=?[NSURLSessionConfiguration?defaultSessionConfiguration];

NSURLSession?*session?=?[NSURLSession?sessionWithConfiguration:config];

NSURLSessionDataTask?*task?=?[session?dataTaskWithURL:[NSURL?URLWithString:@"http://ww3.sinaimg.cn/large/006y8lVagw1faknzht671g30b408c1l2.gif"]?completionHandler:^(NSData?*?_Nullable?data,?NSURLResponse?*?_Nullable?response,?NSError?*?_Nullable?error)?{

if(!error)?{

//緩存到tmp文件夾

NSString?*path?=?[NSHomeDirectory()?stringByAppendingPathComponent:[NSString?stringWithFormat:@"tmp/%@att.%@",@([NSDate?date].timeIntervalSince1970),@"gif"]];

NSError?*err?=?nil;

[data?writeToFile:path?atomically:YES];

UNNotificationAttachment?*gif_attachment?=?[UNNotificationAttachment?attachmentWithIdentifier:@"attachment"URL:[NSURL?fileURLWithPath:path]?options:@{UNNotificationAttachmentOptionsThumbnailClippingRectKey:[NSValue?valueWithCGRect:CGRectMake(0,?0,?1,?1)]}?error:&err];

UNMutableNotificationContent?*content?=?[[UNMutableNotificationContent?alloc]?init];

content.title?=?@"\"Fly?to?the?moon\"";

content.subtitle?=?@"by?Neo";

content.body?=?@"the?wonderful?song?with?you~";

content.badge?=?@0;

NSError?*error?=?nil;

if(gif_attachment)?{

content.attachments?=?@[gif_attachment];

}

if(error)?{

NSLog(@"%@",?error);

}

//設(shè)置為@""以后摹蘑,進(jìn)入app將沒有啟動(dòng)頁(yè)

content.launchImageName?=?@"";

UNNotificationSound?*sound?=?[UNNotificationSound?defaultSound];

content.sound?=?sound;

UNTimeIntervalNotificationTrigger?*time_trigger?=?[UNTimeIntervalNotificationTrigger?triggerWithTimeInterval:1?repeats:NO];

NSString?*requestIdentifer?=?@"time?interval?request";

content.categoryIdentifier?=?@"seeCategory1";

UNNotificationRequest?*request?=?[UNNotificationRequest?requestWithIdentifier:requestIdentifer?content:content?trigger:time_trigger];

[[UNUserNotificationCenter?currentNotificationCenter]?addNotificationRequest:request?withCompletionHandler:^(NSError?*?_Nullable?error)?{

NSLog(@"%@",error);

}];

}

}];

[task?resume];

}


在文件緩存以后,發(fā)起本地通知伴榔。值得注意的一點(diǎn)是纹蝴,形成request發(fā)起以后,如果URL所代表的文件過大踪少,打開通知的交互界面的時(shí)候會(huì)非常慢塘安,甚至有時(shí)候會(huì)出現(xiàn)資源顯示不出來,還有一點(diǎn)是援奢,當(dāng)你在通知觸發(fā)展示以后兼犯,再通過request取出attachment文件的URL的時(shí)候,發(fā)現(xiàn)URL竟然發(fā)生了變化集漾,文件是緩存到一個(gè)叫pushstore的文件夾下切黔,這個(gè)在后面介紹 Notification Service Extension與Notification Content Extension 數(shù)據(jù)共享的時(shí)候會(huì)討論該問題。

iOS10 Notification Service Extension:

Notification Service Extension是Xcode8中加入眾多extension的其中一種具篇,Extension實(shí)際上是App提供了一個(gè)額外插件功能纬霞,以供iOS操作系統(tǒng)調(diào)用,與App是宿主關(guān)系驱显。

Notification Service Extension target


工作流程如下:

Notification Service Extension流程


Notification Service Extension的作用:

使得推送的數(shù)據(jù)在iOS系統(tǒng)展示之前诗芜,經(jīng)過App開發(fā)者的Extension,可以在不啟動(dòng)App的情況下埃疫,完成一些快捷操作邏輯伏恐,比如上面的例子,如果你是個(gè)社交App栓霜,可以在不啟動(dòng)App的情況下翠桦,直接點(diǎn)贊回復(fù),而不用打開App胳蛮,提高效率

雖然iOS10的推送數(shù)據(jù)包已經(jīng)達(dá)到4k销凑,但是對(duì)于一些圖片視頻gif還是無力的,有了Extension仅炊,可以在此下載完畢然后直接展示斗幼,豐富的圖片和視頻可以在此顯示

可以在此Extension中如果要完成1中所述的用戶行為操作,則必須加強(qiáng)安全性茂洒,服務(wù)端可以對(duì)推送的數(shù)據(jù)配合RSA算法用服務(wù)端的私鑰加密孟岛,在Extension中使用服務(wù)端私鑰解密,其實(shí)APNs從SSL數(shù)字安全證書到Json Web Token令牌督勺,已經(jīng)非常安全渠羞,但是大量的App使用第三方諸如JPush的推送服務(wù),來跟APNs交互智哀,業(yè)務(wù)數(shù)據(jù)跑在別人的管道上次询,當(dāng)然有所顧忌,所以瓷叫,這個(gè)地方加密的更多現(xiàn)實(shí)意義是防止業(yè)務(wù)數(shù)據(jù)被第三方服務(wù)商窺探屯吊。

新建一個(gè)target

addtarget_notification_service_extension


這點(diǎn)沒什么好說的,BundleID 就是宿主App的BundleID.這里設(shè)置的ProductName 摹菠,自動(dòng)生成盒卸。

注意使用組織名與team證書。

在新生成的NotificationService文件里有如下方法

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

self.contentHandler?=?contentHandler;

self.bestAttemptContent?=?[request.content?mutableCopy];

self.bestAttemptContent.title?=@"";

//?Modify?the?notification?content?here...

NSDictionary?*?userInfo?=?request.content.userInfo;

NSURLSessionConfiguration?*config?=?[NSURLSessionConfiguration?defaultSessionConfiguration];

NSURLSession?*session?=?[NSURLSession?sessionWithConfiguration:config];

//服務(wù)端與客戶端約定各種資源的url次氨,根據(jù)url資源進(jìn)行下載

NSString?*?imageUrl?=?[userInfo?objectForKey:@"imageUrl"];

NSString?*?gifUrl?=?[userInfo?objectForKey:@"gifUrl"];

NSString?*?typeString?;

NSURL?*?url;

if(imageUrl.length>0)?{

url?=?[NSURL?URLWithString:imageUrl];

typeString?=?@"jpg";

}

if(gifUrl.length>0)?{

url?=?[NSURL?URLWithString:gifUrl];

typeString?=?@"gif";

}

if(url)?{

NSURLRequest?*?urlRequest?=?[NSURLRequest?requestWithURL:url?cachePolicy:NSURLRequestUseProtocolCachePolicy?timeoutInterval:5];

//注意使用DownloadTask蔽介,這點(diǎn)會(huì)詳細(xì)說明

NSURLSessionDownloadTask?*task?=?[session?downloadTaskWithRequest:urlRequest?completionHandler:^(NSURL?*?_Nullable?location,?NSURLResponse?*?_Nullable?response,?NSError?*?_Nullable?error)?{

if(!error)?{

NSString?*path?=?[location.path?stringByAppendingString:[NSString?stringWithFormat:@".%@",typeString]];

NSError?*err?=?nil;

NSURL?*?pathUrl?=?[NSURL?fileURLWithPath:path];

[[NSFileManager?defaultManager]?moveItemAtURL:location?toURL:pathUrl?error:nil];

//下載完畢生成附件,添加到內(nèi)容中

UNNotificationAttachment?*resource_attachment?=?[UNNotificationAttachment?attachmentWithIdentifier:@"attachment"URL:pathUrl?options:nil?error:&err];

if(resource_attachment)?{

self.bestAttemptContent.attachments?=?@[resource_attachment];

}

if(error)?{

NSLog(@"%@",?error);

}

//設(shè)置為@""以后煮寡,進(jìn)入app將沒有啟動(dòng)頁(yè)

self.bestAttemptContent.launchImageName?=?@"";

UNNotificationSound?*sound?=?[UNNotificationSound?defaultSound];

self.bestAttemptContent.sound?=?sound;

//回調(diào)給系統(tǒng)

self.contentHandler(self.bestAttemptContent);

}

else{

self.contentHandler(self.bestAttemptContent);

}

}];

[task?resume];

}

else{

self.contentHandler(self.bestAttemptContent);

}

}

6- (void)serviceExtensionTimeWillExpire {

//?Called?just?before?the?extension?will?be?terminated?by?the?system.

//?Use?this?as?an?opportunity?to?deliver?your?"best?attempt"?at?modified?content,?otherwise?the?original?push?payload?will?be?used.

self.contentHandler(self.bestAttemptContent);

}

WWDC2016上的俄羅斯口音小伙上臺(tái)講Notification Service Extension的時(shí)候虹蓄,明確提到了”You will get a short execution time, which means this is not for long background running tasks.“,但實(shí)際測(cè)試過程中幸撕,Notification Service Extension非常容易崩潰crash和內(nèi)存溢出out of memory薇组。

更加坑的是debug運(yùn)行的時(shí)候和真機(jī)運(yùn)行的時(shí)候,Notification Service Extension性能表現(xiàn)是不一樣的坐儿,真機(jī)運(yùn)行的時(shí)候Notification Service Extension非常容易不起作用律胀,我做了幾次實(shí)驗(yàn),圖片稍大挑童,Notification Service Extension就崩潰了不起作用了累铅,而相同的debug調(diào)試環(huán)境下則沒問題,我覺得他應(yīng)該也提提這個(gè)站叼,比如說你下載資源的時(shí)候最好分段緩存下載娃兽,真機(jī)環(huán)境下NSURLSessionDataTask下載數(shù)據(jù)不好使,必須使用NSURLSessionDownloadTask才可以尽楔,這點(diǎn)很無奈投储。

iOS10 Notification Content Extension:

自定義通知UI


Notification Content Extension是另外一個(gè)擴(kuò)展,其內(nèi)容使用了UserNotificationsUIFramework阔馋,首先還是創(chuàng)建Notification Content Extension的target玛荞。

Notification Content Extension


此時(shí)會(huì)得到Notification Content Extension與MainInterface,storyboard里面含有一個(gè)試圖控制器呕寝,這個(gè)試圖控制器就是Notification點(diǎn)擊后中間顯示的那部分勋眯。這部分你可以自定義UI,注意的是該視圖控制器無法響應(yīng)交互控件,要想使用交互組件客蹋,就必須配合UNNotificationAction和category來對(duì)應(yīng)你的UI部分塞蹭,還有一點(diǎn),Notification Content Extension只能有一個(gè)控制器讶坯,所以你要想定制多種UI番电,就需要代碼判斷加載不同的View來實(shí)現(xiàn)。

自定義UI部分.png



在視圖控制器部分辆琅,代碼如下:

- (void)didReceiveNotification:(UNNotification *)notification {

self.label.text?=?notification.request.content.body;

UNNotificationAttachment?*?attachment?=?notification.request.content.attachments.firstObject;

if(attachment)?{

//開始訪問pushStore的存儲(chǔ)權(quán)限

[attachment.URL?startAccessingSecurityScopedResource];

NSData?*?data?=?[NSData?dataWithContentsOfFile:attachment.URL.path];

[attachment.URL?stopAccessingSecurityScopedResource];

self.imageView.image?=?[UIImage?imageWithData:data];

}

}

-?(void)didReceiveNotificationResponse:(UNNotificationResponse?*)response?completionHandler:(void?(^)(UNNotificationContentExtensionResponseOption?option))completion;{

if([response?isKindOfClass:[UNTextInputNotificationAction?class]])?{

//處理提交文本的邏輯

}

if([response.actionIdentifier?isEqualToString:@"see1"])?{

//處理按鈕3

}

if([response.actionIdentifier?isEqualToString:@"see2"])?{

//處理按鈕2

}

//可根據(jù)action的邏輯回調(diào)的時(shí)候傳入不同的UNNotificationContentExtensionResponseOption

completion(UNNotificationContentExtensionResponseOptionDismiss);

}


加入你有了Service Extension在前面下載好了圖片或者是視頻漱办,在自定義UI部分你想獲取,就可以通過UNNotificationAttachment * attachment = notification.request.content.attachments.firstObject;查找附件來獲取數(shù)據(jù)婉烟,但是必須注意娩井,前面提到的是,形成附件后似袁,文件的實(shí)際存儲(chǔ)被移到了pushStore的一個(gè)系統(tǒng)級(jí)別的緩存文件夾撞牢,此時(shí)需要調(diào)用NSURL在iOS8開始提供的兩個(gè)方法來獲取權(quán)限,提取數(shù)據(jù)叔营。

startAccessingSecurityScopedResource

stopAccessingSecurityScopedResource

點(diǎn)擊按鈕后屋彪,回調(diào)的方法是didReceiveNotificationResponse,在前面已經(jīng)演示過绒尊,在這畜挥,可以不用打開App進(jìn)而完成一些交互動(dòng)作。

Info.plist文件有一些設(shè)置需要表明



1.UNNotificationExtensionCategory改成數(shù)組婴谱,將你自定義的UI支持的categoryIdentifier一一放上蟹但,這樣,APNs推過來的數(shù)據(jù)中category包含哪個(gè)值谭羔,就調(diào)用哪個(gè)UNNotificationCategory設(shè)置好的actions交互組合

2.UNNotificationExtensionInitialContentSizeRatio华糖,自定義內(nèi)容的高度與寬度的比值,當(dāng)然也可以在ViewDidLoad中修改preferredContentSize來完成這一目標(biāo)

3.UNNotificationExtensionDefaultContentHidden瘟裸,決定是否在自定義UI下部顯示通知的原內(nèi)容客叉,默認(rèn)是顯示


Extensions 數(shù)據(jù)共享:

ServiceExtension與ContentExtension配合使用是非常棒的組合,在ServiceExtension中預(yù)先下載好數(shù)據(jù)话告,用戶點(diǎn)擊后在ContentExtension中直接展示兼搏,這樣交互會(huì)比較流暢,有一個(gè)問題是沙郭,如果你想在不打開App的時(shí)候使用自定義的action來與用戶交互佛呻,就必須加ContentExtension,因?yàn)橹挥兴芙邮沼脩酎c(diǎn)擊action的response病线,ServiceExtension是沒有的吓著。

如果你想使你的App在打開的時(shí)候訪問到這些數(shù)據(jù)鲤嫡,同樣可以根據(jù)UNNotificationAttachment來查找,但是更好的方案我個(gè)人覺得可以是用App Group來解決這個(gè)問題绑莺,當(dāng)然App Group的過多討論是偏離本文章的話題的泛范。

動(dòng)態(tài)配置通知交互:

上面我們可以知道,Notification可以配上很多category與action來自定義交互方式紊撕,但都是硬編碼來實(shí)現(xiàn),有時(shí)候我們想讓某個(gè)actionIdentifier對(duì)應(yīng)的按鈕文字改變一下赡突,或者是某個(gè)category對(duì)應(yīng)的actions改變一下对扶,來滿足運(yùn)營(yíng)活動(dòng)的靈活性,需要思考動(dòng)態(tài)配置UNNotificationCategory和UNNotificationAction的問題惭缰。有如下這個(gè)方案浪南,可以把UNNotificationCategory和UNNotificationAction做成配置文件,如下:

{

"NotificationConfig":?{

"UNNotificationCategory":?{

"seeCategory":?{

"identifier":"seeCategory",

"actions":?["see1","see2"],

"options":?0

}

},

"UNNotificationAction":?{

"see1":?{

"identifier":"see1",

"title":"I?love?it~",

"options":?4

},

"see2":?{

"identifier":"see2",

"title":"I?dont't?care~",

"options":?4

}

}

}

}

然后在通知中心設(shè)置categorys的時(shí)候

-(NSSet *)createNotificationCategoryActions{

if(HBCONFIGOBJECT.moduleConfig.userNotificationConfig)?{

//讀取json文件

NSDictionary?*?notificationConfig;

NSDictionary?*?UNNotificationCategorys?=[notificationConfig?objectForKey:@"UNNotificationCategory"];

NSDictionary?*?UNNotificationActions?=?[notificationConfig?objectForKey:@"UNNotificationAction"];

NSMutableSet?*?set?=?[NSMutableSet?set];

for(NSString?*?categoryKeyinUNNotificationCategorys.allKeys)?{

NSDictionary?*?cateDict?=?UNNotificationCategorys[categoryKey];

NSString?*?cateId?=?[cateDict?objectForKey:@"identifier"];

NSArray?*?cateActions?=?[cateDict?objectForKey:@"actions"];

NSNumber?*?cateOptions?=?[cateDict?objectForKey:@"options"];

NSMutableArray?*?actionsArr?=?[[NSMutableArray?alloc]init];

for(NSString?*?actionKeyincateActions)?{

NSDictionary?*?actionDict?=?[UNNotificationActions?objectForKey:actionKey];

if(actionDict)?{

NSString?*?actionId?=?[actionDict?objectForKey:@"identifier"];

NSString?*?actionTitle?=?[actionDict?objectForKey:@"title"];

NSNumber?*?actionOption?=?[actionDict?objectForKey:@"options"];

UNNotificationAction?*?action?=?[UNNotificationAction?actionWithIdentifier:actionId?title:actionTitle?options:actionOption.unsignedIntegerValue];

[actionsArr?addObject:action];

}

}

UNNotificationCategory?*?category?=?[UNNotificationCategory?categoryWithIdentifier:cateId?actions:actionsArr?intentIdentifiers:cateActions?options:cateOptions.unsignedIntegerValue];

[set?addObject:category];

}

returnset;

}

else

{

returnnil;

}

}

這樣就可以任意組合category和actions了漱受,json文件可以在App內(nèi)部做全量更新络凿,在運(yùn)營(yíng)活動(dòng)之前,就下發(fā)好給客戶端昂羡。

有一個(gè)問題是絮记,ContentExtension需要在plist里指定category,所以建議將categoryIdentifier按照一定格式進(jìn)行序列化取名虐先,在ContentExtension提前寫入0-10等很多的category怨愤,方面動(dòng)態(tài)配置的時(shí)候取用。

運(yùn)營(yíng)如何使用通知與推送:

Instagram上iOS10通知的使用


iOS10推出了十分出色的通知以后蛹批,我經(jīng)常使用的Instagram撰洗、Twitter、Facebook等都及時(shí)跟進(jìn)腐芍,做出了非常好的交互差导,我希望微信團(tuán)隊(duì)能在通知上快速預(yù)覽內(nèi)容和回復(fù)上面增加此功能。

其實(shí)猪勇,這個(gè)話題是我非常想討論的设褐,作為工程師,有得天獨(dú)厚的條件深刻理解最新最前沿的技術(shù)泣刹,那么络断,這些技術(shù)如何產(chǎn)生現(xiàn)實(shí)意義,如何使用项玛,在這點(diǎn)上貌笨,工程師是非常具有優(yōu)勢(shì)的,假如你了解硅谷的工程師文化襟沮,你就會(huì)發(fā)現(xiàn)锥惋,硅谷的科技公司很少有產(chǎn)品經(jīng)理的昌腰,大部分出色的功能和優(yōu)質(zhì)的用戶體驗(yàn)是由工程師打造的,詳情可以參考MacTalk的一篇文章《硅谷不需要產(chǎn)品經(jīng)理》膀跌。

真正的工程師文化遭商,不像國(guó)內(nèi)的開發(fā)者認(rèn)為的是在某一技術(shù)領(lǐng)域非常深的理論研究,在國(guó)外的開發(fā)者眼里捅伤,真正的工程師文化是一群善于創(chuàng)造并且有巨大的改變現(xiàn)實(shí)世界的能力的工程師文化劫流,話說回來,現(xiàn)如今丛忆,移動(dòng)端的工程師很多很多祠汇,像本篇這樣的技術(shù)介紹類的文章數(shù)不勝數(shù),技術(shù)水平差不多的工程師非常之多熄诡,你如何脫穎而出可很?這是你需要思考的,我的建議是凰浮,作為工程師我抠,跟你一樣熟悉API和開發(fā)技術(shù)的人多了去了,但是如果你能知道技術(shù)在各種場(chǎng)景下的最佳使用方案袜茧,并且能切實(shí)改變現(xiàn)實(shí)情況菜拓,舉個(gè),iOS10的通知你是了解笛厦,但怎么用才能更好的提升你的App的用戶體驗(yàn)尘惧?更好的提高你的App在某些功能場(chǎng)景下的用戶使用成本?怎么樣才能讓運(yùn)營(yíng)活動(dòng)通過通知提高活躍度递递?如果你有這樣的各種解決方案喷橙,你就是勝出者~比方說,你是社交類App的開發(fā)者登舞,你有一堆技術(shù)解決方案在手贰逾,能夠切實(shí)提升用戶體驗(yàn)的,你是電商類開發(fā)者菠秒,你有通知的技術(shù)使用解決方案能夠更好的支撐運(yùn)營(yíng)活動(dòng)的疙剑。

那iOS10的通知能想到哪些使用場(chǎng)景呢?

運(yùn)營(yíng)活動(dòng)可以配上活動(dòng)海報(bào)或者是動(dòng)圖海報(bào)践叠,在用戶點(diǎn)擊好能更好的查看運(yùn)營(yíng)活動(dòng)詳情

即時(shí)通訊類的App可以通過自定義ContentExtension來在通知上完成回復(fù)消息

比方說言缤,你有個(gè)秒殺活動(dòng),通知一下來禁灼,用戶立馬可以通過iOS10的通知交互完成秒殺預(yù)定管挟,然后再啟動(dòng)App慢慢付款~這個(gè)用戶體驗(yàn)的提升那是相當(dāng)巨大的

比方說,你可以通過推送收取一些用戶對(duì)某個(gè)活動(dòng)或者新版本的反饋意見弄捕?使用TextAction來做

你是否可可以發(fā)個(gè)可視化的賬單給用戶僻孝,在自定義UI上顯示导帝?

……

再來說說技術(shù)方案吧,上面的場(chǎng)景要想實(shí)現(xiàn)穿铆,有個(gè)問題是您单,通知的ServiceExtension和ContentExtension拿到了用戶反饋的信息,那這些信息該怎么辦~方案如下:

最新通知交互方案


這是個(gè)簡(jiǎn)單的單推交互方案荞雏,其中需要由動(dòng)態(tài)化配置Category與actions支撐虐秦,同時(shí)要做好加解密工作。

以上凤优,就是本次討論通知和推送的主要內(nèi)容

文章轉(zhuǎn)載地址www.cocoachina.com/ios/20170126/18618.html

轉(zhuǎn)載Demo地址github.com/Neojoke/UntificationLearn

可以使用工具來測(cè)試推送悦陋,工具地址:https://github.com/noodlewerk/NWPusher

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市别洪,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌柳刮,老刑警劉巖挖垛,帶你破解...
    沈念sama閱讀 219,366評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異秉颗,居然都是意外死亡痢毒,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,521評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門蚕甥,熙熙樓的掌柜王于貴愁眉苦臉地迎上來哪替,“玉大人,你說我怎么就攤上這事菇怀∑静埃” “怎么了?”我有些...
    開封第一講書人閱讀 165,689評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵爱沟,是天一觀的道長(zhǎng)帅霜。 經(jīng)常有香客問我,道長(zhǎng)呼伸,這世上最難降的妖魔是什么身冀? 我笑而不...
    開封第一講書人閱讀 58,925評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮括享,結(jié)果婚禮上搂根,老公的妹妹穿的比我還像新娘。我一直安慰自己铃辖,他們只是感情好剩愧,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,942評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著娇斩,像睡著了一般隙咸。 火紅的嫁衣襯著肌膚如雪沐悦。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,727評(píng)論 1 305
  • 那天五督,我揣著相機(jī)與錄音藏否,去河邊找鬼。 笑死充包,一個(gè)胖子當(dāng)著我的面吹牛副签,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播基矮,決...
    沈念sama閱讀 40,447評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼淆储,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了家浇?” 一聲冷哼從身側(cè)響起本砰,我...
    開封第一講書人閱讀 39,349評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎钢悲,沒想到半個(gè)月后点额,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,820評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡莺琳,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,990評(píng)論 3 337
  • 正文 我和宋清朗相戀三年还棱,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片惭等。...
    茶點(diǎn)故事閱讀 40,127評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡珍手,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出辞做,到底是詐尸還是另有隱情琳要,我是刑警寧澤,帶...
    沈念sama閱讀 35,812評(píng)論 5 346
  • 正文 年R本政府宣布秤茅,位于F島的核電站焙蹭,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏嫂伞。R本人自食惡果不足惜孔厉,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,471評(píng)論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望帖努。 院中可真熱鬧斑唬,春花似錦轴或、人聲如沸秧秉。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,017評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)匙监。三九已至凡橱,卻和暖如春小作,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背稼钩。 一陣腳步聲響...
    開封第一講書人閱讀 33,142評(píng)論 1 272
  • 我被黑心中介騙來泰國(guó)打工顾稀, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人坝撑。 一個(gè)月前我還...
    沈念sama閱讀 48,388評(píng)論 3 373
  • 正文 我出身青樓静秆,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親巡李。 傳聞我的和親對(duì)象是個(gè)殘疾皇子抚笔,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,066評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容

  • 前言在今年6月14號(hào)蘋果WWDC開發(fā)者大會(huì)上,蘋果帶來了新的iOS系統(tǒng)——iOS 10侨拦。蘋果為iOS 10帶來了十...
    sky_kYU閱讀 3,465評(píng)論 5 12
  • iOS 10 came 在今年 6月14號(hào) 蘋果開發(fā)者大會(huì) WWDC 2016 之后殊橙,筆者趕緊就去 apple 的...
    pikacode閱讀 8,505評(píng)論 30 39
  • 二、壇廟 1.祭祀 (1)祭祀的含義祭祀是對(duì)人們向自然狱从、神靈膨蛮、鬼魂、祖先矫夯、繁殖等表示一種意向的活動(dòng)儀式的通稱鸽疾,它的...
    槑槑啊閱讀 682評(píng)論 0 1
  • 每次看到哪個(gè)大神一年讀了一兩百本書递沪,或者一天讀完一本書時(shí),我們總會(huì)滿腹雞血综液,暗下決心款慨,從今天開始我也要一兩天讀完一...
    青芒護(hù)膚閱讀 882評(píng)論 3 3