如何在 iOS 8 中使用 Swift 实现本地通知(下)

   2023-02-09 学习力0
核心提示:在上集中,我们已经构建了一个简单的待办列表应用(to-do list app),这个应用可以在待办项过期时通过本地通知提醒用户。现在,我们要在之前的基础上添加以下功能:应用图标角标上显示过期待办项的数量、通知动作的支持和在不开启应用的情况下编辑和完成待办

在上集中,我们已经构建了一个简单的待办列表应用(to-do list app),这个应用可以在待办项过期时通过本地通知提醒用户。现在,我们要在之前的基础上添加以下功能:应用图标角标上显示过期待办项的数量、通知动作的支持和在不开启应用的情况下编辑和完成待办项。

你可以在这里下载上一集的源代码。

为应用图标添加角标(Badge)

值得注意的是,我们不通过本地通知也可以为应用图标添加角标。AppDelegate 中的 applicationWillResignActive: 方法是可以实现这个功能的。在用户返回主屏幕的时候,看到应用图标时候,会触发这个方法。

func applicationWillResignActive(application: UIApplication) {
var todoItems: [TodoItem] = TodoList.sharedInstance.allItems()
var overdueItems = todoItems.filter({ (todoItem) -> Bool in
return todoItem.deadline.compare(NSDate()) != .OrderedDescending
})
UIApplication.sharedApplication().applicationIconBadgeNumber = overdueItems.count
}

如何在 iOS 8 中使用 Swift 实现本地通知(下)如何在 iOS 8 中使用 Swift 实现本地通知(下)

虽然可行,但是并不能在待办项过期时自动更新角标的值。因为我们不能简单的在通知触发的时候增加角标的值,因此,我们可以为本地通知预设一个“applicationIconBadgeNumber”的属性值。接下来,我们在 TodoList 中写一个为每个通知设置相关角标值的方法。

func setBadgeNumbers() {
var notifications = UIApplication.sharedApplication().scheduledLocalNotifications as! [UILocalNotification]
var todoItems: [TodoItem] = self.allItems()
for notification in notifications {
var overdueItems = todoItems.filter({ (todoItem) -> Bool in
return (todoItem.deadline.compare(notification.fireDate!) != .OrderedDescending)
})
UIApplication.sharedApplication().cancelLocalNotification(notification)
notification.applicationIconBadgeNumber = overdueItems.count
UIApplication.sharedApplication().scheduleLocalNotification(notification)
}
}

虽然没有办法去更新已经计划好时间的通知,但是可以通过删除当前通知然后重新设置的方式来达到同样的效果。

applicationIconBadgeNumber这个属性可以接受的最大值是 2,147,483,647(NSIntegerMax),超过 5 位数的数字就会在图标上被截断,如图所示。而且,设置为零或负数是不会产生效果的。


如何在 iOS 8 中使用 Swift 实现本地通知(下)如何在 iOS 8 中使用 Swift 实现本地通知(下)

我们只是需要在待办列表发生变化的时候调用这个方法。将下面这段代码添加到 TodoList 中的addItem:removeItem:两个方法之后。

self.setBadgeNumbers()

这样,当一个通知被触发的时候,角标的值会被自动更新。

如何让通知反复触发

UILocalNotificaiton实例有个属性,叫repeatInterval,可以让通知以一个固定的时长反复触发。这种方法可以很好的避免通知数量 64 的这个上限,而且一个重复的通知只会被计算一次。

不幸的是,在我们的应用中,只能在repeatIntervalapplicationIconBadgeNumber中选择一个。角标值会在每次通知被触发的时候更新。较旧的通知可能会被新的通知终结,而且恰好新的通知是重复的通知(一个重复的通知只会被计算一次),即导致角标的值会比预计的小。我们可以设置两个通知,一个是重复的通知会显示提醒和播放提示音,另一个是非重复的通知会更新角标值,会发现测试的结果角标值只有预想的一半。

重复通知的最大局限是重复的间隔(repeatInterval)不接受自定义值。你必须提供一个NSCalendarUnit值,例如HourCalendarUnitDayCalendarUnit。假如,如果你希望设置的通知每 30 分钟触发一次,那么可以通过设置两个通知(两者之间间隔 30 分钟,并都以一小时重复触发)来解决。当然了,如果你希望是每 31 分钟触发一次,那就没办法了。

