KMBOTAOutlineView.swift 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570
  1. //
  2. // KMBOTAOutlineView.swift
  3. // PDF Reader Pro
  4. //
  5. // Created by lizhe on 2023/4/2.
  6. //
  7. import Cocoa
  8. protocol KMBOTAOutlineViewDelegate: NSObjectProtocol {
  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. weak 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. var isSearchMode = false
  29. var searchKey = ""
  30. var wholeWords = true {
  31. didSet {
  32. if isValidSearchMode() == false {
  33. return
  34. }
  35. reloadData()
  36. outlineView.expandItem(nil, expandChildren: true)
  37. }
  38. }
  39. var caseSensitive = false {
  40. didSet {
  41. if isValidSearchMode() == false {
  42. return
  43. }
  44. reloadData()
  45. outlineView.expandItem(nil, expandChildren: true)
  46. }
  47. }
  48. override func draw(_ dirtyRect: NSRect) {
  49. super.draw(dirtyRect)
  50. // Drawing code here.
  51. }
  52. override func awakeFromNib() {
  53. super.awakeFromNib()
  54. self.setup()
  55. }
  56. override func setup() {
  57. self.scrollView.backgroundColor(NSColor.km_init(hex: "#F2F9FF"))
  58. self.outlineView.registerForDraggedTypes([NSPasteboard.PasteboardType(rawValue: "kKMPDFViewOutlineDragDataType")])
  59. self.outlineView.delegate = self
  60. self.outlineView.dataSource = self
  61. self.outlineView.selectionHighlightStyle = NSTableView.SelectionHighlightStyle.none;
  62. self.outlineView.allowsMultipleSelection = true
  63. // self.outlineView.indentationPerLevel = 0
  64. outlineView.tocDelegate = self
  65. outlineView.hasImageToolTips = true
  66. }
  67. func reloadData(expandItemType: KMOutlineViewExpandItemType = .none) {
  68. if self.inputData != nil {
  69. //获取数据
  70. var tempData: KMBOTAOutlineItem = KMBOTAOutlineItem()
  71. if self.inputData!.numberOfChildren > 0 {
  72. let outline: CPDFOutline = self.inputData!
  73. tempData = self.addOutlineItem(outlineItem:tempData, outline: outline, expandItemType: expandItemType)
  74. } else {
  75. tempData.outline = CPDFOutline()
  76. if expandItemType == .collapse {
  77. tempData.isItemExpanded = false
  78. } else if (expandItemType == .expand) {
  79. tempData.isItemExpanded = true
  80. }
  81. }
  82. tempData.parent = nil
  83. self.data = tempData
  84. if isValidSearchMode() {
  85. self.reloadSearchChildren(item: self.data)
  86. }
  87. self.outlineView.reloadData()
  88. }
  89. self.delegate?.BOTAOutlineView(self, didReloadData: self.data ?? KMBOTAOutlineItem())
  90. }
  91. // 递归处理
  92. func reloadSearchChildren(item: KMBOTAOutlineItem?) {
  93. guard let theItem = item else {
  94. return
  95. }
  96. // 处理当前 item
  97. let models = self.fetchOutlines(for: theItem, searchString: searchKey)
  98. // 搜索数据
  99. theItem.searchChildren = models
  100. theItem.isItemExpanded = true
  101. if theItem.children.isEmpty { // 递归退出条件
  102. return
  103. }
  104. // 处理 childItem
  105. for childM in theItem.children {
  106. self.reloadSearchChildren(item: childM)
  107. }
  108. }
  109. func addOutlineItem(outlineItem: KMBOTAOutlineItem?, outline: CPDFOutline, expandItemType: KMOutlineViewExpandItemType = .none) -> KMBOTAOutlineItem {
  110. //添加根节点
  111. let item: KMBOTAOutlineItem = KMBOTAOutlineItem()
  112. item.outline = outline
  113. item.parent = outlineItem
  114. var items: [KMBOTAOutlineItem] = []
  115. if outline.numberOfChildren > 0 {
  116. for index in 0...outline.numberOfChildren - 1 {
  117. let children: CPDFOutline = outline.child(at: index)
  118. let childrenItem = self.addOutlineItem(outlineItem: item, outline: children, expandItemType: expandItemType)
  119. if expandItemType == .collapse {
  120. childrenItem.isItemExpanded = false
  121. } else if (expandItemType == .expand) {
  122. childrenItem.isItemExpanded = true
  123. }
  124. items.append(childrenItem)
  125. }
  126. }
  127. item.children = items
  128. return item
  129. }
  130. func updateUI() {
  131. }
  132. func updateLanguage() {
  133. }
  134. func hasContainString(_ searchString: String, rootOutline outline: CPDFOutline) -> Bool {
  135. var label = outline.label ?? ""
  136. var searchLabel = searchString
  137. if caseSensitive == false {
  138. label = label.lowercased()
  139. searchLabel = searchLabel.lowercased()
  140. }
  141. if label.contains(searchLabel) {
  142. if wholeWords {
  143. let words = label.words()
  144. return words.contains(searchLabel)
  145. }
  146. return true
  147. } else {
  148. var subHas = false
  149. for i in 0 ..< outline.numberOfChildren {
  150. if let subOutline = outline.child(at: i) {
  151. subHas = self.hasContainString(searchString, rootOutline: subOutline)
  152. } else {
  153. continue
  154. }
  155. if (subHas) {
  156. break
  157. }
  158. }
  159. return subHas
  160. }
  161. }
  162. func fetchOutlines(for item: KMBOTAOutlineItem, searchString: String) -> [KMBOTAOutlineItem] {
  163. var items: [KMBOTAOutlineItem] = []
  164. for childI in item.children {
  165. if self.hasContainString(searchString, rootOutline: childI.outline) {
  166. items.append(childI)
  167. }
  168. }
  169. return items
  170. }
  171. func isValidSearchMode() -> Bool {
  172. if isSearchMode == false {
  173. return false
  174. }
  175. if searchKey.isEmpty {
  176. return false
  177. }
  178. return true
  179. }
  180. }
  181. //MARK: NSOutlineViewDataSource,NSOutlineViewDelegate
  182. extension KMBOTAOutlineView : NSOutlineViewDataSource,NSOutlineViewDelegate {
  183. func outlineView(_ outlineView: NSOutlineView, numberOfChildrenOfItem item: Any?) -> Int {
  184. guard let rootModel = self.data else {
  185. return 0
  186. }
  187. guard let model = item as? KMBOTAOutlineItem else {
  188. if isValidSearchMode() { // 是否为搜索模块
  189. if self.hasContainString(searchKey, rootOutline: rootModel.outline) == false {
  190. return 0
  191. }
  192. return rootModel.searchChildren.count
  193. }
  194. return Int(rootModel.outline.numberOfChildren)
  195. }
  196. if isValidSearchMode() {
  197. return model.searchChildren.count
  198. }
  199. return model.children.count
  200. }
  201. func outlineView(_ outlineView: NSOutlineView, child index: Int, ofItem item: Any?) -> Any {
  202. guard let rootModel = self.data else {
  203. return ""
  204. }
  205. guard let model = item as? KMBOTAOutlineItem else {
  206. if isValidSearchMode() { // 是否为搜索模块
  207. if self.hasContainString(searchKey, rootOutline: rootModel.outline) == false {
  208. return ""
  209. }
  210. return rootModel.searchChildren.safe_element(for: index) as Any
  211. }
  212. return rootModel.children[index] as Any
  213. }
  214. if isValidSearchMode() {
  215. return model.searchChildren.safe_element(for: index) as Any
  216. }
  217. return model.children[index] as Any
  218. }
  219. func outlineView(_ outlineView: NSOutlineView, isItemExpandable item: Any) -> Bool {
  220. guard let model = item as? KMBOTAOutlineItem else {
  221. return false
  222. }
  223. if isValidSearchMode() {
  224. let datas = self.fetchOutlines(for: model, searchString: searchKey)
  225. return !datas.isEmpty
  226. }
  227. return !model.children.isEmpty
  228. }
  229. func outlineView(_ outlineView: NSOutlineView, shouldExpandItem item: Any) -> Bool {
  230. if let item = item as? KMBOTAOutlineItem {
  231. if !item.isItemExpanded {
  232. item.isItemExpanded = true
  233. outlineView.animator().expandItem(item, expandChildren: true)
  234. return false
  235. }
  236. }
  237. return true
  238. }
  239. func outlineView(_ outlineView: NSOutlineView, shouldCollapseItem item: Any) -> Bool {
  240. if let item = item as? KMBOTAOutlineItem {
  241. if item.isItemExpanded {
  242. item.isItemExpanded = false
  243. outlineView.animator().collapseItem(item, collapseChildren: true)
  244. return false
  245. }
  246. }
  247. return true
  248. }
  249. func outlineView(_ outlineView: NSOutlineView, viewFor tableColumn: NSTableColumn?, item: Any) -> NSView? {
  250. guard let model = item as? KMBOTAOutlineItem else {
  251. return nil
  252. }
  253. let cell : KMBOTAOutlineCellView = KMBOTAOutlineCellView.createFromNib() ?? KMBOTAOutlineCellView.init()
  254. if isValidSearchMode() {
  255. var label = model.outline.label ?? ""
  256. let attri = NSMutableAttributedString(string: label, attributes: [
  257. .font : NSFont.SFProTextRegularFont(13),
  258. .foregroundColor : KMNColorTools.colorText_1()])
  259. var _searchKey = searchKey
  260. if caseSensitive == false {
  261. label = label.lowercased()
  262. _searchKey = _searchKey.lowercased()
  263. }
  264. if wholeWords {
  265. let escapedKey = NSRegularExpression.escapedPattern(for: _searchKey)
  266. let pattern = "\\b\(escapedKey)\\b"
  267. do {
  268. let regex = try NSRegularExpression(pattern: pattern, options: caseSensitive ? [] : [.caseInsensitive])
  269. let matches = regex.matches(in: label, range: NSRange(location: 0, length: label.utf16.count))
  270. for match in matches {
  271. attri.addAttributes([
  272. .font : NSFont.SFProTextBoldFont(13),
  273. .foregroundColor: KMNColorTools.colorPrimary_textLight()
  274. ], range: match.range)
  275. }
  276. } catch {
  277. }
  278. } else {
  279. let ranges = label.ranges(of: _searchKey)
  280. for range in ranges.nsRnage {
  281. attri.addAttributes([.font : NSFont.SFProTextBoldFont(13), .foregroundColor: KMNColorTools.colorPrimary_textLight()], range: range)
  282. }
  283. }
  284. cell.titleLabel.attributedStringValue = attri
  285. cell.iconButton.isHidden = model.searchChildren.isEmpty
  286. } else {
  287. cell.titleLabel.stringValue = model.outline.label ?? ""
  288. if model.outline.numberOfChildren > 0 {
  289. cell.iconButton.isHidden = false
  290. } else {
  291. cell.iconButton.isHidden = true
  292. }
  293. }
  294. cell.model = model
  295. cell.iconAction = { [unowned self] view in
  296. let rowIndex = outlineView.row(forItem: item)
  297. let rowView = outlineView.rowView(atRow: rowIndex, makeIfNecessary: false)
  298. self.didSelectItem(view: (rowView as? KMBOTAOutlineRowView), event: NSEvent())
  299. if self.selectItems.count == 1 {
  300. self.needOpenOrCloseItem(oulineItem: (self.selectItems.first)!)
  301. }
  302. }
  303. return cell
  304. }
  305. func outlineView(_ outlineView: NSOutlineView, rowViewForItem item: Any) -> NSTableRowView? {
  306. let rowView = KMBOTAOutlineRowView()
  307. rowView.model = item as? KMBOTAOutlineItem
  308. rowView.mouseDownCallback = { [unowned self] (view, event) in
  309. self.didSelectItem(view: view, event: event)
  310. }
  311. rowView.rightMouseCallback = { [unowned self] (view, event) in
  312. if !KMOCToolClass.arrayContains(array: self.selectItems, annotation: item) ||
  313. self.selectItems.count == 1 {
  314. self.selectItem(outlineItem: item as! KMBOTAOutlineItem)
  315. }
  316. self.delegate?.BOTAOutlineView(self, rightDidMoseDown: item as! KMBOTAOutlineItem, event: event)
  317. }
  318. rowView.hoverCallback = { [unowned self] (mouseEntered, mouseBox) in
  319. self.outlineView.enumerateAvailableRowViews { view, row in
  320. if view is KMBOTAOutlineRowView {
  321. (view as? KMBOTAOutlineRowView)?.model?.hover = false
  322. (view as? KMBOTAOutlineRowView)?.reloadData()
  323. }
  324. }
  325. if mouseEntered {
  326. rowView.model?.hover = true
  327. } else {
  328. rowView.model?.hover = false
  329. }
  330. }
  331. return rowView
  332. }
  333. func outlineView(_ outlineView: NSOutlineView, heightOfRowByItem item: Any) -> CGFloat {
  334. if item is KMBOTAOutlineItem {
  335. let tempItem: KMBOTAOutlineItem = item as! KMBOTAOutlineItem
  336. let string: NSString = tempItem.outline.label as NSString
  337. let paragraphStyle = NSMutableParagraphStyle()
  338. paragraphStyle.lineHeightMultiple = 1.32
  339. paragraphStyle.alignment = .left
  340. let attributes = [NSAttributedString.Key.paragraphStyle: paragraphStyle,
  341. NSAttributedString.Key.font : NSFont.SFProTextRegularFont(14.0)]
  342. let size = string.boundingRect(with: NSMakeSize(outlineView.frame.size.width - 30, 200), options: NSString.DrawingOptions(rawValue: 3), attributes: attributes)
  343. return max(40, size.height + 16)
  344. }
  345. return 40
  346. }
  347. func outlineView(_ outlineView: NSOutlineView, writeItems items: [Any], to pasteboard: NSPasteboard) -> Bool {
  348. guard let callBack = self.delegate else { return false}
  349. return callBack.BOTAOutlineView(self, writeItems: items, to: pasteboard)
  350. }
  351. func outlineView(_ outlineView: NSOutlineView, validateDrop info: NSDraggingInfo, proposedItem item: Any?, proposedChildIndex index: Int) -> NSDragOperation {
  352. guard let callBack = self.delegate else { return NSDragOperation.init(rawValue: 0)}
  353. return callBack.BOTAOutlineView(self, validateDrop: info, proposedItem: item, proposedChildIndex: index)
  354. }
  355. func outlineView(_ outlineView: NSOutlineView, acceptDrop info: NSDraggingInfo, item: Any?, childIndex index: Int) -> Bool {
  356. guard let callBack = self.delegate else { return false}
  357. return callBack.BOTAOutlineView(self, acceptDrop: info, item: item, childIndex: index)
  358. }
  359. func outlineViewSelectionDidChange(_ notification: Notification) {
  360. // if self.outlineView.selectedRow == -1 {
  361. // self.cancelSelect()
  362. // }
  363. }
  364. }
  365. extension KMBOTAOutlineView: KMTocOutlineViewDelegate {
  366. func outlineView(_ anOutlineView: NSOutlineView, imageContextForItem item: Any?) -> AnyObject? {
  367. if item == nil {
  368. return true as AnyObject
  369. }
  370. if let data = item as? KMBOTAOutlineItem {
  371. return data.outline.destination
  372. }
  373. return nil
  374. }
  375. }
  376. //MARK: Action
  377. extension KMBOTAOutlineView {
  378. @objc func expandAllComments(item: NSMenuItem) {
  379. self.reloadData(expandItemType: .expand)
  380. self.outlineView.reloadData()
  381. self.outlineView.expandItem(nil, expandChildren: true)
  382. }
  383. @objc func collapseAllComments(item: NSMenuItem) {
  384. self.reloadData(expandItemType: .collapse)
  385. self.outlineView.reloadData()
  386. self.outlineView.collapseItem(nil, collapseChildren: true)
  387. }
  388. func selectItem(outlineItem: KMBOTAOutlineItem) {
  389. let index = self.outlineView.row(forItem: outlineItem)
  390. self.outlineView.selectRowIndexes(IndexSet(integer: IndexSet.Element(index)), byExtendingSelection: false)
  391. self.didSelectItem(view: nil, event: NSEvent(), isNeedDelegate: false)
  392. }
  393. func selectIndex(index: Int) {
  394. self.outlineView.selectRowIndexes(IndexSet(integer: IndexSet.Element(index)), byExtendingSelection: false)
  395. self.didSelectItem(view: nil, event: NSEvent(), isNeedDelegate: false)
  396. }
  397. func cancelSelect() {
  398. self.outlineView.deselectAll(nil)
  399. for model in self.selectItems {
  400. model.select = false
  401. self.outlineView.reloadItem(model)
  402. }
  403. }
  404. func selectAll() {
  405. let totalRows = self.outlineView.numberOfRows
  406. //outline全选
  407. var allIndexes = IndexSet()
  408. for row in 0..<totalRows {
  409. allIndexes.insert(row)
  410. }
  411. outlineView.selectRowIndexes(allIndexes, byExtendingSelection: false)
  412. //数据全选
  413. var items: [KMBOTAOutlineItem] = []
  414. for row in 0..<totalRows {
  415. let item: KMBOTAOutlineItem = self.outlineView.item(atRow: row) as! KMBOTAOutlineItem
  416. item.select = true
  417. items.append(item)
  418. //刷新数据
  419. if self.outlineView.rowView(atRow: row, makeIfNecessary: false) != nil {
  420. let rowView: KMBOTAOutlineRowView = self.outlineView.rowView(atRow: row, makeIfNecessary: false) as! KMBOTAOutlineRowView
  421. rowView.reloadData()
  422. self.outlineView.reloadItem(item, reloadChildren: true)
  423. }
  424. }
  425. self.selectItems = items
  426. }
  427. func didSelectItem(view: KMBOTAOutlineRowView?, event: NSEvent, isNeedDelegate: Bool = true) {
  428. //当选中一个时
  429. if view != nil && (self.outlineView.selectedRowIndexes.count == 1 ||
  430. (!event.modifierFlags.contains(NSEvent.ModifierFlags.command) &&
  431. !event.modifierFlags.contains(NSEvent.ModifierFlags.shift))) {
  432. let rowView: KMBOTAOutlineRowView = view!
  433. let index = self.outlineView.row(for: rowView)
  434. self.outlineView.selectRowIndexes(IndexSet(integer: IndexSet.Element(index)), byExtendingSelection: false)
  435. }
  436. //原始数据置空
  437. for item in self.selectItems {
  438. item.select = false
  439. let index = self.outlineView.row(forItem: item)
  440. if index != -1 {
  441. if self.outlineView.rowView(atRow: index, makeIfNecessary: false) != nil {
  442. let rowView: KMBOTAOutlineRowView = self.outlineView.rowView(atRow: index, makeIfNecessary: false) as! KMBOTAOutlineRowView
  443. rowView.reloadData()
  444. }
  445. }
  446. }
  447. //获取最新数据
  448. var items: [KMBOTAOutlineItem] = []
  449. for index in self.outlineView.selectedRowIndexes {
  450. if index != -1 {
  451. let item: KMBOTAOutlineItem = self.outlineView.item(atRow: index) as! KMBOTAOutlineItem
  452. item.select = true
  453. items.append(item)
  454. //刷新数据
  455. if self.outlineView.rowView(atRow: index, makeIfNecessary: false) != nil {
  456. let rowView: KMBOTAOutlineRowView = self.outlineView.rowView(atRow: index, makeIfNecessary: false) as! KMBOTAOutlineRowView
  457. rowView.reloadData()
  458. self.outlineView.reloadItem(item, reloadChildren: true)
  459. }
  460. }
  461. }
  462. self.selectItems = items
  463. // if self.selectItems?.count == 1 {
  464. // self.needOpenOrCloseItem(oulineItem: (self.selectItems?.first)!)
  465. // }
  466. if isNeedDelegate {
  467. self.delegate?.BOTAOutlineView(self, didSelectItem: self.selectItems)
  468. }
  469. }
  470. func needOpenOrCloseItem(oulineItem: KMBOTAOutlineItem) {
  471. //只有一个选中项时开启关闭item
  472. if self.selectItems.count == 1 {
  473. if self.outlineView.isItemExpanded(oulineItem) {
  474. self.outlineView.collapseItem(oulineItem)
  475. oulineItem.isItemExpanded = false
  476. } else {
  477. self.outlineView.expandItem(oulineItem)
  478. oulineItem.isItemExpanded = true
  479. }
  480. let row = self.outlineView.row(forItem: oulineItem)
  481. self.outlineView.reloadData(forRowIndexes: IndexSet(integer: row), columnIndexes: IndexSet(integer: 0))
  482. }
  483. }
  484. }