UILocalNotification本地通知

什么是本地通知
  • 顧名思義,不需要聯(lián)網(wǎng)就能發(fā)送的通知,無需服務(wù)器.
本地通知使用場景
  • 常用來提醒用戶完成一些定時任務(wù),比如
  • 清理垃圾,買衣服,看電影,睡覺,運動等

1. 如何發(fā)送本地推送通知


推送通知也屬于UI的一部分,所以推送通知對象是以UI開頭

方法送通知的代碼方法控制器的-touchesBegan: withEvent: 中測試,比較合適,放到viewDidLoad方法,用戶的注冊請求還沒有完成方法就調(diào)用了

  • 創(chuàng)建本地通知
//  創(chuàng)建本地通知對象
    UILocalNotification *ln = [[UILocalNotification alloc] init];
  • 設(shè)置本地通知屬性(推薦一個一個屬性測試運行)
//  1.設(shè)置通知的內(nèi)容(如果此屬性不設(shè)置是不會發(fā)送通知的)
    ln.alertBody = @"小明,你媽叫你回家吃飯了!";
//  2.設(shè)置通知觸發(fā)的開始時間
    ln.fireDate = [NSDate dateWithTimeIntervalSinceNow:3];
//  3.設(shè)置重復(fù)通知的時間,間隔
    ln.repeatInterval = NSCalendarUnitSecond;
//  4.設(shè)置重復(fù)執(zhí)行使用日歷(用戶設(shè)置的日歷)
    ln.repeatCalendar = [NSCalendar  currentCalendar];
//    NSString * const NSGregorianCalendar; 公歷
//    NSString * const NSChineseCalendar; 農(nóng)歷
//    ln.repeatCalendar = [[NSCalendar alloc] initWithCalendarIdentifier:NSChineseCalendar];
//  5.設(shè)置應(yīng)用圖標右上角的數(shù)字
    ln.applicationIconBadgeNumber = 3;
//  6.設(shè)置點擊推送通知進入界面的時候顯示,加載圖片
    ln.alertLaunchImage = @"";
//  7 設(shè)置通知的音效(只有真機有效)
    local.soundName = UILocalNotificationDefaultSoundName;
//  8 設(shè)置一些額外信息
    local.userInfo = @{@"QQ":@"55555",@"info":@"約了沒"};

//  iOS8.0 以后新增屬性
//  ************************************
//  1.設(shè)置區(qū)域,進入或離開某個區(qū)域的時候觸發(fā)
//    CLLocationCoordinate2D coordinate = CLLocationCoordinate2DMake(40.1,106.1);
//    ln.region = [[CLCircularRegion alloc] initWithCenter:coordinate radius:10.0 identifier:@"ab"];
//  2.設(shè)置進入或離開某個區(qū)域只執(zhí)行一次
//    ln.regionTriggersOnce = YES;
//  ***************************************

//  iOS8.2 新增屬性
//    ln.alertTitle = @"通知標題";


  • 使用應(yīng)用[UIApplication]調(diào)度本地通知
//  讓應(yīng)用調(diào)度通知
    [[UIApplication sharedApplication] scheduleLocalNotification:ln];

2. 本地推送通知頁面跳轉(zhuǎn)


無論應(yīng)用是在前臺,后臺還是已經(jīng)關(guān)閉都能如期接收到本地通知,但是當用戶點擊通知進入應(yīng)用的時候,我們需要根據(jù)不同情況,進行處理

AppDelegate本地通知代理方法


/**
 *  一旦接收到本地通知就會調(diào)用該方法
 *  注意這個方法:應(yīng)用在前臺也會調(diào)用
 *  @param application  應(yīng)用
 *  @param notification 本地通知對象
 */
- (void) application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification
{

//  當應(yīng)用在前臺時候什么都不做
    if (application.applicationState == UIApplicationStateActive) {
        return;
    }

//  當應(yīng)用不再前臺的時候才去跳轉(zhuǎn),這樣用戶體檢更好
    UITabBarController *tbVc = (UITabBarController *)application.keyWindow.rootViewController;
    tbVc.selectedIndex = 1;

}

