2017-02-26 14:00:19 +03:00
|
|
|
/**
|
|
|
|
* 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) {
|
2017-04-26 20:40:42 +03:00
|
|
|
self.emit = emitter.typedEmit()
|
2017-02-26 14:00:19 +03:00
|
|
|
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()
|
|
|
|
})
|
2017-04-22 17:30:40 +03:00
|
|
|
.disposed(by: self.disposeBag)
|
2017-02-26 14:00:19 +03:00
|
|
|
}
|
|
|
|
|
2017-04-22 16:56:13 +03:00
|
|
|
fileprivate let emit: (UuidAction<Action>) -> Void
|
2017-02-26 14:00:19 +03:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2017-04-22 16:56:13 +03:00
|
|
|
self.emit(UuidAction(uuid: self.uuid, action: .open(self.buffers[clickedRow])))
|
2017-02-26 14:00:19 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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
|
|
|
|
}
|
|
|
|
}
|