后台执行通知动作

iOS 8 引进了一个很有用的新特性:通知动作(notification actions),它可以让通知由用户来触发,甚至用户没有打开应用都可以触发。让我们来实现这个功能吧:从通知横幅上直接完成或等会提醒待办项。


如何在 iOS 8 中使用 Swift 实现本地通知(下)如何在 iOS 8 中使用 Swift 实现本地通知(下)

AppDelegate文件里可以这样写:

func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions: [NSObject: AnyObject]?) -> Bool {
let completeAction = UIMutableUserNotificationAction()
completeAction.identifier = "COMPLETE_TODO"
completeAction.title = "Complete"
completeAction.activationMode = .Background
completeAction.authenticationRequired = false
completeAction.destructive = true
let remindAction = UIMutableUserNotificationAction()
remindAction.identifier = "REMIND"
remindAction.title = "Remind in 30 minutes"
remindAction.activationMode = .Background
remindAction.destructive = fal
let todoCategory = UIMutableUserNotificationCategory()
todoCategory.identifier = "TODO_CATEGORY"
todoCategory.setActions([remindAction, completeAction], forContext: .Default)
todoCategory.setActions([completeAction, remindAction], forContext: .Minimal)
application.registerUserNotificationSettings(UIUserNotificationSettings(forTypes: .Alert | .Badge | .Sou categories: NSSet(array: [todoCategory])))
return true
}

值得注意的是,上面的代码中我们调用了todoCategory.setActions()两次,分别设置了动作上下文(action contexts)。如果通知是以横幅(banner)的形式显示,那么通知动作会以迷你形式(minimal context)显示出来。如果通知是以(默认的)提示(alert) 形式显示,通知动作会显示至少 4 个操作。如下图。


如何在 iOS 8 中使用 Swift 实现本地通知(下)如何在 iOS 8 中使用 Swift 实现本地通知(下)

我们传递到setActions:方法里的通知动作的顺序会一一对应的呈现在 UI 里,但是很奇怪的是,迷你形式里通知动作的顺序是以从右向左排列的方式显示的。

下面的代码是为了确保通知的category是为 TodoList 的addItem:方法设置的。

notification.category = "TODO_CATEGORY"

到此,我们已经可以用removeItem:方法来实现完成待办项,现在我们需要在 TodoList 里实现过会提醒的功能。

func scheduleReminderforItem(item: TodoItem) {
var notification = UILocalNotification()
notification.alertBody = "Reminder: Todo Item \"\(item.title)\" Is Overdue"
notification.alertAction = "open"
notification.fireDate = NSDate().dateByAddingTimeInterval(30 * 60)
notification.soundName = UILocalNotificationDefaultSoundName
notification.userInfo = ["title": item.title, "UUID": item.UUID]
notification.category = "TODO_CATEGOR
UIApplication.sharedApplication().scheduleLocalNotification(notification)
}

值得注意的是,我们并没有修改待办项的到期日期(或者试着取消原来的通知,虽然已经被自动删除)。现在,回到AppDelegate里,实现actions:里的处理。

func application(application: UIApplication, handleActionWithIdentifier identifier: String?, forLocalNotification notification: UILocalNotification, completionHandler: () -> Void) {
var item = TodoItem(deadline: notification.fireDate!, title: notification.userInfo!["title"] as String, UU notification.userInfo!["UUID"] as String!)
switch (identifier!) {
case "COMPLETE_TODO":
TodoList.sharedInstance.removeItem(item)
case "REMIND":
TodoList.sharedInstance.scheduleReminderforItem(item)
default:
println("Error: unexpected notification action identifier!")
}
completionHandler()
}

终于可以试着运行这个应用了(为了测试方便,建议将dateByAddingTimeInterval:设置一个较小的值)。


如何在 iOS 8 中使用 Swift 实现本地通知(下)


我们讨论了所有的非地理上(注:上一集讲到的用户进入和离开某个地理区域时,可以触发本地通知的功能)的 UILocalNotification 功能,现在已经有一个功能较全的待办项应用了,所以本系列教程到此结束。你可以在这里下载工程的源代码。

 
反对 0举报 0 评论 0
 

