NetNewsWire/iOS/AppDelegate.swift

475 lines
14 KiB
Swift
Raw Normal View History

//
// AppDelegate.swift
// NetNewsWire
//
// Created by Maurice Parker on 4/8/19.
// Copyright © 2019 Ranchero Software. All rights reserved.
//
import UIKit
import Web
import Account
2024-07-01 03:14:42 +02:00
@preconcurrency import BackgroundTasks
2019-04-26 14:44:00 +02:00
import os.log
import WidgetKit
import Core
import Images
2024-09-30 06:03:24 +02:00
import libxml2
@MainActor var appDelegate: AppDelegate!
@UIApplicationMain
2024-09-30 06:58:39 +02:00
final class AppDelegate: UIResponder, UIApplicationDelegate, UNUserNotificationCenterDelegate, UnreadCountProvider {
2019-06-19 17:26:03 +02:00
private var bgTaskDispatchQueue = DispatchQueue.init(label: "BGTaskScheduler")
private var waitBackgroundUpdateTask = UIBackgroundTaskIdentifier.invalid
2019-06-20 01:09:42 +02:00
private var syncBackgroundUpdateTask = UIBackgroundTaskIdentifier.invalid
2019-05-20 20:51:08 +02:00
var syncTimer: ArticleStatusSyncTimer?
2019-05-20 20:51:08 +02:00
var shuttingDown = false {
didSet {
if shuttingDown {
syncTimer?.shuttingDown = shuttingDown
syncTimer?.invalidate()
}
}
}
2019-06-19 17:26:03 +02:00
2024-05-04 08:10:57 +02:00
nonisolated(unsafe) let log = OSLog(subsystem: Bundle.main.bundleIdentifier!, category: "Application")
2019-06-19 17:26:03 +02:00
var userNotificationManager: UserNotificationManager!
var extensionContainersFile: ExtensionContainersFile!
var extensionFeedAddRequestFile: ExtensionFeedAddRequestFile!
2019-06-19 17:26:03 +02:00
var unreadCount = 0 {
didSet {
if unreadCount != oldValue {
handleUnreadCountDidChange()
}
}
}
var isSyncArticleStatusRunning = false
var isWaitingForSyncTasks = false
2024-03-20 04:33:54 +01:00
override init() {
xmlInitParser()
2024-03-20 04:33:54 +01:00
super.init()
appDelegate = self
NotificationCenter.default.addObserver(self, selector: #selector(unreadCountDidChange(_:)), name: .UnreadCountDidChange, object: nil)
NotificationCenter.default.addObserver(self, selector: #selector(accountRefreshDidFinish(_:)), name: .AccountRefreshDidFinish, object: nil)
}
2019-06-19 17:26:03 +02:00
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
AppDefaults.registerDefaults()
if AppDefaults.shared.isFirstRun {
os_log(.debug, "Is first run.")
}
FaviconGenerator.faviconTemplateImage = AppAsset.faviconTemplateImage
importFeedsIfNeeded()
registerBackgroundTasks()
CacheCleaner.purgeIfNecessary()
2019-09-01 23:54:07 +02:00
initializeDownloaders()
initializeHomeScreenQuickActions()
2019-06-19 17:26:03 +02:00
2024-03-20 04:33:54 +01:00
Task { @MainActor in
self.unreadCount = AccountManager.shared.unreadCount
}
2024-03-20 04:33:54 +01:00
UNUserNotificationCenter.current().requestAuthorization(options:[.badge, .sound, .alert]) { (granted, error) in
if granted {
2024-03-20 04:33:54 +01:00
Task { @MainActor in
2019-04-23 14:48:22 +02:00
UIApplication.shared.registerForRemoteNotifications()
}
}
}
UNUserNotificationCenter.current().delegate = self
userNotificationManager = UserNotificationManager()
extensionContainersFile = ExtensionContainersFile()
extensionFeedAddRequestFile = ExtensionFeedAddRequestFile()
2019-05-20 20:51:08 +02:00
syncTimer = ArticleStatusSyncTimer()
2019-06-19 17:26:03 +02:00
2019-05-20 20:51:08 +02:00
#if DEBUG
2019-06-19 17:26:03 +02:00
syncTimer!.update()
2019-05-20 20:51:08 +02:00
#endif
2021-11-02 11:44:21 +01:00
return true
}
2019-06-19 17:26:03 +02:00
func application(_ application: UIApplication, didReceiveRemoteNotification userInfo: [AnyHashable : Any]) async -> UIBackgroundFetchResult {
resumeDatabaseProcessingIfNecessary()
await AccountManager.shared.receiveRemoteNotification(userInfo: userInfo)
suspendApplication()
return .newData
}
2019-06-28 17:28:02 +02:00
func applicationWillTerminate(_ application: UIApplication) {
shuttingDown = true
}
func applicationDidEnterBackground(_ application: UIApplication) {
ArticleStringFormatter.emptyCaches()
MultilineUILabelSizer.emptyCache()
SingleLineUILabelSizer.emptyCache()
IconImageCache.shared.emptyCache()
AccountManager.shared.emptyCaches()
Task.detached {
await DownloadWithCacheManager.shared.cleanupCache()
}
}
2019-06-28 17:28:02 +02:00
// MARK: Notifications
@objc func unreadCountDidChange(_ note: Notification) {
if note.object is AccountManager {
unreadCount = AccountManager.shared.unreadCount
2019-06-28 17:28:02 +02:00
}
}
2019-06-19 17:26:03 +02:00
2019-06-28 17:28:02 +02:00
@objc func accountRefreshDidFinish(_ note: Notification) {
AppDefaults.shared.lastRefresh = Date()
2019-06-28 17:28:02 +02:00
}
// MARK: - API
func manualRefresh(errorHandler: @escaping (Error) -> ()) {
2024-04-08 02:06:39 +02:00
let sceneDelegates = UIApplication.shared.connectedScenes.compactMap{ $0.delegate as? SceneDelegate }
for sceneDelegate in sceneDelegates {
sceneDelegate.cleanUp(conditional: true)
}
2024-03-26 07:36:27 +01:00
Task { @MainActor in
await AccountManager.shared.refreshAll(errorHandler: errorHandler)
2024-03-26 07:36:27 +01:00
}
}
func resumeDatabaseProcessingIfNecessary() {
if AccountManager.shared.isSuspended {
AccountManager.shared.resumeAll()
os_log("Application processing resumed.", log: self.log, type: .info)
}
}
2019-06-28 17:28:02 +02:00
func prepareAccountsForBackground() {
extensionFeedAddRequestFile.suspend()
2019-05-20 20:51:08 +02:00
syncTimer?.invalidate()
2019-06-19 17:26:03 +02:00
scheduleBackgroundFeedRefresh()
syncArticleStatus()
waitForSyncTasksToFinish()
}
2019-06-19 17:26:03 +02:00
2019-06-28 17:28:02 +02:00
func prepareAccountsForForeground() {
extensionFeedAddRequestFile.resume()
syncTimer?.update()
2024-03-26 07:36:27 +01:00
Task { @MainActor in
if let lastRefresh = AppDefaults.shared.lastRefresh {
if Date() > lastRefresh.addingTimeInterval(15 * 60) {
await AccountManager.shared.refreshAll(errorHandler: ErrorHandler.log)
2024-03-26 07:36:27 +01:00
} else {
await AccountManager.shared.syncArticleStatusAll()
}
2024-03-26 07:36:27 +01:00
} else {
await AccountManager.shared.refreshAll(errorHandler: ErrorHandler.log)
}
}
}
2024-03-26 07:36:27 +01:00
2024-05-04 07:35:20 +02:00
nonisolated func userNotificationCenter(_ center: UNUserNotificationCenter, willPresent notification: UNNotification, withCompletionHandler completionHandler: @escaping (UNNotificationPresentationOptions) -> Void) {
2024-02-26 04:02:29 +01:00
completionHandler([.list, .banner, .badge, .sound])
}
2024-07-01 03:14:42 +02:00
nonisolated func userNotificationCenter(_ center: UNUserNotificationCenter, didReceive response: UNNotificationResponse, withCompletionHandler completionHandler: @escaping () -> Void) {
MainActor.assumeIsolated {
defer { completionHandler() }
let userInfo = response.notification.request.content.userInfo
switch response.actionIdentifier {
case "MARK_AS_READ":
handleMarkAsRead(userInfo: userInfo)
case "MARK_AS_STARRED":
handleMarkAsStarred(userInfo: userInfo)
default:
if let sceneDelegate = response.targetScene?.delegate as? SceneDelegate {
sceneDelegate.handle(response)
DispatchQueue.main.asyncAfter(deadline: .now() + 0.5, execute: {
sceneDelegate.coordinator.dismissIfLaunchingFromExternalAction()
})
}
2020-12-23 13:15:25 +01:00
}
2019-10-03 16:53:21 +02:00
}
}
}
2019-09-01 23:54:07 +02:00
// MARK: App Initialization
private extension AppDelegate {
2019-09-01 23:54:07 +02:00
private func initializeHomeScreenQuickActions() {
let unreadTitle = NSLocalizedString("First Unread", comment: "First Unread")
let unreadIcon = UIApplicationShortcutIcon(systemImageName: "chevron.down.circle")
2019-09-01 23:54:07 +02:00
let unreadItem = UIApplicationShortcutItem(type: "com.ranchero.NetNewsWire.FirstUnread", localizedTitle: unreadTitle, localizedSubtitle: nil, icon: unreadIcon, userInfo: nil)
let searchTitle = NSLocalizedString("Search", comment: "Search")
let searchIcon = UIApplicationShortcutIcon(systemImageName: "magnifyingglass")
let searchItem = UIApplicationShortcutItem(type: "com.ranchero.NetNewsWire.ShowSearch", localizedTitle: searchTitle, localizedSubtitle: nil, icon: searchIcon, userInfo: nil)
2019-09-02 22:45:09 +02:00
let addTitle = NSLocalizedString("Add Feed", comment: "Add Feed")
let addIcon = UIApplicationShortcutIcon(systemImageName: "plus")
let addItem = UIApplicationShortcutItem(type: "com.ranchero.NetNewsWire.ShowAdd", localizedTitle: addTitle, localizedSubtitle: nil, icon: addIcon, userInfo: nil)
2019-09-02 23:05:55 +02:00
UIApplication.shared.shortcutItems = [addItem, searchItem, unreadItem]
2019-09-01 23:54:07 +02:00
}
}
// MARK: Go To Background
private extension AppDelegate {
func waitForSyncTasksToFinish() {
guard !isWaitingForSyncTasks && UIApplication.shared.applicationState == .background else { return }
isWaitingForSyncTasks = true
self.waitBackgroundUpdateTask = UIApplication.shared.beginBackgroundTask { [weak self] in
guard let self = self else { return }
self.completeProcessing(true)
os_log("Accounts wait for progress terminated for running too long.", log: self.log, type: .info)
}
DispatchQueue.main.async { [weak self] in
self?.waitToComplete() { [weak self] suspend in
self?.completeProcessing(suspend)
}
}
}
func waitToComplete(completion: @escaping (Bool) -> Void) {
guard UIApplication.shared.applicationState == .background else {
2022-01-04 23:25:20 +01:00
os_log("App came back to foreground, no longer waiting.", log: self.log, type: .info)
completion(false)
return
}
if AccountManager.shared.refreshInProgress || isSyncArticleStatusRunning {
os_log("Waiting for sync to finish...", log: self.log, type: .info)
DispatchQueue.main.asyncAfter(deadline: .now() + 1.0) { [weak self] in
self?.waitToComplete(completion: completion)
}
} else {
os_log("Refresh progress complete.", log: self.log, type: .info)
completion(true)
}
}
func completeProcessing(_ suspend: Bool) {
if suspend {
suspendApplication()
}
UIApplication.shared.endBackgroundTask(self.waitBackgroundUpdateTask)
self.waitBackgroundUpdateTask = UIBackgroundTaskIdentifier.invalid
isWaitingForSyncTasks = false
}
func syncArticleStatus() {
guard !isSyncArticleStatusRunning else { return }
isSyncArticleStatusRunning = true
let completeProcessing = { [unowned self] in
self.isSyncArticleStatusRunning = false
UIApplication.shared.endBackgroundTask(self.syncBackgroundUpdateTask)
self.syncBackgroundUpdateTask = UIBackgroundTaskIdentifier.invalid
}
self.syncBackgroundUpdateTask = UIApplication.shared.beginBackgroundTask {
completeProcessing()
os_log("Accounts sync processing terminated for running too long.", log: self.log, type: .info)
}
2024-03-20 04:33:54 +01:00
Task { @MainActor in
await AccountManager.shared.syncArticleStatusAll()
completeProcessing()
}
}
func suspendApplication() {
guard UIApplication.shared.applicationState == .background else { return }
AccountManager.shared.suspendNetworkAll()
AccountManager.shared.suspendDatabaseAll()
ArticleThemeDownloader.cleanUp()
CoalescingQueue.standard.performCallsImmediately()
for scene in UIApplication.shared.connectedScenes {
if let sceneDelegate = scene.delegate as? SceneDelegate {
sceneDelegate.suspend()
}
}
os_log("Application processing suspended.", log: self.log, type: .info)
}
}
2019-09-01 23:54:07 +02:00
// MARK: Background Tasks
2019-06-19 17:26:03 +02:00
private extension AppDelegate {
2019-06-28 17:28:02 +02:00
2019-06-19 17:26:03 +02:00
/// Register all background tasks.
func registerBackgroundTasks() {
// Register background feed refresh.
BGTaskScheduler.shared.register(forTaskWithIdentifier: "com.ranchero.NetNewsWire.FeedRefresh", using: nil) { (task) in
self.performBackgroundFeedRefresh(with: task as! BGAppRefreshTask)
}
}
2019-06-19 17:26:03 +02:00
/// Schedules a background app refresh based on `AppDefaults.refreshInterval`.
func scheduleBackgroundFeedRefresh() {
let request = BGAppRefreshTaskRequest(identifier: "com.ranchero.NetNewsWire.FeedRefresh")
request.earliestBeginDate = Date(timeIntervalSinceNow: 15 * 60)
// We send this to a dedicated serial queue because as of 11/05/19 on iOS 13.2 the call to the
// task scheduler can hang indefinitely.
bgTaskDispatchQueue.async {
do {
try BGTaskScheduler.shared.submit(request)
} catch {
2024-05-04 20:05:45 +02:00
Task { @MainActor in
os_log(.error, log: self.log, "Could not schedule app refresh: %@", error.localizedDescription)
}
}
2019-06-19 17:26:03 +02:00
}
}
/// Performs background feed refresh.
/// - Parameter task: `BGAppRefreshTask`
/// - Warning: As of Xcode 11 beta 2, when triggered from the debugger this doesn't work.
func performBackgroundFeedRefresh(with task: BGAppRefreshTask) {
scheduleBackgroundFeedRefresh() // schedule next refresh
os_log("Woken to perform account refresh.", log: self.log, type: .info)
2024-03-20 04:33:54 +01:00
Task { @MainActor in
if AccountManager.shared.isSuspended {
AccountManager.shared.resumeAll()
}
await AccountManager.shared.refreshAll(errorHandler: ErrorHandler.log)
if !AccountManager.shared.isSuspended {
2024-03-26 07:36:27 +01:00
try? WidgetDataEncoder.shared.encodeWidgetData()
self.suspendApplication()
os_log("Account refresh operation completed.", log: self.log, type: .info)
task.setTaskCompleted(success: true)
2019-06-19 17:26:03 +02:00
}
}
2019-06-19 17:26:03 +02:00
// set expiration handler
2019-11-28 01:03:19 +01:00
task.expirationHandler = { [weak task] in
DispatchQueue.main.sync {
self.suspendApplication()
}
2019-06-19 17:26:03 +02:00
os_log("Accounts refresh processing terminated for running too long.", log: self.log, type: .info)
2019-11-28 01:03:19 +01:00
task?.setTaskCompleted(success: false)
2019-06-19 17:26:03 +02:00
}
}
}
2020-12-23 13:15:25 +01:00
// Handle Notification Actions
private extension AppDelegate {
2024-03-19 18:15:30 +01:00
@MainActor func handleMarkAsRead(userInfo: [AnyHashable: Any]) {
guard let articlePathInfo = ArticlePathInfo(userInfo: userInfo) else {
return
2020-12-23 13:15:25 +01:00
}
2024-03-19 18:15:30 +01:00
2020-12-23 13:15:25 +01:00
resumeDatabaseProcessingIfNecessary()
2024-03-19 18:15:30 +01:00
guard let accountID = articlePathInfo.accountID, let account = AccountManager.shared.existingAccount(with: accountID) else {
2020-12-23 13:15:25 +01:00
os_log(.debug, "No account found from notification.")
return
}
guard let articleID = articlePathInfo.articleID else {
os_log(.debug, "No articleID found from notification.")
return
}
2024-03-19 18:15:30 +01:00
Task { @MainActor in
guard let articles = try? await account.articles(for: .articleIDs([articleID])) else {
os_log(.debug, "No article found from search using %@", articleID)
return
2020-12-23 13:15:25 +01:00
}
2024-03-19 18:15:30 +01:00
2024-03-27 02:48:44 +01:00
try? await account.markArticles(articles, statusKey: .read, flag: true)
2024-03-19 18:15:30 +01:00
self.prepareAccountsForBackground()
2024-03-27 02:48:44 +01:00
try? await account.syncArticleStatus()
if !AccountManager.shared.isSuspended {
try? WidgetDataEncoder.shared.encodeWidgetData()
self.prepareAccountsForBackground()
self.suspendApplication()
}
2024-03-19 18:15:30 +01:00
}
2020-12-23 13:15:25 +01:00
}
2024-03-19 18:15:30 +01:00
@MainActor func handleMarkAsStarred(userInfo: [AnyHashable: Any]) {
guard let articlePathInfo = ArticlePathInfo(userInfo: userInfo) else {
return
2020-12-23 13:15:25 +01:00
}
2024-03-19 18:15:30 +01:00
2020-12-23 13:15:25 +01:00
resumeDatabaseProcessingIfNecessary()
2024-03-19 18:15:30 +01:00
guard let accountID = articlePathInfo.accountID, let account = AccountManager.shared.existingAccount(with: accountID) else {
2020-12-23 13:15:25 +01:00
os_log(.debug, "No account found from notification.")
return
}
guard let articleID = articlePathInfo.articleID else {
os_log(.debug, "No articleID found from notification.")
return
}
2024-03-19 18:15:30 +01:00
Task { @MainActor in
guard let articles = try? await account.articles(for: .articleIDs([articleID])) else {
os_log(.debug, "No article found from search using %@", articleID)
return
2020-12-23 13:15:25 +01:00
}
2024-03-19 18:15:30 +01:00
2024-03-27 02:48:44 +01:00
try? await account.markArticles(articles, statusKey: .starred, flag: true)
2024-03-19 18:15:30 +01:00
try? await account.syncArticleStatus()
if !AccountManager.shared.isSuspended {
try? WidgetDataEncoder.shared.encodeWidgetData()
self.prepareAccountsForBackground()
self.suspendApplication()
}
2024-03-19 18:15:30 +01:00
}
2020-12-23 13:15:25 +01:00
}
}