KMBatchProcessingTableView.swift 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532
  1. //
  2. // KMBatchProcessingTableView.swift
  3. // PDF Master
  4. //
  5. // Created by lizhe on 2022/11/18.
  6. //
  7. import Cocoa
  8. class KMBatchTableView: NSTableView {
  9. var mouseDownAction: (() -> Void)?
  10. // override func mouseDown(with event: NSEvent) {
  11. // super.mouseDown(with: event)
  12. // mouseDownAction?()
  13. // }
  14. }
  15. class KMBatchProcessingTableView: NSView {
  16. @IBOutlet var contentView: NSView!
  17. @IBOutlet weak var tableView: KMBatchTableView!
  18. var orderClickRow = -1
  19. let tableRowPasteboardType: NSPasteboard.PasteboardType = NSPasteboard.PasteboardType(rawValue: "private.table-row")
  20. lazy var presenter: KMBatchProcessingTableViewPresenter! = KMBatchProcessingTableViewPresenter()
  21. weak var delegate: KMBatchProcessingTableViewDelegate?
  22. var selectModels: [KMBatchProcessingTableViewModel] = []
  23. var inputType: KMBatchCollectionViewType = .convertPDF {
  24. didSet {
  25. self.reloadData()
  26. }
  27. }
  28. var isDrag = false
  29. /**
  30. @abstract 外部传入数据
  31. @param inputData 文件路劲
  32. */
  33. var inputData: [URL]! {
  34. didSet {
  35. self.presenter.addData(data: inputData)
  36. }
  37. }
  38. //内部使用数据
  39. var data: [KMBatchProcessingTableViewModel] = []
  40. fileprivate var options: KMBatchProcessingTableViewOptions?
  41. override func draw(_ dirtyRect: NSRect) {
  42. super.draw(dirtyRect)
  43. // Drawing code here.
  44. }
  45. // MARK: 初始化
  46. public required init?(coder decoder: NSCoder) {
  47. super.init(coder: decoder)
  48. initContentView()
  49. setup()
  50. }
  51. override init(frame frameRect: NSRect) {
  52. super.init(frame: frameRect)
  53. initContentView()
  54. setup()
  55. }
  56. private func initContentView() {
  57. //绑定xib
  58. let resource = NSNib(nibNamed: String(describing: self.classForCoder.self),
  59. bundle: Bundle(for: self.classForCoder.self))!
  60. resource.instantiate(withOwner: self, topLevelObjects: nil)
  61. addSubview(contentView)
  62. contentView.translatesAutoresizingMaskIntoConstraints = false
  63. NSLayoutConstraint.activate([
  64. contentView.topAnchor.constraint(equalTo: topAnchor),
  65. contentView.leftAnchor.constraint(equalTo: leftAnchor),
  66. contentView.rightAnchor.constraint(equalTo: rightAnchor),
  67. contentView.bottomAnchor.constraint(equalTo: bottomAnchor)])
  68. contentView.updateConstraintsForSubtreeIfNeeded()
  69. }
  70. func setup() {
  71. self.tableView.dataSource = self
  72. self.tableView.delegate = self
  73. // self.tableView.allowsMultipleSelection = true
  74. self.tableView.registerForDraggedTypes([NSPasteboard.PasteboardType.string, NSPasteboard.PasteboardType.fileURL, self.tableRowPasteboardType]) //支持拖拽
  75. self.tableView.setDraggingSourceOperationMask([.copy, .delete], forLocal: false)
  76. self.tableView.mouseDownAction = { [weak self] in
  77. guard let self = self else { return }
  78. self.cancelAllSelect()
  79. }
  80. self.options = .all
  81. self.presenter.initPresenter(view: self, data: [])
  82. }
  83. func reloadData() {
  84. for _ in self.tableView.tableColumns {
  85. self.tableView.removeTableColumn(self.tableView.tableColumns[0])
  86. }
  87. if inputType == .imageToPDF {
  88. self.options = [.number, .name, .dimension, .size, .state]
  89. } else {
  90. self.options = [.number, .name, .order, .size, .state]
  91. }
  92. if (options!.contains(KMBatchProcessingTableViewOptions.number)) {
  93. let column = NSTableColumn()
  94. column.headerCell = KMBatchProcessingColumnHeaderCell.init()
  95. column.title = NSLocalizedString(" ", comment: "")
  96. column.identifier = NSUserInterfaceItemIdentifier(String(KMBatchProcessingTableViewOptions.number.rawValue))
  97. // column.resizingMask = .userResizingMask
  98. column.width = 40
  99. self.tableView.addTableColumn(column)
  100. }
  101. if (options!.contains(KMBatchProcessingTableViewOptions.name)) {
  102. let column = NSTableColumn()
  103. column.headerCell = KMBatchProcessingColumnHeaderCell.init()
  104. column.title = NSLocalizedString("File Name", comment: "")
  105. column.identifier = NSUserInterfaceItemIdentifier(String(KMBatchProcessingTableViewOptions.name.rawValue))
  106. // column.resizingMask = .userResizingMask
  107. if self.inputType == .imageToPDF {
  108. column.width = 220
  109. } else {
  110. column.width = self.canShowOrder() ? 180 : 344
  111. }
  112. self.tableView.addTableColumn(column)
  113. }
  114. if (options!.contains(KMBatchProcessingTableViewOptions.order) && self.canShowOrder()) {
  115. let column = NSTableColumn()
  116. column.headerCell = KMBatchProcessingColumnHeaderCell.init()
  117. column.title = NSLocalizedString("Page Range", comment: "")
  118. column.identifier = NSUserInterfaceItemIdentifier(String(KMBatchProcessingTableViewOptions.order.rawValue))
  119. column.resizingMask = .userResizingMask
  120. column.width = 164
  121. self.tableView.addTableColumn(column)
  122. }
  123. if (options!.contains(KMBatchProcessingTableViewOptions.dimension)) {
  124. let column = NSTableColumn()
  125. column.headerCell = KMBatchProcessingColumnHeaderCell.init()
  126. column.title = NSLocalizedString("Dimension", comment: "")
  127. column.identifier = NSUserInterfaceItemIdentifier(String(KMBatchProcessingTableViewOptions.dimension.rawValue))
  128. column.resizingMask = .userResizingMask
  129. column.width = 164
  130. self.tableView.addTableColumn(column)
  131. }
  132. if (options!.contains(KMBatchProcessingTableViewOptions.size)) {
  133. let column = NSTableColumn()
  134. column.headerCell = KMBatchProcessingColumnHeaderCell.init()
  135. column.title = NSLocalizedString("Size", comment: "")
  136. column.identifier = NSUserInterfaceItemIdentifier(String(KMBatchProcessingTableViewOptions.size.rawValue))
  137. column.resizingMask = .userResizingMask
  138. column.width = 88
  139. self.tableView.addTableColumn(column)
  140. }
  141. if (options!.contains(KMBatchProcessingTableViewOptions.state)) {
  142. let column = NSTableColumn()
  143. column.headerCell = KMBatchProcessingColumnHeaderCell.init()
  144. column.title = NSLocalizedString("Status", comment: "")
  145. column.identifier = NSUserInterfaceItemIdentifier(String(KMBatchProcessingTableViewOptions.state.rawValue))
  146. column.headerCell.textColor = NSColor.red
  147. column.resizingMask = .userResizingMask
  148. column.width = 56
  149. self.tableView.addTableColumn(column)
  150. }
  151. if (options!.contains(KMBatchProcessingTableViewOptions.delete)) {
  152. let column = NSTableColumn()
  153. column.headerCell = KMBatchProcessingColumnHeaderCell.init()
  154. column.title = NSLocalizedString("", comment: "")
  155. column.identifier = NSUserInterfaceItemIdentifier(String(KMBatchProcessingTableViewOptions.delete.rawValue))
  156. column.resizingMask = .userResizingMask
  157. column.width = 30
  158. self.tableView.addTableColumn(column)
  159. }
  160. self.tableView.reloadData()
  161. }
  162. func canShowOrder() -> Bool {
  163. if (self.inputType == .imageToPDF ||
  164. self.inputType == .security ||
  165. self.inputType == .compress) {
  166. return false
  167. } else {
  168. return true
  169. }
  170. }
  171. }
  172. extension KMBatchProcessingTableView: NSTableViewDelegate {
  173. func numberOfRows(in tableView: NSTableView) -> Int {
  174. return self.data.count
  175. }
  176. func tableView(_ tableView: NSTableView, viewFor tableColumn: NSTableColumn?, row: Int) -> NSView? {
  177. var cell: KMBatchProcessingTableCell?
  178. if (tableColumn?.identifier.rawValue == String(KMBatchProcessingTableViewOptions.number.rawValue)) {
  179. cell = KMBatchProcessingNumTableCell.init(frame: CGRect(x: 0, y: 0, width: tableColumn!.width, height:tableView.rowHeight))
  180. } else if (tableColumn?.identifier.rawValue == String(KMBatchProcessingTableViewOptions.name.rawValue)) {
  181. cell = KMBatchProcessingNameTableCell.init(frame: CGRect(x: 0, y: 0, width: tableColumn!.width, height:tableView.rowHeight))
  182. } else if (tableColumn?.identifier.rawValue == String(KMBatchProcessingTableViewOptions.order.rawValue)) {
  183. cell = KMBatchProcessingOrderTableCell.init(frame: CGRect(x: 0, y: 0, width: tableColumn!.width, height:tableView.rowHeight))
  184. if cell != nil {
  185. (cell as? KMBatchProcessingOrderTableCell)!.cellAction = { [weak self] pageRange in
  186. self?.orderClickRow = row
  187. }
  188. }
  189. } else if (tableColumn?.identifier.rawValue == String(KMBatchProcessingTableViewOptions.dimension.rawValue)) {
  190. cell = KMBatchProcessingWidthTableCell.init(frame: CGRect(x: 0, y: 0, width: tableColumn!.width, height:tableView.rowHeight))
  191. } else if (tableColumn?.identifier.rawValue == String(KMBatchProcessingTableViewOptions.size.rawValue)) {
  192. cell = KMBatchProcessingSizeTableCell.init(frame: CGRect(x: 0, y: 0, width: tableColumn!.width, height:tableView.rowHeight))
  193. } else if (tableColumn?.identifier.rawValue == String(KMBatchProcessingTableViewOptions.state.rawValue)) {
  194. cell = KMBatchProcessingStateTableCell.init(frame: CGRect(x: 0, y: 0, width: tableColumn!.width, height:tableView.rowHeight))
  195. } else if (tableColumn?.identifier.rawValue == String(KMBatchProcessingTableViewOptions.delete.rawValue)) {
  196. cell = KMBatchProcessingDeleteTableCell.init(frame: CGRect(x: 0, y: 0, width: tableColumn!.width, height:tableView.rowHeight))
  197. cell?.action = { [weak self] (view) in
  198. guard let self = self else { return }
  199. self.presenter.deleteData(model: view.model)
  200. }
  201. }
  202. if(cell != nil) {
  203. if (self.data.count > row) {
  204. let model = self.data[row]
  205. model.type = self.inputType
  206. model.row = row + 1
  207. cell!.model = model
  208. }
  209. }
  210. return cell
  211. }
  212. func tableView(_ tableView: NSTableView, rowViewForRow row: Int) -> NSTableRowView? {
  213. //悬浮删除
  214. let tableRowView = KMBatchProcessingTableRowView()
  215. tableRowView.selectionHighlightStyle = .none
  216. tableRowView.model = self.data[row]
  217. tableRowView.hoverCallback = { [weak self] (mouseEntered, mouseBox) in
  218. guard let self = self else { return }
  219. for i in 0...self.data.count - 1 {
  220. let model = self.data[i]
  221. if model.hover == true {
  222. model.hover = false
  223. let columnIndex = self.tableView.column(withIdentifier: NSUserInterfaceItemIdentifier(KMBatchProcessingTableViewOptions.delete.rawValue.description))
  224. self.tableView.reloadData(forRowIndexes: IndexSet(integer: i), columnIndexes: IndexSet(integer: columnIndex))
  225. if self.tableView.rowView(atRow: i, makeIfNecessary: false) is KMBatchProcessingTableRowView {
  226. let rowView: KMBatchProcessingTableRowView = self.tableView.rowView(atRow: i, makeIfNecessary: false) as! KMBatchProcessingTableRowView
  227. rowView.reloadData()
  228. }
  229. }
  230. if i == row {
  231. if mouseEntered {
  232. model.hover = true
  233. } else {
  234. model.hover = false
  235. }
  236. let columnIndex = self.tableView.column(withIdentifier: NSUserInterfaceItemIdentifier(KMBatchProcessingTableViewOptions.delete.rawValue.description))
  237. self.tableView.reloadData(forRowIndexes: IndexSet(integer: i), columnIndexes: IndexSet(integer: columnIndex))
  238. if self.tableView.rowView(atRow: i, makeIfNecessary: false) is KMBatchProcessingTableRowView {
  239. let rowView: KMBatchProcessingTableRowView = self.tableView.rowView(atRow: i, makeIfNecessary: false) as! KMBatchProcessingTableRowView
  240. rowView.reloadData()
  241. }
  242. }
  243. }
  244. }
  245. tableRowView.rightMouseCallback = { [weak self] (view, event) in
  246. guard let self = self else { return }
  247. if self.selectModels.count <= 1 {
  248. self.didSelectItem(shouldSelectRow: row)
  249. }
  250. DispatchQueue.main.async {
  251. if self.tableView.rowView(atRow: row, makeIfNecessary: false) != nil {
  252. let rowView = self.tableView.rowView(atRow: row, makeIfNecessary: false)
  253. self.addRightMenu(view: rowView!, event: event)
  254. }
  255. }
  256. }
  257. return tableRowView
  258. }
  259. func tableView(_ tableView: NSTableView, shouldSelect tableColumn: NSTableColumn?) -> Bool {
  260. return true
  261. }
  262. func tableView(_ tableView: NSTableView, didClick tableColumn: NSTableColumn) {
  263. }
  264. func tableViewSelectionDidChange(_ notification: Notification) {
  265. let tableView: NSTableView = notification.object as? NSTableView ?? NSTableView()
  266. if tableView == self.tableView {
  267. // 获取所有选中的行索引
  268. let selectedIndexes = tableView.selectedRowIndexes
  269. if selectedIndexes.isEmpty {
  270. print("没有选中任何行")
  271. self.didSelectItems(indexs: IndexSet())
  272. } else {
  273. // 打印所有选中的行索引
  274. print("选中的行索引: \(selectedIndexes)")
  275. for index in selectedIndexes {
  276. print(index)
  277. }
  278. self.didSelectItems(indexs: selectedIndexes)
  279. }
  280. }
  281. }
  282. func tableView(_ tableView: NSTableView, shouldSelectRow row: Int) -> Bool {
  283. return true
  284. }
  285. func didSelectItems(indexs: IndexSet) {
  286. if !self.isDrag {
  287. for i in 0..<(self.data.count) {
  288. let item = self.data[i]
  289. if item.select {
  290. item.select = false
  291. self.tableView.reloadData(forRowIndexes: IndexSet(integer: i), columnIndexes: IndexSet())
  292. self.reloadRowData(index: i)
  293. }
  294. }
  295. self.selectModels.removeAll()
  296. for index in indexs {
  297. let model = self.data[index]
  298. model.select = true
  299. self.selectModels.append(model)
  300. self.tableView.reloadData(forRowIndexes: IndexSet(integer: index), columnIndexes: IndexSet())
  301. self.reloadRowData(index: index)
  302. }
  303. }
  304. }
  305. func reloadRowData(index: Int) {
  306. let columnIndex = self.tableView.column(withIdentifier: NSUserInterfaceItemIdentifier(KMBatchProcessingTableViewOptions.delete.rawValue.description))
  307. self.tableView.reloadData(forRowIndexes: IndexSet(integer: index), columnIndexes: IndexSet(integer: columnIndex))
  308. if self.tableView.rowView(atRow: index, makeIfNecessary: false) is KMBatchProcessingTableRowView {
  309. let rowView: KMBatchProcessingTableRowView = self.tableView.rowView(atRow: index, makeIfNecessary: false) as! KMBatchProcessingTableRowView
  310. rowView.reloadData()
  311. }
  312. }
  313. func didSelectItem(shouldSelectRow row: Int) {
  314. self.didSelectItems(indexs: IndexSet(integer: row))
  315. }
  316. func cancelAllSelect() {
  317. for i in 0...self.data.count - 1 {
  318. let model = self.data[i]
  319. model.select = false
  320. model.hover = false
  321. let columnIndex = self.tableView.column(withIdentifier: NSUserInterfaceItemIdentifier(KMBatchProcessingTableViewOptions.delete.rawValue.description))
  322. self.tableView.reloadData(forRowIndexes: IndexSet(integer: i), columnIndexes: IndexSet(integer: columnIndex))
  323. if self.tableView.rowView(atRow: i, makeIfNecessary: false) is KMBatchProcessingTableRowView {
  324. let rowView: KMBatchProcessingTableRowView = self.tableView.rowView(atRow: i, makeIfNecessary: false) as! KMBatchProcessingTableRowView
  325. rowView.reloadData()
  326. }
  327. }
  328. }
  329. }
  330. extension KMBatchProcessingTableView: NSTableViewDataSource {
  331. func tableView(_ tableView: NSTableView, heightOfRow row: Int) -> CGFloat {
  332. if self.inputType == .imageToPDF {
  333. return 80
  334. } else {
  335. return 40
  336. }
  337. }
  338. func tableView(_ tableView: NSTableView, writeRowsWith rowIndexes: IndexSet, to pboard: NSPasteboard) -> Bool {
  339. //自定义内部拖拽数据
  340. let data: Data = try! NSKeyedArchiver.archivedData(withRootObject: rowIndexes, requiringSecureCoding: true)
  341. pboard.declareTypes([self.tableRowPasteboardType], owner: self)
  342. pboard.setData(data, forType: self.tableRowPasteboardType)
  343. return true
  344. }
  345. func tableView(_ tableView: NSTableView, validateDrop info: NSDraggingInfo, proposedRow row: Int, proposedDropOperation dropOperation: NSTableView.DropOperation) -> NSDragOperation {
  346. guard dropOperation == .above else {
  347. return []
  348. }
  349. self.isDrag = true
  350. self.cancelAllSelect()
  351. if let source = info.draggingSource as? NSTableView, source === tableView {
  352. tableView.draggingDestinationFeedbackStyle = .gap
  353. } else {
  354. tableView.draggingDestinationFeedbackStyle = .regular
  355. }
  356. return .move
  357. }
  358. func tableView(_ tableView: NSTableView, acceptDrop info: NSDraggingInfo, row: Int, dropOperation: NSTableView.DropOperation) -> Bool {
  359. var result: Bool = false;
  360. let pboard: NSPasteboard = info.draggingPasteboard
  361. if ((pboard.availableType(from: [NSPasteboard.PasteboardType.fileURL])) != nil) {
  362. //获取url
  363. var array: [URL] = []
  364. for item: NSPasteboardItem in pboard.pasteboardItems! {
  365. let string: String = item.string(forType: NSPasteboard.PasteboardType.fileURL)!
  366. let url = NSURL(string: string)
  367. array.append(url! as URL)
  368. }
  369. //添加url
  370. self.presenter.insetData(data: array, index: row)
  371. result = true
  372. } else if ((pboard.availableType(from: [self.tableRowPasteboardType])) != nil) {
  373. //获取初始数据
  374. let rowData: Data = pboard.data(forType: self.tableRowPasteboardType)!
  375. let rowIndexes: NSIndexSet = try!NSKeyedUnarchiver.unarchivedObject(ofClass: NSIndexSet.self, from: rowData)!
  376. if info.draggingSource as? NSTableView == tableView {
  377. //数据处理
  378. self.data.move(with: IndexSet(rowIndexes), to: row)
  379. /**
  380. 数据刷新
  381. */
  382. tableView.beginUpdates()
  383. tableView.reloadData()
  384. tableView.endUpdates()
  385. }
  386. result = true
  387. }
  388. self.isDrag = false
  389. return result
  390. }
  391. func tableView(_ tableView: NSTableView, draggingSession session: NSDraggingSession, endedAt screenPoint: NSPoint, operation: NSDragOperation) {
  392. // Handle items dragged to Trash
  393. tableView.draggingDestinationFeedbackStyle = .none
  394. DispatchQueue.main.async {
  395. self.isDrag = false
  396. }
  397. }
  398. }
  399. extension KMBatchProcessingTableView: KMBatchProcessingTableViewPresenterDelegate {
  400. func showData(presenter: KMBatchProcessingTableViewPresenter, data: [KMBatchProcessingTableViewModel]) {
  401. self.data = data
  402. self.tableView.reloadData()
  403. self.delegate?.reloadData(data: data)
  404. }
  405. }
  406. extension Array {
  407. mutating func move(from start: Index, to end: Index) {
  408. guard (0..<count) ~= start, (0...count) ~= end else { return }
  409. if start == end { return }
  410. let targetIndex = start < end ? end - 1 : end
  411. insert(remove(at: start), at: targetIndex)
  412. }
  413. mutating func move(with indexes: IndexSet, to toIndex: Index) {
  414. let movingData = indexes.map{ self[$0] }
  415. let targetIndex = toIndex - indexes.filter{ $0 < toIndex }.count
  416. for (i, e) in indexes.enumerated() {
  417. remove(at: e - i)
  418. }
  419. insert(contentsOf: movingData, at: targetIndex)
  420. }
  421. }
  422. protocol KMBatchProcessingTableViewDelegate: NSObject {
  423. func reloadData(data: [KMBatchProcessingTableViewModel])
  424. }
  425. extension KMBatchProcessingTableView {
  426. func addRightMenu(view: NSView, event: NSEvent) {
  427. let menus = NSMenu()
  428. menus.addItem(withTitle: NSLocalizedString("Show in Finder", comment: ""), action: #selector(showInFinder), target: self)
  429. menus.addItem(withTitle: NSLocalizedString("Remove", comment: ""), action: #selector(removeItem), target: self)
  430. menus.addItem(withTitle: NSLocalizedString("Select All", comment: ""), action: #selector(selectAllFiles), target: self)
  431. let point = view.convert(event.locationInWindow, from: nil)
  432. menus.popUp(positioning: nil, at: point, in: view)
  433. }
  434. @objc func showInFinder() {
  435. if self.selectModels.count != 0 {
  436. for model in self.selectModels {
  437. if FileManager.default.fileExists(atPath: model.filePath) {
  438. NSWorkspace.shared.activateFileViewerSelecting([URL(fileURLWithPath: model.filePath)])
  439. }
  440. }
  441. }
  442. }
  443. @objc func removeItem() {
  444. self.presenter.deleteData(models: self.selectModels)
  445. }
  446. @objc func selectAllFiles() {
  447. for model in data {
  448. model.select = true
  449. self.selectModels.append(model)
  450. }
  451. self.tableView.reloadData()
  452. }
  453. }
  454. extension KMBatchProcessingTableView{
  455. func canSelect(row: Int) -> Bool {
  456. if row < data.count {
  457. if !self.isDrag && (self.orderClickRow != row && data[row].pageRange == .custom) {
  458. return true
  459. } else {
  460. return false
  461. }
  462. } else {
  463. return false
  464. }
  465. }
  466. }