免责声明:本文仅代表作者个人观点,与乐学笔记(本网)无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
    本网站有部分内容均转载自其它媒体,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责,若因作品内容、知识产权、版权和其他问题,请及时提供相关证明等材料并与我们留言联系,本网站将在规定时间内给予删除等相关处理.

  • swift 命令行工具初探
    亲爱的同学们好,今天我们要介绍这么一个东西。相信有过解释型语言(PHP,Ruby,等)使用经验的同学会更加熟悉,就是 Swift 也为我们提供了命令行运行工具,俗称 REPL。好了,我们进入正题,在安装好 Swift 开发环境的机器上,打开命令行,输入 swift 命令,就进
    03-16
  • [Swift]冒泡排序 | Bubble sort
    [Swift]冒泡排序 | Bubble sort
    ★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★➤微信公众号:山青咏芝(shanqingyongzhi)➤博客园地址:山青咏芝(https://www.cnblogs.com/strengthen/)➤GitHub地址:https://github.com/strengthen/LeetCode➤原文
    03-08
  • [Swift] 自定义在 SwiftUI 中实现的可搜索的外观
    [Swift] 自定义在 SwiftUI 中实现的可搜索的外
    首先我找遍了,似乎找不到任何信息......(我遇到了许多根本不起作用的事情......)终于在详细的英文文献中找到了,我会保留下来,希望以后有机会。关于 SwiftUI 中的可搜索searchable是iOS15新增的易于实现的搜索字段。关于这种情况有一个参数placement,您
    03-08
  • [Swift] 检测 SwiftUI ScrollView 中的偏移量
    [Swift] 检测 SwiftUI ScrollView 中的偏移量
    首先你可以用ScrollViewReader做一些可以从iOS14使用的事情。但是,我不能做我想做的事情,所以我想我还能做些什么。再次,可重复使用我尝试过了。执行我将首先发布实现的图像。 (Swift Playgrounds 演示)您可以像这样根据滚动获取坐标。让我们看看实现。1.
    03-08
  • Swift3.0 UICollectionView 删除,拖动
    Swift3.0 UICollectionView 删除,拖动
    UICollectionView实现了一下常见的新闻分类.  附有效果图 近期一直在深入学习swift,实现了CollectionView item的头东与删除,用的都是系统的一些函数方法,看起来比较直观. 第一步:class HotViewController: UIViewController,UICollectionViewDelegate,UICo
    02-09
  • swift -懒加载创建view
     // 只有外界访问到headerView的时候才会去执行闭包, 然后将闭包的返回值赋值给headerView    // 注意: 一定要记住闭包后面需要写上(), 代表执行闭包    //懒加载创建UIView    lazy var headerView: UIView = {        let view = UIView()
    02-09
  • Swift--非常好用的适合局部的代码约束
    // 哪个控件的哪个属性等于(大于、小于)另外一个控件的哪个属性乘以多少再加上多少 eg:let widthContraint = NSLayoutConstraint(item: messageLabel, attribute: NSLayoutAttribute.Width, relatedBy: NSLayoutRelation.Equal, toItem: nil, attribute: NSLa
    02-09
  • iOS打包framework - Swift完整项目打包Framework,嵌入OC项目使用
    iOS打包framework - Swift完整项目打包Framewor
    场景说明:-之前做的App,使用Swift框架语言,混合编程,内含少部分OC代码。-需要App整体功能打包成静态库,完整移植到另一个App使用,该App使用OC。-所以涉及到一个语言互转的处理,以及一些AppDelegate的代码减除变化。 --------------------------------
    02-09
  • Swift -- 官方文档Swift-Guides的学习笔记
    在经历的一段时间的郁闷之后,我发现感情都是虚伪的,只有代码是真实的(呸)因为看了swift语法之后依然不会用swift,然后我非常作死的跑去看官方文档,就是xcode里自带的help》documentation and API reference其中的swift里的guide这里主要总结一下里面每一
    02-09
  • Swift - 进度条(UIProgressView)的用法
     1,创建进度条1234var progressView=UIProgressView(progressViewStyle:UIProgressViewStyle.Default)progressView.center=self.view.centerprogressView.progress=0.5 //默认进度50%self.view.addSubview(progressView); 2,设置进度,同时有动画效果1p
    02-09
点击排行