2017-05-27 19:43:27 +02:00
|
|
|
//
|
|
|
|
// SidebarTreeControllerDelegate.swift
|
2018-08-29 07:18:24 +02:00
|
|
|
// NetNewsWire
|
2017-05-27 19:43:27 +02:00
|
|
|
//
|
|
|
|
// Created by Brent Simmons on 7/24/16.
|
|
|
|
// Copyright © 2016 Ranchero Software, LLC. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
import RSTree
|
2018-07-24 03:29:08 +02:00
|
|
|
import Articles
|
2017-09-24 21:24:44 +02:00
|
|
|
import Account
|
2017-05-27 19:43:27 +02:00
|
|
|
|
2019-11-15 03:11:41 +01:00
|
|
|
final class WebFeedTreeControllerDelegate: TreeControllerDelegate {
|
2017-05-27 19:43:27 +02:00
|
|
|
|
2019-11-28 19:40:33 +01:00
|
|
|
private var filterExceptions = Set<FeedIdentifier>()
|
2019-11-22 17:55:54 +01:00
|
|
|
var isReadFiltered = false
|
2019-11-21 22:55:50 +01:00
|
|
|
|
2019-11-28 19:40:33 +01:00
|
|
|
func addFilterException(_ feedID: FeedIdentifier) {
|
|
|
|
filterExceptions.insert(feedID)
|
|
|
|
}
|
|
|
|
|
|
|
|
func resetFilterExceptions() {
|
|
|
|
filterExceptions = Set<FeedIdentifier>()
|
|
|
|
}
|
|
|
|
|
2017-05-27 19:43:27 +02:00
|
|
|
func treeController(treeController: TreeController, childNodesFor node: Node) -> [Node]? {
|
|
|
|
if node.isRoot {
|
|
|
|
return childNodesForRootNode(node)
|
|
|
|
}
|
2017-11-19 21:59:37 +01:00
|
|
|
if node.representedObject is Container {
|
|
|
|
return childNodesForContainerNode(node)
|
2017-05-27 19:43:27 +02:00
|
|
|
}
|
2017-12-16 20:16:32 +01:00
|
|
|
if node.representedObject is SmartFeedsController {
|
|
|
|
return childNodesForSmartFeeds(node)
|
|
|
|
}
|
2017-11-19 21:59:37 +01:00
|
|
|
|
2017-05-27 19:43:27 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-15 03:11:41 +01:00
|
|
|
private extension WebFeedTreeControllerDelegate {
|
2017-05-27 19:43:27 +02:00
|
|
|
|
2017-10-20 06:38:50 +02:00
|
|
|
func childNodesForRootNode(_ rootNode: Node) -> [Node]? {
|
2019-11-21 22:55:50 +01:00
|
|
|
var topLevelNodes = [Node]()
|
|
|
|
|
2019-12-02 00:54:34 +01:00
|
|
|
let smartFeedsNode = rootNode.existingOrNewChildNode(with: SmartFeedsController.shared)
|
|
|
|
smartFeedsNode.canHaveChildNodes = true
|
|
|
|
smartFeedsNode.isGroupItem = true
|
|
|
|
topLevelNodes.append(smartFeedsNode)
|
2017-10-08 06:41:21 +02:00
|
|
|
|
2019-11-21 22:55:50 +01:00
|
|
|
topLevelNodes.append(contentsOf: sortedAccountNodes(rootNode))
|
|
|
|
|
|
|
|
return topLevelNodes
|
2017-11-19 01:56:36 +01:00
|
|
|
}
|
|
|
|
|
2017-12-16 20:16:32 +01:00
|
|
|
func childNodesForSmartFeeds(_ parentNode: Node) -> [Node] {
|
2019-11-21 22:55:50 +01:00
|
|
|
return SmartFeedsController.shared.smartFeeds.compactMap { (feed) -> Node? in
|
2021-04-27 02:23:51 +02:00
|
|
|
// All Smart Feeds should remain visible despite the Hide Read Feeds setting
|
2019-11-21 22:55:50 +01:00
|
|
|
return parentNode.existingOrNewChildNode(with: feed as AnyObject)
|
|
|
|
}
|
2017-10-20 06:52:45 +02:00
|
|
|
}
|
2017-10-20 06:38:50 +02:00
|
|
|
|
2017-11-19 21:59:37 +01:00
|
|
|
func childNodesForContainerNode(_ containerNode: Node) -> [Node]? {
|
|
|
|
let container = containerNode.representedObject as! Container
|
2017-10-20 06:52:45 +02:00
|
|
|
|
2018-09-17 02:54:42 +02:00
|
|
|
var children = [AnyObject]()
|
2019-11-21 22:55:50 +01:00
|
|
|
|
|
|
|
for webFeed in container.topLevelWebFeeds {
|
2019-11-28 19:40:33 +01:00
|
|
|
if let feedID = webFeed.feedID, !(!filterExceptions.contains(feedID) && isReadFiltered && webFeed.unreadCount == 0) {
|
2019-11-21 22:55:50 +01:00
|
|
|
children.append(webFeed)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-17 02:54:42 +02:00
|
|
|
if let folders = container.folders {
|
2019-11-21 22:55:50 +01:00
|
|
|
for folder in folders {
|
2019-11-28 19:40:33 +01:00
|
|
|
if let feedID = folder.feedID, !(!filterExceptions.contains(feedID) && isReadFiltered && folder.unreadCount == 0) {
|
2019-11-21 22:55:50 +01:00
|
|
|
children.append(folder)
|
|
|
|
}
|
|
|
|
}
|
2018-09-17 02:54:42 +02:00
|
|
|
}
|
|
|
|
|
2017-05-27 19:43:27 +02:00
|
|
|
var updatedChildNodes = [Node]()
|
2017-10-08 06:41:21 +02:00
|
|
|
|
2018-09-17 02:54:42 +02:00
|
|
|
children.forEach { (representedObject) in
|
2017-10-08 06:41:21 +02:00
|
|
|
|
2017-10-20 06:52:45 +02:00
|
|
|
if let existingNode = containerNode.childNodeRepresentingObject(representedObject) {
|
2017-05-27 19:43:27 +02:00
|
|
|
if !updatedChildNodes.contains(existingNode) {
|
|
|
|
updatedChildNodes += [existingNode]
|
2017-10-20 06:38:50 +02:00
|
|
|
return
|
2017-05-27 19:43:27 +02:00
|
|
|
}
|
|
|
|
}
|
2017-10-20 06:38:50 +02:00
|
|
|
|
2017-10-20 06:52:45 +02:00
|
|
|
if let newNode = self.createNode(representedObject: representedObject, parent: containerNode) {
|
2017-05-27 19:43:27 +02:00
|
|
|
updatedChildNodes += [newNode]
|
|
|
|
}
|
|
|
|
}
|
2017-10-20 06:38:50 +02:00
|
|
|
|
2017-11-19 01:56:36 +01:00
|
|
|
return updatedChildNodes.sortedAlphabeticallyWithFoldersAtEnd()
|
2017-05-27 19:43:27 +02:00
|
|
|
}
|
2017-10-20 06:52:45 +02:00
|
|
|
|
2017-10-08 06:41:21 +02:00
|
|
|
func createNode(representedObject: Any, parent: Node) -> Node? {
|
2019-11-15 03:11:41 +01:00
|
|
|
if let webFeed = representedObject as? WebFeed {
|
|
|
|
return createNode(webFeed: webFeed, parent: parent)
|
2017-05-27 19:43:27 +02:00
|
|
|
}
|
2019-11-21 22:55:50 +01:00
|
|
|
|
2017-05-27 19:43:27 +02:00
|
|
|
if let folder = representedObject as? Folder {
|
|
|
|
return createNode(folder: folder, parent: parent)
|
|
|
|
}
|
2019-11-21 22:55:50 +01:00
|
|
|
|
2017-11-19 01:56:36 +01:00
|
|
|
if let account = representedObject as? Account {
|
|
|
|
return createNode(account: account, parent: parent)
|
|
|
|
}
|
|
|
|
|
2017-05-27 19:43:27 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-11-15 03:11:41 +01:00
|
|
|
func createNode(webFeed: WebFeed, parent: Node) -> Node {
|
|
|
|
return parent.createChildNode(webFeed)
|
2017-05-27 19:43:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func createNode(folder: Folder, parent: Node) -> Node {
|
2017-11-19 01:56:36 +01:00
|
|
|
let node = parent.createChildNode(folder)
|
2017-05-27 19:43:27 +02:00
|
|
|
node.canHaveChildNodes = true
|
|
|
|
return node
|
|
|
|
}
|
2017-11-19 01:56:36 +01:00
|
|
|
|
|
|
|
func createNode(account: Account, parent: Node) -> Node {
|
|
|
|
let node = parent.createChildNode(account)
|
|
|
|
node.canHaveChildNodes = true
|
|
|
|
node.isGroupItem = true
|
|
|
|
return node
|
|
|
|
}
|
|
|
|
|
|
|
|
func sortedAccountNodes(_ parent: Node) -> [Node] {
|
2019-11-21 22:55:50 +01:00
|
|
|
let nodes = AccountManager.shared.sortedActiveAccounts.compactMap { (account) -> Node? in
|
2017-12-18 21:43:18 +01:00
|
|
|
let accountNode = parent.existingOrNewChildNode(with: account)
|
|
|
|
accountNode.canHaveChildNodes = true
|
|
|
|
accountNode.isGroupItem = true
|
|
|
|
return accountNode
|
|
|
|
}
|
2019-05-02 02:07:44 +02:00
|
|
|
return nodes
|
2017-11-19 01:56:36 +01:00
|
|
|
}
|
|
|
|
|
2017-11-05 07:05:20 +01:00
|
|
|
func nodeInArrayRepresentingObject(_ nodes: [Node], _ representedObject: AnyObject) -> Node? {
|
2017-05-27 19:43:27 +02:00
|
|
|
for oneNode in nodes {
|
2017-11-05 07:05:20 +01:00
|
|
|
if oneNode.representedObject === representedObject {
|
2017-05-27 19:43:27 +02:00
|
|
|
return oneNode
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|