// // SettingService.swift // Mastodon // // Created by MainasuK Cirno on 2021-4-25. // import os.log import UIKit import Combine import CoreDataStack import MastodonSDK import MastodonAsset import MastodonLocalization import MastodonCommon public final class SettingService { var disposeBag = Set() private var currentSettingUpdateSubscription: AnyCancellable? // input weak var apiService: APIService? weak var authenticationService: AuthenticationService? weak var notificationService: NotificationService? // output let settingFetchedResultController: SettingFetchedResultController public let currentSetting = CurrentValueSubject(nil) init( apiService: APIService, authenticationService: AuthenticationService, notificationService: NotificationService ) { self.apiService = apiService self.authenticationService = authenticationService self.notificationService = notificationService self.settingFetchedResultController = SettingFetchedResultController( managedObjectContext: authenticationService.managedObjectContext, additionalPredicate: nil ) // create setting (if non-exist) for authenticated users authenticationService.$mastodonAuthenticationBoxes .compactMap { [weak self] mastodonAuthenticationBoxes -> AnyPublisher<[MastodonAuthenticationBox], Never>? in guard let self = self else { return nil } guard let authenticationService = self.authenticationService else { return nil } let managedObjectContext = authenticationService.backgroundManagedObjectContext return managedObjectContext.performChanges { for authenticationBox in mastodonAuthenticationBoxes { let domain = authenticationBox.domain let userID = authenticationBox.userID _ = APIService.CoreData.createOrMergeSetting( into: managedObjectContext, property: Setting.Property( domain: domain, userID: userID ) ) } // end for } .map { _ in mastodonAuthenticationBoxes } .eraseToAnyPublisher() } .sink { _ in // do nothing } .store(in: &disposeBag) // bind current setting Publishers.CombineLatest( authenticationService.$mastodonAuthenticationBoxes, settingFetchedResultController.settings ) .sink { [weak self] mastodonAuthenticationBoxes, settings in guard let self = self else { return } guard let activeMastodonAuthenticationBox = mastodonAuthenticationBoxes.first else { return } let currentSetting = settings.first(where: { setting in return setting.domain == activeMastodonAuthenticationBox.domain && setting.userID == activeMastodonAuthenticationBox.userID }) self.currentSetting.value = currentSetting } .store(in: &disposeBag) // observe current setting currentSetting .receive(on: DispatchQueue.main) .sink { [weak self] setting in guard let self = self else { return } guard let setting = setting else { self.currentSettingUpdateSubscription = nil return } SettingService.updatePreference(setting: setting) self.currentSettingUpdateSubscription = ManagedObjectObserver.observe(object: setting) .sink(receiveCompletion: { _ in // do nothing }, receiveValue: { change in guard case .update(let object) = change.changeType, let setting = object as? Setting else { return } SettingService.updatePreference(setting: setting) }) } .store(in: &disposeBag) let logger = Logger(subsystem: "Notification", category: "SettingService") Publishers.CombineLatest3( notificationService.deviceToken, currentSetting.eraseToAnyPublisher(), authenticationService.$mastodonAuthenticationBoxes ) .compactMap { [weak self] deviceToken, setting, mastodonAuthenticationBoxes -> AnyPublisher, Error>? in guard let self = self else { return nil } guard let deviceToken = deviceToken else { return nil } guard let setting = setting else { return nil } guard let authenticationBox = mastodonAuthenticationBoxes.first else { return nil } guard let subscription = setting.activeSubscription else { return nil } guard setting.domain == authenticationBox.domain, setting.userID == authenticationBox.userID else { return nil } let _viewModel = self.notificationService?.dequeueNotificationViewModel( mastodonAuthenticationBox: authenticationBox ) guard let viewModel = _viewModel else { return nil } let queryData = Mastodon.API.Subscriptions.QueryData( policy: subscription.policy, alerts: Mastodon.API.Subscriptions.QueryData.Alerts( favourite: subscription.alert.favourite, follow: subscription.alert.follow, reblog: subscription.alert.reblog, mention: subscription.alert.mention, poll: subscription.alert.poll ) ) let query = viewModel.createSubscribeQuery( deviceToken: deviceToken, queryData: queryData, mastodonAuthenticationBox: authenticationBox ) return apiService.createSubscription( subscriptionObjectID: subscription.objectID, query: query, mastodonAuthenticationBox: authenticationBox ) } .debounce(for: .seconds(3), scheduler: DispatchQueue.main) // limit subscribe request emit time interval .sink(receiveValue: { [weak self] publisher in guard let self = self else { return } publisher .sink { completion in switch completion { case .failure(let error): logger.log(level: .debug, "\((#file as NSString).lastPathComponent, privacy: .public)[\(#line, privacy: .public)], \(#function, privacy: .public): [Push Notification] subscribe failure: \(error.localizedDescription)") case .finished: logger.log(level: .debug, "\((#file as NSString).lastPathComponent, privacy: .public)[\(#line, privacy: .public)], \(#function, privacy: .public): [Push Notification] subscribe success") } } receiveValue: { response in logger.log(level: .debug, "\((#file as NSString).lastPathComponent, privacy: .public)[\(#line, privacy: .public)], \(#function, privacy: .public): subscribe response: \(response.value.endpoint)") } .store(in: &self.disposeBag) }) .store(in: &disposeBag) } } extension SettingService { public static func openSettingsAlertController(title: String, message: String) -> UIAlertController { let alertController = UIAlertController(title: title, message: message, preferredStyle: .alert) let settingAction = UIAlertAction(title: L10n.Common.Controls.Actions.settings, style: .default) { _ in guard let url = URL(string: UIApplication.openSettingsURLString) else { return } UIApplication.shared.open(url, options: [:], completionHandler: nil) } alertController.addAction(settingAction) let cancelAction = UIAlertAction(title: L10n.Common.Controls.Actions.cancel, style: .cancel, handler: nil) alertController.addAction(cancelAction) return alertController } } extension SettingService { static func updatePreference(setting: Setting) { // set theme let themeName: ThemeName = .system if UserDefaults.shared.currentThemeNameRawValue != themeName.rawValue { ThemeService.shared.set(themeName: themeName) os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s: update theme style", ((#file as NSString).lastPathComponent), #line, #function) } // set avatar mode if UserDefaults.shared.preferredStaticAvatar != setting.preferredStaticAvatar { UserDefaults.shared.preferredStaticAvatar = setting.preferredStaticAvatar } // set emoji mode if UserDefaults.shared.preferredStaticEmoji != setting.preferredStaticEmoji { UserDefaults.shared.preferredStaticEmoji = setting.preferredStaticEmoji } // set browser if UserDefaults.shared.preferredUsingDefaultBrowser != setting.preferredUsingDefaultBrowser { UserDefaults.shared.preferredUsingDefaultBrowser = setting.preferredUsingDefaultBrowser } } }