1
1
mirror of https://github.com/qvacua/vimr.git synced 2024-11-24 03:25:03 +03:00
vimr/VimR/Context.swift

172 lines
5.5 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-02-28 12:43:45 +03:00
class Context {
2017-01-25 00:39:19 +03:00
let stateSource: Observable<AppState>
2017-04-22 16:56:13 +03:00
let actionEmitter = ActionEmitter()
init(baseServerUrl: URL, state: AppState) {
2017-02-20 20:04:45 +03:00
2017-02-28 12:42:34 +03:00
self.appState = state
self.stateSource = self.stateSubject.asObservable()
2017-04-02 18:27:12 +03:00
self.httpService = HttpServerService(port: baseServerUrl.port!)
2017-03-29 21:10:11 +03:00
let openQuicklyReducer = OpenQuicklyReducer()
let markdownReducer = MarkdownReducer(baseServerUrl: baseServerUrl)
2017-02-28 12:20:25 +03:00
let prefService = PrefService()
let htmlPreviewToolReducer = HtmlPreviewToolReducer(baseServerUrl: baseServerUrl)
2017-02-28 12:20:25 +03:00
let previewService = PreviewService()
2017-02-28 12:42:34 +03:00
// AppState
2017-01-22 16:22:05 +03:00
Observable
.of(
2017-04-22 12:31:42 +03:00
self.actionSourceForAppState()
.reduce(by: AppDelegateReducer(baseServerUrl: baseServerUrl))
.filterMapPair(),
self.actionSourceForAppState()
.reduce(by: UiRootReducer())
.reduce(by: openQuicklyReducer.forMainWindow)
.filter { $0.modified }
.apply(to: prefService.forMainWindow)
.map { $0.state },
self.actionSourceForAppState()
.reduce(by: FileMonitorReducer())
.filterMapPair(),
self.actionSourceForAppState()
.reduce(by: openQuicklyReducer.forOpenQuicklyWindow)
.filterMapPair()
2017-01-22 16:22:05 +03:00
)
.merge()
.subscribe(onNext: self.emitAppState)
.disposed(by: self.disposeBag)
2017-02-28 12:42:34 +03:00
// MainWindow.State
2017-02-12 13:24:15 +03:00
Observable
2017-02-24 02:24:30 +03:00
.of(
2017-04-22 12:31:42 +03:00
self.actionSourceForMainWindow()
.reduce(by: MainWindowReducer())
.reduce(by: markdownReducer.forMainWindow)
.filter { $0.modified }
.apply(to: previewService.forMainWindow)
.apply(to: self.httpService.forMainWindow)
.map { $0.state },
self.actionSourceForMainWindow()
.reduce(by: PreviewToolReducer(baseServerUrl: baseServerUrl))
.filterMapPair(),
self.actionSourceForMainWindow()
.reduce(by: htmlPreviewToolReducer)
.filter { $0.modified }
.apply(to: self.httpService.forHtmlPreviewTool)
.map { $0.state },
self.actionSourceForMainWindow()
.reduce(by: FileBrowserReducer())
.filterMapPair(),
self.actionSourceForMainWindow()
.reduce(by: OpenedFileListReducer())
.reduce(by: markdownReducer.forOpenedFileList)
.filter { $0.modified }
.apply(to: previewService.forOpenedFileList)
.map { $0.state }
2017-02-24 02:24:30 +03:00
)
2017-02-12 13:24:15 +03:00
.merge()
.subscribe(onNext: self.emitAppState)
.disposed(by: self.disposeBag)
2017-02-28 12:42:34 +03:00
// Preferences
2017-02-28 00:45:26 +03:00
Observable
.of(
2017-04-22 12:31:42 +03:00
self.actionSourceForAppState()
.reduce(by: PrefWindowReducer())
.filterMapPair(),
self.actionSourceForAppState()
.reduce(by: GeneralPrefReducer())
.filterMapPair(),
self.actionSourceForAppState()
.reduce(by: AppearancePrefReducer())
.filterMapPair(),
self.actionSourceForAppState()
.reduce(by: AdvancedPrefReducer())
.filterMapPair()
2017-04-22 17:03:45 +03:00
)
2017-02-28 00:45:26 +03:00
.merge()
.apply(to: prefService.forPrefPanes)
.subscribe(onNext: self.emitAppState)
.disposed(by: self.disposeBag)
2017-02-24 02:24:30 +03:00
2017-02-03 00:39:05 +03:00
#if DEBUG
// actionSource.debug().subscribe().disposed(by: self.disposeBag)
// stateSource.debug().subscribe().disposed(by: self.disposeBag)
2017-02-03 00:39:05 +03:00
#endif
}
deinit {
self.stateSubject.onCompleted()
}
2017-04-02 18:27:12 +03:00
fileprivate let httpService: HttpServerService
2017-03-24 19:47:08 +03:00
2017-01-25 00:39:19 +03:00
fileprivate let stateSubject = PublishSubject<AppState>()
2017-01-22 16:22:05 +03:00
fileprivate let scheduler = SerialDispatchQueueScheduler(qos: .userInitiated)
fileprivate let disposeBag = DisposeBag()
2017-02-06 20:57:50 +03:00
fileprivate var appState: AppState
2017-02-28 12:42:34 +03:00
fileprivate func emitAppState(_ mainWindow: UuidState<MainWindow.State>) {
self.appState.mainWindows[mainWindow.uuid] = mainWindow.payload
self.stateSubject.onNext(self.appState)
self.cleanUpAppState()
}
fileprivate func emitAppState(_ appState: AppState) {
self.appState = appState
self.stateSubject.onNext(self.appState)
self.cleanUpAppState()
}
fileprivate func cleanUpAppState() {
self.appState.mainWindows.keys.forEach { uuid in
2017-03-31 20:32:01 +03:00
self.appState.mainWindows[uuid]?.viewToBeFocused = nil
self.appState.mainWindows[uuid]?.urlsToOpen.removeAll()
}
}
2017-04-22 17:03:45 +03:00
fileprivate func actionSourceForAppState<ActionType>() -> Observable<StateActionPair<AppState, ActionType>> {
2017-04-22 12:31:42 +03:00
return self.actionEmitter.observable
2017-04-22 17:03:45 +03:00
.mapOmittingNil { $0 as? ActionType }
2017-04-22 12:31:42 +03:00
.map { self.appStateActionPair(for: $0) }
}
2017-04-22 17:03:45 +03:00
fileprivate func actionSourceForMainWindow<ActionType>()
-> Observable<StateActionPair<UuidState<MainWindow.State>, ActionType>> {
2017-04-22 12:31:42 +03:00
return self.actionEmitter.observable
2017-04-22 17:03:45 +03:00
.mapOmittingNil { $0 as? UuidAction<ActionType> }
2017-04-22 12:31:42 +03:00
.mapOmittingNil { self.mainWindowStateActionPair(for: $0) }
}
2017-02-28 12:42:34 +03:00
fileprivate func appStateActionPair<ActionType>(for action: ActionType) -> StateActionPair<AppState, ActionType> {
return StateActionPair(state: self.appState, action: action, modified: false)
}
fileprivate func mainWindowStateActionPair<ActionType>(for action: UuidAction<ActionType>)
-> StateActionPair<UuidState<MainWindow.State>, ActionType>? {
guard let mainWindowState = self.appState.mainWindows[action.uuid] else {
return nil
}
return StateActionPair(state: UuidState(uuid: action.uuid, state: mainWindowState),
action: action.payload,
modified: false)
}
}