但是當應(yīng)用已經(jīng)退出的時候,點擊通知進入本應(yīng)用時候,不在調(diào)用application:didReceiveLocalNotification:的代理方法,難道當應(yīng)用退出后,用戶再進入應(yīng)用我們就不再跳轉(zhuǎn)指定界面了嗎?為了更好用戶體驗,我此時也應(yīng)該讓應(yīng)用跳轉(zhuǎn)到指定的界面,怎么才能實現(xiàn)這個功能呢? 我們知道當應(yīng)用程序啟動的時候一定會調(diào)用application: didFinishLaunchingWithOptions:的代理方法,在這里我們能拿到本地通知信息,也可以跳轉(zhuǎn)相應(yīng)的界面

//  如果是點擊本地通知進來的那么launchOptions[UIApplicationLaunchOptionsLocalNotificationKey]就會有內(nèi)容
    if(launchOptions[UIApplicationLaunchOptionsLocalNotificationKey])
    {
        //頁面跳轉(zhuǎn)
        UITabBarController *tbVc = (UITabBarController *)self.window.rootViewController;
        tbVc.selectedIndex = 1;
    }

一個應(yīng)用可能要各種不同的通but知,點擊不同的通知可以跳轉(zhuǎn)不同界面,這個有該怎么做呢?

  1. 在發(fā)送通知時候,設(shè)置userInfo屬性
//  7.設(shè)置應(yīng)用信息
    ln.userInfo = @{@"pageKey":@"friend"};
  1. 在AppDelegate本地通知代理方法中進行判斷

/**
 *  一旦接收到本地通知就會調(diào)用該方法
 *  注意這個方法:應(yīng)用在前臺也會調(diào)用
 *  @param application  應(yīng)用
 *  @param notification 本地通知對象
 */
- (void) application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification
{

//  當應(yīng)用在前臺時候什么都不做
    if (application.applicationState == UIApplicationStateActive) {
        return;
    }
//  當應(yīng)用不再前臺的時候才去跳轉(zhuǎn),這樣用戶體檢更好
//  獲取tabBarController
    UITabBarController *tbVc = (UITabBarController *)self.window.rootViewController;


//  獲取用戶設(shè)置的跳轉(zhuǎn)頁
    NSString *page = notification.userInfo[@"pageKey"];
//  如果是朋友圈
    if ([page isEqualToString:@"session"]) {
        tbVc.selectedIndex = 1;
    }else{
//       否則跳轉(zhuǎn)到好友
        tbVc.selectedIndex = 0;
    }
}

  • 測試launchOptions[UIApplicationLaunchOptionsLocalNotificationKey]中的內(nèi)容
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    // Override point for customization after application launch.
//  獲取UIApplicationLaunchOptionsLocalNotificationKey對應(yīng)內(nèi)容
    id obj = launchOptions[UIApplicationLaunchOptionsLocalNotificationKey];
//  獲取控制器(注意此時需要通過self.window,通過application.keyWindow無法獲取到,因為此時的window還沒有成為keyWindow)
    UITabBarController *tbVc = (UITabBarController *)self.window.rootViewController;
//  獲取索引為0的控制(注意此時tbVc.selectedViewController為nil)
    UIViewController *vc = tbVc.viewControllers[0];

//  創(chuàng)建一個文本
    UILabel *label = [[UILabel alloc] init];
    label.backgroundColor = [UIColor brownColor];
//  設(shè)置text為UIApplicationLaunchOptionsLocalNotificationKey對應(yīng)的內(nèi)容
    label.text =  [obj description] ;
    label.frame = CGRectMake(10, 100, 300, 400);
    label.numberOfLines = 0;
//  添加到控制器的View上
    [vc.view addSubview:label];


    return YES;
}

顯示結(jié)果:


