173 lines
4.6 KiB
Swift
173 lines
4.6 KiB
Swift
//
|
|
// SidebarTreeControllerDelegate.swift
|
|
// NetNewsWire
|
|
//
|
|
// Created by Brent Simmons on 7/24/16.
|
|
// Copyright © 2016 Ranchero Software, LLC. All rights reserved.
|
|
//
|
|
|
|
import Foundation
|
|
import RSTree
|
|
import Articles
|
|
import Account
|
|
|
|
final class WebFeedTreeControllerDelegate: TreeControllerDelegate {
|
|
|
|
private var filterExceptions = Set<FeedIdentifier>()
|
|
var isReadFiltered = false
|
|
|
|
func addFilterException(_ feedID: FeedIdentifier) {
|
|
filterExceptions.insert(feedID)
|
|
}
|
|
|
|
func resetFilterExceptions() {
|
|
filterExceptions = Set<FeedIdentifier>()
|
|
}
|
|
|
|
func treeController(treeController: TreeController, childNodesFor node: Node) -> [Node]? {
|
|
if node.isRoot {
|
|
return childNodesForRootNode(node)
|
|
}
|
|
if node.representedObject is Container {
|
|
return childNodesForContainerNode(node)
|
|
}
|
|
if node.representedObject is SmartFeedsController {
|
|
return childNodesForSmartFeeds(node)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
}
|
|
|
|
private extension WebFeedTreeControllerDelegate {
|
|
|
|
func childNodesForRootNode(_ rootNode: Node) -> [Node]? {
|
|
var topLevelNodes = [Node]()
|
|
|
|
// Check to see if we should show the SmartFeeds top level by checking the unreadFeed
|
|
if !(isReadFiltered && SmartFeedsController.shared.unreadFeed.unreadCount == 0) {
|
|
let smartFeedsNode = rootNode.existingOrNewChildNode(with: SmartFeedsController.shared)
|
|
smartFeedsNode.canHaveChildNodes = true
|
|
smartFeedsNode.isGroupItem = true
|
|
topLevelNodes.append(smartFeedsNode)
|
|
}
|
|
|
|
topLevelNodes.append(contentsOf: sortedAccountNodes(rootNode))
|
|
|
|
return topLevelNodes
|
|
}
|
|
|
|
func childNodesForSmartFeeds(_ parentNode: Node) -> [Node] {
|
|
return SmartFeedsController.shared.smartFeeds.compactMap { (feed) -> Node? in
|
|
if let feedID = feed.feedID, !filterExceptions.contains(feedID) && isReadFiltered && feed.unreadCount == 0 {
|
|
return nil
|
|
}
|
|
return parentNode.existingOrNewChildNode(with: feed as AnyObject)
|
|
}
|
|
}
|
|
|
|
func childNodesForContainerNode(_ containerNode: Node) -> [Node]? {
|
|
let container = containerNode.representedObject as! Container
|
|
|
|
var children = [AnyObject]()
|
|
|
|
for webFeed in container.topLevelWebFeeds {
|
|
if let feedID = webFeed.feedID, !(!filterExceptions.contains(feedID) && isReadFiltered && webFeed.unreadCount == 0) {
|
|
children.append(webFeed)
|
|
}
|
|
}
|
|
|
|
if let folders = container.folders {
|
|
for folder in folders {
|
|
if let feedID = folder.feedID, !(!filterExceptions.contains(feedID) && isReadFiltered && folder.unreadCount == 0) {
|
|
children.append(folder)
|
|
}
|
|
}
|
|
}
|
|
|
|
var updatedChildNodes = [Node]()
|
|
|
|
children.forEach { (representedObject) in
|
|
|
|
if let existingNode = containerNode.childNodeRepresentingObject(representedObject) {
|
|
if !updatedChildNodes.contains(existingNode) {
|
|
updatedChildNodes += [existingNode]
|
|
return
|
|
}
|
|
}
|
|
|
|
if let newNode = self.createNode(representedObject: representedObject, parent: containerNode) {
|
|
updatedChildNodes += [newNode]
|
|
}
|
|
}
|
|
|
|
return updatedChildNodes.sortedAlphabeticallyWithFoldersAtEnd()
|
|
}
|
|
|
|
func createNode(representedObject: Any, parent: Node) -> Node? {
|
|
if let webFeed = representedObject as? WebFeed {
|
|
return createNode(webFeed: webFeed, parent: parent)
|
|
}
|
|
|
|
if let folder = representedObject as? Folder {
|
|
return createNode(folder: folder, parent: parent)
|
|
}
|
|
|
|
if let account = representedObject as? Account {
|
|
return createNode(account: account, parent: parent)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func createNode(webFeed: WebFeed, parent: Node) -> Node {
|
|
return parent.createChildNode(webFeed)
|
|
}
|
|
|
|
func createNode(folder: Folder, parent: Node) -> Node {
|
|
let node = parent.createChildNode(folder)
|
|
node.canHaveChildNodes = true
|
|
return node
|
|
}
|
|
|
|
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] {
|
|
|
|
var accountFilterException: String? = nil
|
|
switch filterExceptions.first {
|
|
case .folder(let accountID, _), .webFeed(let accountID, _):
|
|
accountFilterException = accountID
|
|
default:
|
|
break
|
|
}
|
|
|
|
let nodes = AccountManager.shared.sortedActiveAccounts.compactMap { (account) -> Node? in
|
|
|
|
if isReadFiltered && account.accountID != accountFilterException && account.unreadCount == 0 {
|
|
return nil
|
|
}
|
|
|
|
let accountNode = parent.existingOrNewChildNode(with: account)
|
|
accountNode.canHaveChildNodes = true
|
|
accountNode.isGroupItem = true
|
|
return accountNode
|
|
}
|
|
return nodes
|
|
}
|
|
|
|
func nodeInArrayRepresentingObject(_ nodes: [Node], _ representedObject: AnyObject) -> Node? {
|
|
for oneNode in nodes {
|
|
if oneNode.representedObject === representedObject {
|
|
return oneNode
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
}
|