1
1
mirror of https://github.com/qvacua/vimr.git synced 2025-01-04 11:53:38 +03:00
vimr/VimR/States.swift

415 lines
12 KiB
Swift
Raw Normal View History

2017-01-22 16:22:05 +03:00
/**
* Tae Won Ha - http://taewon.de - @hataewon
* See LICENSE
*/
import Foundation
import RxSwift
2017-03-01 00:04:12 +03:00
struct AppState: SerializableState {
enum AfterLastWindowAction: String {
case doNothing = "do-nothing"
case hide = "hide"
case quit = "quit"
}
2017-03-01 18:59:23 +03:00
static let `default` = AppState()
2017-02-27 23:17:40 +03:00
var openNewMainWindowOnLaunch = true
var openNewMainWindowOnReactivation = true
var afterLastWindowAction = AfterLastWindowAction.doNothing
2017-02-28 13:10:04 +03:00
var useSnapshotUpdate = false
2017-02-28 00:45:26 +03:00
var preferencesOpen = Marked(false)
2017-03-01 18:59:23 +03:00
var mainWindowTemplate = MainWindow.State.default
var currentMainWindowUuid: String?
2017-01-22 16:22:05 +03:00
var mainWindows: [String: MainWindow.State] = [:]
var openQuickly = OpenQuicklyWindow.State.default
2017-03-01 18:59:23 +03:00
init() {
2017-05-14 13:06:21 +03:00
}
2017-03-01 00:04:12 +03:00
init?(dict: [String: Any]) {
2017-03-01 18:59:23 +03:00
guard let openOnLaunch = PrefUtils.bool(from: dict, for: Keys.openNewOnLaunch),
let openOnReactivation = PrefUtils.bool(from: dict, for: Keys.openNewOnReactivation),
let useSnapshot = PrefUtils.bool(from: dict, for: Keys.useSnapshotUpdateChannel)
else {
return nil
}
self.openNewMainWindowOnLaunch = openOnLaunch
self.openNewMainWindowOnReactivation = openOnReactivation
let lastWindowActionString = PrefUtils.string(from: dict, for: Keys.afterLastWindowAction)
?? AfterLastWindowAction.doNothing.rawValue
self.afterLastWindowAction = AfterLastWindowAction(rawValue: lastWindowActionString) ?? .doNothing
2017-03-01 18:59:23 +03:00
self.useSnapshotUpdate = useSnapshot
let openQuicklyDict = dict[Keys.OpenQuickly.key] as? [String: Any] ?? [:]
self.openQuickly = OpenQuicklyWindow.State(dict: openQuicklyDict) ?? OpenQuicklyWindow.State.default
let mainWindowDict = dict[Keys.MainWindow.key] as? [String: Any] ?? [:]
self.mainWindowTemplate = MainWindow.State(dict: mainWindowDict) ?? MainWindow.State.default
2017-03-01 00:04:12 +03:00
}
func dict() -> [String: Any] {
return [
Keys.openNewOnLaunch: self.openNewMainWindowOnLaunch,
Keys.openNewOnReactivation: self.openNewMainWindowOnReactivation,
Keys.afterLastWindowAction: self.afterLastWindowAction.rawValue,
2017-03-01 00:04:12 +03:00
Keys.useSnapshotUpdateChannel: self.useSnapshotUpdate,
Keys.OpenQuickly.key: self.openQuickly.dict(),
Keys.MainWindow.key: self.mainWindowTemplate.dict(),
]
}
}
extension OpenQuicklyWindow {
2017-03-01 00:04:12 +03:00
struct State: SerializableState {
static let `default` = State()
2017-05-14 13:06:21 +03:00
let root = FileItem(URL(fileURLWithPath: "/", isDirectory: true))
var flatFileItems = Observable<[FileItem]>.empty()
var cwd = FileUtils.userHomeUrl
2017-02-28 00:45:26 +03:00
var ignorePatterns = Set(["*/.git", "*.o", "*.d", "*.dia"].map(FileItemIgnorePattern.init))
var ignoreToken = Token()
var open = false
2017-03-01 00:04:12 +03:00
init() {
}
init?(dict: [String: Any]) {
guard let patternsString = PrefUtils.string(from: dict, for: Keys.OpenQuickly.ignorePatterns) else {
return nil
}
self.ignorePatterns = FileItemIgnorePattern.from(string: patternsString)
}
func dict() -> [String: Any] {
return [
Keys.OpenQuickly.ignorePatterns: FileItemIgnorePattern.toString(self.ignorePatterns)
]
}
}
}
2017-02-06 00:39:55 +03:00
struct PreviewState {
2017-02-03 00:39:05 +03:00
2017-02-07 00:54:22 +03:00
static let `default` = PreviewState()
2017-02-06 20:36:57 +03:00
enum Status {
case none
case notSaved
case error
case markdown
}
var status = Status.none
2017-02-06 00:39:55 +03:00
var buffer: URL?
var html: URL?
var server: URL?
2017-02-07 00:54:22 +03:00
2017-02-12 13:24:15 +03:00
var updateDate: Date
2017-02-07 00:54:22 +03:00
2017-02-12 15:41:12 +03:00
var editorPosition = Marked(Position.beginning)
var previewPosition = Marked(Position.beginning)
var ignoreNextForward = false
2017-03-31 09:05:42 +03:00
var ignoreNextReverse = false
2017-02-12 18:40:49 +03:00
var forceNextReverse = false
2017-02-07 00:54:22 +03:00
init(status: Status = .none,
buffer: URL? = nil,
html: URL? = nil,
server: URL? = nil,
2017-02-28 00:45:26 +03:00
updateDate: Date = Date()) {
2017-02-07 00:54:22 +03:00
self.status = status
self.buffer = buffer
self.html = html
self.server = server
self.updateDate = updateDate
}
2017-02-03 00:39:05 +03:00
}
2017-03-24 19:47:08 +03:00
struct HtmlPreviewState {
static let `default` = HtmlPreviewState()
var htmlFile: URL?
var server: Marked<URL>?
2017-03-24 19:47:08 +03:00
}
2017-03-01 00:04:12 +03:00
struct AppearanceState: SerializableState {
2017-02-28 11:53:27 +03:00
static let `default` = AppearanceState()
var font = NSFont.userFixedPitchFont(ofSize: 13)!
var linespacing: CGFloat = 1
var usesLigatures = false
2017-03-01 00:04:12 +03:00
var usesTheme = true
var theme = Marked(Theme.default)
2017-03-01 00:04:12 +03:00
init() {
}
init?(dict: [String: Any]) {
guard let editorFontName = dict[Keys.Appearance.editorFontName] as? String,
let fEditorFontSize = PrefUtils.float(from: dict, for: Keys.Appearance.editorFontSize),
let fEditorLinespacing = PrefUtils.float(from: dict, for: Keys.Appearance.editorLinespacing),
let editorUsesLigatures = PrefUtils.bool(from: dict, for: Keys.Appearance.editorUsesLigatures)
else {
return nil
}
self.usesTheme = PrefUtils.bool(from: dict, for: Keys.Appearance.usesTheme, default: true)
2017-03-01 00:04:12 +03:00
self.font = PrefUtils.saneFont(editorFontName, fontSize: CGFloat(fEditorFontSize))
self.linespacing = CGFloat(fEditorLinespacing)
self.usesLigatures = editorUsesLigatures
}
func dict() -> [String: Any] {
return [
Keys.Appearance.usesTheme: self.usesTheme,
2017-03-01 00:04:12 +03:00
Keys.Appearance.editorFontName: self.font.fontName,
Keys.Appearance.editorFontSize: Float(self.font.pointSize),
Keys.Appearance.editorLinespacing: Float(self.linespacing),
Keys.Appearance.editorUsesLigatures: self.usesLigatures,
]
}
2017-02-28 11:53:27 +03:00
}
extension MainWindow {
2017-03-01 18:59:23 +03:00
struct State: SerializableState {
static let `default` = State(isAllToolsVisible: true, isToolButtonsVisible: true)
var isAllToolsVisible = true
var isToolButtonsVisible = true
var activeTools = [
Tools.fileBrowser: true,
Tools.openedFilesList: true,
Tools.preview: true,
Tools.htmlPreview: true,
]
2017-01-22 16:22:05 +03:00
////// transient
2017-05-14 13:06:21 +03:00
var lastFileSystemUpdate = Marked(FileUtils.userHomeUrl)
2017-03-01 00:04:12 +03:00
var tools = WorkspaceToolState.default
2017-04-10 21:38:17 +03:00
var orderedTools = WorkspaceToolState.orderedDefault
2017-02-28 17:52:48 +03:00
2017-02-06 00:39:55 +03:00
var preview = PreviewState.default
var htmlPreview = HtmlPreviewState.default
2017-02-07 00:54:22 +03:00
var previewTool = PreviewTool.State.default
var fileBrowserShowHidden = false
2017-01-22 16:22:05 +03:00
// neovim
var uuid = UUID().uuidString
2017-01-22 16:22:05 +03:00
var currentBuffer: NeoVimBuffer?
var buffers = [NeoVimBuffer]()
var cwd = FileUtils.userHomeUrl
var isDirty = false
2017-02-28 11:53:27 +03:00
var appearance = AppearanceState.default
2017-02-28 13:10:04 +03:00
var useInteractiveZsh = false
var nvimArgs: [String]?
2017-06-12 19:37:42 +03:00
var cliPipePath: String?
// to be cleaned
var urlsToOpen = [URL: OpenMode]()
var currentBufferToSet: NeoVimBuffer?
var cwdToSet: URL?
2017-03-31 20:32:01 +03:00
var viewToBeFocused: FocusableView? = FocusableView.neoVimView
init(isAllToolsVisible: Bool, isToolButtonsVisible: Bool) {
self.isAllToolsVisible = isAllToolsVisible
self.isToolButtonsVisible = isToolButtonsVisible
}
2017-03-01 00:04:12 +03:00
init?(dict: [String: Any]) {
guard let isAllToolsVisible = PrefUtils.bool(from: dict, for: Keys.MainWindow.allToolsVisible),
let isToolButtonsVisible = PrefUtils.bool(from: dict, for: Keys.MainWindow.toolButtonsVisible),
2017-04-10 21:38:17 +03:00
let orderedToolsAsString = dict[Keys.MainWindow.orderedTools] as? [String],
2017-03-01 00:04:12 +03:00
let isShowHidden = PrefUtils.bool(from: dict, for: Keys.MainWindow.isShowHidden)
else {
return nil
}
// Stay compatible with 168
self.useInteractiveZsh = PrefUtils.bool(from: dict, for: Keys.MainWindow.useInteractiveZsh, default: false)
2017-03-01 00:04:12 +03:00
self.isAllToolsVisible = isAllToolsVisible
self.isToolButtonsVisible = isToolButtonsVisible
let appearanceDict = dict[Keys.Appearance.key] as? [String: Any] ?? [:]
self.appearance = AppearanceState(dict: appearanceDict) ?? AppearanceState.default
2017-04-10 21:38:17 +03:00
self.orderedTools = orderedToolsAsString.flatMap { MainWindow.Tools(rawValue: $0) }
let missingOrderedTools = MainWindow.Tools.all.subtracting(self.orderedTools)
self.orderedTools.append(contentsOf: missingOrderedTools)
2017-04-10 21:38:17 +03:00
// To stay compatible with 168 we do not guard against nil activeToolsAsString.
let activeToolsAsString = dict[Keys.MainWindow.activeTools] as? [String: Bool] ?? [:]
self.activeTools = activeToolsAsString.flatMapToDict { (key, value) in
guard let toolId = MainWindow.Tools(rawValue: key) else {
return nil
}
return (toolId, value)
}
let missingActiveTools = MainWindow.Tools.all.subtracting(self.activeTools.keys)
missingActiveTools.forEach { self.activeTools[$0] = true }
2017-03-01 00:04:12 +03:00
let workspaceToolsDict = dict[Keys.WorkspaceTool.key] as? [String: [String: Any]] ?? [:]
let toolKeys = workspaceToolsDict.keys.flatMap { MainWindow.Tools(rawValue: $0) }
let missingToolKeys = MainWindow.Tools.all.subtracting(toolKeys)
2017-04-10 23:15:48 +03:00
var tools = Array(toolKeys).toDict { tool in
2017-04-10 23:15:48 +03:00
return WorkspaceToolState(dict: workspaceToolsDict[tool.rawValue]!) ?? WorkspaceToolState.default[tool]!
2017-03-01 00:04:12 +03:00
}
missingToolKeys.forEach { missingTool in
tools[missingTool] = WorkspaceToolState.default[missingTool]!
2017-03-01 00:04:12 +03:00
}
self.tools = tools
2017-03-01 00:04:12 +03:00
let previewToolDict = dict[Keys.PreviewTool.key] as? [String: Any] ?? [:]
self.previewTool = PreviewTool.State(dict: previewToolDict) ?? PreviewTool.State.default
self.fileBrowserShowHidden = isShowHidden
}
func dict() -> [String: Any] {
return [
Keys.MainWindow.allToolsVisible: self.isAllToolsVisible,
Keys.MainWindow.toolButtonsVisible: self.isToolButtonsVisible,
Keys.Appearance.key: self.appearance.dict(),
Keys.WorkspaceTool.key: self.tools.mapToDict { ($0.rawValue, $1.dict()) },
2017-03-01 00:04:12 +03:00
2017-04-10 21:38:17 +03:00
Keys.MainWindow.orderedTools: self.orderedTools.map { $0.rawValue },
Keys.MainWindow.activeTools: self.activeTools.mapToDict { ($0.rawValue, $1) },
2017-04-10 21:38:17 +03:00
2017-03-01 00:04:12 +03:00
Keys.PreviewTool.key: self.previewTool.dict(),
Keys.MainWindow.isShowHidden: self.fileBrowserShowHidden,
Keys.MainWindow.useInteractiveZsh: self.useInteractiveZsh,
2017-03-01 00:04:12 +03:00
]
}
}
}
2017-03-01 00:04:12 +03:00
struct WorkspaceToolState: SerializableState {
2017-02-28 17:52:48 +03:00
2017-03-01 00:04:12 +03:00
static let `default` = [
MainWindow.Tools.fileBrowser: WorkspaceToolState(location: .left, dimension: 200, open: true),
MainWindow.Tools.openedFilesList: WorkspaceToolState(location: .left, dimension: 200, open: false),
MainWindow.Tools.preview: WorkspaceToolState(location: .right, dimension: 250, open: false),
2017-04-10 23:15:48 +03:00
MainWindow.Tools.htmlPreview: WorkspaceToolState(location: .right, dimension: 500, open: false),
2017-03-01 00:04:12 +03:00
]
2017-02-28 17:52:48 +03:00
2017-04-10 21:38:17 +03:00
static let `orderedDefault` = [
MainWindow.Tools.fileBrowser,
MainWindow.Tools.openedFilesList,
MainWindow.Tools.preview,
2017-04-10 23:15:48 +03:00
MainWindow.Tools.htmlPreview,
2017-04-10 21:38:17 +03:00
]
2017-02-28 17:52:48 +03:00
var location = WorkspaceBarLocation.left
var dimension = CGFloat(200)
var open = false
2017-03-01 00:04:12 +03:00
init(location: WorkspaceBarLocation, dimension: CGFloat, open: Bool) {
self.location = location
self.dimension = dimension
self.open = open
}
init?(dict: [String: Any]) {
guard let locationRawValue = dict[Keys.WorkspaceTool.location] as? String,
let isOpen = PrefUtils.bool(from: dict, for: Keys.WorkspaceTool.open),
let fDimension = PrefUtils.float(from: dict, for: Keys.WorkspaceTool.dimension)
else {
return nil
}
guard let location = WorkspaceBarLocation(rawValue: locationRawValue) else {
2017-03-01 00:04:12 +03:00
return nil
}
self.location = location
self.dimension = CGFloat(fDimension)
self.open = isOpen
}
func dict() -> [String: Any] {
return [
Keys.WorkspaceTool.location: self.location.rawValue,
2017-03-01 00:04:12 +03:00
Keys.WorkspaceTool.open: self.open,
Keys.WorkspaceTool.dimension: Float(self.dimension),
]
}
2017-02-28 17:52:48 +03:00
}
2017-02-07 00:54:22 +03:00
extension PreviewTool {
2017-03-01 00:04:12 +03:00
struct State: SerializableState {
2017-02-07 00:54:22 +03:00
static let `default` = State()
var isForwardSearchAutomatically = false
var isReverseSearchAutomatically = false
var isRefreshOnWrite = true
2017-03-01 00:04:12 +03:00
init() {
}
init?(dict: [String: Any]) {
guard let isForward = PrefUtils.bool(from: dict, for: Keys.PreviewTool.forwardSearchAutomatically),
let isReverse = PrefUtils.bool(from: dict, for: Keys.PreviewTool.reverseSearchAutomatically),
let isRefreshOnWrite = PrefUtils.bool(from: dict, for: Keys.PreviewTool.refreshOnWrite)
else {
return nil
}
self.isRefreshOnWrite = isRefreshOnWrite
self.isForwardSearchAutomatically = isForward
self.isReverseSearchAutomatically = isReverse
}
func dict() -> [String: Any] {
return [
Keys.PreviewTool.forwardSearchAutomatically: self.isForwardSearchAutomatically,
Keys.PreviewTool.reverseSearchAutomatically: self.isReverseSearchAutomatically,
Keys.PreviewTool.refreshOnWrite: self.isRefreshOnWrite,
]
}
2017-02-07 00:54:22 +03:00
}
}