336729BE-B7DA-4B83-9515-E10864F3FADE.png
  • 我們從中可以出他是一個UILocalNotification對象
  • 所以我們?nèi)〕鯱ILocalNotification對象,剩下的做法與接收到本地通知代理方法中處理相同,所以我們把它提取為一個公用的方法
/**
 *  根據(jù)通知跳轉(zhuǎn)到不同頁面
 */
- (void) jumpToPageWithLocalNotification:(UILocalNotification *) notification
{
    //  獲取tabBarController
    UITabBarController *tbVc = (UITabBarController *)self.window.rootViewController;

    //  獲取用戶設(shè)置的跳轉(zhuǎn)頁
    NSString *page = notification.userInfo[@"pageKey"];
    //  如果是朋友圈
    if ([page isEqualToString:@"session"]) {
        tbVc.selectedIndex = 0;
    }else{
        //       否則跳轉(zhuǎn)到好友
        tbVc.selectedIndex = 1;
    }
}
  • 在didReceiveLocalNotification方法中
/**
 *  一旦接收到本地通知就會調(diào)用該方法
 *  注意這個方法:應(yīng)用在前臺也會調(diào)用
 *  @param application  應(yīng)用
 *  @param notification 本地通知對象
 */
- (void) application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification
{

//  當應(yīng)用在前臺時候什么都不做
    if (application.applicationState == UIApplicationStateActive) {
        return;
    }
//  當應(yīng)用不再前臺的時候才去跳轉(zhuǎn),這樣用戶體檢更好
    [self jumpToPageWithLocalNotification:notification];
}

在didFinishLaunchingWithOptions方法中

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    // Override point for customization after application launch.
//  如果是點擊本地通知進來的那么launchOptions[UIApplicationLaunchOptionsLocalNotificationKey]就會有內(nèi)容
    UILocalNotification *notifcation = launchOptions[UIApplicationLaunchOptionsLocalNotificationKey];
