mirror of
https://github.com/qvacua/vimr.git
synced 2024-12-28 08:13:17 +03:00
230 lines
5.7 KiB
Swift
230 lines
5.7 KiB
Swift
/**
|
|
* Tae Won Ha - http://taewon.de - @hataewon
|
|
* See LICENSE
|
|
*/
|
|
|
|
import Cocoa
|
|
import CocoaMarkdown
|
|
|
|
extension NSImage {
|
|
|
|
func tinting(with color: NSColor) -> NSImage {
|
|
|
|
let result: NSImage = self.copy() as! NSImage
|
|
|
|
result.lockFocus()
|
|
color.set()
|
|
NSRectFillUsingOperation(CGRect(origin: .zero, size: self.size), .sourceAtop)
|
|
result.unlockFocus()
|
|
|
|
return result
|
|
}
|
|
}
|
|
|
|
extension NSButton {
|
|
|
|
var boolState: Bool {
|
|
get {
|
|
return self.state == NSOnState ? true : false
|
|
}
|
|
|
|
set {
|
|
self.state = newValue ? NSOnState : NSOffState
|
|
}
|
|
}
|
|
}
|
|
|
|
extension NSMenuItem {
|
|
|
|
var boolState: Bool {
|
|
get {
|
|
return self.state == NSOnState ? true : false
|
|
}
|
|
|
|
set {
|
|
self.state = newValue ? NSOnState : NSOffState
|
|
}
|
|
}
|
|
}
|
|
|
|
extension NSAttributedString {
|
|
|
|
func draw(at point: CGPoint, angle: CGFloat) {
|
|
var translation = AffineTransform.identity
|
|
var rotation = AffineTransform.identity
|
|
|
|
translation.translate(x: point.x, y: point.y)
|
|
rotation.rotate(byRadians: angle)
|
|
|
|
(translation as NSAffineTransform).concat()
|
|
(rotation as NSAffineTransform).concat()
|
|
|
|
self.draw(at: CGPoint.zero)
|
|
|
|
rotation.invert()
|
|
translation.invert()
|
|
|
|
(rotation as NSAffineTransform).concat()
|
|
(translation as NSAffineTransform).concat()
|
|
}
|
|
|
|
// From https://developer.apple.com/library/mac/qa/qa1487/_index.html
|
|
static func link(withUrl url: URL, text: String, font: NSFont? = nil) -> NSAttributedString {
|
|
let attrString = NSMutableAttributedString(string: text)
|
|
let range = NSRange(location: 0, length: attrString.length)
|
|
|
|
attrString.beginEditing()
|
|
if font != nil {
|
|
attrString.addAttribute(NSFontAttributeName, value: font!, range: range)
|
|
}
|
|
attrString.addAttribute(NSLinkAttributeName, value: url.absoluteString, range: range)
|
|
attrString.addAttribute(NSForegroundColorAttributeName, value: NSColor.blue, range: range)
|
|
attrString.addAttribute(NSUnderlineStyleAttributeName,
|
|
value: NSNumber(value: NSUnderlineStyle.styleSingle.rawValue as Int), range: range)
|
|
attrString.endEditing()
|
|
|
|
return attrString
|
|
}
|
|
|
|
var wholeRange: NSRange {
|
|
return NSRange(location: 0, length: self.length)
|
|
}
|
|
|
|
static func infoLabel(markdown: String) -> NSAttributedString {
|
|
let size = NSFont.smallSystemFontSize()
|
|
let document = CMDocument(data: markdown.data(using: .utf8), options: .normalize)
|
|
|
|
let attrs = CMTextAttributes()
|
|
attrs?.textAttributes = [
|
|
NSFontAttributeName: NSFont.systemFont(ofSize: size),
|
|
NSForegroundColorAttributeName: NSColor.gray,
|
|
]
|
|
attrs?.inlineCodeAttributes = [
|
|
NSFontAttributeName: NSFont.userFixedPitchFont(ofSize: size)!,
|
|
NSForegroundColorAttributeName: NSColor.gray,
|
|
]
|
|
|
|
let renderer = CMAttributedStringRenderer(document: document, attributes: attrs)
|
|
renderer?.register(CMHTMLStrikethroughTransformer())
|
|
renderer?.register(CMHTMLSuperscriptTransformer())
|
|
renderer?.register(CMHTMLUnderlineTransformer())
|
|
|
|
guard let result = renderer?.render() else {
|
|
preconditionFailure("Wrong markdown: \(markdown)")
|
|
}
|
|
|
|
return result
|
|
}
|
|
}
|
|
|
|
extension NSView {
|
|
|
|
func removeAllSubviews() {
|
|
self.subviews.forEach { $0.removeFromSuperview() }
|
|
}
|
|
|
|
func removeAllConstraints() {
|
|
self.removeConstraints(self.constraints)
|
|
}
|
|
|
|
var isFirstResponder: Bool {
|
|
return self.window?.firstResponder == self
|
|
}
|
|
|
|
func beFirstResponder() {
|
|
self.window?.makeFirstResponder(self)
|
|
}
|
|
}
|
|
|
|
extension NSTableView {
|
|
|
|
static func standardTableView() -> NSTableView {
|
|
let tableView = NSTableView(frame: CGRect.zero)
|
|
|
|
let column = NSTableColumn(identifier: "name")
|
|
column.isEditable = false
|
|
|
|
tableView.addTableColumn(column)
|
|
tableView.rowSizeStyle = .default
|
|
tableView.sizeLastColumnToFit()
|
|
tableView.allowsEmptySelection = false
|
|
tableView.allowsMultipleSelection = false
|
|
tableView.headerView = nil
|
|
tableView.focusRingType = .none
|
|
|
|
return tableView
|
|
}
|
|
|
|
static func standardSourceListTableView() -> NSTableView {
|
|
let tableView = self.standardTableView()
|
|
tableView.selectionHighlightStyle = .sourceList
|
|
|
|
return tableView
|
|
}
|
|
}
|
|
|
|
extension NSOutlineView {
|
|
|
|
static func standardOutlineView() -> NSOutlineView {
|
|
let outlineView = NSOutlineView(frame: CGRect.zero)
|
|
NSOutlineView.configure(toStandard: outlineView)
|
|
return outlineView
|
|
}
|
|
|
|
static func configure(toStandard outlineView: NSOutlineView) {
|
|
let column = NSTableColumn(identifier: "name")
|
|
column.resizingMask = .autoresizingMask
|
|
column.isEditable = false
|
|
|
|
outlineView.addTableColumn(column)
|
|
outlineView.outlineTableColumn = column
|
|
outlineView.allowsEmptySelection = false
|
|
outlineView.allowsMultipleSelection = false
|
|
outlineView.headerView = nil
|
|
outlineView.focusRingType = .none
|
|
}
|
|
|
|
/**
|
|
The selected item. When the selection is empty, then returns `nil`. When multiple items are selected, then returns
|
|
the last selected item.
|
|
*/
|
|
var selectedItem: Any? {
|
|
if self.selectedRow < 0 {
|
|
return nil
|
|
}
|
|
|
|
return self.item(atRow: self.selectedRow)
|
|
}
|
|
|
|
var clickedItem: Any? {
|
|
if self.clickedRow < 0 {
|
|
return nil
|
|
}
|
|
|
|
return self.item(atRow: self.clickedRow)
|
|
}
|
|
|
|
func toggle(item: Any) {
|
|
if self.isItemExpanded(item) {
|
|
self.collapseItem(item)
|
|
} else {
|
|
self.expandItem(item)
|
|
}
|
|
}
|
|
}
|
|
|
|
extension NSScrollView {
|
|
|
|
static func standardScrollView() -> NSScrollView {
|
|
let scrollView = NSScrollView(frame: CGRect.zero)
|
|
|
|
scrollView.translatesAutoresizingMaskIntoConstraints = false
|
|
scrollView.hasVerticalScroller = true
|
|
scrollView.hasHorizontalScroller = true
|
|
scrollView.autohidesScrollers = true
|
|
scrollView.borderType = .bezelBorder
|
|
|
|
return scrollView
|
|
}
|
|
}
|