// // AccountManager.swift // Evergreen // // Created by Brent Simmons on 7/18/15. // Copyright © 2015 Ranchero Software, LLC. All rights reserved. // import Foundation import RSCore import Data let AccountsDidChangeNotification = "AccountsDidChangeNotification" private let localAccountFolderName = "OnMyMac" private let localAccountIdentifier = "OnMyMac" public final class AccountManager: UnreadCountProvider { public static let shared = AccountManager() public let localAccount: Account private let accountsFolder = RSDataSubfolder(nil, "Accounts")! private var accountsDictionary = [String: Account]() public var unreadCount = 0 { didSet { if unreadCount != oldValue { postUnreadCountDidChangeNotification() } } } public var accounts: [Account] { get { return Array(accountsDictionary.values) } } public var sortedAccounts: [Account] { get { return accountsSortedByName() } } public var refreshInProgress: Bool { get { for account in accounts { if account.refreshInProgress { return true } } return false } } public var combinedRefreshProgress: CombinedRefreshProgress { get { let downloadProgressArray = accounts.map { $0.refreshProgress } return CombinedRefreshProgress(downloadProgressArray: downloadProgressArray) } } public init() { // The local "On My Mac" account must always exist, even if it's empty. let localAccountFolder = (accountsFolder as NSString).appendingPathComponent("OnMyMac") do { try FileManager.default.createDirectory(atPath: localAccountFolder, withIntermediateDirectories: true, attributes: nil) } catch { assertionFailure("Could not create folder for OnMyMac account.") abort() } let localAccountSettingsFile = accountFilePathWithFolder(localAccountFolder) localAccount = Account(dataFolder: localAccountFolder, settingsFile: localAccountSettingsFile, type: .onMyMac, accountID: localAccountIdentifier)! accountsDictionary[localAccount.accountID] = localAccount readNonLocalAccountsFromDisk() NotificationCenter.default.addObserver(self, selector: #selector(unreadCountDidChange(_:)), name: .UnreadCountDidChange, object: nil) } // MARK: API public func existingAccount(with accountID: String) -> Account? { return accountsDictionary[accountID] } public func refreshAll() { accounts.forEach { $0.refreshAll() } } public func anyAccountHasAtLeastOneFeed() -> Bool { for account in accounts { if account.hasAtLeastOneFeed { return true } } return false } public func anyAccountHasFeedWithURL(_ urlString: String) -> Bool { for account in accounts { if let _ = account.existingFeed(withURL: urlString) { return true } } return false } func updateUnreadCount() { let updatedUnreadCount = calculateUnreadCount(accounts) if updatedUnreadCount != unreadCount { unreadCount = updatedUnreadCount } } // MARK: Notifications @objc dynamic func unreadCountDidChange(_ notification: Notification) { guard let _ = notification.object as? Account else { return } updateUnreadCount() } // MARK: Private private func createAccount(_ accountSpecifier: AccountSpecifier) -> Account? { return nil } private func createAccount(_ filename: String) -> Account? { let folderPath = (accountsFolder as NSString).appendingPathComponent(filename) if let accountSpecifier = AccountSpecifier(folderPath: folderPath) { return createAccount(accountSpecifier) } return nil } private func readNonLocalAccountsFromDisk() { var filenames: [String]? do { filenames = try FileManager.default.contentsOfDirectory(atPath: accountsFolder) } catch { print("Error reading Accounts folder: \(error)") return } filenames?.forEach { (oneFilename) in guard oneFilename != localAccountFolderName else { return } if let oneAccount = createAccount(oneFilename) { accountsDictionary[oneAccount.accountID] = oneAccount } } } private func accountsSortedByName() -> [Account] { // LocalAccount is first. return accounts.sorted { (account1, account2) -> Bool in if account1 === localAccount { return true } if account2 === localAccount { return false } //TODO: Use localizedCaseInsensitiveCompare: return account1.nameForDisplay < account2.nameForDisplay } } } private let accountDataFileName = "AccountData.plist" private func accountFilePathWithFolder(_ folderPath: String) -> String { return NSString(string: folderPath).appendingPathComponent(accountDataFileName) } private struct AccountSpecifier { let type: String let identifier: String let folderPath: String let folderName: String let dataFilePath: String init?(folderPath: String) { self.folderPath = folderPath self.folderName = NSString(string: folderPath).lastPathComponent let nameComponents = self.folderName.components(separatedBy: "-") let satisfyCompilerFolderName = self.folderName assert(nameComponents.count == 2, "Can’t determine account info from \(satisfyCompilerFolderName)") if nameComponents.count != 2 { return nil } self.type = nameComponents[0] self.identifier = nameComponents[1] self.dataFilePath = accountFilePathWithFolder(self.folderPath) } }