// // ArticleThemesManager.sqift // NetNewsWire // // Created by Brent Simmons on 9/26/15. // Copyright © 2015 Ranchero Software, LLC. All rights reserved. // #if os(macOS) import AppKit #else import UIKit #endif import RSCore public extension Notification.Name { static let ArticleThemeNamesDidChangeNotification = Notification.Name("ArticleThemeNamesDidChangeNotification") static let CurrentArticleThemeDidChangeNotification = Notification.Name("CurrentArticleThemeDidChangeNotification") } final class ArticleThemesManager { static var shared: ArticleThemesManager! private let folderPath: String var currentThemeName: String { get { return AppDefaults.shared.currentThemeName ?? AppDefaults.defaultThemeName } set { if newValue != currentThemeName { AppDefaults.shared.currentThemeName = newValue updateCurrentTheme() } } } var currentTheme: ArticleTheme { didSet { NotificationCenter.default.post(name: .CurrentArticleThemeDidChangeNotification, object: self) } } var themeNames = [AppDefaults.defaultThemeName] { didSet { NotificationCenter.default.post(name: .ArticleThemeNamesDidChangeNotification, object: self) } } init(folderPath: String) { self.folderPath = folderPath do { try FileManager.default.createDirectory(atPath: folderPath, withIntermediateDirectories: true, attributes: nil) } catch { assertionFailure("Could not create folder for Themes.") abort() } currentTheme = ArticleTheme.defaultTheme updateThemeNames() updateCurrentTheme() #if os(macOS) NotificationCenter.default.addObserver(self, selector: #selector(applicationDidBecomeActive(_:)), name: NSApplication.didBecomeActiveNotification, object: nil) #else NotificationCenter.default.addObserver(self, selector: #selector(applicationDidBecomeActive(_:)), name: UIApplication.didBecomeActiveNotification, object: nil) #endif } // MARK: API func themeExists(filename: String) -> Bool { let filenameLastPathComponent = (filename as NSString).lastPathComponent let toFilename = (folderPath as NSString).appendingPathComponent(filenameLastPathComponent) return FileManager.default.fileExists(atPath: toFilename) } func importTheme(filename: String) throws { let filenameLastPathComponent = (filename as NSString).lastPathComponent let toFilename = (folderPath as NSString).appendingPathComponent(filenameLastPathComponent) if FileManager.default.fileExists(atPath: toFilename) { try FileManager.default.removeItem(atPath: toFilename) } try FileManager.default.copyItem(atPath: filename, toPath: toFilename) updateThemeNames() } // MARK: Notifications @objc dynamic func applicationDidBecomeActive(_ note: Notification) { updateThemeNames() updateCurrentTheme() } } // MARK : Private private extension ArticleThemesManager { func updateThemeNames() { let updatedThemeNames = allThemePaths(folderPath).map { ArticleTheme.themeNameForPath($0) } let sortedThemeNames = updatedThemeNames.sorted(by: { $0.compare($1, options: .caseInsensitive) == .orderedAscending }) if sortedThemeNames != themeNames { themeNames = sortedThemeNames } } func articleThemeWithThemeName(_ themeName: String) -> ArticleTheme? { if themeName == AppDefaults.defaultThemeName { return ArticleTheme.defaultTheme } guard let path = pathForThemeName(themeName, folder: folderPath) else { return nil } return ArticleTheme(path: path) } func defaultArticleTheme() -> ArticleTheme { return articleThemeWithThemeName(AppDefaults.defaultThemeName)! } func updateCurrentTheme() { var themeName = currentThemeName if !themeNames.contains(themeName) { themeName = AppDefaults.defaultThemeName currentThemeName = AppDefaults.defaultThemeName } var articleTheme = articleThemeWithThemeName(themeName) if articleTheme == nil { articleTheme = defaultArticleTheme() currentThemeName = AppDefaults.defaultThemeName } if let articleTheme = articleTheme, articleTheme != currentTheme { currentTheme = articleTheme } } func allThemePaths(_ folder: String) -> [String] { let filepaths = FileManager.default.filePaths(inFolder: folder) return filepaths?.filter { $0.hasSuffix(ArticleTheme.nnwThemeSuffix) } ?? [] } func pathForThemeName(_ themeName: String, folder: String) -> String? { for onePath in allThemePaths(folder) { if ArticleTheme.pathIsPathForThemeName(themeName, path: onePath) { return onePath } } return nil } }