// // 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 final class UserNotificationManager: Sendable { init() { NotificationCenter.default.addObserver(self, selector: #selector(accountDidDownloadArticles(_:)), name: .AccountDidDownloadArticles, object: nil) NotificationCenter.default.addObserver(self, selector: #selector(statusesDidChange(_:)), name: .StatusesDidChange, object: nil) registerCategoriesAndActions() } @objc func accountDidDownloadArticles(_ note: Notification) { guard let articles = note.userInfo?[Account.UserInfoKey.newArticles] as? Set
else { return } Task { @MainActor in for article in articles { if !article.status.read, let feed = article.feed, feed.shouldSendUserNotificationForNewArticles ?? false { sendNotification(feed: feed, article: article) } } } } @objc func statusesDidChange(_ note: Notification) { if let statuses = note.userInfo?[Account.UserInfoKey.statuses] as? Set, !statuses.isEmpty { let identifiers = statuses.filter({ $0.read }).map { "articleID:\($0.articleID)" } UNUserNotificationCenter.current().removeDeliveredNotifications(withIdentifiers: identifiers) return } if let articleIDs = note.userInfo?[Account.UserInfoKey.articleIDs] as? Set, 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) } } } private extension UserNotificationManager { @MainActor func sendNotification(feed: Feed, article: Article) { let content = UNMutableNotificationContent() content.title = feed.nameForDisplay if !ArticleStringFormatter.truncatedTitle(article).isEmpty { content.subtitle = ArticleStringFormatter.truncatedTitle(article) } content.body = ArticleStringFormatter.truncatedSummary(article) content.threadIdentifier = feed.feedID content.sound = UNNotificationSound.default content.userInfo = [UserInfoKey.articlePath: article.pathUserInfo] content.categoryIdentifier = "NEW_ARTICLE_NOTIFICATION_CATEGORY" if let attachment = thumbnailAttachment(for: article, feed: feed) { content.attachments.append(attachment) } let request = UNNotificationRequest.init(identifier: "articleID:\(article.articleID)", content: content, trigger: nil) UNUserNotificationCenter.current().add(request) } /// 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` /// - feed: `Feed` /// - Returns: A `UNNotifcationAttachment` if an icon is available. Otherwise nil. /// - Warning: In certain scenarios, this will return the `faviconTemplateImage`. @MainActor func thumbnailAttachment(for article: Article, feed: Feed) -> UNNotificationAttachment? { if let imageURL = article.iconImageUrl(feed: feed) { let thumbnail = try? UNNotificationAttachment(identifier: feed.feedID, url: imageURL, options: nil) 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: "", options: []) UNUserNotificationCenter.current().setNotificationCategories([newArticleCategory]) } }