// // AppDelegate.swift // Mastodon // // Created by MainasuK Cirno on 2021/1/22. // import UIKit import UserNotifications import AVFoundation import MastodonCore import MastodonUI @main class AppDelegate: UIResponder, UIApplicationDelegate { let appContext = AppContext() func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool { AuthenticationServiceProvider.shared.restore() AppSecret.default.register() // configure appearance ThemeService.shared.apply() // configure AudioSession try? AVAudioSession.sharedInstance().setCategory(.ambient) // Update app version info. See: `Settings.bundle` UserDefaults.standard.setValue(UIApplication.appVersion(), forKey: "Mastodon.appVersion") UserDefaults.standard.setValue(UIApplication.appBuild(), forKey: "Mastodon.appBundle") // Setup notification UNUserNotificationCenter.current().delegate = self application.registerForRemoteNotifications() // increase app process count var count = UserDefaults.shared.processCompletedCount count += 1 // Int64. could ignore overflow here UserDefaults.shared.processCompletedCount = count return true } // MARK: UISceneSession Lifecycle func application(_ application: UIApplication, configurationForConnecting connectingSceneSession: UISceneSession, options: UIScene.ConnectionOptions) -> UISceneConfiguration { // Called when a new scene session is being created. // Use this method to select a configuration to create the new scene with. return UISceneConfiguration(name: "Default Configuration", sessionRole: connectingSceneSession.role) } func application(_ application: UIApplication, didDiscardSceneSessions sceneSessions: Set) { // Called when the user discards a scene session. // If any sessions were discarded while the application was not running, this will be called shortly after application:didFinishLaunchingWithOptions. // Use this method to release any resources that were specific to the discarded scenes, as they will not return. } func application(_ app: UIApplication, open url: URL, options: [UIApplication.OpenURLOptionsKey : Any] = [:]) -> Bool { return true } } extension AppDelegate { func application(_ application: UIApplication, didRegisterForRemoteNotificationsWithDeviceToken deviceToken: Data) { appContext.notificationService.deviceToken.value = deviceToken } } // MARK: - UNUserNotificationCenterDelegate extension AppDelegate: UNUserNotificationCenterDelegate { // notification present in the foreground func userNotificationCenter( _ center: UNUserNotificationCenter, willPresent notification: UNNotification, withCompletionHandler completionHandler: @escaping (UNNotificationPresentationOptions) -> Void ) { guard let pushNotification = AppDelegate.mastodonPushNotification(from: notification) else { completionHandler([]) return } let accessToken = pushNotification.accessToken UserDefaults.shared.increaseNotificationCount(accessToken: accessToken) appContext.notificationService.applicationIconBadgeNeedsUpdate.send() appContext.notificationService.handle(pushNotification: pushNotification) completionHandler([.sound]) } // notification present in the background (or resume from background) func application(_ application: UIApplication, didReceiveRemoteNotification userInfo: [AnyHashable : Any]) async -> UIBackgroundFetchResult { let shortcutItems = try? await appContext.notificationService.unreadApplicationShortcutItems() UIApplication.shared.shortcutItems = shortcutItems return .noData } // response to user action for notification (e.g. redirect to post) func userNotificationCenter( _ center: UNUserNotificationCenter, didReceive response: UNNotificationResponse, withCompletionHandler completionHandler: @escaping () -> Void ) { guard let pushNotification = AppDelegate.mastodonPushNotification(from: response.notification) else { completionHandler() return } appContext.notificationService.handle(pushNotification: pushNotification) appContext.notificationService.requestRevealNotificationPublisher.send(pushNotification) completionHandler() } private static func mastodonPushNotification(from notification: UNNotification) -> MastodonPushNotification? { guard let plaintext = notification.request.content.userInfo["plaintext"] as? Data, let mastodonPushNotification = try? JSONDecoder().decode(MastodonPushNotification.self, from: plaintext) else { return nil } return mastodonPushNotification } } extension AppContext { static var shared: AppContext { let appDelegate = UIApplication.shared.delegate as! AppDelegate return appDelegate.appContext } }