2016-06-05 10:49:14 +03:00
|
|
|
/**
|
|
|
|
* Tae Won Ha - http://taewon.de - @hataewon
|
|
|
|
* See LICENSE
|
|
|
|
*/
|
|
|
|
|
|
|
|
import Cocoa
|
2016-06-18 12:43:37 +03:00
|
|
|
|
2016-06-19 11:07:03 +03:00
|
|
|
/// Contiguous piece of cells of a row that has the same attributes.
|
|
|
|
private struct RowRun: CustomStringConvertible {
|
2016-06-18 12:43:37 +03:00
|
|
|
|
|
|
|
let row: Int
|
|
|
|
let range: Range<Int>
|
|
|
|
let attrs: CellAttributes
|
|
|
|
|
|
|
|
var description: String {
|
2016-06-19 11:07:03 +03:00
|
|
|
return "RowRun<\(row): \(range)\n\(attrs)>"
|
2016-06-18 12:43:37 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-05 00:34:09 +03:00
|
|
|
public class NeoVimView: NSView, NSUserInterfaceValidations {
|
2016-07-29 19:25:32 +03:00
|
|
|
|
|
|
|
public static let minFontSize = CGFloat(4)
|
|
|
|
public static let maxFontSize = CGFloat(128)
|
|
|
|
public static let defaultFont = NSFont(name: "Menlo", size: 13)!
|
2016-07-14 22:53:20 +03:00
|
|
|
|
2016-07-10 15:14:02 +03:00
|
|
|
public let uuid = NSUUID().UUIDString
|
2016-07-17 15:41:53 +03:00
|
|
|
public weak var delegate: NeoVimViewDelegate?
|
2016-07-30 10:49:39 +03:00
|
|
|
|
2016-07-16 01:07:50 +03:00
|
|
|
private let agent: NeoVimAgent
|
2016-07-30 10:49:39 +03:00
|
|
|
private let drawer: TextDrawer
|
2016-08-03 22:35:24 +03:00
|
|
|
private let fontManager = NSFontManager.sharedFontManager()
|
2016-08-04 21:01:03 +03:00
|
|
|
private let pasteboard = NSPasteboard.generalPasteboard()
|
2016-07-10 15:14:02 +03:00
|
|
|
|
2016-08-03 22:30:41 +03:00
|
|
|
public private(set) var mode = Mode.Normal
|
|
|
|
|
2016-07-16 01:07:50 +03:00
|
|
|
private let grid = Grid()
|
|
|
|
|
|
|
|
private var markedText: String?
|
2016-06-24 22:08:34 +03:00
|
|
|
|
2016-07-02 22:38:52 +03:00
|
|
|
/// We store the last marked text because Cocoa's text input system does the following:
|
|
|
|
/// 하 -> hanja popup -> insertText(하) -> attributedSubstring...() -> setMarkedText(下) -> ...
|
|
|
|
/// We want to return "하" in attributedSubstring...()
|
2016-07-16 01:07:50 +03:00
|
|
|
private var lastMarkedText: String?
|
2016-07-02 22:38:52 +03:00
|
|
|
|
2016-07-16 01:07:50 +03:00
|
|
|
private var markedPosition = Position.null
|
|
|
|
private var keyDownDone = true
|
2016-07-14 00:08:45 +03:00
|
|
|
|
2016-07-16 01:07:50 +03:00
|
|
|
private var lastClickedCellPosition = Position.null
|
2016-06-06 19:25:03 +03:00
|
|
|
|
2016-07-16 01:07:50 +03:00
|
|
|
private var xOffset = CGFloat(0)
|
|
|
|
private var yOffset = CGFloat(0)
|
|
|
|
private var cellSize = CGSize.zero
|
|
|
|
private var descent = CGFloat(0)
|
|
|
|
private var leading = CGFloat(0)
|
2016-07-16 19:38:54 +03:00
|
|
|
|
|
|
|
private let maxScrollDeltaX = 30
|
|
|
|
private let maxScrollDeltaY = 30
|
|
|
|
private let scrollLimiterX = CGFloat(20)
|
|
|
|
private let scrollLimiterY = CGFloat(20)
|
|
|
|
private var scrollGuardCounterX = 5
|
|
|
|
private var scrollGuardCounterY = 5
|
|
|
|
private let scrollGuardYield = 5
|
2016-07-29 23:01:09 +03:00
|
|
|
|
|
|
|
private var isCurrentlyPinching = false
|
|
|
|
private var pinchTargetScale = CGFloat(1)
|
|
|
|
private var pinchImage = NSImage()
|
2016-07-30 19:31:33 +03:00
|
|
|
|
2016-07-31 00:04:20 +03:00
|
|
|
public var usesLigatures = false {
|
2016-07-30 19:31:33 +03:00
|
|
|
didSet {
|
2016-07-31 00:04:20 +03:00
|
|
|
self.drawer.usesLigatures = self.usesLigatures
|
2016-07-30 19:31:33 +03:00
|
|
|
self.needsDisplay = true
|
|
|
|
}
|
|
|
|
}
|
2016-06-15 00:50:25 +03:00
|
|
|
|
2016-07-29 23:01:09 +03:00
|
|
|
private var _font = NeoVimView.defaultFont
|
2016-07-28 20:37:39 +03:00
|
|
|
public var font: NSFont {
|
|
|
|
get {
|
|
|
|
return self._font
|
|
|
|
}
|
|
|
|
|
|
|
|
set {
|
|
|
|
guard newValue.fixedPitch else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-07-30 10:49:39 +03:00
|
|
|
let size = newValue.pointSize
|
|
|
|
guard size >= NeoVimView.minFontSize && size <= NeoVimView.maxFontSize else {
|
2016-07-29 23:01:09 +03:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-07-28 20:37:39 +03:00
|
|
|
self._font = newValue
|
2016-07-10 15:14:02 +03:00
|
|
|
self.drawer.font = self.font
|
|
|
|
self.cellSize = self.drawer.cellSize
|
|
|
|
self.descent = self.drawer.descent
|
|
|
|
self.leading = self.drawer.leading
|
|
|
|
|
2016-08-01 08:46:04 +03:00
|
|
|
self.resizeNeoVimUiTo(size: self.bounds.size)
|
2016-07-10 15:14:02 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-09 23:52:59 +03:00
|
|
|
override init(frame rect: NSRect = CGRect.zero) {
|
2016-07-30 19:31:33 +03:00
|
|
|
self.drawer = TextDrawer(font: self._font, useLigatures: false)
|
2016-07-10 15:14:02 +03:00
|
|
|
self.agent = NeoVimAgent(uuid: self.uuid)
|
2016-07-14 22:53:20 +03:00
|
|
|
|
2016-06-05 10:49:14 +03:00
|
|
|
super.init(frame: rect)
|
2016-06-18 12:43:37 +03:00
|
|
|
|
2016-06-15 23:11:35 +03:00
|
|
|
self.wantsLayer = true
|
2016-06-18 12:43:37 +03:00
|
|
|
self.cellSize = self.drawer.cellSize
|
2016-06-19 14:39:20 +03:00
|
|
|
self.descent = self.drawer.descent
|
|
|
|
self.leading = self.drawer.leading
|
2016-07-09 23:52:59 +03:00
|
|
|
|
|
|
|
// We cannot set bridge in init since self is not available before super.init()...
|
|
|
|
self.agent.bridge = self
|
|
|
|
self.agent.establishLocalServer()
|
|
|
|
}
|
2016-07-17 15:41:53 +03:00
|
|
|
|
2016-08-12 15:54:37 +03:00
|
|
|
required public init?(coder: NSCoder) {
|
|
|
|
fatalError("init(coder:) has not been implemented")
|
|
|
|
}
|
2016-06-15 23:11:35 +03:00
|
|
|
|
2016-08-04 21:01:03 +03:00
|
|
|
@IBAction public func debug1(sender: AnyObject!) {
|
|
|
|
NSLog("DEBUG-1")
|
|
|
|
}
|
|
|
|
|
2016-06-27 19:42:37 +03:00
|
|
|
public func debugInfo() {
|
|
|
|
Swift.print(self.grid)
|
|
|
|
}
|
2016-08-12 15:54:37 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - API
|
|
|
|
extension NeoVimView {
|
|
|
|
|
|
|
|
public func hasDirtyDocs() -> Bool {
|
|
|
|
return self.agent.hasDirtyDocs()
|
|
|
|
}
|
2016-08-13 00:07:16 +03:00
|
|
|
|
|
|
|
public func isCurrentBufferDirty() -> Bool {
|
|
|
|
return self.agent.buffers().filter { $0.current }.first?.dirty ?? true
|
|
|
|
}
|
2016-08-12 15:54:37 +03:00
|
|
|
|
|
|
|
public func newTab() {
|
|
|
|
switch self.mode {
|
|
|
|
case .Normal:
|
|
|
|
self.agent.vimInput(":tabe<CR>")
|
|
|
|
default:
|
|
|
|
self.agent.vimInput("<Esc>:tabe<CR>")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func open(urls urls: [NSURL]) {
|
|
|
|
let currentBufferIsTransient = self.agent.buffers().filter { $0.current }.first?.transient ?? false
|
|
|
|
|
|
|
|
urls.enumerate().forEach { (idx, url) in
|
|
|
|
if idx == 0 && currentBufferIsTransient {
|
|
|
|
self.open(url, cmd: ":e")
|
|
|
|
} else {
|
|
|
|
self.open(url, cmd: ":tabe")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func openInNewTab(urls urls: [NSURL]) {
|
|
|
|
urls.forEach { self.open($0, cmd: ":tabe") }
|
|
|
|
}
|
2016-08-13 00:07:16 +03:00
|
|
|
|
|
|
|
public func closeCurrentTab() {
|
|
|
|
switch self.mode {
|
|
|
|
case .Normal:
|
|
|
|
self.agent.vimInput(":q<CR>")
|
|
|
|
default:
|
|
|
|
self.agent.vimInput("<Esc>:q<CR>")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func closeCurrentTabWithoutSaving() {
|
|
|
|
switch self.mode {
|
|
|
|
case .Normal:
|
|
|
|
self.agent.vimInput(":q!<CR>")
|
|
|
|
default:
|
|
|
|
self.agent.vimInput("<Esc>:q!<CR>")
|
|
|
|
}
|
|
|
|
}
|
2016-08-12 15:54:37 +03:00
|
|
|
|
2016-08-12 19:00:05 +03:00
|
|
|
public func closeAllWindowsWithoutSaving() {
|
|
|
|
switch self.mode {
|
|
|
|
case .Normal:
|
|
|
|
self.agent.vimInput(":qa!<CR>")
|
|
|
|
default:
|
|
|
|
self.agent.vimInput("<Esc>:qa!<CR>")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-12 15:54:37 +03:00
|
|
|
private func open(url: NSURL, cmd: String) {
|
|
|
|
guard let path = url.path else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let escapedFileName = self.agent.escapedFileNames([path])[0]
|
|
|
|
|
|
|
|
switch self.mode {
|
|
|
|
case .Normal:
|
|
|
|
self.agent.vimInput("\(cmd) \(escapedFileName)<CR>")
|
|
|
|
default:
|
|
|
|
self.agent.vimInput("<Esc>\(cmd) \(escapedFileName)<CR>")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Resizing
|
|
|
|
extension NeoVimView {
|
2016-06-27 19:42:37 +03:00
|
|
|
|
2016-07-06 20:11:53 +03:00
|
|
|
override public func setFrameSize(newSize: NSSize) {
|
|
|
|
super.setFrameSize(newSize)
|
|
|
|
|
|
|
|
// initial resizing is done when grid has data
|
|
|
|
guard self.grid.hasData else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if self.inLiveResize {
|
2016-08-01 08:46:04 +03:00
|
|
|
// TODO: Turn off live resizing for now.
|
2016-07-06 20:11:53 +03:00
|
|
|
// self.resizeNeoVimUiTo(size: newSize)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// There can be cases where the frame is resized not by live resizing, eg when the window is resized by window
|
|
|
|
// management tools. Thus, we make sure that the resize call is made when this happens.
|
|
|
|
self.resizeNeoVimUiTo(size: newSize)
|
|
|
|
}
|
2016-07-04 23:06:39 +03:00
|
|
|
|
|
|
|
override public func viewDidEndLiveResize() {
|
|
|
|
super.viewDidEndLiveResize()
|
|
|
|
self.resizeNeoVimUiTo(size: self.bounds.size)
|
|
|
|
}
|
|
|
|
|
2016-07-16 01:07:50 +03:00
|
|
|
private func resizeNeoVimUiTo(size size: CGSize) {
|
2016-07-10 14:51:00 +03:00
|
|
|
// NSLog("\(#function): \(size)")
|
2016-07-04 23:06:39 +03:00
|
|
|
let discreteSize = Size(width: Int(floor(size.width / self.cellSize.width)),
|
|
|
|
height: Int(floor(size.height / self.cellSize.height)))
|
|
|
|
|
|
|
|
self.xOffset = floor((size.width - self.cellSize.width * CGFloat(discreteSize.width)) / 2)
|
|
|
|
self.yOffset = floor((size.height - self.cellSize.height * CGFloat(discreteSize.height)) / 2)
|
|
|
|
|
2016-07-09 23:52:59 +03:00
|
|
|
self.agent.resizeToWidth(Int32(discreteSize.width), height: Int32(discreteSize.height))
|
2016-07-04 23:06:39 +03:00
|
|
|
}
|
2016-08-12 15:54:37 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Drawing
|
|
|
|
extension NeoVimView {
|
2016-07-04 23:06:39 +03:00
|
|
|
|
2016-06-15 23:11:35 +03:00
|
|
|
override public func drawRect(dirtyUnionRect: NSRect) {
|
|
|
|
guard self.grid.hasData else {
|
|
|
|
return
|
|
|
|
}
|
2016-06-27 19:42:37 +03:00
|
|
|
|
2016-07-04 23:06:39 +03:00
|
|
|
if self.inLiveResize {
|
|
|
|
NSColor.lightGrayColor().set()
|
|
|
|
dirtyUnionRect.fill()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-07-03 20:12:25 +03:00
|
|
|
// NSLog("\(#function): \(dirtyUnionRect)")
|
2016-06-08 19:17:12 +03:00
|
|
|
let context = NSGraphicsContext.currentContext()!.CGContext
|
2016-07-29 23:01:09 +03:00
|
|
|
|
|
|
|
if self.isCurrentlyPinching {
|
|
|
|
let boundsSize = self.bounds.size
|
|
|
|
let targetSize = CGSize(width: boundsSize.width * self.pinchTargetScale,
|
|
|
|
height: boundsSize.height * self.pinchTargetScale)
|
|
|
|
self.pinchImage.drawInRect(CGRect(origin: self.bounds.origin, size: targetSize))
|
|
|
|
return
|
|
|
|
}
|
2016-06-08 19:17:12 +03:00
|
|
|
|
|
|
|
CGContextSetTextMatrix(context, CGAffineTransformIdentity);
|
|
|
|
CGContextSetTextDrawingMode(context, .Fill);
|
|
|
|
|
2016-06-16 19:36:26 +03:00
|
|
|
let dirtyRects = self.rectsBeingDrawn()
|
2016-07-31 00:04:20 +03:00
|
|
|
// NSLog("\(dirtyRects)")
|
2016-06-18 12:43:37 +03:00
|
|
|
|
2016-06-19 11:07:03 +03:00
|
|
|
self.rowRunIntersecting(rects: dirtyRects).forEach { rowFrag in
|
2016-06-29 19:32:33 +03:00
|
|
|
// For background drawing we don't filter out the put(0, 0)s: in some cases only the put(0, 0)-cells should be
|
2016-07-01 22:24:42 +03:00
|
|
|
// redrawn. => FIXME: probably we have to consider this also when drawing further down, ie when the range starts
|
|
|
|
// with '0'...
|
2016-07-05 21:13:46 +03:00
|
|
|
self.drawBackground(positions: rowFrag.range.map { self.pointInViewFor(row: rowFrag.row, column: $0) },
|
2016-06-27 20:04:27 +03:00
|
|
|
background: rowFrag.attrs.background)
|
|
|
|
|
2016-06-08 19:17:12 +03:00
|
|
|
let positions = rowFrag.range
|
|
|
|
// filter out the put(0, 0)s (after a wide character)
|
|
|
|
.filter { self.grid.cells[rowFrag.row][$0].string.characters.count > 0 }
|
2016-07-05 21:13:46 +03:00
|
|
|
.map { self.pointInViewFor(row: rowFrag.row, column: $0) }
|
2016-06-15 00:50:25 +03:00
|
|
|
|
2016-06-27 20:04:27 +03:00
|
|
|
if positions.isEmpty {
|
|
|
|
return
|
|
|
|
}
|
2016-06-18 12:43:37 +03:00
|
|
|
|
2016-06-16 19:36:26 +03:00
|
|
|
let string = self.grid.cells[rowFrag.row][rowFrag.range].reduce("") { $0 + $1.string }
|
2016-06-19 14:39:20 +03:00
|
|
|
let glyphPositions = positions.map { CGPoint(x: $0.x, y: $0.y + self.descent + self.leading) }
|
2016-06-18 12:43:37 +03:00
|
|
|
self.drawer.drawString(string,
|
|
|
|
positions: UnsafeMutablePointer(glyphPositions), positionsCount: positions.count,
|
|
|
|
highlightAttrs: rowFrag.attrs,
|
|
|
|
context: context)
|
2016-06-08 19:17:12 +03:00
|
|
|
}
|
2016-06-29 20:54:06 +03:00
|
|
|
|
2016-08-07 19:14:02 +03:00
|
|
|
self.drawCursor(self.grid.foreground)
|
2016-06-29 20:54:06 +03:00
|
|
|
}
|
|
|
|
|
2016-08-07 19:14:02 +03:00
|
|
|
private func drawCursor(foreground: UInt32) {
|
2016-06-29 20:54:06 +03:00
|
|
|
// FIXME: for now do some rudimentary cursor drawing
|
2016-08-07 17:15:33 +03:00
|
|
|
let cursorPosition = self.mode == .Cmdline ? self.grid.putPosition : self.grid.screenCursor
|
2016-07-11 20:37:06 +03:00
|
|
|
// NSLog("\(#function): \(cursorPosition)")
|
2016-06-29 20:54:06 +03:00
|
|
|
|
2016-07-05 21:13:46 +03:00
|
|
|
var cursorRect = self.cellRectFor(row: cursorPosition.row, column: cursorPosition.column)
|
2016-06-29 21:06:31 +03:00
|
|
|
if self.grid.isNextCellEmpty(cursorPosition) {
|
|
|
|
let nextPosition = self.grid.nextCellPosition(cursorPosition)
|
2016-07-05 21:13:46 +03:00
|
|
|
cursorRect = cursorRect.union(self.cellRectFor(row: nextPosition.row, column:nextPosition.column))
|
2016-06-29 20:54:06 +03:00
|
|
|
}
|
|
|
|
|
2016-08-07 19:14:02 +03:00
|
|
|
ColorUtils.colorIgnoringAlpha(foreground).set()
|
2016-06-29 20:54:06 +03:00
|
|
|
NSRectFillUsingOperation(cursorRect, .CompositeDifference)
|
2016-06-15 23:11:35 +03:00
|
|
|
}
|
2016-06-08 19:17:12 +03:00
|
|
|
|
2016-06-18 12:43:37 +03:00
|
|
|
private func drawBackground(positions positions: [CGPoint], background: UInt32) {
|
2016-08-07 19:14:02 +03:00
|
|
|
if background == self.grid.background {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-07-03 22:59:03 +03:00
|
|
|
ColorUtils.colorIgnoringAlpha(background).set()
|
2016-08-01 22:33:52 +03:00
|
|
|
// NSColor(calibratedRed: CGFloat(drand48()), green: CGFloat(drand48()), blue: CGFloat(drand48()), alpha: 1.0).set()
|
2016-06-16 19:36:26 +03:00
|
|
|
let backgroundRect = CGRect(
|
|
|
|
x: positions[0].x, y: positions[0].y,
|
2016-07-13 21:16:04 +03:00
|
|
|
width: CGFloat(positions.count) * self.cellSize.width, height: self.cellSize.height
|
2016-06-16 19:36:26 +03:00
|
|
|
)
|
|
|
|
backgroundRect.fill()
|
|
|
|
}
|
|
|
|
|
2016-06-19 11:07:03 +03:00
|
|
|
private func rowRunIntersecting(rects rects: [CGRect]) -> [RowRun] {
|
2016-06-15 23:11:35 +03:00
|
|
|
return rects
|
2016-07-06 20:02:30 +03:00
|
|
|
.map { rect -> (Range<Int>, Range<Int>) in
|
2016-06-19 14:39:20 +03:00
|
|
|
// Get all Regions that intersects with the given rects. There can be overlaps between the Regions, but for the
|
|
|
|
// time being we ignore them; probably not necessary to optimize them away.
|
2016-07-06 20:02:30 +03:00
|
|
|
let region = self.regionFor(rect: rect)
|
|
|
|
return (region.rowRange, region.columnRange)
|
2016-06-19 14:39:20 +03:00
|
|
|
}
|
2016-07-06 20:02:30 +03:00
|
|
|
.map { self.rowRunsFor(rowRange: $0, columnRange: $1) } // All RowRuns for all Regions grouped by their row range.
|
|
|
|
.flatMap { $0 } // Flattened RowRuns for all Regions.
|
|
|
|
}
|
|
|
|
|
|
|
|
private func rowRunsFor(rowRange rowRange: Range<Int>, columnRange: Range<Int>) -> [RowRun] {
|
|
|
|
return rowRange
|
|
|
|
.map { (row) -> [RowRun] in
|
|
|
|
let rowCells = self.grid.cells[row]
|
|
|
|
let startIdx = columnRange.startIndex
|
|
|
|
|
|
|
|
var result = [ RowRun(row: row, range: startIdx...startIdx, attrs: rowCells[startIdx].attrs) ]
|
|
|
|
columnRange.forEach { idx in
|
|
|
|
if rowCells[idx].attrs == result.last!.attrs {
|
|
|
|
let last = result.popLast()!
|
|
|
|
result.append(RowRun(row: row, range: last.range.startIndex...idx, attrs: last.attrs))
|
|
|
|
} else {
|
|
|
|
result.append(RowRun(row: row, range: idx...idx, attrs: rowCells[idx].attrs))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result // All RowRuns for a row in a Region.
|
|
|
|
} // All RowRuns for all rows in a Region grouped by row.
|
|
|
|
.flatMap { $0 } // Flattened RowRuns for a Region.
|
|
|
|
}
|
|
|
|
|
|
|
|
private func regionFor(rect rect: CGRect) -> Region {
|
2016-08-01 08:46:04 +03:00
|
|
|
let cellWidth = self.cellSize.width
|
|
|
|
let cellHeight = self.cellSize.height
|
|
|
|
|
2016-07-06 20:02:30 +03:00
|
|
|
let rowStart = max(
|
2016-08-01 08:46:04 +03:00
|
|
|
Int(floor((self.bounds.height - self.yOffset - (rect.origin.y + rect.size.height)) / cellHeight)), 0
|
2016-07-06 20:02:30 +03:00
|
|
|
)
|
|
|
|
let rowEnd = min(
|
2016-08-03 23:33:33 +03:00
|
|
|
Int(ceil((self.bounds.height - self.yOffset - rect.origin.y) / cellHeight)) - 1, self.grid.size.height - 1
|
2016-07-06 20:02:30 +03:00
|
|
|
)
|
|
|
|
let columnStart = max(
|
2016-08-01 08:46:04 +03:00
|
|
|
Int(floor((rect.origin.x - self.xOffset) / cellWidth)), 0
|
2016-07-06 20:02:30 +03:00
|
|
|
)
|
|
|
|
let columnEnd = min(
|
2016-08-01 08:46:04 +03:00
|
|
|
Int(ceil((rect.origin.x - self.xOffset + rect.size.width) / cellWidth)) - 1, self.grid.size.width - 1
|
2016-07-06 20:02:30 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
return Region(top: rowStart, bottom: rowEnd, left: columnStart, right: columnEnd)
|
2016-06-05 10:49:14 +03:00
|
|
|
}
|
2016-07-01 22:24:42 +03:00
|
|
|
|
2016-07-16 01:07:50 +03:00
|
|
|
private func pointInViewFor(position position: Position) -> CGPoint {
|
2016-07-05 21:13:46 +03:00
|
|
|
return self.pointInViewFor(row: position.row, column: position.column)
|
2016-07-01 22:24:42 +03:00
|
|
|
}
|
2016-06-05 10:49:14 +03:00
|
|
|
|
2016-07-16 01:07:50 +03:00
|
|
|
private func pointInViewFor(row row: Int, column: Int) -> CGPoint {
|
2016-06-08 19:17:12 +03:00
|
|
|
return CGPoint(
|
2016-07-31 00:04:20 +03:00
|
|
|
x: self.xOffset + CGFloat(column) * self.cellSize.width,
|
2016-08-01 08:46:04 +03:00
|
|
|
y: self.bounds.size.height - self.yOffset - CGFloat(row) * self.cellSize.height - self.cellSize.height
|
2016-06-08 19:17:12 +03:00
|
|
|
)
|
|
|
|
}
|
2016-06-24 22:08:34 +03:00
|
|
|
|
2016-07-16 01:07:50 +03:00
|
|
|
private func cellRectFor(row row: Int, column: Int) -> CGRect {
|
2016-07-05 21:13:46 +03:00
|
|
|
return CGRect(origin: self.pointInViewFor(row: row, column: column), size: self.cellSize)
|
2016-06-24 22:08:34 +03:00
|
|
|
}
|
|
|
|
|
2016-07-16 01:07:50 +03:00
|
|
|
private func regionRectFor(region region: Region) -> CGRect {
|
2016-07-04 20:04:49 +03:00
|
|
|
let top = CGFloat(region.top)
|
|
|
|
let bottom = CGFloat(region.bottom)
|
|
|
|
let left = CGFloat(region.left)
|
|
|
|
let right = CGFloat(region.right)
|
|
|
|
|
|
|
|
let width = right - left + 1
|
|
|
|
let height = bottom - top + 1
|
|
|
|
|
2016-07-31 00:04:20 +03:00
|
|
|
let cellWidth = self.cellSize.width
|
|
|
|
let cellHeight = self.cellSize.height
|
|
|
|
|
2016-07-04 20:04:49 +03:00
|
|
|
return CGRect(
|
2016-07-31 00:04:20 +03:00
|
|
|
x: self.xOffset + left * cellWidth,
|
|
|
|
y: self.bounds.size.height - self.yOffset - top * cellHeight - height * cellHeight,
|
|
|
|
width: width * cellWidth,
|
|
|
|
height: height * cellHeight
|
2016-07-04 20:04:49 +03:00
|
|
|
)
|
2016-06-27 19:42:37 +03:00
|
|
|
}
|
2016-06-27 20:04:27 +03:00
|
|
|
|
2016-07-16 01:07:50 +03:00
|
|
|
private func wrapNamedKeys(string: String) -> String {
|
2016-06-27 20:04:27 +03:00
|
|
|
return "<\(string)>"
|
|
|
|
}
|
|
|
|
|
2016-07-16 01:07:50 +03:00
|
|
|
private func vimPlainString(string: String) -> String {
|
2016-07-12 00:24:40 +03:00
|
|
|
return string.stringByReplacingOccurrencesOfString("<", withString: self.wrapNamedKeys("lt"))
|
2016-06-27 20:04:27 +03:00
|
|
|
}
|
2016-06-05 10:49:14 +03:00
|
|
|
}
|
2016-07-16 01:07:50 +03:00
|
|
|
|
2016-08-05 00:34:09 +03:00
|
|
|
// MARK: - NSUserInterfaceValidationsProtocol
|
|
|
|
extension NeoVimView {
|
|
|
|
|
|
|
|
public func validateUserInterfaceItem(item: NSValidatedUserInterfaceItem) -> Bool {
|
2016-08-09 09:22:10 +03:00
|
|
|
let canUndoOrRedo = self.mode == .Insert || self.mode == .Replace || self.mode == .Normal || self.mode == .Visual
|
|
|
|
let canCopyOrCut = self.mode == .Normal || self.mode == .Visual
|
|
|
|
let canPaste = self.pasteboard.stringForType(NSPasteboardTypeString) != nil
|
|
|
|
let canDelete = self.mode == .Visual || self.mode == .Normal
|
|
|
|
let canSelectAll = self.mode == .Insert || self.mode == .Replace || self.mode == .Normal || self.mode == .Visual
|
2016-08-05 00:34:09 +03:00
|
|
|
|
|
|
|
switch item.action() {
|
2016-08-09 09:22:10 +03:00
|
|
|
case NSSelectorFromString("undo:"), NSSelectorFromString("redo:"):
|
|
|
|
return canUndoOrRedo
|
2016-08-05 00:34:09 +03:00
|
|
|
case NSSelectorFromString("copy:"), NSSelectorFromString("cut:"):
|
|
|
|
return canCopyOrCut
|
2016-08-09 09:22:10 +03:00
|
|
|
case NSSelectorFromString("paste:"):
|
|
|
|
return canPaste
|
|
|
|
case NSSelectorFromString("delete:"):
|
|
|
|
return canDelete
|
|
|
|
case NSSelectorFromString("selectAll:"):
|
|
|
|
return canSelectAll
|
2016-08-05 00:34:09 +03:00
|
|
|
default:
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-04 21:01:03 +03:00
|
|
|
// MARK: - Edit Menu Items
|
|
|
|
extension NeoVimView {
|
|
|
|
|
2016-08-09 09:22:10 +03:00
|
|
|
@IBAction func undo(sender: AnyObject!) {
|
|
|
|
switch self.mode {
|
|
|
|
case .Insert, .Replace:
|
|
|
|
self.agent.vimInput("<Esc>ui")
|
|
|
|
case .Normal, .Visual:
|
|
|
|
self.agent.vimInput("u")
|
|
|
|
default:
|
2016-08-04 21:01:03 +03:00
|
|
|
return
|
|
|
|
}
|
2016-08-09 09:22:10 +03:00
|
|
|
}
|
2016-08-04 21:01:03 +03:00
|
|
|
|
2016-08-09 09:22:10 +03:00
|
|
|
@IBAction func redo(sender: AnyObject!) {
|
|
|
|
switch self.mode {
|
|
|
|
case .Insert, .Replace:
|
|
|
|
self.agent.vimInput("<Esc><C-r>i")
|
|
|
|
case .Normal, .Visual:
|
|
|
|
self.agent.vimInput("<C-r>")
|
|
|
|
default:
|
|
|
|
return
|
|
|
|
}
|
2016-08-04 21:01:03 +03:00
|
|
|
}
|
|
|
|
|
2016-08-09 09:22:10 +03:00
|
|
|
@IBAction func cut(sender: AnyObject!) {
|
|
|
|
switch self.mode {
|
2016-08-09 19:50:35 +03:00
|
|
|
case .Visual, .Normal:
|
2016-08-09 09:22:10 +03:00
|
|
|
self.agent.vimInput("\"+d")
|
|
|
|
default:
|
2016-08-04 21:01:03 +03:00
|
|
|
return
|
|
|
|
}
|
2016-08-09 09:22:10 +03:00
|
|
|
}
|
2016-08-04 21:01:03 +03:00
|
|
|
|
2016-08-09 09:22:10 +03:00
|
|
|
@IBAction func copy(sender: AnyObject!) {
|
|
|
|
switch self.mode {
|
2016-08-09 19:50:35 +03:00
|
|
|
case .Visual, .Normal:
|
2016-08-09 09:22:10 +03:00
|
|
|
self.agent.vimInput("\"+y")
|
|
|
|
default:
|
|
|
|
return
|
|
|
|
}
|
2016-08-04 21:01:03 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
@IBAction func paste(sender: AnyObject!) {
|
|
|
|
guard let content = self.pasteboard.stringForType(NSPasteboardTypeString) else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
switch self.mode {
|
2016-08-09 09:22:10 +03:00
|
|
|
case .Cmdline, .Insert, .Replace:
|
2016-08-04 21:01:03 +03:00
|
|
|
self.agent.vimInput(self.vimPlainString(content))
|
2016-08-09 09:22:10 +03:00
|
|
|
case .Normal, .Visual:
|
|
|
|
self.agent.vimInput("\"+p")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@IBAction func delete(sender: AnyObject!) {
|
|
|
|
switch self.mode {
|
|
|
|
case .Normal, .Visual:
|
|
|
|
self.agent.vimInput("x")
|
|
|
|
default:
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@IBAction public override func selectAll(sender: AnyObject?) {
|
|
|
|
switch self.mode {
|
|
|
|
case .Insert, .Visual:
|
|
|
|
self.agent.vimInput("<Esc>ggVG")
|
2016-08-04 21:01:03 +03:00
|
|
|
default:
|
2016-08-09 09:22:10 +03:00
|
|
|
self.agent.vimInput("ggVG")
|
2016-08-04 21:01:03 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-16 01:07:50 +03:00
|
|
|
// MARK: - Key Events
|
|
|
|
extension NeoVimView: NSTextInputClient {
|
|
|
|
|
|
|
|
override public func keyDown(event: NSEvent) {
|
|
|
|
self.keyDownDone = false
|
|
|
|
|
|
|
|
let context = NSTextInputContext.currentInputContext()!
|
|
|
|
let cocoaHandledEvent = context.handleEvent(event)
|
|
|
|
if self.keyDownDone && cocoaHandledEvent {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// NSLog("\(#function): \(event)")
|
|
|
|
|
|
|
|
let modifierFlags = event.modifierFlags
|
|
|
|
let capslock = modifierFlags.contains(.AlphaShiftKeyMask)
|
|
|
|
let shift = modifierFlags.contains(.ShiftKeyMask)
|
|
|
|
let chars = event.characters!
|
|
|
|
let charsIgnoringModifiers = shift || capslock ? event.charactersIgnoringModifiers!.lowercaseString
|
|
|
|
: event.charactersIgnoringModifiers!
|
|
|
|
|
|
|
|
if KeyUtils.isSpecial(key: charsIgnoringModifiers) {
|
|
|
|
if let vimModifiers = self.vimModifierFlags(modifierFlags) {
|
|
|
|
self.agent.vimInput(self.wrapNamedKeys(vimModifiers + KeyUtils.namedKeyFrom(key: charsIgnoringModifiers)))
|
|
|
|
} else {
|
|
|
|
self.agent.vimInput(self.wrapNamedKeys(KeyUtils.namedKeyFrom(key: charsIgnoringModifiers)))
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if let vimModifiers = self.vimModifierFlags(modifierFlags) {
|
|
|
|
self.agent.vimInput(self.wrapNamedKeys(vimModifiers + charsIgnoringModifiers))
|
|
|
|
} else {
|
|
|
|
self.agent.vimInput(self.vimPlainString(chars))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
self.keyDownDone = true
|
|
|
|
}
|
|
|
|
|
|
|
|
public func insertText(aString: AnyObject, replacementRange: NSRange) {
|
|
|
|
// NSLog("\(#function): \(replacementRange): '\(aString)'")
|
|
|
|
|
|
|
|
switch aString {
|
|
|
|
case let string as String:
|
|
|
|
self.agent.vimInput(self.vimPlainString(string))
|
|
|
|
case let attributedString as NSAttributedString:
|
|
|
|
self.agent.vimInput(self.vimPlainString(attributedString.string))
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// unmarkText()
|
|
|
|
self.lastMarkedText = self.markedText
|
|
|
|
self.markedText = nil
|
|
|
|
self.markedPosition = Position.null
|
|
|
|
self.keyDownDone = true
|
|
|
|
}
|
|
|
|
|
|
|
|
public override func doCommandBySelector(aSelector: Selector) {
|
|
|
|
// NSLog("\(#function): \(aSelector)");
|
|
|
|
|
|
|
|
// FIXME: handle when ㅎ -> delete
|
|
|
|
|
|
|
|
if self.respondsToSelector(aSelector) {
|
|
|
|
Swift.print("\(#function): calling \(aSelector)")
|
|
|
|
self.performSelector(aSelector, withObject: self)
|
|
|
|
self.keyDownDone = true
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// NSLog("\(#function): "\(aSelector) not implemented, forwarding input to vim")
|
|
|
|
self.keyDownDone = false
|
|
|
|
}
|
|
|
|
|
|
|
|
public func setMarkedText(aString: AnyObject, selectedRange: NSRange, replacementRange: NSRange) {
|
|
|
|
if self.markedText == nil {
|
|
|
|
self.markedPosition = self.grid.putPosition
|
|
|
|
}
|
|
|
|
|
|
|
|
// eg 하 -> hanja popup, cf comment for self.lastMarkedText
|
|
|
|
if replacementRange.length > 0 {
|
|
|
|
self.agent.deleteCharacters(replacementRange.length)
|
|
|
|
}
|
|
|
|
|
|
|
|
switch aString {
|
|
|
|
case let string as String:
|
|
|
|
self.markedText = string
|
|
|
|
case let attributedString as NSAttributedString:
|
|
|
|
self.markedText = attributedString.string
|
|
|
|
default:
|
|
|
|
self.markedText = String(aString) // should not occur
|
|
|
|
}
|
|
|
|
|
|
|
|
// NSLog("\(#function): \(self.markedText), \(selectedRange), \(replacementRange)")
|
|
|
|
|
|
|
|
self.agent.vimInputMarkedText(self.markedText!)
|
|
|
|
self.keyDownDone = true
|
|
|
|
}
|
|
|
|
|
|
|
|
public func unmarkText() {
|
|
|
|
// NSLog("\(#function): ")
|
|
|
|
self.markedText = nil
|
|
|
|
self.markedPosition = Position.null
|
|
|
|
self.keyDownDone = true
|
|
|
|
|
|
|
|
// TODO: necessary?
|
|
|
|
self.setNeedsDisplayInRect(self.cellRectFor(row: self.grid.putPosition.row, column: self.grid.putPosition.column))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Return the current selection (or the position of the cursor with empty-length range). For example when you enter
|
|
|
|
/// "Cmd-Ctrl-Return" you'll get the Emoji-popup at the rect by firstRectForCharacterRange(actualRange:) where the
|
|
|
|
/// first range is the result of this method.
|
|
|
|
public func selectedRange() -> NSRange {
|
|
|
|
// When the app starts and the Hangul input method is selected, this method gets called very early...
|
|
|
|
guard self.grid.hasData else {
|
|
|
|
// NSLog("\(#function): not found")
|
|
|
|
return NSRange(location: NSNotFound, length: 0)
|
|
|
|
}
|
|
|
|
|
|
|
|
let result = NSRange(location: self.grid.singleIndexFrom(self.grid.putPosition), length: 0)
|
|
|
|
// NSLog("\(#function): \(result)")
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
public func markedRange() -> NSRange {
|
|
|
|
// FIXME: do we have to handle positions at the column borders?
|
|
|
|
if let markedText = self.markedText {
|
|
|
|
let result = NSRange(location: self.grid.singleIndexFrom(self.markedPosition),
|
|
|
|
length: markedText.characters.count)
|
|
|
|
// NSLog("\(#function): \(result)")
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
NSLog("\(#function): returning empty range")
|
|
|
|
return NSRange(location: NSNotFound, length: 0)
|
|
|
|
}
|
|
|
|
|
|
|
|
public func hasMarkedText() -> Bool {
|
|
|
|
// NSLog("\(#function)")
|
|
|
|
return self.markedText != nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: take into account the "return nil"-case
|
|
|
|
// FIXME: just fix me, PLEASE...
|
|
|
|
public func attributedSubstringForProposedRange(aRange: NSRange, actualRange: NSRangePointer) -> NSAttributedString? {
|
|
|
|
// NSLog("\(#function): \(aRange), \(actualRange[0])")
|
|
|
|
if aRange.location == NSNotFound {
|
|
|
|
// NSLog("\(#function): range not found: returning nil")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
guard let lastMarkedText = self.lastMarkedText else {
|
|
|
|
// NSLog("\(#function): no last marked text: returning nil")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// we only support last marked text, thus fill dummy characters when Cocoa asks for more characters than marked...
|
|
|
|
let fillCount = aRange.length - lastMarkedText.characters.count
|
|
|
|
guard fillCount >= 0 else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
let fillChars = Array(0..<fillCount).reduce("") { (result, _) in return result + " " }
|
|
|
|
|
|
|
|
// NSLog("\(#function): \(aRange), \(actualRange[0]): \(fillChars + lastMarkedText)")
|
|
|
|
return NSAttributedString(string: fillChars + lastMarkedText)
|
|
|
|
}
|
|
|
|
|
|
|
|
public func validAttributesForMarkedText() -> [String] {
|
|
|
|
return []
|
|
|
|
}
|
|
|
|
|
|
|
|
public func firstRectForCharacterRange(aRange: NSRange, actualRange: NSRangePointer) -> NSRect {
|
|
|
|
let position = self.grid.positionFromSingleIndex(aRange.location)
|
|
|
|
|
|
|
|
// NSLog("\(#function): \(aRange),\(actualRange[0]) -> \(position.row):\(position.column)")
|
|
|
|
|
|
|
|
let resultInSelf = self.cellRectFor(row: position.row, column: position.column)
|
|
|
|
let result = self.window?.convertRectToScreen(self.convertRect(resultInSelf, toView: nil))
|
|
|
|
|
|
|
|
return result!
|
|
|
|
}
|
|
|
|
|
|
|
|
public func characterIndexForPoint(aPoint: NSPoint) -> Int {
|
|
|
|
// NSLog("\(#function): \(aPoint)")
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
private func vimModifierFlags(modifierFlags: NSEventModifierFlags) -> String? {
|
|
|
|
var result = ""
|
|
|
|
|
|
|
|
let control = modifierFlags.contains(.ControlKeyMask)
|
|
|
|
let option = modifierFlags.contains(.AlternateKeyMask)
|
|
|
|
let command = modifierFlags.contains(.CommandKeyMask)
|
|
|
|
|
|
|
|
if control {
|
|
|
|
result += "C-"
|
|
|
|
}
|
|
|
|
|
|
|
|
if option {
|
|
|
|
result += "M-"
|
|
|
|
}
|
|
|
|
|
|
|
|
if command {
|
|
|
|
result += "D-"
|
|
|
|
}
|
|
|
|
|
|
|
|
if result.characters.count > 0 {
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-29 19:25:32 +03:00
|
|
|
// MARK: - Gesture Events
|
|
|
|
extension NeoVimView {
|
|
|
|
|
|
|
|
override public func magnifyWithEvent(event: NSEvent) {
|
|
|
|
let factor = 1 + event.magnification
|
2016-07-29 23:01:09 +03:00
|
|
|
let pinchTargetScale = self.pinchTargetScale * factor
|
|
|
|
let resultingFontSize = round(pinchTargetScale * self._font.pointSize)
|
|
|
|
if resultingFontSize >= NeoVimView.minFontSize && resultingFontSize <= NeoVimView.maxFontSize {
|
|
|
|
self.pinchTargetScale = pinchTargetScale
|
2016-07-29 19:25:32 +03:00
|
|
|
}
|
|
|
|
|
2016-07-29 23:01:09 +03:00
|
|
|
switch event.phase {
|
|
|
|
case NSEventPhase.Began:
|
|
|
|
let pinchImageRep = self.bitmapImageRepForCachingDisplayInRect(self.bounds)!
|
|
|
|
self.cacheDisplayInRect(self.bounds, toBitmapImageRep: pinchImageRep)
|
|
|
|
self.pinchImage = NSImage()
|
|
|
|
self.pinchImage.addRepresentation(pinchImageRep)
|
|
|
|
|
|
|
|
self.isCurrentlyPinching = true
|
|
|
|
self.needsDisplay = true
|
|
|
|
|
|
|
|
case NSEventPhase.Ended, NSEventPhase.Cancelled:
|
|
|
|
self.isCurrentlyPinching = false
|
|
|
|
self.font = self.fontManager.convertFont(self._font, toSize: resultingFontSize)
|
|
|
|
self.pinchTargetScale = 1
|
|
|
|
|
|
|
|
default:
|
|
|
|
self.needsDisplay = true
|
2016-07-29 19:25:32 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-16 01:07:50 +03:00
|
|
|
// MARK: - Mouse Events
|
|
|
|
extension NeoVimView {
|
|
|
|
|
|
|
|
override public func mouseDown(event: NSEvent) {
|
|
|
|
self.mouse(event: event, vimName:"LeftMouse")
|
|
|
|
}
|
|
|
|
|
|
|
|
override public func mouseUp(event: NSEvent) {
|
|
|
|
self.mouse(event: event, vimName:"LeftRelease")
|
|
|
|
}
|
|
|
|
|
|
|
|
override public func mouseDragged(event: NSEvent) {
|
|
|
|
self.mouse(event: event, vimName:"LeftDrag")
|
|
|
|
}
|
|
|
|
|
|
|
|
override public func scrollWheel(event: NSEvent) {
|
|
|
|
let (deltaX, deltaY) = (event.scrollingDeltaX, event.scrollingDeltaY)
|
|
|
|
if deltaX == 0 && deltaY == 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let cellPosition = self.cellPositionFor(event: event)
|
|
|
|
let (vimInputX, vimInputY) = self.vimScrollInputFor(deltaX: deltaX, deltaY: deltaY,
|
|
|
|
modifierFlags: event.modifierFlags,
|
|
|
|
cellPosition: cellPosition)
|
|
|
|
|
|
|
|
let (absDeltaX, absDeltaY) = (abs(deltaX), abs(deltaY))
|
|
|
|
|
|
|
|
// The absolute delta values can get very very big when you use two finger scrolling on the trackpad:
|
|
|
|
// Cap them using heuristic values...
|
2016-07-16 19:38:54 +03:00
|
|
|
let numX = deltaX != 0 ? max(1, min(Int(absDeltaX / self.scrollLimiterX), self.maxScrollDeltaX)) : 0
|
|
|
|
let numY = deltaY != 0 ? max(1, min(Int(absDeltaY / self.scrollLimiterY), self.maxScrollDeltaY)) : 0
|
2016-07-16 01:07:50 +03:00
|
|
|
|
|
|
|
for i in 0..<max(numX, numY) {
|
|
|
|
if i < numX {
|
|
|
|
self.throttleScrollX(absDelta: absDeltaX, vimInput: vimInputX)
|
|
|
|
}
|
|
|
|
|
|
|
|
if i < numY {
|
|
|
|
self.throttleScrollY(absDelta: absDeltaY, vimInput: vimInputY)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func cellPositionFor(event event: NSEvent) -> Position {
|
|
|
|
let location = self.convertPoint(event.locationInWindow, fromView: nil)
|
|
|
|
let cellPosition = Position(
|
|
|
|
row: min(Int(floor(location.x / self.cellSize.width)), self.grid.size.width - 1),
|
|
|
|
column: min(Int(floor((self.bounds.height - location.y) / self.cellSize.height)), self.grid.size.height - 1)
|
|
|
|
)
|
|
|
|
|
|
|
|
return cellPosition
|
|
|
|
}
|
|
|
|
|
|
|
|
private func mouse(event event: NSEvent, vimName: String) {
|
|
|
|
let cellPosition = self.cellPositionFor(event: event)
|
|
|
|
guard self.shouldFireVimInputFor(event: event, newCellPosition: cellPosition) else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let vimMouseLocation = self.wrapNamedKeys("\(cellPosition.row),\(cellPosition.column)")
|
|
|
|
let vimClickCount = self.vimClickCountFrom(event: event)
|
|
|
|
|
|
|
|
let result: String
|
|
|
|
if let vimModifiers = self.vimModifierFlags(event.modifierFlags) {
|
|
|
|
result = self.wrapNamedKeys("\(vimModifiers)\(vimClickCount)\(vimName)") + vimMouseLocation
|
|
|
|
} else {
|
|
|
|
result = self.wrapNamedKeys("\(vimClickCount)\(vimName)") + vimMouseLocation
|
|
|
|
}
|
|
|
|
|
|
|
|
// NSLog("\(#function): \(result)")
|
|
|
|
self.agent.vimInput(result)
|
|
|
|
}
|
|
|
|
|
|
|
|
private func shouldFireVimInputFor(event event:NSEvent, newCellPosition: Position) -> Bool {
|
|
|
|
let type = event.type
|
|
|
|
guard type == .LeftMouseDragged || type == .RightMouseDragged || type == .OtherMouseDragged else {
|
|
|
|
self.lastClickedCellPosition = newCellPosition
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
if self.lastClickedCellPosition == newCellPosition {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
self.lastClickedCellPosition = newCellPosition
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
private func vimClickCountFrom(event event: NSEvent) -> String {
|
|
|
|
let clickCount = event.clickCount
|
|
|
|
|
|
|
|
guard 2 <= clickCount && clickCount <= 4 else {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
switch event.type {
|
|
|
|
case .LeftMouseDown, .LeftMouseUp, .RightMouseDown, .RightMouseUp:
|
|
|
|
return "\(clickCount)-"
|
|
|
|
default:
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func vimScrollEventNamesFor(deltaX deltaX: CGFloat, deltaY: CGFloat) -> (String, String) {
|
|
|
|
let typeY: String
|
|
|
|
if deltaY > 0 {
|
|
|
|
typeY = "ScrollWheelUp"
|
|
|
|
} else {
|
|
|
|
typeY = "ScrollWheelDown"
|
|
|
|
}
|
|
|
|
|
|
|
|
let typeX: String
|
|
|
|
if deltaX < 0 {
|
|
|
|
typeX = "ScrollWheelRight"
|
|
|
|
} else {
|
|
|
|
typeX = "ScrollWheelLeft"
|
|
|
|
}
|
|
|
|
|
|
|
|
return (typeX, typeY)
|
|
|
|
}
|
|
|
|
|
|
|
|
private func vimScrollInputFor(deltaX deltaX: CGFloat, deltaY: CGFloat,
|
|
|
|
modifierFlags: NSEventModifierFlags,
|
|
|
|
cellPosition: Position) -> (String, String)
|
|
|
|
{
|
|
|
|
let vimMouseLocation = self.wrapNamedKeys("\(cellPosition.row),\(cellPosition.column)")
|
|
|
|
|
|
|
|
let (typeX, typeY) = self.vimScrollEventNamesFor(deltaX: deltaX, deltaY: deltaY)
|
|
|
|
let resultX: String
|
|
|
|
let resultY: String
|
|
|
|
if let vimModifiers = self.vimModifierFlags(modifierFlags) {
|
|
|
|
resultX = self.wrapNamedKeys("\(vimModifiers)\(typeX)") + vimMouseLocation
|
|
|
|
resultY = self.wrapNamedKeys("\(vimModifiers)\(typeY)") + vimMouseLocation
|
|
|
|
} else {
|
|
|
|
resultX = self.wrapNamedKeys("\(typeX)") + vimMouseLocation
|
|
|
|
resultY = self.wrapNamedKeys("\(typeY)") + vimMouseLocation
|
|
|
|
}
|
|
|
|
|
|
|
|
return (resultX, resultY)
|
|
|
|
}
|
|
|
|
|
|
|
|
private func throttleScrollX(absDelta absDeltaX: CGFloat, vimInput: String) {
|
|
|
|
if absDeltaX == 0 {
|
|
|
|
self.scrollGuardCounterX = self.scrollGuardYield - 1
|
|
|
|
} else if absDeltaX <= 2 {
|
|
|
|
// Poor man's throttle for scroll value = 1 or 2
|
|
|
|
if self.scrollGuardCounterX % self.scrollGuardYield == 0 {
|
|
|
|
self.agent.vimInput(vimInput)
|
|
|
|
self.scrollGuardCounterX = 1
|
|
|
|
} else {
|
|
|
|
self.scrollGuardCounterX += 1
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
self.agent.vimInput(vimInput)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func throttleScrollY(absDelta absDeltaY: CGFloat, vimInput: String) {
|
|
|
|
if absDeltaY == 0 {
|
|
|
|
self.scrollGuardCounterY = self.scrollGuardYield - 1
|
|
|
|
} else if absDeltaY <= 2 {
|
|
|
|
// Poor man's throttle for scroll value = 1 or 2
|
|
|
|
if self.scrollGuardCounterY % self.scrollGuardYield == 0 {
|
|
|
|
self.agent.vimInput(vimInput)
|
|
|
|
self.scrollGuardCounterY = 1
|
|
|
|
} else {
|
|
|
|
self.scrollGuardCounterY += 1
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
self.agent.vimInput(vimInput)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - NeoVimUiBridgeProtocol
|
|
|
|
extension NeoVimView: NeoVimUiBridgeProtocol {
|
|
|
|
|
|
|
|
public func neoVimUiIsReady() {
|
|
|
|
DispatchUtils.gui {
|
2016-08-01 08:46:04 +03:00
|
|
|
self.resizeNeoVimUiTo(size: self.bounds.size)
|
2016-07-27 00:40:20 +03:00
|
|
|
self.delegate?.neoVimReady()
|
2016-07-16 01:07:50 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func resizeToWidth(width: Int32, height: Int32) {
|
|
|
|
DispatchUtils.gui {
|
2016-08-13 00:44:16 +03:00
|
|
|
// NSLog("\(#function): \(width):\(height)")
|
2016-07-16 01:07:50 +03:00
|
|
|
self.grid.resize(Size(width: Int(width), height: Int(height)))
|
|
|
|
self.needsDisplay = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func clear() {
|
|
|
|
DispatchUtils.gui {
|
|
|
|
self.grid.clear()
|
|
|
|
self.needsDisplay = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func eolClear() {
|
|
|
|
DispatchUtils.gui {
|
|
|
|
self.grid.eolClear()
|
|
|
|
|
|
|
|
let origin = self.pointInViewFor(position: self.grid.putPosition)
|
|
|
|
let size = CGSize(
|
|
|
|
width: CGFloat(self.grid.region.right - self.grid.putPosition.column + 1) * self.cellSize.width,
|
|
|
|
height: self.cellSize.height
|
|
|
|
)
|
|
|
|
let rect = CGRect(origin: origin, size: size)
|
|
|
|
self.setNeedsDisplayInRect(rect)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func gotoPosition(position: Position, screenCursor: Position) {
|
|
|
|
DispatchUtils.gui {
|
|
|
|
// NSLog("\(#function): \(position), \(screenCursor)")
|
|
|
|
|
2016-08-01 22:33:52 +03:00
|
|
|
// Because neovim fills blank space with "Space" and when we enter "Space" we don't get the puts, thus we have to
|
|
|
|
// redraw the put position.
|
|
|
|
if self.usesLigatures {
|
|
|
|
self.setNeedsDisplay(region: self.grid.regionOfWord(at: self.grid.putPosition))
|
|
|
|
self.setNeedsDisplay(region: self.grid.regionOfWord(at: screenCursor))
|
|
|
|
} else {
|
|
|
|
self.setNeedsDisplay(cellPosition: self.grid.putPosition)
|
|
|
|
self.setNeedsDisplay(screenCursor: position)
|
|
|
|
}
|
2016-07-16 01:07:50 +03:00
|
|
|
|
2016-08-01 22:57:06 +03:00
|
|
|
self.setNeedsDisplay(cellPosition: self.grid.nextCellPosition(self.grid.putPosition))
|
|
|
|
|
2016-08-07 17:15:33 +03:00
|
|
|
// Redraw where the cursor has been till now, ie remove the current cursor.
|
|
|
|
self.setNeedsDisplay(cellPosition: self.grid.screenCursor)
|
|
|
|
|
2016-07-16 01:07:50 +03:00
|
|
|
self.grid.goto(position)
|
|
|
|
self.grid.moveCursor(screenCursor)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func updateMenu() {
|
|
|
|
}
|
|
|
|
|
|
|
|
public func busyStart() {
|
|
|
|
}
|
|
|
|
|
|
|
|
public func busyStop() {
|
|
|
|
}
|
|
|
|
|
|
|
|
public func mouseOn() {
|
|
|
|
}
|
|
|
|
|
|
|
|
public func mouseOff() {
|
|
|
|
}
|
|
|
|
|
2016-08-03 22:30:41 +03:00
|
|
|
public func modeChange(mode: Mode) {
|
2016-08-11 22:19:03 +03:00
|
|
|
// NSLog("mode changed to: %02x", mode.rawValue)
|
2016-08-03 22:30:41 +03:00
|
|
|
self.mode = mode
|
2016-07-16 01:07:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
public func setScrollRegionToTop(top: Int32, bottom: Int32, left: Int32, right: Int32) {
|
|
|
|
DispatchUtils.gui {
|
|
|
|
let region = Region(top: Int(top), bottom: Int(bottom), left: Int(left), right: Int(right))
|
|
|
|
self.grid.setScrollRegion(region)
|
|
|
|
self.setNeedsDisplay(region: region)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func scroll(count: Int32) {
|
|
|
|
DispatchUtils.gui {
|
|
|
|
self.grid.scroll(Int(count))
|
|
|
|
self.setNeedsDisplay(region: self.grid.region)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func highlightSet(attrs: CellAttributes) {
|
|
|
|
DispatchUtils.gui {
|
|
|
|
self.grid.attrs = attrs
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func put(string: String) {
|
|
|
|
DispatchUtils.gui {
|
|
|
|
let curPos = self.grid.putPosition
|
|
|
|
// NSLog("\(#function): \(curPos) -> \(string)")
|
|
|
|
self.grid.put(string)
|
2016-07-31 00:04:20 +03:00
|
|
|
|
|
|
|
if self.usesLigatures {
|
|
|
|
if string == " " {
|
|
|
|
self.setNeedsDisplay(cellPosition: curPos)
|
|
|
|
} else {
|
2016-07-31 21:53:51 +03:00
|
|
|
self.setNeedsDisplay(region: self.grid.regionOfWord(at: curPos))
|
2016-07-31 00:04:20 +03:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
self.setNeedsDisplay(cellPosition: curPos)
|
|
|
|
}
|
|
|
|
|
2016-07-16 01:07:50 +03:00
|
|
|
// When the cursor is in the command line, then we need this...
|
|
|
|
self.setNeedsDisplay(cellPosition: self.grid.nextCellPosition(curPos))
|
|
|
|
self.setNeedsDisplay(screenCursor: self.grid.screenCursor)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func putMarkedText(markedText: String) {
|
|
|
|
DispatchUtils.gui {
|
|
|
|
NSLog("\(#function): '\(markedText)'")
|
|
|
|
let curPos = self.grid.putPosition
|
|
|
|
self.grid.putMarkedText(markedText)
|
|
|
|
|
|
|
|
self.setNeedsDisplay(position: curPos)
|
|
|
|
// When the cursor is in the command line, then we need this...
|
|
|
|
self.setNeedsDisplay(cellPosition: self.grid.nextCellPosition(curPos))
|
|
|
|
if markedText.characters.count == 0 {
|
|
|
|
self.setNeedsDisplay(position: self.grid.previousCellPosition(curPos))
|
|
|
|
}
|
|
|
|
self.setNeedsDisplay(screenCursor: self.grid.screenCursor)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func unmarkRow(row: Int32, column: Int32) {
|
|
|
|
DispatchUtils.gui {
|
|
|
|
let position = Position(row: Int(row), column: Int(column))
|
|
|
|
|
|
|
|
NSLog("\(#function): \(position)")
|
|
|
|
|
|
|
|
self.grid.unmarkCell(position)
|
|
|
|
self.setNeedsDisplay(position: position)
|
|
|
|
|
|
|
|
self.setNeedsDisplay(screenCursor: self.grid.screenCursor)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func bell() {
|
|
|
|
DispatchUtils.gui {
|
|
|
|
NSBeep()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func visualBell() {
|
|
|
|
}
|
|
|
|
|
|
|
|
public func flush() {
|
|
|
|
// NSLog("\(#function)")
|
|
|
|
}
|
|
|
|
|
|
|
|
public func updateForeground(fg: Int32) {
|
|
|
|
DispatchUtils.gui {
|
|
|
|
self.grid.foreground = UInt32(bitPattern: fg)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func updateBackground(bg: Int32) {
|
|
|
|
DispatchUtils.gui {
|
|
|
|
self.grid.background = UInt32(bitPattern: bg)
|
|
|
|
self.layer?.backgroundColor = ColorUtils.colorIgnoringAlpha(self.grid.background).CGColor
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func updateSpecial(sp: Int32) {
|
|
|
|
DispatchUtils.gui {
|
|
|
|
self.grid.special = UInt32(bitPattern: sp)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func suspend() {
|
|
|
|
}
|
|
|
|
|
|
|
|
public func setTitle(title: String) {
|
|
|
|
DispatchUtils.gui {
|
2016-07-28 23:13:30 +03:00
|
|
|
self.delegate?.setTitle(title)
|
2016-07-16 01:07:50 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func setIcon(icon: String) {
|
|
|
|
}
|
|
|
|
|
|
|
|
public func stop() {
|
2016-07-17 15:41:53 +03:00
|
|
|
DispatchUtils.gui {
|
|
|
|
self.delegate?.neoVimStopped()
|
2016-08-12 19:00:05 +03:00
|
|
|
self.agent.quit()
|
2016-07-17 15:41:53 +03:00
|
|
|
}
|
2016-07-16 01:07:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
private func setNeedsDisplay(region region: Region) {
|
|
|
|
self.setNeedsDisplayInRect(self.regionRectFor(region: region))
|
|
|
|
}
|
|
|
|
|
|
|
|
private func setNeedsDisplay(cellPosition position: Position) {
|
|
|
|
self.setNeedsDisplay(position: position)
|
|
|
|
|
|
|
|
if self.grid.isCellEmpty(position) {
|
|
|
|
self.setNeedsDisplay(position: self.grid.previousCellPosition(position))
|
|
|
|
}
|
|
|
|
|
|
|
|
if self.grid.isNextCellEmpty(position) {
|
|
|
|
self.setNeedsDisplay(position: self.grid.nextCellPosition(position))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func setNeedsDisplay(position position: Position) {
|
|
|
|
self.setNeedsDisplay(row: position.row, column: position.column)
|
|
|
|
}
|
|
|
|
|
|
|
|
private func setNeedsDisplay(row row: Int, column: Int) {
|
|
|
|
// Swift.print("\(#function): \(row):\(column)")
|
|
|
|
self.setNeedsDisplayInRect(self.cellRectFor(row: row, column: column))
|
|
|
|
}
|
|
|
|
|
|
|
|
private func setNeedsDisplay(screenCursor position: Position) {
|
|
|
|
self.setNeedsDisplay(position: position)
|
|
|
|
if self.grid.isNextCellEmpty(position) {
|
|
|
|
self.setNeedsDisplay(position: self.grid.nextCellPosition(position))
|
|
|
|
}
|
|
|
|
}
|
2016-07-30 10:49:39 +03:00
|
|
|
}
|