2020-06-28 21:21:43 +02:00
|
|
|
//
|
|
|
|
// SidebarModel.swift
|
|
|
|
// NetNewsWire
|
|
|
|
//
|
|
|
|
// Created by Maurice Parker on 6/28/20.
|
|
|
|
// Copyright © 2020 Ranchero Software. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Foundation
|
2020-07-11 19:47:13 +02:00
|
|
|
import Combine
|
2020-06-29 23:58:10 +02:00
|
|
|
import RSCore
|
2020-06-29 00:43:20 +02:00
|
|
|
import Account
|
|
|
|
|
|
|
|
protocol SidebarModelDelegate: class {
|
2020-06-29 13:16:48 +02:00
|
|
|
func unreadCount(for: Feed) -> Int
|
2020-06-29 00:43:20 +02:00
|
|
|
}
|
2020-06-28 21:21:43 +02:00
|
|
|
|
2020-07-12 01:22:47 +02:00
|
|
|
class SidebarModel: ObservableObject, UndoableCommandRunner {
|
2020-06-28 21:21:43 +02:00
|
|
|
|
2020-06-29 00:43:20 +02:00
|
|
|
weak var delegate: SidebarModelDelegate?
|
|
|
|
|
2020-06-29 13:16:48 +02:00
|
|
|
@Published var sidebarItems = [SidebarItem]()
|
2020-07-11 19:47:13 +02:00
|
|
|
@Published var selectedFeedIdentifiers = Set<FeedIdentifier>()
|
|
|
|
@Published var selectedFeedIdentifier: FeedIdentifier? = .none
|
|
|
|
@Published var selectedFeeds = [Feed]()
|
2020-07-12 17:52:42 +02:00
|
|
|
@Published var isReadFiltered = false
|
2020-06-29 13:16:48 +02:00
|
|
|
|
2020-07-18 22:20:15 +02:00
|
|
|
private var cancellables = Set<AnyCancellable>()
|
2020-07-11 19:47:13 +02:00
|
|
|
|
2020-07-12 16:54:39 +02:00
|
|
|
private let rebuildSidebarItemsQueue = CoalescingQueue(name: "Rebuild The Sidebar Items", interval: 0.5)
|
|
|
|
|
2020-07-12 01:22:47 +02:00
|
|
|
var undoManager: UndoManager?
|
|
|
|
var undoableCommands = [UndoableCommand]()
|
|
|
|
|
2020-06-29 23:58:10 +02:00
|
|
|
init() {
|
2020-07-18 23:53:30 +02:00
|
|
|
subscribeToUnreadCountInitialization()
|
|
|
|
subscribeToUnreadCountChanges()
|
|
|
|
subscribeToRebuildSidebarItemsEvents()
|
|
|
|
subscribeToSelectedFeedChanges()
|
|
|
|
subscribeToReadFilterChanges()
|
|
|
|
}
|
2020-07-19 20:23:08 +02:00
|
|
|
|
|
|
|
// MARK: API
|
|
|
|
|
|
|
|
func goToNextUnread() {
|
2020-07-19 22:24:59 +02:00
|
|
|
guard let startFeed = selectedFeeds.first ?? sidebarItems.first?.children.first?.feed else { return }
|
2020-07-19 20:23:08 +02:00
|
|
|
|
2020-07-19 22:24:59 +02:00
|
|
|
if !goToNextUnread(startingAt: startFeed) {
|
|
|
|
if let firstFeed = sidebarItems.first?.children.first?.feed {
|
|
|
|
goToNextUnread(startingAt: firstFeed)
|
2020-07-19 20:23:08 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-07-18 23:53:30 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: Private
|
|
|
|
|
|
|
|
private extension SidebarModel {
|
|
|
|
|
|
|
|
// MARK: Subscriptions
|
|
|
|
|
|
|
|
func subscribeToUnreadCountInitialization() {
|
2020-07-18 22:20:15 +02:00
|
|
|
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)
|
2020-07-18 23:53:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func subscribeToUnreadCountChanges() {
|
2020-07-18 22:20:15 +02:00
|
|
|
NotificationCenter.default.publisher(for: .UnreadCountDidChange)
|
|
|
|
.filter { _ in AccountManager.shared.isUnreadCountsInitialized }
|
|
|
|
.sink { [weak self] _ in
|
|
|
|
self?.queueRebuildSidebarItems()
|
|
|
|
}.store(in: &cancellables)
|
2020-07-18 23:53:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func subscribeToRebuildSidebarItemsEvents() {
|
2020-07-18 22:20:15 +02:00
|
|
|
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)
|
2020-07-18 23:53:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func subscribeToSelectedFeedChanges() {
|
2020-07-19 00:10:45 +02:00
|
|
|
$selectedFeedIdentifiers.map { [weak self] feedIDs in
|
|
|
|
feedIDs.compactMap { self?.findFeed($0) }
|
|
|
|
}
|
|
|
|
.assign(to: $selectedFeeds)
|
2020-07-11 19:47:13 +02:00
|
|
|
|
2020-07-19 04:02:38 +02:00
|
|
|
$selectedFeedIdentifier.compactMap { [weak self] feedID in
|
2020-07-19 00:10:45 +02:00
|
|
|
if let feedID = feedID, let feed = self?.findFeed(feedID) {
|
|
|
|
return [feed]
|
|
|
|
} else {
|
2020-07-19 04:02:38 +02:00
|
|
|
return nil
|
2020-07-11 19:47:13 +02:00
|
|
|
}
|
2020-07-19 00:10:45 +02:00
|
|
|
}
|
|
|
|
.assign(to: $selectedFeeds)
|
2020-07-18 23:53:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func subscribeToReadFilterChanges() {
|
2020-07-18 22:20:15 +02:00
|
|
|
$isReadFiltered.sink { [weak self] filter in
|
|
|
|
self?.rebuildSidebarItems(isReadFiltered: filter)
|
|
|
|
}.store(in: &cancellables)
|
2020-06-29 23:58:10 +02:00
|
|
|
}
|
|
|
|
|
2020-07-19 20:23:08 +02:00
|
|
|
// MARK: Sidebar Building
|
2020-07-12 02:56:42 +02:00
|
|
|
|
2020-07-04 16:33:27 +02:00
|
|
|
func sort(_ folders: Set<Folder>) -> [Folder] {
|
|
|
|
return folders.sorted(by: { $0.nameForDisplay.localizedStandardCompare($1.nameForDisplay) == .orderedAscending })
|
|
|
|
}
|
|
|
|
|
|
|
|
func sort(_ feeds: Set<WebFeed>) -> [Feed] {
|
|
|
|
return feeds.sorted(by: { $0.nameForDisplay.localizedStandardCompare($1.nameForDisplay) == .orderedAscending })
|
|
|
|
}
|
|
|
|
|
2020-07-12 16:54:39 +02:00
|
|
|
func queueRebuildSidebarItems() {
|
2020-07-12 17:52:42 +02:00
|
|
|
rebuildSidebarItemsQueue.add(self, #selector(rebuildSidebarItemsWithCurrentValues))
|
2020-07-12 16:54:39 +02:00
|
|
|
}
|
2020-07-12 17:52:42 +02:00
|
|
|
|
|
|
|
@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)
|
2020-07-12 16:54:39 +02:00
|
|
|
|
2020-07-12 17:52:42 +02:00
|
|
|
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<Folder>()) {
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2020-07-19 20:23:08 +02:00
|
|
|
// MARK:
|
|
|
|
|
|
|
|
func findFeed(_ feedID: FeedIdentifier) -> Feed? {
|
|
|
|
switch feedID {
|
|
|
|
case .smartFeed:
|
|
|
|
return SmartFeedsController.shared.find(by: feedID)
|
|
|
|
default:
|
|
|
|
return AccountManager.shared.existingFeed(with: feedID)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@discardableResult
|
2020-07-19 22:24:59 +02:00
|
|
|
func goToNextUnread(startingAt: Feed) -> Bool {
|
|
|
|
|
|
|
|
var foundStartFeed = false
|
|
|
|
var nextSidebarItem: SidebarItem? = nil
|
|
|
|
for section in sidebarItems {
|
|
|
|
if nextSidebarItem == nil {
|
|
|
|
section.visit { sidebarItem in
|
|
|
|
if !foundStartFeed && sidebarItem.feed?.feedID == startingAt.feedID {
|
|
|
|
foundStartFeed = true
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if foundStartFeed && sidebarItem.unreadCount > 0 {
|
|
|
|
nextSidebarItem = sidebarItem
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
2020-07-19 20:23:08 +02:00
|
|
|
}
|
|
|
|
}
|
2020-07-19 22:24:59 +02:00
|
|
|
|
|
|
|
if let nextFeedID = nextSidebarItem?.feed?.feedID {
|
|
|
|
select(nextFeedID)
|
|
|
|
return true
|
|
|
|
}
|
2020-07-19 20:23:08 +02:00
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func select(_ feedID: FeedIdentifier) {
|
|
|
|
selectedFeedIdentifiers = Set([feedID])
|
|
|
|
selectedFeedIdentifier = feedID
|
|
|
|
}
|
|
|
|
|
2020-06-29 23:58:10 +02:00
|
|
|
}
|