KMBOTAOutlineView.swift 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344
  1. //
  2. // KMBOTAOutlineView.swift
  3. // PDF Reader Pro
  4. //
  5. // Created by lizhe on 2023/4/2.
  6. //
  7. import Cocoa
  8. protocol KMBOTAOutlineViewDelegate {
  9. func BOTAOutlineView(_ outlineView: KMBOTAOutlineView, didReloadData: KMBOTAOutlineItem)
  10. func BOTAOutlineView(_ outlineView: KMBOTAOutlineView, didSelectItem: [KMBOTAOutlineItem])
  11. func BOTAOutlineView(_ outlineView: KMBOTAOutlineView, rightDidMoseDown: KMBOTAOutlineItem, event: NSEvent)
  12. func BOTAOutlineView(_ outlineView: KMBOTAOutlineView, writeItems items: [Any], to pasteboard: NSPasteboard) -> Bool
  13. func BOTAOutlineView(_ outlineView: KMBOTAOutlineView, validateDrop info: NSDraggingInfo, proposedItem item: Any?, proposedChildIndex index: Int) -> NSDragOperation
  14. func BOTAOutlineView(_ outlineView: KMBOTAOutlineView, acceptDrop info: NSDraggingInfo, item: Any?, childIndex index: Int) -> Bool
  15. }
  16. class KMBOTAOutlineView: BaseXibView {
  17. @IBOutlet weak var outlineView: KMOutlineView!
  18. @IBOutlet weak var scrollView: NSScrollView!
  19. var delegate: KMBOTAOutlineViewDelegate?
  20. var inputData: CPDFOutline? {
  21. didSet {
  22. self.reloadData(expandItemType: .none)
  23. }
  24. }
  25. var data: KMBOTAOutlineItem?
  26. var selectItems: [KMBOTAOutlineItem]?
  27. var dragPDFOutline: KMBOTAOutlineItem!
  28. override func draw(_ dirtyRect: NSRect) {
  29. super.draw(dirtyRect)
  30. // Drawing code here.
  31. }
  32. override func awakeFromNib() {
  33. super.awakeFromNib()
  34. self.setup()
  35. }
  36. func setup() {
  37. self.scrollView.backgroundColor(NSColor.km_init(hex: "#F2F9FF"))
  38. self.outlineView.registerForDraggedTypes([NSPasteboard.PasteboardType(rawValue: "kKMPDFViewOutlineDragDataType")])
  39. self.outlineView.delegate = self
  40. self.outlineView.dataSource = self
  41. self.outlineView.selectionHighlightStyle = NSTableView.SelectionHighlightStyle.none;
  42. self.outlineView.allowsMultipleSelection = true
  43. // self.outlineView.indentationPerLevel = 0
  44. }
  45. func reloadData(expandItemType: KMOutlineViewExpandItemType = .none) {
  46. if self.inputData != nil {
  47. //获取数据
  48. var tempData: KMBOTAOutlineItem = KMBOTAOutlineItem()
  49. if self.inputData!.numberOfChildren > 0 {
  50. let outline: CPDFOutline = self.inputData!
  51. tempData = self.addOutlineItem(outlineItem:tempData, outline: outline, expandItemType: expandItemType)
  52. } else {
  53. tempData.outline = CPDFOutline()
  54. if expandItemType == .collapse {
  55. tempData.isItemExpanded = false
  56. } else if (expandItemType == .expand) {
  57. tempData.isItemExpanded = true
  58. }
  59. }
  60. tempData.parent = nil
  61. self.data = tempData
  62. self.outlineView.reloadData()
  63. }
  64. self.delegate?.BOTAOutlineView(self, didReloadData: self.data ?? KMBOTAOutlineItem())
  65. }
  66. func addOutlineItem(outlineItem: KMBOTAOutlineItem?, outline: CPDFOutline, expandItemType: KMOutlineViewExpandItemType = .none) -> KMBOTAOutlineItem {
  67. //添加根节点
  68. let item: KMBOTAOutlineItem = KMBOTAOutlineItem()
  69. item.outline = outline
  70. item.parent = outlineItem
  71. var items: [KMBOTAOutlineItem] = []
  72. if outline.numberOfChildren > 0 {
  73. for index in 0...outline.numberOfChildren - 1 {
  74. let children: CPDFOutline = outline.child(at: index)
  75. let childrenItem = self.addOutlineItem(outlineItem: item, outline: children, expandItemType: expandItemType)
  76. if expandItemType == .collapse {
  77. childrenItem.isItemExpanded = false
  78. } else if (expandItemType == .expand) {
  79. childrenItem.isItemExpanded = true
  80. }
  81. items.append(childrenItem)
  82. }
  83. }
  84. item.children = items
  85. return item
  86. }
  87. func updateUI() {
  88. }
  89. func updateLanguage() {
  90. }
  91. }
  92. //MARK: NSOutlineViewDataSource,NSOutlineViewDelegate
  93. extension KMBOTAOutlineView : NSOutlineViewDataSource,NSOutlineViewDelegate {
  94. func outlineView(_ outlineView: NSOutlineView, numberOfChildrenOfItem item: Any?) -> Int {
  95. let outline = item as? KMBOTAOutlineItem
  96. if outline != nil {
  97. return outline?.children.count ?? 0
  98. } else {
  99. return Int(self.data?.outline.numberOfChildren ?? 0)
  100. }
  101. }
  102. func outlineView(_ outlineView: NSOutlineView, child index: Int, ofItem item: Any?) -> Any {
  103. let outline = item as? KMBOTAOutlineItem
  104. if outline != nil {
  105. return outline?.children[index] as Any
  106. } else {
  107. return self.data?.children[index] as Any
  108. }
  109. }
  110. func outlineView(_ outlineView: NSOutlineView, isItemExpandable item: Any) -> Bool {
  111. let outline = item as? KMBOTAOutlineItem
  112. if outline != nil && outline?.children.count != 0 {
  113. return true
  114. }
  115. return false
  116. }
  117. func outlineView(_ outlineView: NSOutlineView, shouldExpandItem item: Any) -> Bool {
  118. if let item = item as? KMBOTAOutlineItem {
  119. if !item.isItemExpanded {
  120. item.isItemExpanded = true
  121. outlineView.animator().expandItem(item, expandChildren: true)
  122. return false
  123. }
  124. }
  125. return true
  126. }
  127. func outlineView(_ outlineView: NSOutlineView, shouldCollapseItem item: Any) -> Bool {
  128. if let item = item as? KMBOTAOutlineItem {
  129. if item.isItemExpanded {
  130. item.isItemExpanded = false
  131. outlineView.animator().collapseItem(item, collapseChildren: true)
  132. return false
  133. }
  134. }
  135. return true
  136. }
  137. func outlineView(_ outlineView: NSOutlineView, viewFor tableColumn: NSTableColumn?, item: Any) -> NSView? {
  138. let cell : KMBOTAOutlineCellView = KMBOTAOutlineCellView.init()
  139. cell.model = item as? KMBOTAOutlineItem
  140. cell.iconAction = { [unowned self] view in
  141. let rowIndex = outlineView.row(forItem: item)
  142. let rowView = outlineView.rowView(atRow: rowIndex, makeIfNecessary: false)
  143. self.didSelectItem(view: (rowView as? KMBOTAOutlineRowView), event: NSEvent())
  144. if self.selectItems?.count == 1 {
  145. self.needOpenOrCloseItem(oulineItem: (self.selectItems?.first)!)
  146. }
  147. }
  148. return cell
  149. }
  150. func outlineView(_ outlineView: NSOutlineView, rowViewForItem item: Any) -> NSTableRowView? {
  151. let rowView = KMBOTAOutlineRowView()
  152. rowView.model = item as? KMBOTAOutlineItem
  153. rowView.mouseDownCallback = { [unowned self] (view, event) in
  154. self.didSelectItem(view: view, event: event)
  155. }
  156. rowView.rightMouseCallback = { [unowned self] (view, event) in
  157. if !KMOCToolClass.arrayContains(array: self.selectItems, annotation: item) ||
  158. self.selectItems!.count == 1 {
  159. self.selectItem(outlineItem: item as! KMBOTAOutlineItem)
  160. }
  161. self.delegate?.BOTAOutlineView(self, rightDidMoseDown: item as! KMBOTAOutlineItem, event: event)
  162. }
  163. rowView.hoverCallback = { [unowned self] (mouseEntered, mouseBox) in
  164. self.outlineView.enumerateAvailableRowViews { view, row in
  165. if view is KMBOTAOutlineRowView {
  166. (view as? KMBOTAOutlineRowView)?.model.hover = false
  167. (view as? KMBOTAOutlineRowView)?.reloadData()
  168. }
  169. }
  170. if mouseEntered {
  171. rowView.model.hover = true
  172. } else {
  173. rowView.model.hover = false
  174. }
  175. }
  176. return rowView
  177. }
  178. func outlineView(_ outlineView: NSOutlineView, heightOfRowByItem item: Any) -> CGFloat {
  179. if item is KMBOTAOutlineItem {
  180. let tempItem: KMBOTAOutlineItem = item as! KMBOTAOutlineItem
  181. let string: NSString = tempItem.outline.label as NSString
  182. let paragraphStyle = NSMutableParagraphStyle()
  183. paragraphStyle.lineHeightMultiple = 1.32
  184. paragraphStyle.alignment = .left
  185. let attributes = [NSAttributedString.Key.paragraphStyle: paragraphStyle,
  186. NSAttributedString.Key.font : NSFont.SFProTextRegularFont(14.0)]
  187. let size = string.boundingRect(with: NSMakeSize(outlineView.frame.size.width - 30, 200), options: NSString.DrawingOptions(rawValue: 3), attributes: attributes)
  188. return max(40, size.height + 16)
  189. }
  190. return 40
  191. }
  192. func outlineView(_ outlineView: NSOutlineView, writeItems items: [Any], to pasteboard: NSPasteboard) -> Bool {
  193. guard let callBack = self.delegate else { return false}
  194. return callBack.BOTAOutlineView(self, writeItems: items, to: pasteboard)
  195. }
  196. func outlineView(_ outlineView: NSOutlineView, validateDrop info: NSDraggingInfo, proposedItem item: Any?, proposedChildIndex index: Int) -> NSDragOperation {
  197. guard let callBack = self.delegate else { return NSDragOperation.init(rawValue: 0)}
  198. return callBack.BOTAOutlineView(self, validateDrop: info, proposedItem: item, proposedChildIndex: index)
  199. }
  200. func outlineView(_ outlineView: NSOutlineView, acceptDrop info: NSDraggingInfo, item: Any?, childIndex index: Int) -> Bool {
  201. guard let callBack = self.delegate else { return false}
  202. return callBack.BOTAOutlineView(self, acceptDrop: info, item: item, childIndex: index)
  203. }
  204. func outlineViewSelectionDidChange(_ notification: Notification) {
  205. // if self.outlineView.selectedRow == -1 {
  206. // self.cancelSelect()
  207. // }
  208. }
  209. }
  210. //MARK: Action
  211. extension KMBOTAOutlineView {
  212. @objc func expandAllComments(item: NSMenuItem) {
  213. self.reloadData(expandItemType: .expand)
  214. self.outlineView.reloadData()
  215. self.outlineView.expandItem(nil, expandChildren: true)
  216. }
  217. @objc func collapseAllComments(item: NSMenuItem) {
  218. self.reloadData(expandItemType: .collapse)
  219. self.outlineView.reloadData()
  220. self.outlineView.collapseItem(nil, collapseChildren: true)
  221. }
  222. func selectItem(outlineItem: KMBOTAOutlineItem) {
  223. let index = self.outlineView.row(forItem: outlineItem)
  224. self.outlineView.selectRowIndexes(IndexSet(integer: IndexSet.Element(index)), byExtendingSelection: false)
  225. self.didSelectItem(view: nil, event: NSEvent(), isNeedDelegate: false)
  226. }
  227. func selectIndex(index: Int) {
  228. self.outlineView.selectRowIndexes(IndexSet(integer: IndexSet.Element(index)), byExtendingSelection: false)
  229. self.didSelectItem(view: nil, event: NSEvent(), isNeedDelegate: false)
  230. }
  231. func cancelSelect() {
  232. guard let items = self.selectItems else { return }
  233. self.outlineView.deselectAll(nil)
  234. for model in items {
  235. model.select = false
  236. self.outlineView.reloadItem(model)
  237. }
  238. }
  239. func didSelectItem(view: KMBOTAOutlineRowView?, event: NSEvent, isNeedDelegate: Bool = true) {
  240. //当选中一个时
  241. if view != nil && (self.outlineView.selectedRowIndexes.count == 1 ||
  242. (!event.modifierFlags.contains(NSEvent.ModifierFlags.command) &&
  243. !event.modifierFlags.contains(NSEvent.ModifierFlags.shift))) {
  244. let rowView: KMBOTAOutlineRowView = view!
  245. let index = self.outlineView.row(for: rowView)
  246. self.outlineView.selectRowIndexes(IndexSet(integer: IndexSet.Element(index)), byExtendingSelection: false)
  247. }
  248. //原始数据置空
  249. if self.selectItems != nil {
  250. for item in self.selectItems! {
  251. item.select = false
  252. let index = self.outlineView.row(forItem: item)
  253. if index != -1 {
  254. if self.outlineView.rowView(atRow: index, makeIfNecessary: false) != nil {
  255. let rowView: KMBOTAOutlineRowView = self.outlineView.rowView(atRow: index, makeIfNecessary: false) as! KMBOTAOutlineRowView
  256. rowView.reloadData()
  257. }
  258. }
  259. }
  260. }
  261. //获取最新数据
  262. var items: [KMBOTAOutlineItem] = []
  263. for index in self.outlineView.selectedRowIndexes {
  264. if index != -1 {
  265. let item: KMBOTAOutlineItem = self.outlineView.item(atRow: index) as! KMBOTAOutlineItem
  266. item.select = true
  267. items.append(item)
  268. //刷新数据
  269. if self.outlineView.rowView(atRow: index, makeIfNecessary: false) != nil {
  270. let rowView: KMBOTAOutlineRowView = self.outlineView.rowView(atRow: index, makeIfNecessary: false) as! KMBOTAOutlineRowView
  271. rowView.reloadData()
  272. self.outlineView.reloadItem(item, reloadChildren: true)
  273. }
  274. }
  275. }
  276. self.selectItems = items
  277. // if self.selectItems?.count == 1 {
  278. // self.needOpenOrCloseItem(oulineItem: (self.selectItems?.first)!)
  279. // }
  280. if self.selectItems != nil && isNeedDelegate {
  281. self.delegate?.BOTAOutlineView(self, didSelectItem: self.selectItems!)
  282. }
  283. }
  284. func needOpenOrCloseItem(oulineItem: KMBOTAOutlineItem) {
  285. //只有一个选中项时开启关闭item
  286. if self.selectItems?.count == 1 {
  287. if self.outlineView.isItemExpanded(oulineItem) {
  288. self.outlineView.collapseItem(oulineItem)
  289. oulineItem.isItemExpanded = false
  290. } else {
  291. self.outlineView.expandItem(oulineItem)
  292. oulineItem.isItemExpanded = true
  293. }
  294. let row = self.outlineView.row(forItem: oulineItem)
  295. self.outlineView.reloadData(forRowIndexes: IndexSet(integer: row), columnIndexes: IndexSet(integer: 0))
  296. }
  297. }
  298. }