mirror of
https://github.com/metabolist/metatext.git
synced 2025-01-08 04:55:25 +00:00
111 lines
4.2 KiB
Swift
111 lines
4.2 KiB
Swift
// Copyright © 2020 Metabolist. All rights reserved.
|
|
|
|
import Combine
|
|
import Foundation
|
|
import ServiceLayer
|
|
|
|
public final class RootViewModel: ObservableObject {
|
|
@Published public private(set) var navigationViewModel: NavigationViewModel?
|
|
|
|
@Published private var mostRecentlyUsedIdentityID: UUID?
|
|
private let environment: AppEnvironment
|
|
private let allIdentitiesService: AllIdentitiesService
|
|
private let userNotificationService: UserNotificationService
|
|
private let registerForRemoteNotifications: () -> AnyPublisher<Data, Error>
|
|
private var cancellables = Set<AnyCancellable>()
|
|
|
|
public init(environment: AppEnvironment,
|
|
registerForRemoteNotifications: @escaping () -> AnyPublisher<Data, Error>) throws {
|
|
self.environment = environment
|
|
allIdentitiesService = try AllIdentitiesService(environment: environment)
|
|
userNotificationService = UserNotificationService(environment: environment)
|
|
self.registerForRemoteNotifications = registerForRemoteNotifications
|
|
|
|
allIdentitiesService.immediateMostRecentlyUsedIdentityIDObservation()
|
|
.replaceError(with: nil)
|
|
.assign(to: &$mostRecentlyUsedIdentityID)
|
|
|
|
identitySelected(id: mostRecentlyUsedIdentityID, immediate: true)
|
|
|
|
allIdentitiesService.identitiesCreated
|
|
.sink { [weak self] in self?.identitySelected(id: $0) }
|
|
.store(in: &cancellables)
|
|
|
|
userNotificationService.isAuthorized()
|
|
.filter { $0 }
|
|
.zip(registerForRemoteNotifications())
|
|
.map { $1 }
|
|
.flatMap(allIdentitiesService.updatePushSubscriptions(deviceToken:))
|
|
.sink { _ in } receiveValue: { _ in }
|
|
.store(in: &cancellables)
|
|
}
|
|
}
|
|
|
|
public extension RootViewModel {
|
|
func identitySelected(id: UUID?) {
|
|
identitySelected(id: id, immediate: false)
|
|
}
|
|
|
|
func deleteIdentity(id: UUID) {
|
|
allIdentitiesService.deleteIdentity(id: id)
|
|
.sink { _ in } receiveValue: { _ in }
|
|
.store(in: &cancellables)
|
|
}
|
|
|
|
func addIdentityViewModel() -> AddIdentityViewModel {
|
|
AddIdentityViewModel(
|
|
allIdentitiesService: allIdentitiesService,
|
|
instanceURLService: InstanceURLService(environment: environment))
|
|
}
|
|
}
|
|
|
|
private extension RootViewModel {
|
|
func identitySelected(id: UUID?, immediate: Bool) {
|
|
navigationViewModel?.presentingSecondaryNavigation = false
|
|
|
|
guard
|
|
let id = id,
|
|
let identityService = try? allIdentitiesService.identityService(id: id) else {
|
|
navigationViewModel = nil
|
|
|
|
return
|
|
}
|
|
|
|
let observation = identityService.observation(immediate: immediate)
|
|
.catch { [weak self] _ -> Empty<Identity, Never> in
|
|
DispatchQueue.main.async {
|
|
self?.identitySelected(id: self?.mostRecentlyUsedIdentityID, immediate: false)
|
|
}
|
|
|
|
return Empty()
|
|
}
|
|
.share()
|
|
|
|
observation
|
|
.filter { [weak self] in $0.id != self?.navigationViewModel?.identification.identity.id }
|
|
.map { [weak self] in
|
|
let identification = Identification(
|
|
identity: $0,
|
|
observation: observation.eraseToAnyPublisher(),
|
|
service: identityService)
|
|
|
|
if let self = self {
|
|
identification.service.updateLastUse()
|
|
.sink { _ in } receiveValue: { _ in }
|
|
.store(in: &self.cancellables)
|
|
|
|
self.userNotificationService.isAuthorized()
|
|
.filter { $0 }
|
|
.zip(self.registerForRemoteNotifications())
|
|
.filter { identification.identity.lastRegisteredDeviceToken != $1 }
|
|
.map { ($1, identification.identity.pushSubscriptionAlerts) }
|
|
.flatMap(identification.service.createPushSubscription(deviceToken:alerts:))
|
|
.sink { _ in } receiveValue: { _ in }
|
|
.store(in: &self.cancellables)
|
|
}
|
|
|
|
return NavigationViewModel(identification: identification)
|
|
}
|
|
.assign(to: &$navigationViewModel)
|
|
}
|
|
}
|