forked from zelo72/mastodon-ios
170 lines
7.7 KiB
Swift
170 lines
7.7 KiB
Swift
//
|
|
// HomeTimelineViewModel+Diffable.swift
|
|
// Mastodon
|
|
//
|
|
// Created by sxiaojian on 2021/2/7.
|
|
//
|
|
|
|
import os.log
|
|
import UIKit
|
|
import CoreData
|
|
import CoreDataStack
|
|
|
|
extension HomeTimelineViewModel {
|
|
|
|
func setupDiffableDataSource(
|
|
for tableView: UITableView,
|
|
dependency: NeedsDependency,
|
|
timelinePostTableViewCellDelegate: StatusTableViewCellDelegate,
|
|
timelineMiddleLoaderTableViewCellDelegate: TimelineMiddleLoaderTableViewCellDelegate
|
|
) {
|
|
let timestampUpdatePublisher = Timer.publish(every: 1.0, on: .main, in: .common)
|
|
.autoconnect()
|
|
.share()
|
|
.eraseToAnyPublisher()
|
|
|
|
diffableDataSource = StatusSection.tableViewDiffableDataSource(
|
|
for: tableView,
|
|
dependency: dependency,
|
|
managedObjectContext: fetchedResultsController.managedObjectContext,
|
|
timestampUpdatePublisher: timestampUpdatePublisher,
|
|
timelinePostTableViewCellDelegate: timelinePostTableViewCellDelegate,
|
|
timelineMiddleLoaderTableViewCellDelegate: timelineMiddleLoaderTableViewCellDelegate
|
|
)
|
|
}
|
|
|
|
}
|
|
|
|
// MARK: - NSFetchedResultsControllerDelegate
|
|
extension HomeTimelineViewModel: NSFetchedResultsControllerDelegate {
|
|
|
|
func controllerWillChangeContent(_ controller: NSFetchedResultsController<NSFetchRequestResult>) {
|
|
os_log("%{public}s[%{public}ld], %{public}s", ((#file as NSString).lastPathComponent), #line, #function)
|
|
}
|
|
|
|
func controller(_ controller: NSFetchedResultsController<NSFetchRequestResult>, didChangeContentWith snapshot: NSDiffableDataSourceSnapshotReference) {
|
|
os_log("%{public}s[%{public}ld], %{public}s", ((#file as NSString).lastPathComponent), #line, #function)
|
|
|
|
guard let tableView = self.tableView else { return }
|
|
guard let navigationBar = self.contentOffsetAdjustableTimelineViewControllerDelegate?.navigationBar() else { return }
|
|
|
|
guard let diffableDataSource = self.diffableDataSource else { return }
|
|
let oldSnapshot = diffableDataSource.snapshot()
|
|
|
|
let predicate = fetchedResultsController.fetchRequest.predicate
|
|
let parentManagedObjectContext = fetchedResultsController.managedObjectContext
|
|
let managedObjectContext = NSManagedObjectContext(concurrencyType: .privateQueueConcurrencyType)
|
|
managedObjectContext.parent = parentManagedObjectContext
|
|
|
|
managedObjectContext.perform {
|
|
var shouldAddBottomLoader = false
|
|
|
|
let timelineIndexes: [HomeTimelineIndex] = {
|
|
let request = HomeTimelineIndex.sortedFetchRequest
|
|
request.returnsObjectsAsFaults = false
|
|
request.predicate = predicate
|
|
do {
|
|
return try managedObjectContext.fetch(request)
|
|
} catch {
|
|
assertionFailure(error.localizedDescription)
|
|
return []
|
|
}
|
|
}()
|
|
|
|
// that's will be the most fastest fetch because of upstream just update and no modify needs consider
|
|
|
|
var oldSnapshotAttributeDict: [NSManagedObjectID : Item.StatusTimelineAttribute] = [:]
|
|
|
|
for item in oldSnapshot.itemIdentifiers {
|
|
guard case let .homeTimelineIndex(objectID, attribute) = item else { continue }
|
|
oldSnapshotAttributeDict[objectID] = attribute
|
|
}
|
|
|
|
var newTimelineItems: [Item] = []
|
|
|
|
for (i, timelineIndex) in timelineIndexes.enumerated() {
|
|
let toot = timelineIndex.toot.reblog ?? timelineIndex.toot
|
|
let isStatusTextSensitive: Bool = {
|
|
guard let spoilerText = toot.spoilerText, !spoilerText.isEmpty else { return false }
|
|
return true
|
|
}()
|
|
let attribute = oldSnapshotAttributeDict[timelineIndex.objectID] ?? Item.StatusTimelineAttribute(isStatusTextSensitive: isStatusTextSensitive, isStatusSensitive: toot.sensitive)
|
|
|
|
// append new item into snapshot
|
|
newTimelineItems.append(.homeTimelineIndex(objectID: timelineIndex.objectID, attribute: attribute))
|
|
|
|
let isLast = i == timelineIndexes.count - 1
|
|
switch (isLast, timelineIndex.hasMore) {
|
|
case (false, true):
|
|
newTimelineItems.append(.homeMiddleLoader(upperTimelineIndexAnchorObjectID: timelineIndex.objectID))
|
|
case (true, true):
|
|
shouldAddBottomLoader = true
|
|
default:
|
|
break
|
|
}
|
|
} // end for
|
|
|
|
var newSnapshot = NSDiffableDataSourceSnapshot<StatusSection, Item>()
|
|
newSnapshot.appendSections([.main])
|
|
newSnapshot.appendItems(newTimelineItems, toSection: .main)
|
|
|
|
let endSnapshot = CACurrentMediaTime()
|
|
|
|
DispatchQueue.main.async {
|
|
if shouldAddBottomLoader, !(self.loadoldestStateMachine.currentState is LoadOldestState.NoMore) {
|
|
newSnapshot.appendItems([.bottomLoader], toSection: .main)
|
|
}
|
|
|
|
guard let difference = self.calculateReloadSnapshotDifference(navigationBar: navigationBar, tableView: tableView, oldSnapshot: oldSnapshot, newSnapshot: newSnapshot) else {
|
|
diffableDataSource.apply(newSnapshot)
|
|
self.isFetchingLatestTimeline.value = false
|
|
return
|
|
}
|
|
|
|
diffableDataSource.apply(newSnapshot, animatingDifferences: false) {
|
|
tableView.scrollToRow(at: difference.targetIndexPath, at: .top, animated: false)
|
|
tableView.contentOffset.y = tableView.contentOffset.y - difference.offset
|
|
self.isFetchingLatestTimeline.value = false
|
|
}
|
|
|
|
let end = CACurrentMediaTime()
|
|
os_log("%{public}s[%{public}ld], %{public}s: calculate home timeline layout cost %.2fs", ((#file as NSString).lastPathComponent), #line, #function, end - endSnapshot)
|
|
}
|
|
} // end perform
|
|
}
|
|
|
|
private struct Difference<T> {
|
|
let item: T
|
|
let sourceIndexPath: IndexPath
|
|
let targetIndexPath: IndexPath
|
|
let offset: CGFloat
|
|
}
|
|
|
|
private func calculateReloadSnapshotDifference<T: Hashable>(
|
|
navigationBar: UINavigationBar,
|
|
tableView: UITableView,
|
|
oldSnapshot: NSDiffableDataSourceSnapshot<StatusSection, T>,
|
|
newSnapshot: NSDiffableDataSourceSnapshot<StatusSection, T>
|
|
) -> Difference<T>? {
|
|
guard oldSnapshot.numberOfItems != 0 else { return nil }
|
|
|
|
// old snapshot not empty. set source index path to first item if not match
|
|
let sourceIndexPath = UIViewController.topVisibleTableViewCellIndexPath(in: tableView, navigationBar: navigationBar) ?? IndexPath(row: 0, section: 0)
|
|
|
|
guard sourceIndexPath.row < oldSnapshot.itemIdentifiers(inSection: .main).count else { return nil }
|
|
|
|
let timelineItem = oldSnapshot.itemIdentifiers(inSection: .main)[sourceIndexPath.row]
|
|
guard let itemIndex = newSnapshot.itemIdentifiers(inSection: .main).firstIndex(of: timelineItem) else { return nil }
|
|
let targetIndexPath = IndexPath(row: itemIndex, section: 0)
|
|
|
|
let offset = UIViewController.tableViewCellOriginOffsetToWindowTop(in: tableView, at: sourceIndexPath, navigationBar: navigationBar)
|
|
return Difference(
|
|
item: timelineItem,
|
|
sourceIndexPath: sourceIndexPath,
|
|
targetIndexPath: targetIndexPath,
|
|
offset: offset
|
|
)
|
|
}
|
|
|
|
}
|