// // DiffableDataSource.TableView.swift // CoreStore // // Copyright © 2018 John Rommel Estropia // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in all // copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE // SOFTWARE. // #if canImport(UIKit) import UIKit import CoreData // MARK: - DiffableDataSource extension DiffableDataSource { // MARK: - TableView open class TableView: NSObject, UITableViewDataSource { // MARK: Open @nonobjc open var defaultRowAnimation: UITableView.RowAnimation = .automatic // MARK: Public public typealias ObjectType = O @nonobjc public init(tableView: UITableView, dataStack: DataStack, cellProvider: @escaping (UITableView, IndexPath, O) -> UITableViewCell?) { self.tableView = tableView self.cellProvider = cellProvider self.dataStack = dataStack super.init() // if #available(iOS 13.0, tvOS 13.0, watchOS 6.0, macOS 10.15, *) { // // self.rawDataSource = UITableViewDiffableDataSource( // tableView: tableView, // cellProvider: { [weak self] (tableView, indexPath, objectID) -> UITableViewCell? in // // guard let self = self else { // // return nil // } // guard let object = self.dataStack.fetchExisting(objectID) as O? else { // // return nil // } // return self.cellProvider(tableView, indexPath, object) // } // ) // } // else { self.rawDataSource = Internals.DiffableDataUIDispatcher(dataStack: dataStack) // } tableView.dataSource = self } public func apply(_ snapshot: ListSnapshot, animatingDifferences: Bool = true) { let diffableSnapshot = snapshot.diffableSnapshot // if #available(iOS 13.0, tvOS 13.0, watchOS 6.0, macOS 10.15, *) { // // self.modernDataSource.apply( // diffableSnapshot as! NSDiffableDataSourceSnapshot, // animatingDifferences: animatingDifferences, // completion: nil // ) // } // else { self.legacyDataSource.apply( diffableSnapshot as! Internals.DiffableDataSourceSnapshot, view: self.tableView, animatingDifferences: animatingDifferences, performUpdates: { tableView, changeset, setSections in tableView.reload( using: changeset, with: self.defaultRowAnimation, setData: setSections ) } ) // } } public func itemIdentifier(for indexPath: IndexPath) -> O.ObjectID? { // if #available(iOS 13.0, tvOS 13.0, watchOS 6.0, macOS 10.15, *) { // // return self.modernDataSource.itemIdentifier(for: indexPath) // } // else { return self.legacyDataSource.itemIdentifier(for: indexPath) // } } public func indexPath(for itemIdentifier: O.ObjectID) -> IndexPath? { // if #available(iOS 13.0, tvOS 13.0, watchOS 6.0, macOS 10.15, *) { // // return self.modernDataSource.indexPath(for: itemIdentifier) // } // else { return self.legacyDataSource.indexPath(for: itemIdentifier) // } } // MARK: - UITableViewDataSource @objc public dynamic func numberOfSections(in tableView: UITableView) -> Int { // if #available(iOS 13.0, tvOS 13.0, watchOS 6.0, macOS 10.15, *) { // // return self.modernDataSource.numberOfSections(in: tableView) // } // else { return self.legacyDataSource.numberOfSections() // } } @objc public dynamic func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { // if #available(iOS 13.0, tvOS 13.0, watchOS 6.0, macOS 10.15, *) { // // return self.modernDataSource.tableView(tableView, numberOfRowsInSection: section) // } // else { return self.legacyDataSource.numberOfItems(inSection: section) // } } @objc open dynamic func tableView(_ tableView: UITableView, titleForHeaderInSection section: Int) -> String? { // if #available(iOS 13.0, tvOS 13.0, watchOS 6.0, macOS 10.15, *) { // // return self.modernDataSource.snapshot().sectionIdentifiers[section] // } // else { return self.legacyDataSource.sectionIdentifier(inSection: section) // } } @objc open dynamic func tableView(_ tableView: UITableView, titleForFooterInSection section: Int) -> String? { // if #available(iOS 13.0, tvOS 13.0, watchOS 6.0, macOS 10.15, *) { // // return self.modernDataSource.tableView(tableView, titleForFooterInSection: section) // } // else { return nil // } } @objc open dynamic func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { // if #available(iOS 13.0, tvOS 13.0, watchOS 6.0, macOS 10.15, *) { // // return self.modernDataSource.tableView(tableView, cellForRowAt: indexPath) // } // else { guard let objectID = self.legacyDataSource.itemIdentifier(for: indexPath) else { Internals.abort("Object at \(Internals.typeName(IndexPath.self)) \(indexPath) already removed from list") } guard let object = self.dataStack.fetchExisting(objectID) as O? else { Internals.abort("Object at \(Internals.typeName(IndexPath.self)) \(indexPath) has been deleted") } guard let cell = self.cellProvider(tableView, indexPath, object) else { Internals.abort("\(Internals.typeName(UITableViewDataSource.self)) returned a `nil` cell for \(Internals.typeName(IndexPath.self)) \(indexPath)") } return cell // } } // MARK: Private private weak var tableView: UITableView? private let dataStack: DataStack private let cellProvider: (UITableView, IndexPath, O) -> UITableViewCell? private var rawDataSource: Any! // @available(iOS 13.0, tvOS 13.0, watchOS 6.0, macOS 10.15, *) // private var modernDataSource: UITableViewDiffableDataSource { // // return self.rawDataSource as! UITableViewDiffableDataSource // } private var legacyDataSource: Internals.DiffableDataUIDispatcher { return self.rawDataSource as! Internals.DiffableDataUIDispatcher } } } // MARK: - UITableView extension UITableView { // MARK: FilePrivate @nonobjc fileprivate func reload( using stagedChangeset: Internals.DiffableDataUIDispatcher.StagedChangeset, with animation: @autoclosure () -> RowAnimation, interrupt: ((Internals.DiffableDataUIDispatcher.Changeset) -> Bool)? = nil, setData: (C) -> Void ) { self.reload( using: stagedChangeset, deleteSectionsAnimation: animation(), insertSectionsAnimation: animation(), reloadSectionsAnimation: animation(), deleteRowsAnimation: animation(), insertRowsAnimation: animation(), reloadRowsAnimation: animation(), interrupt: interrupt, setData: setData ) } @nonobjc fileprivate func reload( using stagedChangeset: Internals.DiffableDataUIDispatcher.StagedChangeset, deleteSectionsAnimation: @autoclosure () -> RowAnimation, insertSectionsAnimation: @autoclosure () -> RowAnimation, reloadSectionsAnimation: @autoclosure () -> RowAnimation, deleteRowsAnimation: @autoclosure () -> RowAnimation, insertRowsAnimation: @autoclosure () -> RowAnimation, reloadRowsAnimation: @autoclosure () -> RowAnimation, interrupt: ((Internals.DiffableDataUIDispatcher.Changeset) -> Bool)? = nil, setData: (C) -> Void ) { if case .none = window, let data = stagedChangeset.last?.data { setData(data) self.reloadData() return } for changeset in stagedChangeset { if let interrupt = interrupt, interrupt(changeset), let data = stagedChangeset.last?.data { setData(data) self.reloadData() return } self.cs_performBatchUpdates { setData(changeset.data) if !changeset.sectionDeleted.isEmpty { self.deleteSections(IndexSet(changeset.sectionDeleted), with: deleteSectionsAnimation()) } if !changeset.sectionInserted.isEmpty { self.insertSections(IndexSet(changeset.sectionInserted), with: insertSectionsAnimation()) } if !changeset.sectionUpdated.isEmpty { self.reloadSections(IndexSet(changeset.sectionUpdated), with: reloadSectionsAnimation()) } for (source, target) in changeset.sectionMoved { self.moveSection(source, toSection: target) } if !changeset.elementDeleted.isEmpty { self.deleteRows(at: changeset.elementDeleted.map { IndexPath(row: $0.element, section: $0.section) }, with: deleteRowsAnimation()) } if !changeset.elementInserted.isEmpty { self.insertRows(at: changeset.elementInserted.map { IndexPath(row: $0.element, section: $0.section) }, with: insertRowsAnimation()) } if !changeset.elementUpdated.isEmpty { self.reloadRows(at: changeset.elementUpdated.map { IndexPath(row: $0.element, section: $0.section) }, with: reloadRowsAnimation()) } for (source, target) in changeset.elementMoved { self.moveRow(at: IndexPath(row: source.element, section: source.section), to: IndexPath(row: target.element, section: target.section)) } } } } @nonobjc private func cs_performBatchUpdates(_ updates: () -> Void) { if #available(iOS 11.0, tvOS 11.0, *) { self.performBatchUpdates(updates) } else { self.beginUpdates() updates() self.endUpdates() } } } #endif