// // UserFetchedResultsController.swift // Mastodon // // Created by MainasuK Cirno on 2021-7-7. // import os.log import UIKit import Combine import CoreData import CoreDataStack import MastodonSDK final class UserFetchedResultsController: NSObject { var disposeBag = Set() let fetchedResultsController: NSFetchedResultsController // input let domain = CurrentValueSubject(nil) let userIDs = CurrentValueSubject<[Mastodon.Entity.Account.ID], Never>([]) // output let _objectIDs = CurrentValueSubject<[NSManagedObjectID], Never>([]) @Published var records: [ManagedObjectRecord] = [] init(managedObjectContext: NSManagedObjectContext, domain: String?, additionalTweetPredicate: NSPredicate?) { self.domain.value = domain ?? "" self.fetchedResultsController = { let fetchRequest = MastodonUser.sortedFetchRequest fetchRequest.predicate = MastodonUser.predicate(domain: domain ?? "", ids: []) fetchRequest.returnsObjectsAsFaults = false fetchRequest.fetchBatchSize = 20 let controller = NSFetchedResultsController( fetchRequest: fetchRequest, managedObjectContext: managedObjectContext, sectionNameKeyPath: nil, cacheName: nil ) return controller }() super.init() // debounce output to prevent UI update issues _objectIDs .throttle(for: 0.1, scheduler: DispatchQueue.main, latest: true) .map { objectIDs in objectIDs.map { ManagedObjectRecord(objectID: $0) } } .assign(to: &$records) fetchedResultsController.delegate = self Publishers.CombineLatest( self.domain.removeDuplicates(), self.userIDs.removeDuplicates() ) .receive(on: DispatchQueue.main) .sink { [weak self] domain, ids in guard let self = self else { return } var predicates = [MastodonUser.predicate(domain: domain ?? "", ids: ids)] if let additionalPredicate = additionalTweetPredicate { predicates.append(additionalPredicate) } self.fetchedResultsController.fetchRequest.predicate = NSCompoundPredicate(andPredicateWithSubpredicates: predicates) do { try self.fetchedResultsController.performFetch() } catch { assertionFailure(error.localizedDescription) } } .store(in: &disposeBag) } } extension UserFetchedResultsController { public func append(userIDs: [Mastodon.Entity.Account.ID]) { var result = self.userIDs.value for userID in userIDs where !result.contains(userID) { result.append(userID) } self.userIDs.value = result } } // MARK: - NSFetchedResultsControllerDelegate extension UserFetchedResultsController: NSFetchedResultsControllerDelegate { func controller(_ controller: NSFetchedResultsController, didChangeContentWith snapshot: NSDiffableDataSourceSnapshotReference) { os_log("%{public}s[%{public}ld], %{public}s", ((#file as NSString).lastPathComponent), #line, #function) let indexes = userIDs.value let objects = fetchedResultsController.fetchedObjects ?? [] let items: [NSManagedObjectID] = objects .compactMap { object in indexes.firstIndex(of: object.id).map { index in (index, object) } } .sorted { $0.0 < $1.0 } .map { $0.1.objectID } self._objectIDs.value = items } }