1
1
mirror of https://github.com/qvacua/vimr.git synced 2024-11-28 11:35:35 +03:00
vimr/VimR/OpenedFileList.swift
2017-04-26 19:40:42 +02:00

149 lines
4.0 KiB
Swift

/**
* Tae Won Ha - http://taewon.de - @hataewon
* See LICENSE
*/
import Cocoa
import RxSwift
import PureLayout
class OpenedFileList: NSView,
UiComponent,
NSTableViewDataSource,
NSTableViewDelegate {
typealias StateType = MainWindow.State
enum Action {
case open(NeoVimBuffer)
}
required init(source: Observable<StateType>, emitter: ActionEmitter, state: StateType) {
self.emit = emitter.typedEmit()
self.uuid = state.uuid
self.genericIcon = FileUtils.icon(forType: "public.data")
super.init(frame: .zero)
self.bufferList.dataSource = self
self.bufferList.delegate = self
self.bufferList.target = self
self.bufferList.doubleAction = #selector(OpenedFileList.doubleClickAction)
self.addViews()
source
.observeOn(MainScheduler.instance)
.subscribe(onNext: { state in
let buffers = state.buffers.removingDuplicatesPreservingFromBeginning()
if self.buffers == buffers {
return
}
self.buffers = buffers
self.bufferList.reloadData()
self.adjustFileViewWidth()
})
.disposed(by: self.disposeBag)
}
fileprivate let emit: (UuidAction<Action>) -> Void
fileprivate let disposeBag = DisposeBag()
fileprivate let uuid: String
fileprivate let bufferList = NSTableView.standardTableView()
fileprivate let genericIcon: NSImage
fileprivate var buffers = [NeoVimBuffer]()
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
fileprivate func addViews() {
let scrollView = NSScrollView.standardScrollView()
scrollView.borderType = .noBorder
scrollView.documentView = self.bufferList
self.addSubview(scrollView)
scrollView.autoPinEdgesToSuperviewEdges()
}
fileprivate func adjustFileViewWidth() {
let maxWidth = self.buffers.reduce(CGFloat(0)) { (curMaxWidth, buffer) in
return max(self.text(for: buffer).size().width, curMaxWidth)
}
let column = self.bufferList.tableColumns[0]
column.minWidth = maxWidth + ImageAndTextTableCell.widthWithoutText
column.maxWidth = column.minWidth
}
}
// MARK: - Actions
extension OpenedFileList {
func doubleClickAction(_ sender: Any?) {
let clickedRow = self.bufferList.clickedRow
guard clickedRow >= 0 && clickedRow < self.buffers.count else {
return
}
self.emit(UuidAction(uuid: self.uuid, action: .open(self.buffers[clickedRow])))
}
}
// MARK: - NSTableViewDataSource
extension OpenedFileList {
@objc(numberOfRowsInTableView:)
func numberOfRows(in tableView: NSTableView) -> Int {
return self.buffers.count
}
}
// MARK: - NSTableViewDelegate
extension OpenedFileList {
@objc(tableView: viewForTableColumn:row:)
func tableView(_ tableView: NSTableView, viewFor tableColumn: NSTableColumn?, row: Int) -> NSView? {
let cachedCell = (tableView.make(withIdentifier: "buffer-list-row", owner: self) as? ImageAndTextTableCell)?.reset()
let cell = cachedCell ?? ImageAndTextTableCell(withIdentifier: "buffer-list-row")
let buffer = self.buffers[row]
cell.attributedText = self.text(for: buffer)
cell.image = self.icon(for: buffer)
return cell
}
fileprivate func text(for buffer: NeoVimBuffer) -> NSAttributedString {
guard let name = buffer.name else {
return NSAttributedString(string: "No Name")
}
guard let url = buffer.url else {
return NSAttributedString(string: name)
}
let pathInfo = url.pathComponents.dropFirst().dropLast().reversed().joined(separator: " / ") + " /"
let rowText = NSMutableAttributedString(string: "\(name)\(pathInfo)")
rowText.addAttribute(NSForegroundColorAttributeName,
value: NSColor.lightGray,
range: NSRange(location: name.characters.count, length: pathInfo.characters.count + 3))
return rowText
}
fileprivate func icon(for buffer: NeoVimBuffer) -> NSImage? {
if let url = buffer.url {
return FileUtils.icon(forUrl: url)
}
return self.genericIcon
}
}