2017-05-27 19:43:27 +02:00
|
|
|
|
//
|
|
|
|
|
// AccountManager.swift
|
|
|
|
|
// Evergreen
|
|
|
|
|
//
|
|
|
|
|
// Created by Brent Simmons on 7/18/15.
|
|
|
|
|
// Copyright © 2015 Ranchero Software, LLC. All rights reserved.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
|
import RSCore
|
2017-09-17 21:20:32 +02:00
|
|
|
|
import Data
|
2017-05-27 19:43:27 +02:00
|
|
|
|
|
|
|
|
|
let AccountsDidChangeNotification = "AccountsDidChangeNotification"
|
|
|
|
|
|
|
|
|
|
private let localAccountFolderName = "OnMyMac"
|
|
|
|
|
private let localAccountIdentifier = "OnMyMac"
|
|
|
|
|
|
2017-09-17 21:34:10 +02:00
|
|
|
|
public final class AccountManager: UnreadCountProvider {
|
2017-05-27 19:43:27 +02:00
|
|
|
|
|
2017-09-17 21:34:10 +02:00
|
|
|
|
public static let sharedInstance = AccountManager()
|
|
|
|
|
public let localAccount: Account
|
2017-05-27 19:43:27 +02:00
|
|
|
|
private let accountsFolder = RSDataSubfolder(nil, "Accounts")!
|
|
|
|
|
private var accountsDictionary = [String: Account]()
|
2017-09-17 21:20:32 +02:00
|
|
|
|
|
2017-09-17 21:34:10 +02:00
|
|
|
|
public var unreadCount = 0 {
|
2017-05-27 19:43:27 +02:00
|
|
|
|
didSet {
|
|
|
|
|
postUnreadCountDidChangeNotification()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var accounts: [Account] {
|
|
|
|
|
get {
|
|
|
|
|
return Array(accountsDictionary.values)
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-09-17 21:20:32 +02:00
|
|
|
|
|
2017-09-17 21:54:08 +02:00
|
|
|
|
public var sortedAccounts: [Account] {
|
2017-05-27 19:43:27 +02:00
|
|
|
|
get {
|
|
|
|
|
return accountsSortedByName()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-09-17 21:34:10 +02:00
|
|
|
|
public var refreshInProgress: Bool {
|
2017-05-27 19:43:27 +02:00
|
|
|
|
get {
|
|
|
|
|
for oneAccount in accountsDictionary.values {
|
|
|
|
|
if oneAccount.refreshInProgress {
|
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return false
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-09-17 21:34:10 +02:00
|
|
|
|
public init() {
|
2017-05-27 19:43:27 +02:00
|
|
|
|
|
|
|
|
|
// 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)
|
2017-09-17 21:20:32 +02:00
|
|
|
|
localAccount = Account(dataFolder: localAccountFolder, settingsFile: localAccountSettingsFile, type: .onMyMac, accountID: localAccountIdentifier)!
|
|
|
|
|
accountsDictionary[localAccount.accountID] = localAccount
|
2017-05-27 19:43:27 +02:00
|
|
|
|
|
|
|
|
|
readNonLocalAccountsFromDisk()
|
|
|
|
|
|
|
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(unreadCountDidChange(_:)), name: .UnreadCountDidChange, object: nil)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// MARK: API
|
|
|
|
|
|
2017-09-18 02:56:04 +02:00
|
|
|
|
public func existingAccount(with accountID: String) -> Account? {
|
2017-05-27 19:43:27 +02:00
|
|
|
|
|
2017-09-17 21:20:32 +02:00
|
|
|
|
return accountsDictionary[accountID]
|
2017-05-27 19:43:27 +02:00
|
|
|
|
}
|
|
|
|
|
|
2017-09-17 21:34:10 +02:00
|
|
|
|
public func refreshAll() {
|
2017-05-27 19:43:27 +02:00
|
|
|
|
|
2017-09-17 21:20:32 +02:00
|
|
|
|
accounts.forEach { (account) in
|
|
|
|
|
account.refreshAll()
|
2017-05-27 19:43:27 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func anyAccountHasAtLeastOneFeed() -> Bool {
|
|
|
|
|
|
2017-09-17 21:20:32 +02:00
|
|
|
|
for account in accounts {
|
|
|
|
|
if account.hasAtLeastOneFeed() {
|
2017-05-27 19:43:27 +02:00
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false
|
|
|
|
|
}
|
|
|
|
|
|
2017-09-19 07:00:35 +02:00
|
|
|
|
public func anyAccountHasFeedWithURL(_ urlString: String) -> Bool {
|
2017-05-27 19:43:27 +02:00
|
|
|
|
|
2017-09-17 21:20:32 +02:00
|
|
|
|
for account in accounts {
|
|
|
|
|
if let _ = account.existingFeed(withURL: urlString) {
|
2017-05-27 19:43:27 +02:00
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return false
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func updateUnreadCount() {
|
|
|
|
|
|
|
|
|
|
let updatedUnreadCount = calculateUnreadCount(accounts)
|
|
|
|
|
if updatedUnreadCount != unreadCount {
|
|
|
|
|
unreadCount = updatedUnreadCount
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// MARK: Notifications
|
|
|
|
|
|
2017-09-17 21:20:32 +02:00
|
|
|
|
@objc dynamic func unreadCountDidChange(_ notification: Notification) {
|
2017-05-27 19:43:27 +02:00
|
|
|
|
|
|
|
|
|
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) {
|
2017-09-17 21:20:32 +02:00
|
|
|
|
accountsDictionary[oneAccount.accountID] = oneAccount
|
2017-05-27 19:43:27 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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)
|
|
|
|
|
}
|
|
|
|
|
|
2017-09-19 07:00:35 +02:00
|
|
|
|
public func accountWithID(_ accountID: String) -> Account? {
|
2017-09-17 21:54:08 +02:00
|
|
|
|
|
|
|
|
|
// Shortcut.
|
2017-09-18 02:56:04 +02:00
|
|
|
|
return AccountManager.sharedInstance.existingAccount(with: accountID)
|
2017-09-17 21:54:08 +02:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-27 19:43:27 +02:00
|
|
|
|
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)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|