// // SidebarModel.swift // NetNewsWire // // Created by Maurice Parker on 6/28/20. // Copyright © 2020 Ranchero Software. All rights reserved. // import Foundation import Combine import RSCore import Account protocol SidebarModelDelegate: class { func unreadCount(for: Feed) -> Int } class SidebarModel: ObservableObject, UndoableCommandRunner { weak var delegate: SidebarModelDelegate? @Published var sidebarItems = [SidebarItem]() @Published var selectedFeedIdentifiers = Set() @Published var selectedFeedIdentifier: FeedIdentifier? = .none @Published var selectedFeeds = [Feed]() @Published var isReadFiltered = false private var cancellables = Set() private let rebuildSidebarItemsQueue = CoalescingQueue(name: "Rebuild The Sidebar Items", interval: 0.5) var undoManager: UndoManager? var undoableCommands = [UndoableCommand]() init() { subscribeToUnreadCountInitialization() subscribeToUnreadCountChanges() subscribeToRebuildSidebarItemsEvents() subscribeToSelectedFeedChanges() subscribeToReadFilterChanges() } } // MARK: Private private extension SidebarModel { // MARK: Subscriptions func subscribeToUnreadCountInitialization() { NotificationCenter.default.publisher(for: .UnreadCountDidInitialize) .filter { $0.object is AccountManager } .sink { [weak self] note in guard let self = self else { return } self.rebuildSidebarItems(isReadFiltered: self.isReadFiltered) }.store(in: &cancellables) } func subscribeToUnreadCountChanges() { NotificationCenter.default.publisher(for: .UnreadCountDidChange) .filter { _ in AccountManager.shared.isUnreadCountsInitialized } .sink { [weak self] _ in self?.queueRebuildSidebarItems() }.store(in: &cancellables) } func subscribeToRebuildSidebarItemsEvents() { let chidrenDidChangePublisher = NotificationCenter.default.publisher(for: .ChildrenDidChange) let batchUpdateDidPerformPublisher = NotificationCenter.default.publisher(for: .BatchUpdateDidPerform) let displayNameDidChangePublisher = NotificationCenter.default.publisher(for: .DisplayNameDidChange) let accountStateDidChangePublisher = NotificationCenter.default.publisher(for: .AccountStateDidChange) let userDidAddAccountPublisher = NotificationCenter.default.publisher(for: .UserDidAddAccount) let userDidDeleteAccountPublisher = NotificationCenter.default.publisher(for: .UserDidDeleteAccount) let sidebarRebuildPublishers = chidrenDidChangePublisher.merge(with: batchUpdateDidPerformPublisher, displayNameDidChangePublisher, accountStateDidChangePublisher, userDidAddAccountPublisher, userDidDeleteAccountPublisher) sidebarRebuildPublishers.sink { [weak self] _ in guard let self = self else { return } self.rebuildSidebarItems(isReadFiltered: self.isReadFiltered) }.store(in: &cancellables) } func subscribeToSelectedFeedChanges() { $selectedFeedIdentifiers.map { [weak self] feedIDs in feedIDs.compactMap { self?.findFeed($0) } } .assign(to: $selectedFeeds) $selectedFeedIdentifier.map { [weak self] feedID in if let feedID = feedID, let feed = self?.findFeed(feedID) { return [feed] } else { return [Feed]() } } .assign(to: $selectedFeeds) } func subscribeToReadFilterChanges() { $isReadFiltered.sink { [weak self] filter in self?.rebuildSidebarItems(isReadFiltered: filter) }.store(in: &cancellables) } // MARK: func findFeed(_ feedID: FeedIdentifier) -> Feed? { switch feedID { case .smartFeed: return SmartFeedsController.shared.find(by: feedID) default: return AccountManager.shared.existingFeed(with: feedID) } } func sort(_ folders: Set) -> [Folder] { return folders.sorted(by: { $0.nameForDisplay.localizedStandardCompare($1.nameForDisplay) == .orderedAscending }) } func sort(_ feeds: Set) -> [Feed] { return feeds.sorted(by: { $0.nameForDisplay.localizedStandardCompare($1.nameForDisplay) == .orderedAscending }) } func queueRebuildSidebarItems() { rebuildSidebarItemsQueue.add(self, #selector(rebuildSidebarItemsWithCurrentValues)) } @objc func rebuildSidebarItemsWithCurrentValues() { rebuildSidebarItems(isReadFiltered: isReadFiltered) } func rebuildSidebarItems(isReadFiltered: Bool) { guard let delegate = delegate else { return } var items = [SidebarItem]() var smartFeedControllerItem = SidebarItem(SmartFeedsController.shared) for feed in SmartFeedsController.shared.smartFeeds { // It looks like SwiftUI loses its mind when the last element in a section is removed. Don't filter // the smartfeeds yet or we crash about everytime because Starred is almost always filtered // if !isReadFiltered || feed.unreadCount > 0 { smartFeedControllerItem.addChild(SidebarItem(feed, unreadCount: delegate.unreadCount(for: feed))) // } } items.append(smartFeedControllerItem) for account in AccountManager.shared.sortedActiveAccounts { var accountItem = SidebarItem(account) for webFeed in sort(account.topLevelWebFeeds) { if !isReadFiltered || webFeed.unreadCount > 0 { accountItem.addChild(SidebarItem(webFeed, unreadCount: delegate.unreadCount(for: webFeed))) } } for folder in sort(account.folders ?? Set()) { if !isReadFiltered || folder.unreadCount > 0 { var folderItem = SidebarItem(folder, unreadCount: delegate.unreadCount(for: folder)) for webFeed in sort(folder.topLevelWebFeeds) { if !isReadFiltered || webFeed.unreadCount > 0 { folderItem.addChild(SidebarItem(webFeed, unreadCount: delegate.unreadCount(for: webFeed))) } } accountItem.addChild(folderItem) } } items.append(accountItem) } sidebarItems = items } }