//  如果存在通知
    if(notifcation){
        [self jumpToPageWithLocalNotification:notifcation];
    }

    return YES;
}```

###3. iOS8的不同點
---
你如果把上面的程序運行在iOS8上,會爆出如下錯誤

>預(yù)習01-本地推送通知[掌握][615:7847] Attempting to schedule a local notification <UIConcreteLocalNotification: 0x7f935182b150>{fire date = Monday, July 13, 2015 at 9:02:25 AM China Standard Time, time zone = (null), repeat interval = 0, repeat count = UILocalNotificationInfiniteRepeatCount, next fire date = Monday, July 13, 2015 at 9:02:25 AM China Standard Time, user info = {
    pageKey = friend;
}} **with an alert but haven't received permission from the user to display alerts**

也就是在iOS8上要發(fā)送本地通知需要 **請求用戶權(quán)限**
如何請求用戶權(quán)限呢?一般在新版有變化的地方,在頭文件中都會有相應(yīng)的說明,所以點擊到scheduleLocalNotification:方法中,看看有沒有我們需要信息
![4D110E8B-9D15-4460-B093-6970621F3229.png](http://upload-images.jianshu.io/upload_images/1253942-62c168e5061a56c9.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
點擊進去,我們看到
![B3F79CF3-8DCE-4E4D-8F0A-3BD39EF08AF1.png](http://upload-images.jianshu.io/upload_images/1253942-6174fc97ba719369.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
意思就是說:在iOS8.0以后,在調(diào)度通知之前你需要使用UIApplication的對象方法registerUseNotificationSetting:來請求用戶授權(quán).

這種請求權(quán)限的代碼一般放在didFinishLaunchingWithOptions:方法中,在用戶不卸載的情況下,只需要請求一次,下次在運行就不用請求了!

// 1.如果是iOS8請求用戶權(quán)限
if ([UIDevice currentDevice].systemVersion.doubleValue >= 8.0) {

    /*
     UIUserNotificationType:

     UIUserNotificationTypeBadge   = 1 << 0, // 接收到通知可更改程序的應(yīng)用圖標
     UIUserNotificationTypeSound   = 1 << 1, // 接收到通知可播放聲音
     UIUserNotificationTypeAlert   = 1 << 2, // 接收到通知課提示內(nèi)容
     如果你需要使用多個類型,可以使用 "|" 來連接
     */

// 向用戶請求通知權(quán)限
// categories暫時傳入nil
UIUserNotificationSettings *setting = [UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeBadge|UIUserNotificationTypeSound|UIUserNotificationTypeAlert categories:nil];

    [application registerUserNotificationSettings:setting];
}
運行程序

![21E244DC-087A-4679-9D62-BFDEC648672C.png](http://upload-images.jianshu.io/upload_images/1253942-ee6ddc53a049ddb5.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
測試點擊通知,進入應(yīng)用,也沒問題

--------
接下來,我們說說-[UIUserNotificationSettings settingsForTypes:categories:] 中的 **categories**

* categories可以讓我們發(fā)送通知之前預(yù)定義一些通知也就是通知上可以顯示按鈕,他需要是一個裝有UIUserNotificationCategory類的對象的NSSet的對象.
但是官方推薦我們使用它的子類UIMutableUserNotificationCategory,來動態(tài)的添加通知的行為按鈕,iOS8支持前臺和后臺的兩種行為.
* 通知Action按鈕以長條展示如圖
![notifcation_action_button.gif](http://upload-images.jianshu.io/upload_images/1253942-22b7dbce7618a358.gif?imageMogr2/auto-orient/strip)
* 通知Action按鈕以AlertView展示如圖
![notification_action_allertview.gif](http://upload-images.jianshu.io/upload_images/1253942-4a6cf2c9fda8767c.gif?imageMogr2/auto-orient/strip)

1. 注冊分類,并在分類中添加不同的行為
由于注冊用戶通知設(shè)置代碼量比較大我們實現(xiàn)一個新的方法registerUserNotification

  • (void) registerUserNotification
    {
    // 向用戶請求通知權(quán)限
    /*
    UIUserNotificationType:用戶通知的類型

    UIUserNotificationTypeBadge = 1 << 0, // 接收到通知可更改程序的應(yīng)用圖標
    UIUserNotificationTypeSound = 1 << 1, // 接收到通知可播放聲音
    UIUserNotificationTypeAlert = 1 << 2, // 接收到通知課提示內(nèi)容
    如果你需要使用多個類型,可以使用 "|" 來連接
    */
    // 1.設(shè)置用戶通知權(quán)限類型
    UIUserNotificationType types = UIUserNotificationTypeBadge|UIUserNotificationTypeSound|UIUserNotificationTypeAlert;

// 2.創(chuàng)建通知的行為按鈕

// 2.1創(chuàng)建第一個行為
UIMutableUserNotificationAction *action1 = [[UIMutableUserNotificationAction alloc] init];
// 2.1.1 設(shè)置行為的唯一標示
action1.identifier = UIMutableUserNotificationActionBackground;
// 2.1.2 設(shè)置通知按鈕的的標題
action1.title = @"后臺";
// 以什么樣模式運行應(yīng)用
// UIUserNotificationActivationModeForeground, // 當應(yīng)用在前臺的時候觸發(fā)
// UIUserNotificationActivationModeBackground // 即使應(yīng)用不在前臺也觸發(fā)
action1.activationMode = UIUserNotificationActivationModeBackground;
// 2.1.3 是否只有鎖屏的鎖屏狀態(tài)下才能顯示
action1.authenticationRequired = NO;
// 2.1.4 按鈕的性質(zhì)
action1.destructive = NO;

// 2.1創(chuàng)建第一個行為
UIMutableUserNotificationAction *action2 = [[UIMutableUserNotificationAction alloc] init];
// 2.1.1 設(shè)置行為的唯一標示
action2.identifier =UIMutableUserNotificationActionForeground;
// 2.1.2 設(shè)置通知按鈕的的標題
action2.title = @"前臺";
// 以什么樣模式運行應(yīng)用
// UIUserNotificationActivationModeForeground, // 當應(yīng)用在前臺的時候觸發(fā)
// UIUserNotificationActivationModeBackground // 即使應(yīng)用不在前臺也觸發(fā)
action2.activationMode = UIUserNotificationActivationModeForeground;
// 2.1.3 用戶必須輸入密碼才能執(zhí)行
action2.authenticationRequired = YES;
// 2.1.4 按鈕的性質(zhì)(沒有效果)
action2.destructive = YES;

// 3.創(chuàng)建用戶通知分類
UIMutableUserNotificationCategory *category = [[UIMutableUserNotificationCategory alloc] init];
// 3.1 設(shè)置類別的唯一標識
category.identifier = @"myCategory";
// 3.2 設(shè)置通知的按鈕
// Context:
// UIUserNotificationActionContextDefault, //默認上下文(情景)下的英文(通常都是)
// UIUserNotificationActionContextMinimal //通知內(nèi)容區(qū)域受限情況下內(nèi)容
[category setActions:@[action1,action2] forContext:UIUserNotificationActionContextDefault];

// 4.創(chuàng)建用戶通知的設(shè)置信息
UIUserNotificationSettings *setting = [UIUserNotificationSettings settingsForTypes:types categories:[NSSet setWithObject:category]];

// 注冊設(shè)置
[[UIApplication sharedApplication] registerUserNotificationSettings:setting];
}


* 在發(fā)送本地推送通知時候指定通知的分類標示

// 9.設(shè)置通知的類別
ln.category = @"myCategory";


* 監(jiān)聽點擊通知按鈕的行為,在AppDelegate中實現(xiàn)監(jiān)聽通知按鈕點擊方法

/**

  • 當用戶點擊通知上定制的按鈕執(zhí)行的行為(注意:不點擊行為按鈕,不會進入該方法)
  • @param application 應(yīng)用
  • @param identifier 行為標識符
  • @param notification 本地通知
  • @param completionHandler 完成回調(diào)
    */
  • (void)application:(UIApplication *)application handleActionWithIdentifier:(NSString )identifier forLocalNotification:(UILocalNotification )notification completionHandler:(void (^)())completionHandler
    {
    // 處理不同行為
    if ([identifier isEqualToString:UIMutableUserNotificationActionBackground]) {
    NSLog(@"后臺運行程序");
    }else if ([identifier isEqualToString:UIMutableUserNotificationActionForeground]){
    NSLog(@"前臺運行程序");
    }else{
    NSLog(@"其他");
    }
    /

    You should call the completion handler as soon as you've finished handling the action.
    當任務(wù)處理完畢時候,你應(yīng)該盡快的調(diào)用completion的block.
    */

// 在當任務(wù)完成的時候,調(diào)用任務(wù)完成的block
completionHandler();
}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 應(yīng)用程序必須進行適當配置,才可以接受本地或遠程通知。配置過程在iOS和OS X略有不同,但基本原理是相同的。在啟動...
    shenzhenboy閱讀 1,477評論 1 2
  • 這兩天在做一個日程提醒功能,用UILocalNotification做日程的提醒功能,相關(guān)知識如下: 一、本地通知...
    七維樹閱讀 1,214評論 1 1
  • 極光推送: 1.JPush當前版本是1.8.2,其SDK的開發(fā)除了正常的功能完善和擴展外也緊隨蘋果官方的步伐,SD...
    Isspace閱讀 6,887評論 10 16
  • 項目中用到了本地通知功能,便到網(wǎng)上搜羅了一翻,各式各樣的說法,經(jīng)過自己的驗證,整理如下 首先要說明的是IOS最多允...
    CrazySteven閱讀 716評論 3 2
  • 躁動的夏日仿佛一夜之間來到了大地上,即使隔三差五的還來點小寒流,但不容置疑,它真的來了。剛剛結(jié)束或者說放棄了一段本...
    季鴻宇閱讀 235評論 0 0

友情鏈接更多精彩內(nèi)容