metatext/Data Sources/TableViewDataSource.swift
2022-11-09 22:21:39 -08:00

90 lines
4 KiB
Swift

// Copyright © 2020 Metabolist. All rights reserved.
import UIKit
import ViewModels
final class TableViewDataSource: UITableViewDiffableDataSource<CollectionSection.Identifier, CollectionItem> {
private let updateQueue =
DispatchQueue(label: "com.metabolist.metatext.collection-data-source.update-queue")
private let viewModel: CollectionViewModel
init(tableView: UITableView, viewModel: CollectionViewModel) {
self.viewModel = viewModel
for cellClass in CollectionItem.cellClasses {
tableView.register(cellClass, forCellReuseIdentifier: String(describing: cellClass))
}
super.init(tableView: tableView) { tableView, indexPath, item in
let cell = tableView.dequeueReusableCell(
withIdentifier: String(describing: item.cellClass),
for: indexPath)
switch (cell, viewModel.viewModel(indexPath: indexPath)) {
case let (statusCell as StatusTableViewCell, statusViewModel as StatusViewModel):
statusCell.viewModel = statusViewModel
case let (accountCell as AccountTableViewCell, accountViewModel as AccountViewModel):
accountCell.viewModel = accountViewModel
case let (loadMoreCell as LoadMoreTableViewCell, loadMoreViewModel as LoadMoreViewModel):
loadMoreCell.viewModel = loadMoreViewModel
case let (notificationCell as NotificationTableViewCell, notificationViewModel as NotificationViewModel):
notificationCell.viewModel = notificationViewModel
case let (conversationCell as ConversationTableViewCell, conversationViewModel as ConversationViewModel):
conversationCell.viewModel = conversationViewModel
case let (tagCell as TagTableViewCell, tagViewModel as TagViewModel):
tagCell.viewModel = tagViewModel
case let (announcementCell as AnnouncementTableViewCell, announcementViewModel as AnnouncementViewModel):
announcementCell.viewModel = announcementViewModel
case let (_, moreResultsViewModel as MoreResultsViewModel):
var configuration = cell.defaultContentConfiguration()
let statusWord = viewModel.identityContext.appPreferences.statusWord
configuration.text = moreResultsViewModel.scope.moreDescription(statusWord: statusWord)
cell.contentConfiguration = configuration
cell.accessoryType = .disclosureIndicator
default:
break
}
return cell
}
}
override func apply(_ snapshot: NSDiffableDataSourceSnapshot<CollectionSection.Identifier, CollectionItem>,
animatingDifferences: Bool = true,
completion: (() -> Void)? = nil) {
updateQueue.async {
super.apply(snapshot, animatingDifferences: animatingDifferences, completion: completion)
}
}
override func applySnapshotUsingReloadData(
_ snapshot: NSDiffableDataSourceSnapshot<CollectionSection.Identifier, CollectionItem>,
completion: (() -> Void)? = nil) {
updateQueue.async {
super.applySnapshotUsingReloadData(snapshot, completion: completion)
}
}
override func tableView(_ tableView: UITableView, titleForHeaderInSection section: Int) -> String? {
let currentSnapshot = snapshot()
let section = currentSnapshot.sectionIdentifiers[section]
if currentSnapshot.numberOfItems(inSection: section) > 0,
let searchScope = section.searchScope {
return searchScope.title(statusWord: viewModel.identityContext.appPreferences.statusWord)
}
return nil
}
}
extension TableViewDataSource {
func indexPath(itemId: CollectionItem.Id) -> IndexPath? {
guard let item = snapshot().itemIdentifiers.first(where: { $0.itemId == itemId }) else { return nil }
return indexPath(for: item)
}
}