Forked from darrarski/FetchedResultsPublisher.swift
Last active
January 30, 2022 13:42
-
-
Save eivindml/11dc16ca0edef8c9242d6c76f9f63ffc to your computer and use it in GitHub Desktop.
Swift-Combine-CoreData-Fetched-Results-Publisher
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import Combine | |
import CoreData | |
public final class FetchedResultsPublisher<ResultType>: Publisher where ResultType: NSFetchRequestResult { | |
public init(request: NSFetchRequest<ResultType>, context: NSManagedObjectContext) { | |
self.request = request | |
self.context = context | |
} | |
let request: NSFetchRequest<ResultType> | |
let context: NSManagedObjectContext | |
// MARK: - Publisher | |
public typealias Output = [ResultType] | |
public typealias Failure = NSError | |
public func receive<S>(subscriber: S) where S: Subscriber, S.Failure == Failure, S.Input == Output { | |
subscriber.receive(subscription: FetchedResultsSubscription( | |
subscriber: subscriber, | |
request: request, | |
context: context | |
)) | |
} | |
} | |
final class FetchedResultsSubscription<SubscriberType, ResultType>: NSObject, Subscription, NSFetchedResultsControllerDelegate where SubscriberType: Subscriber, SubscriberType.Input == [ResultType], SubscriberType.Failure == NSError, ResultType: NSFetchRequestResult { | |
init(subscriber: SubscriberType, request: NSFetchRequest<ResultType>, context: NSManagedObjectContext) { | |
self.subscriber = subscriber | |
self.request = request | |
self.context = context | |
} | |
private(set) var subscriber: SubscriberType? | |
private(set) var request: NSFetchRequest<ResultType>? | |
private(set) var context: NSManagedObjectContext? | |
private(set) var controller: NSFetchedResultsController<ResultType>? | |
// MARK: - Subscription | |
func request(_ demand: Subscribers.Demand) { | |
guard demand > 0, | |
let subscriber = subscriber, | |
let request = request, | |
let context = context else { return } | |
controller = NSFetchedResultsController( | |
fetchRequest: request, | |
managedObjectContext: context, | |
sectionNameKeyPath: nil, | |
cacheName: nil | |
) | |
controller?.delegate = self | |
do { | |
try controller?.performFetch() | |
if let fetchedObjects = controller?.fetchedObjects { | |
_ = subscriber.receive(fetchedObjects) | |
} | |
} catch { | |
subscriber.receive(completion: .failure(error as NSError)) | |
} | |
} | |
// MARK: - Cancellable | |
func cancel() { | |
subscriber = nil | |
controller = nil | |
request = nil | |
context = nil | |
} | |
// MARK: - NSFetchedResultsControllerDelegate | |
func controllerDidChangeContent(_ controller: NSFetchedResultsController<NSFetchRequestResult>) { | |
guard let subscriber = subscriber, | |
controller == self.controller else { return } | |
if let fetchedObjects = self.controller?.fetchedObjects { | |
_ = subscriber.receive(fetchedObjects) | |
} | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment