// // HomeViewController.swift // Reader // // Created by Shadowfacts on 11/25/21. // import UIKit import CoreData class HomeViewController: UIViewController { let fervorController: FervorController private var collectionView: UICollectionView! private var dataSource: UICollectionViewDiffableDataSource! private var groupResultsController: NSFetchedResultsController! private var feedResultsController: NSFetchedResultsController! init(fervorController: FervorController) { self.fervorController = fervorController super.init(nibName: nil, bundle: nil) } required init?(coder: NSCoder) { fatalError("init(coder:) has not been implemented") } override func viewDidLoad() { super.viewDidLoad() // todo: account info title = "Reader" var config = UICollectionLayoutListConfiguration(appearance: .grouped) config.headerMode = .supplementary config.backgroundColor = .appBackground config.separatorConfiguration.topSeparatorVisibility = .visible config.separatorConfiguration.bottomSeparatorVisibility = .hidden config.itemSeparatorHandler = { indexPath, defaultConfig in var config = defaultConfig if indexPath.section == 0 && indexPath.row == 0 { config.topSeparatorVisibility = .hidden } return config } let layout = UICollectionViewCompositionalLayout.list(using: config) collectionView = UICollectionView(frame: view.bounds, collectionViewLayout: layout) collectionView.autoresizingMask = [.flexibleWidth, .flexibleHeight] collectionView.delegate = self view.addSubview(collectionView) dataSource = createDataSource() var snapshot = NSDiffableDataSourceSnapshot() snapshot.appendSections([.all, .groups, .feeds]) snapshot.appendItems([.unread, .all], toSection: .all) dataSource.apply(snapshot, animatingDifferences: false) let groupReq = Group.fetchRequest() groupReq.sortDescriptors = [NSSortDescriptor(key: "title", ascending: true)] groupResultsController = NSFetchedResultsController(fetchRequest: groupReq, managedObjectContext: fervorController.persistentContainer.viewContext, sectionNameKeyPath: nil, cacheName: nil) groupResultsController.delegate = self try! groupResultsController.performFetch() let feedReq = Feed.fetchRequest() feedReq.sortDescriptors = [NSSortDescriptor(key: "title", ascending: true)] feedResultsController = NSFetchedResultsController(fetchRequest: feedReq, managedObjectContext: fervorController.persistentContainer.viewContext, sectionNameKeyPath: nil, cacheName: nil) feedResultsController.delegate = self try! feedResultsController.performFetch() } override func viewWillAppear(_ animated: Bool) { super.viewWillAppear(animated) if let indexPaths = collectionView.indexPathsForSelectedItems { for indexPath in indexPaths { collectionView.deselectItem(at: indexPath, animated: true) } } var snapshot = dataSource.snapshot() // reconfigure so that unread counts update snapshot.reconfigureItems(snapshot.itemIdentifiers) dataSource.apply(snapshot) } private func createDataSource() -> UICollectionViewDiffableDataSource { let sectionHeaderCell = UICollectionView.SupplementaryRegistration(elementKind: UICollectionView.elementKindSectionHeader) { supplementaryView, elementKind, indexPath in let section = self.dataSource.sectionIdentifier(for: indexPath.section)! var config = supplementaryView.defaultContentConfiguration() config.text = section.title supplementaryView.contentConfiguration = config } let listCell = UICollectionView.CellRegistration { cell, indexPath, item in var config = UIListContentConfiguration.valueCell() config.text = item.title if let req = item.countFetchRequest, let count = try? self.fervorController.persistentContainer.viewContext.count(for: req) { config.secondaryText = "\(count)" config.secondaryTextProperties.color = .tintColor } cell.contentConfiguration = config cell.accessories = [.disclosureIndicator()] } let dataSource = UICollectionViewDiffableDataSource(collectionView: collectionView) { collectionView, indexPath, item in return collectionView.dequeueConfiguredReusableCell(using: listCell, for: indexPath, item: item) } dataSource.supplementaryViewProvider = { collectionView, elementKind, indexPath in if elementKind == UICollectionView.elementKindSectionHeader { return collectionView.dequeueConfiguredReusableSupplementary(using: sectionHeaderCell, for: indexPath) } else { return nil } } return dataSource } } extension HomeViewController { enum Section: Hashable { case all case groups case feeds var title: String { switch self { case .all: return "" case .groups: return "Groups" case .feeds: return "Feeds" } } } enum Item: Hashable { case unread case all case group(Group) case feed(Feed) var title: String { switch self { case .unread: return "Unread Articles" case .all: return "All Articles" case let .group(group): return group.title case let .feed(feed): return feed.title! } } var fetchRequest: NSFetchRequest { let req = Reader.Item.fetchRequest() switch self { case .unread: req.predicate = NSPredicate(format: "read = NO") case .all: break case .group(let group): req.predicate = NSPredicate(format: "feed in %@", group.feeds!) case .feed(let feed): req.predicate = NSPredicate(format: "feed = %@", feed) } return req } var countFetchRequest: NSFetchRequest? { let req = Reader.Item.fetchRequest() switch self { case .unread: req.predicate = NSPredicate(format: "read = NO") case .all: return nil case .group(let group): req.predicate = NSPredicate(format: "read = NO AND feed in %@", group.feeds!) case .feed(let feed): req.predicate = NSPredicate(format: "read = NO AND feed = %@", feed) } return req } } } extension HomeViewController: NSFetchedResultsControllerDelegate { func controller(_ controller: NSFetchedResultsController, didChangeContentWith snapshot: NSDiffableDataSourceSnapshotReference) { var snapshot = dataSource.snapshot() if controller == groupResultsController { if snapshot.sectionIdentifiers.contains(.groups) { snapshot.deleteItems(snapshot.itemIdentifiers(inSection: .groups)) } snapshot.appendItems(controller.fetchedObjects!.map { .group($0 as! Group) }, toSection: .groups) } else if controller == feedResultsController { if snapshot.sectionIdentifiers.contains(.feeds) { snapshot.deleteItems(snapshot.itemIdentifiers(inSection: .feeds)) } snapshot.appendItems(controller.fetchedObjects!.map { .feed($0 as! Feed) }, toSection: .feeds) } dataSource.apply(snapshot, animatingDifferences: false) } } extension HomeViewController: UICollectionViewDelegate { func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) { guard let item = dataSource.itemIdentifier(for: indexPath) else { return } let vc = ItemsViewController(fetchRequest: item.fetchRequest, fervorController: fervorController) vc.title = item.title show(vc, sender: nil) } func collectionView(_ collectionView: UICollectionView, contextMenuConfigurationForItemAt indexPath: IndexPath, point: CGPoint) -> UIContextMenuConfiguration? { guard let item = dataSource.itemIdentifier(for: indexPath) else { return nil } return UIContextMenuConfiguration(identifier: nil, previewProvider: { return ItemsViewController(fetchRequest: item.fetchRequest, fervorController: self.fervorController) }, actionProvider: nil) } func collectionView(_ collectionView: UICollectionView, willPerformPreviewActionForMenuWith configuration: UIContextMenuConfiguration, animator: UIContextMenuInteractionCommitAnimating) { guard let vc = animator.previewViewController else { return } animator.preferredCommitStyle = .pop animator.addCompletion { self.show(vc, sender: nil) } } }