NetNewsWire/Shared/Commands/DeleteCommand.swift
2024-06-08 11:58:30 -07:00

308 lines
5.9 KiB
Swift

//
// DeleteFromSidebarCommand.swift
// NetNewsWire
//
// Created by Brent Simmons on 11/4/17.
// Copyright © 2017 Ranchero Software. All rights reserved.
//
import Foundation
import Tree
import Account
import Articles
import Core
@MainActor final class DeleteCommand: UndoableCommand {
let treeController: TreeController?
let undoManager: UndoManager
let undoActionName: String
var redoActionName: String {
return undoActionName
}
let errorHandler: (Error) -> ()
private let itemSpecifiers: [SidebarItemSpecifier]
init?(nodesToDelete: [Node], treeController: TreeController? = nil, undoManager: UndoManager, errorHandler: @escaping (Error) -> ()) {
guard DeleteCommand.canDelete(nodesToDelete) else {
return nil
}
guard let actionName = DeleteActionName.name(for: nodesToDelete) else {
return nil
}
self.treeController = treeController
self.undoActionName = actionName
self.undoManager = undoManager
self.errorHandler = errorHandler
let itemSpecifiers = nodesToDelete.compactMap{ SidebarItemSpecifier(node: $0, errorHandler: errorHandler) }
guard !itemSpecifiers.isEmpty else {
return nil
}
self.itemSpecifiers = itemSpecifiers
}
func perform() {
Task {
for itemSpecifier in itemSpecifiers {
await itemSpecifier.delete()
}
treeController?.rebuild()
registerUndo()
}
}
func undo() {
for itemSpecifier in itemSpecifiers {
itemSpecifier.restore()
}
registerRedo()
}
static func canDelete(_ nodes: [Node]) -> Bool {
// Return true if all nodes are feeds and folders.
// Any other type: return false.
if nodes.isEmpty {
return false
}
for node in nodes {
if let _ = node.representedObject as? Feed {
continue
}
if let _ = node.representedObject as? Folder {
continue
}
return false
}
return true
}
}
// Remember as much as we can now about the items being deleted,
// so they can be restored to the correct place.
@MainActor private struct SidebarItemSpecifier {
private weak var account: Account?
private let parentFolder: Folder?
private let folder: Folder?
private let feed: Feed?
private let path: ContainerPath
private let errorHandler: (Error) -> ()
private var container: Container? {
if let parentFolder = parentFolder {
return parentFolder
}
if let account = account {
return account
}
return nil
}
init?(node: Node, errorHandler: @escaping (Error) -> ()) {
var account: Account?
self.parentFolder = node.parentFolder()
if let feed = node.representedObject as? Feed {
self.feed = feed
self.folder = nil
account = feed.account
}
else if let folder = node.representedObject as? Folder {
self.feed = nil
self.folder = folder
account = folder.account
}
else {
return nil
}
if account == nil {
return nil
}
self.account = account!
self.path = ContainerPath(account: account!, folders: node.containingFolders())
self.errorHandler = errorHandler
}
func delete() async {
if let feed {
guard let container = path.resolveContainer() else {
return
}
BatchUpdate.shared.start()
do {
try await account?.removeFeed(feed, from: container)
BatchUpdate.shared.end()
} catch {
BatchUpdate.shared.end()
errorHandler(error)
}
} else if let folder {
BatchUpdate.shared.start()
do {
try await account?.removeFolder(folder)
BatchUpdate.shared.end()
} catch {
BatchUpdate.shared.end()
errorHandler(error)
}
}
}
func restore() {
if let _ = feed {
restoreFeed()
}
else if let _ = folder {
restoreFolder()
}
}
private func restoreFeed() {
guard let account = account, let feed = feed, let container = path.resolveContainer() else {
return
}
BatchUpdate.shared.start()
Task {
do {
try await account.restoreFeed(feed, container: container)
BatchUpdate.shared.end()
} catch {
BatchUpdate.shared.end()
self.errorHandler(error)
}
}
}
private func restoreFolder() {
guard let account, let folder else {
return
}
Task {
BatchUpdate.shared.start()
do {
try await account.restoreFolder(folder)
BatchUpdate.shared.end()
} catch {
BatchUpdate.shared.end()
self.errorHandler(error)
}
}
}
private func checkResult(_ result: Result<Void, Error>) {
switch result {
case .success:
break
case .failure(let error):
errorHandler(error)
}
}
}
private extension Node {
func parentFolder() -> Folder? {
guard let parentNode = self.parent else {
return nil
}
if parentNode.isRoot {
return nil
}
if let folder = parentNode.representedObject as? Folder {
return folder
}
return nil
}
func containingFolders() -> [Folder] {
var nomad = self.parent
var folders = [Folder]()
while nomad != nil {
if let folder = nomad!.representedObject as? Folder {
folders += [folder]
}
else {
break
}
nomad = nomad!.parent
}
return folders.reversed()
}
}
private struct DeleteActionName {
private static let deleteFeed = NSLocalizedString("Delete Feed", comment: "command")
private static let deleteFeeds = NSLocalizedString("Delete Feeds", comment: "command")
private static let deleteFolder = NSLocalizedString("Delete Folder", comment: "command")
private static let deleteFolders = NSLocalizedString("Delete Folders", comment: "command")
private static let deleteFeedsAndFolders = NSLocalizedString("Delete Feeds and Folders", comment: "command")
@MainActor static func name(for nodes: [Node]) -> String? {
var numberOfFeeds = 0
var numberOfFolders = 0
for node in nodes {
if let _ = node.representedObject as? Feed {
numberOfFeeds += 1
}
else if let _ = node.representedObject as? Folder {
numberOfFolders += 1
}
else {
return nil // Delete only Feeds and Folders.
}
}
if numberOfFolders < 1 {
return numberOfFeeds == 1 ? deleteFeed : deleteFeeds
}
if numberOfFeeds < 1 {
return numberOfFolders == 1 ? deleteFolder : deleteFolders
}
return deleteFeedsAndFolders
}
}