1
1
mirror of https://github.com/qvacua/vimr.git synced 2024-12-22 05:01:50 +03:00
vimr/VimR/MainWindowComponent.swift

521 lines
16 KiB
Swift
Raw Normal View History

/**
* Tae Won Ha - http://taewon.de - @hataewon
* See LICENSE
*/
import Cocoa
import PureLayout
2016-07-21 20:28:58 +03:00
import RxSwift
enum MainWindowAction {
case becomeKey(mainWindow: MainWindowComponent)
2016-09-01 21:10:40 +03:00
case openQuickly(mainWindow: MainWindowComponent)
2016-09-27 19:02:05 +03:00
case changeCwd(mainWindow: MainWindowComponent)
case close(mainWindow: MainWindowComponent, mainWindowPrefData: MainWindowPrefData)
}
2016-11-19 15:29:18 +03:00
struct MainWindowPrefData: StandardPrefData {
2016-11-19 15:29:18 +03:00
fileprivate static let isAllToolsVisible = "is-all-tools-visible"
fileprivate static let isToolButtonsVisible = "is-tool-buttons-visible"
fileprivate static let toolPrefDatas = "tool-pref-datas"
2016-11-19 15:29:18 +03:00
static let `default` = MainWindowPrefData(isAllToolsVisible: true,
isToolButtonsVisible: true,
toolPrefDatas: [ ToolPrefData.defaults[.fileBrowser]! ])
var isAllToolsVisible: Bool
var isToolButtonsVisible: Bool
var toolPrefDatas: [ToolPrefData]
init(isAllToolsVisible: Bool, isToolButtonsVisible: Bool, toolPrefDatas: [ToolPrefData]) {
self.isAllToolsVisible = isAllToolsVisible
self.isToolButtonsVisible = isToolButtonsVisible
self.toolPrefDatas = toolPrefDatas
}
init?(dict: [String: Any]) {
guard let isAllToolsVisible = PrefUtils.bool(from: dict, for: MainWindowPrefData.isAllToolsVisible),
let isToolButtonsVisible = PrefUtils.bool(from: dict, for: MainWindowPrefData.isToolButtonsVisible),
let toolDataDicts = dict[MainWindowPrefData.toolPrefDatas] as? [[String: Any]]
else {
return nil
}
2016-11-19 15:29:18 +03:00
// Add default tool pref data for missing identifiers.
let toolDatas = toolDataDicts.map { ToolPrefData(dict: $0) }.flatMap { $0 }
let missingToolDatas = Set(ToolIdentifier.all)
.subtracting(toolDatas.map { $0.identifier })
.map { ToolPrefData.defaults[$0] }
.flatMap { $0 }
2016-09-27 19:02:05 +03:00
2016-11-19 15:29:18 +03:00
self.init(isAllToolsVisible: isAllToolsVisible,
isToolButtonsVisible: isToolButtonsVisible,
toolPrefDatas: [ toolDatas, missingToolDatas ].flatMap { $0 })
}
func dict() -> [String: Any] {
return [
MainWindowPrefData.isAllToolsVisible: self.isAllToolsVisible,
MainWindowPrefData.isToolButtonsVisible: self.isToolButtonsVisible,
MainWindowPrefData.toolPrefDatas: self.toolPrefDatas.map { $0.dict() },
]
}
func toolPrefData(for identifier: ToolIdentifier) -> ToolPrefData {
guard let tool = self.toolPrefDatas.filter({ $0.identifier == identifier }).first else {
preconditionFailure("[ERROR] No tool for \(identifier) found!")
}
return tool
}
2016-09-27 19:02:05 +03:00
}
2016-10-03 16:03:18 +03:00
class MainWindowComponent: WindowComponent, NSWindowDelegate, NSUserInterfaceValidations, WorkspaceDelegate {
2016-07-21 20:28:58 +03:00
2016-09-27 01:17:53 +03:00
fileprivate static let nibName = "MainWindow"
2016-07-21 20:28:58 +03:00
2016-09-25 18:50:33 +03:00
fileprivate var defaultEditorFont: NSFont
2016-10-27 09:21:29 +03:00
// fileprivate var usesLigatures: Bool
2016-07-27 00:40:20 +03:00
2016-10-02 15:11:39 +03:00
fileprivate var _cwd: URL = FileUtils.userHomeUrl
2016-09-27 01:17:53 +03:00
fileprivate let fontManager = NSFontManager.shared()
fileprivate let fileItemService: FileItemService
fileprivate let workspace: Workspace
fileprivate let neoVimView: NeoVimView
2016-11-18 20:54:14 +03:00
fileprivate var tools = [ToolIdentifier: WorkspaceToolComponent]()
2016-10-02 15:07:12 +03:00
2016-09-27 01:17:53 +03:00
// MARK: - API
var uuid: String {
return self.neoVimView.uuid
}
2016-07-21 20:28:58 +03:00
2016-09-25 18:50:33 +03:00
var cwd: URL {
2016-09-03 00:35:18 +03:00
get {
2016-09-07 21:12:18 +03:00
self._cwd = self.neoVimView.cwd
return self._cwd
2016-09-03 00:35:18 +03:00
}
set {
2016-09-07 21:12:18 +03:00
let oldValue = self._cwd
if oldValue == newValue {
return
}
self._cwd = newValue
2016-09-03 00:35:18 +03:00
self.neoVimView.cwd = newValue
2016-09-07 21:12:18 +03:00
self.fileItemService.unmonitor(url: oldValue)
self.fileItemService.monitor(url: newValue)
2016-09-03 00:35:18 +03:00
}
}
2016-09-24 17:31:14 +03:00
// TODO: Consider an option object for cwd, urls, etc...
/**
The init() method does not show the window. Call MainWindowComponent.show() to do so.
*/
2016-09-24 17:31:14 +03:00
init(source: Observable<Any>,
fileItemService: FileItemService,
2016-09-25 18:50:33 +03:00
cwd: URL,
urls: [URL] = [],
2016-09-24 17:31:14 +03:00
initialData: PrefData)
{
self.neoVimView = NeoVimView(frame: CGRect.zero,
2016-09-25 09:55:26 +03:00
config: NeoVimView.Config(useInteractiveZsh: initialData.advanced.useInteractiveZsh))
2016-09-24 17:31:14 +03:00
self.neoVimView.translatesAutoresizingMaskIntoConstraints = false
self.workspace = Workspace(mainView: self.neoVimView)
self.defaultEditorFont = initialData.appearance.editorFont
2016-09-07 21:12:18 +03:00
self.fileItemService = fileItemService
self._cwd = cwd
2016-07-21 20:28:58 +03:00
2016-09-27 01:17:53 +03:00
super.init(source: source, nibName: MainWindowComponent.nibName)
2016-07-21 20:28:58 +03:00
self.window.delegate = self
2016-09-27 19:02:05 +03:00
self.workspace.delegate = self
// FIXME: We do not use [self.sink, source].toMergedObservables. If we do so, then self.sink seems to live as long
// as source, i.e. forever. Thus, self (MainWindowComponent) does not get deallocated. Not nice...
2016-09-27 19:02:05 +03:00
let fileBrowser = FileBrowserComponent(source: self.sink, fileItemService: fileItemService)
2016-11-18 20:54:14 +03:00
let fileBrowserTool = WorkspaceToolComponent(title: "Files",
viewComponent: fileBrowser,
toolIdentifier: .fileBrowser,
minimumDimension: 100)
2016-09-27 19:02:05 +03:00
self.tools[.fileBrowser] = fileBrowserTool
2016-07-24 21:32:07 +03:00
2016-10-02 15:07:12 +03:00
self.addReactions()
2016-09-07 21:12:18 +03:00
self.neoVimView.delegate = self
self.neoVimView.font = self.defaultEditorFont
2016-10-27 09:21:29 +03:00
self.neoVimView.usesLigatures = initialData.appearance.editorUsesLigatures
self.neoVimView.linespacing = initialData.appearance.editorLinespacing
self.neoVimView.cwd = cwd // This will publish the MainWindowAction.changeCwd action for the file browser.
self.neoVimView.open(urls: urls)
2016-09-07 21:12:18 +03:00
// We don't call self.fileItemService.monitor(url: cwd) here since self.neoVimView.cwd = cwd causes the call
// cwdChanged() and in that function we do monitor(...).
// By default the tool buttons are shown and no tools are shown.
let mainWindowData = initialData.mainWindow
2016-11-19 15:29:18 +03:00
let fileBrowserToolData = mainWindowData.toolPrefData(for: .fileBrowser)
self.workspace.append(tool: fileBrowserTool, location: fileBrowserToolData.location)
fileBrowserTool.dimension = CGFloat(fileBrowserToolData.dimension)
if !mainWindowData.isAllToolsVisible {
self.toggleAllTools(self)
}
if !mainWindowData.isToolButtonsVisible {
self.toggleToolButtons(self)
}
2016-11-19 15:29:18 +03:00
if fileBrowserToolData.isVisible {
2016-10-03 16:03:18 +03:00
fileBrowserTool.toggle()
}
2016-07-21 20:28:58 +03:00
self.window.makeFirstResponder(self.neoVimView)
2016-07-27 00:40:20 +03:00
}
2016-09-25 18:50:33 +03:00
func open(urls: [URL]) {
2016-08-25 00:06:39 +03:00
self.neoVimView.open(urls: urls)
2016-10-03 16:03:18 +03:00
self.window.makeFirstResponder(self.neoVimView)
2016-08-25 00:06:39 +03:00
}
func isDirty() -> Bool {
return self.neoVimView.hasDirtyDocs()
}
func closeAllNeoVimWindows() {
self.neoVimView.closeAllWindows()
}
func closeAllNeoVimWindowsWithoutSaving() {
self.neoVimView.closeAllWindowsWithoutSaving()
}
2016-10-02 15:07:12 +03:00
// MARK: - Private
fileprivate func addReactions() {
self.tools.values
2016-10-02 15:07:12 +03:00
.map { $0.sink }
.toMergedObservables()
.subscribe(onNext: { [unowned self] action in
switch action {
2016-11-12 18:42:10 +03:00
2016-10-02 15:07:12 +03:00
case let FileBrowserAction.open(url: url):
2016-11-12 18:42:10 +03:00
self.neoVimView.open(urls: [url])
case let FileBrowserAction.openInNewTab(url: url):
self.neoVimView.openInNewTab(urls: [url])
case let FileBrowserAction.openInCurrentTab(url: url):
self.neoVimView.openInCurrentTab(url: url)
case let FileBrowserAction.openInHorizontalSplit(url: url):
self.neoVimView.openInHorizontalSplit(urls: [url])
case let FileBrowserAction.openInVerticalSplit(url: url):
self.neoVimView.openInVerticalSplit(urls: [url])
case let FileBrowserAction.setAsWorkingDirectory(url: url):
self.neoVimView.cwd = url
case let FileBrowserAction.setParentAsWorkingDirectory(url: url):
self.neoVimView.cwd = url.parent
2016-11-12 18:42:10 +03:00
2016-10-02 15:07:12 +03:00
default:
NSLog("unrecognized action: \(action)")
return
}
2016-11-12 18:42:10 +03:00
self.window.makeFirstResponder(self.neoVimView)
2016-10-02 15:07:12 +03:00
})
.addDisposableTo(self.disposeBag)
}
2016-09-27 01:17:53 +03:00
// MARK: - WindowComponent
override func addViews() {
self.window.contentView?.addSubview(self.workspace)
self.workspace.autoPinEdgesToSuperviewEdges()
2016-07-21 20:28:58 +03:00
}
2016-07-24 21:32:07 +03:00
2016-09-25 18:50:33 +03:00
override func subscription(source: Observable<Any>) -> Disposable {
return source
2016-07-24 21:32:07 +03:00
.filter { $0 is PrefData }
.map { ($0 as! PrefData).appearance }
2016-08-14 16:38:41 +03:00
.filter { [unowned self] appearanceData in
2016-09-25 18:50:33 +03:00
!appearanceData.editorFont.isEqual(to: self.neoVimView.font)
2016-08-14 16:38:41 +03:00
|| appearanceData.editorUsesLigatures != self.neoVimView.usesLigatures
2016-10-27 09:21:29 +03:00
|| appearanceData.editorLinespacing != self.neoVimView.linespacing
2016-08-14 16:38:41 +03:00
}
.observeOn(MainScheduler.instance)
2016-09-25 19:10:07 +03:00
.subscribe(onNext: { [unowned self] appearance in
self.neoVimView.usesLigatures = appearance.editorUsesLigatures
self.neoVimView.font = appearance.editorFont
2016-10-27 09:21:29 +03:00
self.neoVimView.linespacing = appearance.editorLinespacing
2016-09-25 19:10:07 +03:00
})
2016-07-24 21:32:07 +03:00
}
2016-07-21 20:28:58 +03:00
}
2016-09-27 19:02:05 +03:00
// MARK: - WorkspaceDelegate
extension MainWindowComponent {
func resizeWillStart(workspace: Workspace) {
self.neoVimView.enterResizeMode()
}
func resizeDidEnd(workspace: Workspace) {
self.neoVimView.exitResizeMode()
}
}
// MARK: - File Menu Item Actions
2016-08-11 22:19:03 +03:00
extension MainWindowComponent {
2016-09-27 01:17:53 +03:00
@IBAction func newTab(_ sender: Any?) {
2016-08-11 22:19:03 +03:00
self.neoVimView.newTab()
}
2016-08-11 23:37:41 +03:00
2016-09-27 01:17:53 +03:00
@IBAction func openDocument(_ sender: Any?) {
2016-08-11 22:19:03 +03:00
let panel = NSOpenPanel()
panel.canChooseDirectories = true
panel.allowsMultipleSelection = true
2016-09-25 18:50:33 +03:00
panel.beginSheetModal(for: self.window) { result in
2016-08-11 22:19:03 +03:00
guard result == NSFileHandlingPanelOKButton else {
return
}
let urls = panel.urls
if self.neoVimView.allBuffers().count == 1 {
let isTransient = self.neoVimView.allBuffers().first?.isTransient ?? false
if isTransient {
self.neoVimView.cwd = FileUtils.commonParent(of: urls)
}
}
self.neoVimView.open(urls: urls)
2016-08-20 20:02:16 +03:00
}
}
2016-09-27 01:17:53 +03:00
@IBAction func openQuickly(_ sender: Any?) {
2016-09-07 21:12:18 +03:00
self.publish(event: MainWindowAction.openQuickly(mainWindow: self))
}
2016-09-27 01:17:53 +03:00
@IBAction func saveDocument(_ sender: Any?) {
guard let curBuf = self.neoVimView.currentBuffer() else {
return
}
2016-08-20 20:02:16 +03:00
if curBuf.fileName == nil {
self.savePanelSheet { self.neoVimView.saveCurrentTab(url: $0) }
return
}
self.neoVimView.saveCurrentTab()
}
2016-09-27 01:17:53 +03:00
@IBAction func saveDocumentAs(_ sender: Any?) {
if self.neoVimView.currentBuffer() == nil {
return
}
2016-08-20 20:02:16 +03:00
self.savePanelSheet { url in
self.neoVimView.saveCurrentTab(url: url)
if self.neoVimView.isCurrentBufferDirty() {
self.neoVimView.openInNewTab(urls: [url])
} else {
self.neoVimView.openInCurrentTab(url: url)
}
}
}
2016-09-25 18:50:33 +03:00
fileprivate func savePanelSheet(action: @escaping (URL) -> Void) {
2016-08-20 20:02:16 +03:00
let panel = NSSavePanel()
2016-09-25 18:50:33 +03:00
panel.beginSheetModal(for: self.window) { result in
2016-08-20 20:02:16 +03:00
guard result == NSFileHandlingPanelOKButton else {
return
}
let showAlert: () -> Void = {
let alert = NSAlert()
2016-09-25 18:50:33 +03:00
alert.addButton(withTitle: "OK")
2016-08-20 20:02:16 +03:00
alert.messageText = "Invalid File Name"
alert.informativeText = "The file name you have entered cannot be used. Please use a different name."
2016-09-25 18:50:33 +03:00
alert.alertStyle = .warning
2016-08-20 20:02:16 +03:00
alert.runModal()
}
2016-09-25 18:50:33 +03:00
guard let url = panel.url else {
2016-08-20 20:02:16 +03:00
showAlert()
return
}
action(url)
}
}
2016-08-11 22:19:03 +03:00
}
2016-10-03 16:03:18 +03:00
// MARK: - Tools Menu Item Actions
extension MainWindowComponent {
@IBAction func toggleAllTools(_ sender: Any?) {
self.workspace.toggleAllTools()
2016-10-03 16:03:18 +03:00
self.focusNeoVimView(self)
}
@IBAction func toggleToolButtons(_ sender: Any?) {
self.workspace.toggleToolButtons()
}
2016-10-03 16:03:18 +03:00
@IBAction func toggleFileBrowser(_ sender: Any?) {
let fileBrowserTool = self.tools[.fileBrowser]!
if fileBrowserTool.isSelected {
if fileBrowserTool.viewComponent.isFirstResponder {
fileBrowserTool.toggle()
self.focusNeoVimView(self)
2016-10-03 16:03:18 +03:00
} else {
fileBrowserTool.viewComponent.beFirstResponder()
}
return
}
fileBrowserTool.toggle()
fileBrowserTool.viewComponent.beFirstResponder()
}
@IBAction func focusNeoVimView(_ sender: Any?) {
self.window.makeFirstResponder(self.neoVimView)
}
}
// MARK: - Font Menu Item Actions
extension MainWindowComponent {
2016-09-27 01:17:53 +03:00
@IBAction func resetFontSize(_ sender: Any?) {
self.neoVimView.font = self.defaultEditorFont
}
2016-09-27 01:17:53 +03:00
@IBAction func makeFontBigger(_ sender: Any?) {
let curFont = self.neoVimView.font
2016-11-19 15:29:18 +03:00
let font = self.fontManager.convert(curFont, toSize: min(curFont.pointSize + 1, NeoVimView.maxFontSize))
self.neoVimView.font = font
}
2016-09-27 01:17:53 +03:00
@IBAction func makeFontSmaller(_ sender: Any?) {
let curFont = self.neoVimView.font
2016-11-19 15:29:18 +03:00
let font = self.fontManager.convert(curFont, toSize: max(curFont.pointSize - 1, NeoVimView.minFontSize))
self.neoVimView.font = font
}
}
2016-07-21 20:28:58 +03:00
// MARK: - NeoVimViewDelegate
2016-09-07 21:12:18 +03:00
extension MainWindowComponent: NeoVimViewDelegate {
2016-07-21 20:28:58 +03:00
2016-09-27 01:17:53 +03:00
func set(title: String) {
self.window.title = title
2016-07-21 20:28:58 +03:00
}
2016-07-27 00:40:20 +03:00
2016-09-27 01:17:53 +03:00
func set(dirtyStatus: Bool) {
self.windowController.setDocumentEdited(dirtyStatus)
}
2016-09-07 21:12:18 +03:00
func cwdChanged() {
let old = self._cwd
self._cwd = self.neoVimView.cwd
self.fileItemService.unmonitor(url: old)
self.fileItemService.monitor(url: self._cwd)
2016-09-27 19:02:05 +03:00
self.publish(event: MainWindowAction.changeCwd(mainWindow: self))
2016-09-07 21:12:18 +03:00
}
2016-07-21 20:28:58 +03:00
func neoVimStopped() {
self.windowController.close()
}
}
// MARK: - NSWindowDelegate
extension MainWindowComponent {
2016-08-25 00:06:39 +03:00
2016-09-25 18:50:33 +03:00
func windowDidBecomeKey(_: Notification) {
self.publish(event: MainWindowAction.becomeKey(mainWindow: self))
2016-08-25 00:06:39 +03:00
}
2016-07-21 20:28:58 +03:00
2016-09-25 18:50:33 +03:00
func windowWillClose(_ notification: Notification) {
2016-09-07 21:12:18 +03:00
self.fileItemService.unmonitor(url: self._cwd)
let fileBrowser = self.tools[.fileBrowser]!
2016-11-19 15:29:18 +03:00
let fileBrowserData = ToolPrefData(identifier: .fileBrowser,
location: fileBrowser.location,
isVisible: fileBrowser.isSelected,
dimension: fileBrowser.dimension)
let prefData = MainWindowPrefData(isAllToolsVisible: self.workspace.isAllToolsVisible,
isToolButtonsVisible: self.workspace.isToolButtonsVisible,
2016-11-19 15:29:18 +03:00
toolPrefDatas: [ fileBrowserData ])
self.publish(event: MainWindowAction.close(mainWindow: self, mainWindowPrefData: prefData))
}
2016-09-25 18:50:33 +03:00
func windowShouldClose(_ sender: Any) -> Bool {
if self.neoVimView.isCurrentBufferDirty() {
let alert = NSAlert()
2016-09-25 18:50:33 +03:00
alert.addButton(withTitle: "Cancel")
alert.addButton(withTitle: "Discard and Close")
alert.messageText = "The current buffer has unsaved changes!"
2016-09-25 18:50:33 +03:00
alert.alertStyle = .warning
alert.beginSheetModal(for: self.window, completionHandler: { response in
if response == NSAlertSecondButtonReturn {
self.neoVimView.closeCurrentTabWithoutSaving()
}
2016-09-25 18:50:33 +03:00
})
return false
}
self.neoVimView.closeCurrentTab()
return false
}
2016-09-07 21:12:18 +03:00
}
2016-10-03 16:03:18 +03:00
// MARK: - NSUserInterfaceValidationsProtocol
extension MainWindowComponent {
public func validateUserInterfaceItem(_ item: NSValidatedUserInterfaceItem) -> Bool {
let canSave = self.neoVimView.currentBuffer() != nil
let canSaveAs = canSave
let canOpen = canSave
let canOpenQuickly = canSave
let canFocusNeoVimView = self.window.firstResponder != self.neoVimView
guard let action = item.action else {
2016-10-03 16:03:18 +03:00
return true
}
switch action {
case #selector(focusNeoVimView(_:)):
return canFocusNeoVimView
case #selector(openDocument(_:)):
return canOpen
case #selector(openQuickly(_:)):
return canOpenQuickly
case #selector(saveDocument(_:)):
return canSave
2016-10-03 16:03:18 +03:00
case #selector(saveDocumentAs(_:)):
return canSaveAs
default:
return true
}
2016-10-03 16:03:18 +03:00
}
}