2019-10-03 02:42:16 +02:00
|
|
|
//
|
|
|
|
// NotificationManager.swift
|
|
|
|
// NetNewsWire
|
|
|
|
//
|
|
|
|
// Created by Maurice Parker on 10/2/19.
|
|
|
|
// Copyright © 2019 Ranchero Software. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
import Account
|
|
|
|
import Articles
|
|
|
|
import UserNotifications
|
|
|
|
|
2024-04-08 06:32:47 +02:00
|
|
|
final class UserNotificationManager: Sendable {
|
|
|
|
|
|
|
|
init() {
|
2019-10-03 02:42:16 +02:00
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(accountDidDownloadArticles(_:)), name: .AccountDidDownloadArticles, object: nil)
|
2019-10-04 18:20:57 +02:00
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(statusesDidChange(_:)), name: .StatusesDidChange, object: nil)
|
2020-12-23 12:57:33 +01:00
|
|
|
registerCategoriesAndActions()
|
2019-10-03 02:42:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@objc func accountDidDownloadArticles(_ note: Notification) {
|
|
|
|
guard let articles = note.userInfo?[Account.UserInfoKey.newArticles] as? Set<Article> else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-03-20 07:05:30 +01:00
|
|
|
Task { @MainActor in
|
|
|
|
for article in articles {
|
2024-05-26 08:05:38 +02:00
|
|
|
if !article.status.read, let feed = article.feed, feed.shouldSendUserNotificationForNewArticles ?? false {
|
2024-03-20 07:05:30 +01:00
|
|
|
sendNotification(feed: feed, article: article)
|
|
|
|
}
|
2019-10-03 02:42:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-04 18:20:57 +02:00
|
|
|
@objc func statusesDidChange(_ note: Notification) {
|
2022-02-27 18:55:59 +01:00
|
|
|
if let statuses = note.userInfo?[Account.UserInfoKey.statuses] as? Set<ArticleStatus>, !statuses.isEmpty {
|
|
|
|
let identifiers = statuses.filter({ $0.read }).map { "articleID:\($0.articleID)" }
|
|
|
|
UNUserNotificationCenter.current().removeDeliveredNotifications(withIdentifiers: identifiers)
|
2019-10-04 18:20:57 +02:00
|
|
|
return
|
|
|
|
}
|
2022-02-27 18:55:59 +01:00
|
|
|
|
|
|
|
if let articleIDs = note.userInfo?[Account.UserInfoKey.articleIDs] as? Set<String>,
|
|
|
|
let statusKey = note.userInfo?[Account.UserInfoKey.statusKey] as? ArticleStatus.Key,
|
|
|
|
let flag = note.userInfo?[Account.UserInfoKey.statusFlag] as? Bool,
|
|
|
|
statusKey == .read,
|
|
|
|
flag == true {
|
|
|
|
let identifiers = articleIDs.map { "articleID:\($0)" }
|
|
|
|
UNUserNotificationCenter.current().removeDeliveredNotifications(withIdentifiers: identifiers)
|
|
|
|
}
|
2019-10-04 18:20:57 +02:00
|
|
|
}
|
|
|
|
|
2019-10-03 02:42:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private extension UserNotificationManager {
|
|
|
|
|
2024-03-20 07:05:30 +01:00
|
|
|
@MainActor func sendNotification(feed: Feed, article: Article) {
|
2019-10-03 02:42:16 +02:00
|
|
|
let content = UNMutableNotificationContent()
|
2019-10-03 16:53:21 +02:00
|
|
|
|
2024-02-26 08:12:21 +01:00
|
|
|
content.title = feed.nameForDisplay
|
2020-05-15 13:53:44 +02:00
|
|
|
if !ArticleStringFormatter.truncatedTitle(article).isEmpty {
|
|
|
|
content.subtitle = ArticleStringFormatter.truncatedTitle(article)
|
2019-10-03 16:53:21 +02:00
|
|
|
}
|
2020-05-15 13:53:44 +02:00
|
|
|
content.body = ArticleStringFormatter.truncatedSummary(article)
|
2024-02-26 08:12:21 +01:00
|
|
|
content.threadIdentifier = feed.feedID
|
2020-05-15 14:32:33 +02:00
|
|
|
content.sound = UNNotificationSound.default
|
|
|
|
content.userInfo = [UserInfoKey.articlePath: article.pathUserInfo]
|
2020-12-23 14:18:51 +01:00
|
|
|
content.categoryIdentifier = "NEW_ARTICLE_NOTIFICATION_CATEGORY"
|
2024-02-26 08:12:21 +01:00
|
|
|
if let attachment = thumbnailAttachment(for: article, feed: feed) {
|
2020-12-23 12:57:33 +01:00
|
|
|
content.attachments.append(attachment)
|
|
|
|
}
|
2020-05-15 14:32:33 +02:00
|
|
|
|
|
|
|
let request = UNNotificationRequest.init(identifier: "articleID:\(article.articleID)", content: content, trigger: nil)
|
2019-10-03 02:42:16 +02:00
|
|
|
UNUserNotificationCenter.current().add(request)
|
|
|
|
}
|
|
|
|
|
2020-12-23 12:57:33 +01:00
|
|
|
/// Determine if there is an available icon for the article. This will then move it to the caches directory and make it avialble for the notification.
|
|
|
|
/// - Parameters:
|
|
|
|
/// - article: `Article`
|
2024-02-26 08:12:21 +01:00
|
|
|
/// - feed: `Feed`
|
2020-12-23 12:57:33 +01:00
|
|
|
/// - Returns: A `UNNotifcationAttachment` if an icon is available. Otherwise nil.
|
|
|
|
/// - Warning: In certain scenarios, this will return the `faviconTemplateImage`.
|
2024-03-20 07:05:30 +01:00
|
|
|
@MainActor func thumbnailAttachment(for article: Article, feed: Feed) -> UNNotificationAttachment? {
|
2024-02-26 08:12:21 +01:00
|
|
|
if let imageURL = article.iconImageUrl(feed: feed) {
|
|
|
|
let thumbnail = try? UNNotificationAttachment(identifier: feed.feedID, url: imageURL, options: nil)
|
2020-12-23 12:57:33 +01:00
|
|
|
return thumbnail
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func registerCategoriesAndActions() {
|
|
|
|
let readAction = UNNotificationAction(identifier: "MARK_AS_READ", title: NSLocalizedString("Mark as Read", comment: "Mark as Read"), options: [])
|
|
|
|
let starredAction = UNNotificationAction(identifier: "MARK_AS_STARRED", title: NSLocalizedString("Mark as Starred", comment: "Mark as Starred"), options: [])
|
|
|
|
let openAction = UNNotificationAction(identifier: "OPEN_ARTICLE", title: NSLocalizedString("Open", comment: "Open"), options: [.foreground])
|
|
|
|
|
|
|
|
let newArticleCategory =
|
|
|
|
UNNotificationCategory(identifier: "NEW_ARTICLE_NOTIFICATION_CATEGORY",
|
|
|
|
actions: [openAction, readAction, starredAction],
|
|
|
|
intentIdentifiers: [],
|
|
|
|
hiddenPreviewsBodyPlaceholder: "",
|
2021-01-24 17:21:34 +01:00
|
|
|
options: [])
|
2020-12-23 12:57:33 +01:00
|
|
|
|
|
|
|
UNUserNotificationCenter.current().setNotificationCategories([newArticleCategory])
|
|
|
|
}
|
2019-10-03 02:42:16 +02:00
|
|
|
}
|