KMBookmarkController.swift 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888
  1. //
  2. // KMBookmarkController.swift
  3. // PDF Reader Pro
  4. //
  5. // Created by lizhe on 2024/2/5.
  6. //
  7. import Cocoa
  8. private let kLabelIdentifier = NSUserInterfaceItemIdentifier("label")
  9. private let kFileIdentifier = NSUserInterfaceItemIdentifier("file")
  10. private let kPageIdentifier = NSUserInterfaceItemIdentifier("page")
  11. let kTextWithIconStringKey = "string";
  12. let kTextWithIconImageKey = "image";
  13. let kBookmarksToolbarIdentifier = "BookmarksToolbarIdentifier"
  14. let kBookmarksNewFolderToolbarItemIdentifier = "BookmarksNewFolderToolbarItemIdentifier"
  15. let kBookmarksNewSeparatorToolbarItemIdentifier = "BookmarksNewSeparatorToolbarItemIdentifier"
  16. let kBookmarksDeleteToolbarItemIdentifier = "BookmarksDeleteToolbarItemIdentifier"
  17. let kPasteboardTypeBookmarkRows = NSPasteboard.PasteboardType(rawValue: "pasteboard.bookmarkrows")
  18. class KMBookmarkController: NSWindowController {
  19. @IBOutlet weak var bookmarkOutlineView: KMBookmarkOutlineView!
  20. @IBOutlet weak var outlineView: KMCustomOutlineView!
  21. var draggedBookmarks: [KMBookmark] = []
  22. var recentDocuments: [[String: Any]] {
  23. get {
  24. return KMBookmarkManager.manager.recentDocuments
  25. }
  26. set {
  27. }
  28. }
  29. var bookmarkRoot: KMRootBookmark {
  30. get {
  31. return KMBookmarkManager.manager.rootBookmark
  32. }
  33. set {
  34. }
  35. }
  36. var toolbarItems: [String: NSToolbarItem] = [:]
  37. override func windowDidLoad() {
  38. super.windowDidLoad()
  39. setupToolbar()
  40. outlineView.delegate = self
  41. outlineView.dataSource = self
  42. outlineView.registerForDraggedTypes([kPasteboardTypeBookmarkRows, .fileURL, .string])
  43. outlineView.doubleAction = #selector(doubleClickBookmark(_:))
  44. bookmarkOutlineView.data = self.bookmarkRoot
  45. }
  46. func updateStatus() {
  47. let row = outlineView.selectedRow
  48. var message = ""
  49. if row != -1 {
  50. if let bookmark = outlineView.item(atRow: row) as? KMBookmark {
  51. switch bookmark.bookmarkType {
  52. case .bookmark:
  53. message = bookmark.fileURL?.path ?? ""
  54. case .folder:
  55. let count = bookmark.children.count
  56. message = count == 1 ? NSLocalizedString("1 item", comment: "Bookmark folder description") : String(format: NSLocalizedString("%ld items", comment: "Bookmark folder description"), count)
  57. default:
  58. break
  59. }
  60. }
  61. }
  62. // statusBar.leftStringValue = message
  63. }
  64. //
  65. static func showBookmarkController() -> KMBookmarkController {
  66. let controller = KMBookmarkController.init(windowNibName: "KMBookmarkController")
  67. NSWindow.currentWindow().addChildWindow(controller.window!, ordered: NSWindow.OrderingMode.above)
  68. controller.window?.center()
  69. return controller
  70. }
  71. //
  72. //
  73. // //MARK: Recent Documents
  74. func recentDocumentInfo(at fileURL: URL) -> [String: Any]? {
  75. let path = fileURL.path
  76. for info in recentDocuments {
  77. // if let aliasData = info[ALIASDATA_KEY] as? Data,
  78. // let alias = SKAlias(aliasData),
  79. // alias.fileURLNoUI?.path.caseInsensitiveCompare(path) == .orderedSame {
  80. // return info
  81. // }
  82. }
  83. return nil
  84. }
  85. //
  86. func addRecentDocument(for fileURL: URL, pageIndex: UInt, scaleFactor factor: CGFloat, snapshots setups: [Any]?) {
  87. // if let info = recentDocumentInfo(at: fileURL) {
  88. // recentDocuments.removeObject(info)
  89. // }
  90. //
  91. // if let alias = SKAlias(url: fileURL) {
  92. // var bm: [String: Any] = [
  93. // PAGEINDEX_KEY: pageIndex,
  94. // SCALE_KEY: factor,
  95. // ALIASDATA_KEY: alias.data,
  96. // ALIAS_KEY: alias,
  97. // SNAPSHOTS_KEY: setups ?? []
  98. // ]
  99. // recentDocuments.insert(bm, at: 0)
  100. // if recentDocuments.count > maxRecentDocumentsCount {
  101. // recentDocuments.removeLastObject()
  102. // }
  103. // }
  104. }
  105. //
  106. // func pageIndex(forRecentDocumentAt fileURL: URL) -> UInt {
  107. // guard let fileURL = fileURL else { return UInt.max }
  108. // if let pageIndex = recentDocumentInfo(at: fileURL)?[PAGEINDEX_KEY] as? UInt {
  109. // return pageIndex
  110. // }
  111. // return UInt.max
  112. // }
  113. //
  114. // func scaleFactor(forRecentDocumentAt fileURL: URL) -> CGFloat {
  115. // guard let fileURL = fileURL else { return 0 }
  116. // if let scaleFactor = recentDocumentInfo(at: fileURL)?[SCALE_KEY] as? CGFloat {
  117. // return scaleFactor
  118. // }
  119. // return 0
  120. // }
  121. //
  122. // func snapshots(forRecentDocumentAt fileURL: URL) -> [Any]? {
  123. // guard let fileURL = fileURL else { return nil }
  124. // if let setups = recentDocumentInfo(at: fileURL)?[SNAPSHOTS_KEY] as? [Any], !setups.isEmpty {
  125. // return setups
  126. // }
  127. // return nil
  128. // }
  129. //
  130. // //MARK: Bookmarks support
  131. func getInsertionFolder(_ bookmarkPtr: inout KMBookmark?, childIndex indexPtr: inout Int) {
  132. let rowIndex = outlineView.clickedRow
  133. var indexes = outlineView.selectedRowIndexes
  134. if rowIndex != -1 && !indexes.contains(rowIndex) {
  135. indexes = IndexSet(integer: rowIndex)
  136. }
  137. let rowIdx = indexes.last ?? NSNotFound
  138. var item = KMBookmarkManager.manager.rootBookmark
  139. var idx = item.children.count
  140. if rowIdx != NSNotFound {
  141. if let selectedItem = outlineView.item(atRow: rowIdx) as? KMBookmark {
  142. if outlineView.isItemExpanded(selectedItem) {
  143. item = selectedItem as! KMRootBookmark
  144. idx = item.children.count
  145. } else if let parent = selectedItem.parent, let itemIdx = parent.children.firstIndex(of: selectedItem) {
  146. item = parent as! KMRootBookmark
  147. idx = itemIdx + 1
  148. }
  149. }
  150. }
  151. bookmarkPtr = item
  152. indexPtr = idx
  153. }
  154. @IBAction func openBookmark(_ sender: Any) {
  155. if let bookmark = (sender as AnyObject).representedObject as? KMBookmark {
  156. bookmark.open()
  157. }
  158. }
  159. @IBAction func doubleClickBookmark(_ sender: Any) {
  160. let row = outlineView.clickedRow
  161. if let bm = (row != -1 ? outlineView.item(atRow: row) : nil) as? KMBookmark,
  162. [KMBookmarkType.bookmark, .session].contains(bm.bookmarkType) {
  163. bm.open()
  164. }
  165. }
  166. func deleteBookmarks(bookmarks: [KMBookmark]) {
  167. for item in minimumCoverForBookmarks(bookmarks).reversed() {
  168. guard let parent = item.parent, let itemIndex = parent.children.firstIndex(of: item) else { continue }
  169. parent.removeObjectFromChildren(index: itemIndex)
  170. }
  171. bookmarkOutlineView.reloadData()
  172. }
  173. @IBAction func insertBookmarkFolder(_ sender: Any) {
  174. let folder = KMFolderBookmark.folderBookmark(label: NSLocalizedString("Folder", comment: "default folder name"))
  175. var item: KMBookmark?
  176. var idx: Int = 0
  177. getInsertionFolder(&item, childIndex: &idx)
  178. item?.insert(child: folder, atIndex: idx)
  179. bookmarkOutlineView.outlineView.reloadData()
  180. let row = bookmarkOutlineView.outlineView.row(forItem: folder)
  181. if row > 0 {
  182. bookmarkOutlineView.outlineView.selectRowIndexes(IndexSet(integer: row), byExtendingSelection: false)
  183. bookmarkOutlineView.outlineView.editColumn(0, row: row, with: nil, select: true)
  184. }
  185. }
  186. @IBAction func insertBookmarkSeparator(_ sender: Any) {
  187. let separator = KMSeparatorBookmark()
  188. var item: KMBookmark?
  189. var idx: Int = 0
  190. getInsertionFolder(&item, childIndex: &idx)
  191. item?.insert(child: separator, atIndex: idx)
  192. let row = outlineView.row(forItem: separator)
  193. outlineView.selectRowIndexes(IndexSet(integer: row), byExtendingSelection: false)
  194. }
  195. @IBAction func addBookmark(_ sender: Any) {
  196. let openPanel = NSOpenPanel()
  197. var types = [String]()
  198. for docClass in NSDocumentController.shared.documentClassNames {
  199. if let docClass = NSClassFromString(docClass) as? NSDocument.Type {
  200. types += docClass.readableTypes
  201. }
  202. }
  203. openPanel.allowsMultipleSelection = true
  204. openPanel.canChooseDirectories = true
  205. openPanel.allowedFileTypes = types
  206. openPanel.beginSheetModal(for: self.window!) { (result) in
  207. guard result == .OK else { return }
  208. let newBookmarks = KMBookmark.bookmarks(urls: openPanel.urls)
  209. if newBookmarks != nil {
  210. var item: KMBookmark?
  211. var index: Int = 0
  212. self.getInsertionFolder(&item, childIndex: &index)
  213. var indexes = IndexSet(integersIn: Int(index)..<Int(index + newBookmarks.count))
  214. item?.mutableArrayValue(forKey: "children").insert(newBookmarks, at: indexes)
  215. if item == self.bookmarkRoot || self.outlineView.isItemExpanded(item) {
  216. if item != self.bookmarkRoot {
  217. indexes.shift(startingAt: 0, by: self.outlineView.row(forItem: item) + 1)
  218. }
  219. self.outlineView.selectRowIndexes(indexes, byExtendingSelection: false)
  220. }
  221. }
  222. }
  223. }
  224. @IBAction func deleteBookmark(_ sender: Any) {
  225. print("deleteBookmark")
  226. guard let bookmark = bookmarkOutlineView.outlineView.selectedItem() as? KMBookmark else { return }
  227. self.deleteBookmarks(bookmarks: [bookmark])
  228. }
  229. //
  230. // @IBAction func toggleStatusBar(_ sender: Any) {
  231. // UserDefaults.standard.set(!statusBar.isVisible, forKey: SKShowBookmarkStatusBarKey)
  232. // statusBar.toggle(below: outlineView.enclosingScrollView, animate: sender != nil)
  233. // }
  234. //
  235. // func clickedBookmarks() -> [Any]? {
  236. // let row = outlineView.clickedRow
  237. // guard row != -1 else { return nil }
  238. // var indexes = outlineView.selectedRowIndexes
  239. // if !indexes.contains(row) {
  240. // indexes = IndexSet(integer: row)
  241. // }
  242. // return indexes.compactMap { outlineView.item(atRow: $0) }
  243. // }
  244. //
  245. // @IBAction func deleteBookmarks(_ sender: Any) {
  246. // guard let items = clickedBookmarks() as? [KMBookmark] else { return }
  247. // for item in items.reversed() {
  248. // if let parent = item.parent, let itemIndex = parent.children.index(of: item) {
  249. // parent.removeObject(fromChildrenAtIndex: itemIndex)
  250. // }
  251. // }
  252. // }
  253. //
  254. // @IBAction func openBookmarks(_ sender: Any) {
  255. // guard let items = clickedBookmarks() as? [KMBookmark] else { return }
  256. // for item in items.reversed() {
  257. // item.open()
  258. // }
  259. // }
  260. //
  261. // @IBAction func previewBookmarks(_ sender: Any) {
  262. // if QLPreviewPanel.sharedPreviewPanelExists() && QLPreviewPanel.shared().isVisible {
  263. // QLPreviewPanel.shared().orderOut(nil)
  264. // } else if let row = outlineView.clickedRow {
  265. // outlineView.selectRowIndexes(IndexSet(integer: row), byExtendingSelection: false)
  266. // QLPreviewPanel.shared().makeKeyAndOrderFront(nil)
  267. // }
  268. // }
  269. //
  270. //
  271. // // MARK: - NSMenu delegate methods
  272. // func addItemForBookmark(_ bookmark: KMBookmark, toMenu menu: NSMenu, isFolder: Bool, isAlternate: Bool) {
  273. // var item: NSMenuItem?
  274. // if isFolder {
  275. // item = menu.addItem(withSubmenuAndTitle: bookmark.label)
  276. // item?.submenu?.delegate = self
  277. // } else {
  278. // item = menu.addItem(withTitle: bookmark.label, action: #selector(openBookmark(_:)), target: self)
  279. // }
  280. // item?.representedObject = bookmark
  281. // if isAlternate {
  282. // item?.keyEquivalentModifierMask = .alternate
  283. // item?.isAlternate = true
  284. // item?.setImageAndSize(bookmark.alternateIcon)
  285. // } else {
  286. // item?.setImageAndSize(bookmark.icon)
  287. // }
  288. // }
  289. //
  290. // func menuNeedsUpdate(_ menu: NSMenu) {
  291. // if menu == outlineView.menu {
  292. // let row = outlineView.clickedRow
  293. // menu.removeAllItems()
  294. // if row != -1 {
  295. // menu.addItem(withTitle: NSLocalizedString("Remove", comment: "Menu item title"), action: #selector(deleteBookmarks(_:)), target: self)
  296. // menu.addItem(withTitle: NSLocalizedString("Open", comment: "Menu item title"), action: #selector(openBookmarks(_:)), target: self)
  297. // menu.addItem(withTitle: NSLocalizedString("Quick Look", comment: "Menu item title"), action: #selector(previewBookmarks(_:)), target: self)
  298. // menu.addItem(.separator())
  299. // }
  300. // menu.addItem(withTitle: NSLocalizedString("New Folder", comment: "Menu item title"), action: #selector(insertBookmarkFolder(_:)), target: self)
  301. // menu.addItem(withTitle: NSLocalizedString("New Separator", comment: "Menu item title"), action: #selector(insertBookmarkSeparator(_:)), target: self)
  302. // } else {
  303. // guard let supermenu = menu.supermenu, let idx = supermenu.indexOfItem(withSubmenu: menu), let bm = (supermenu == NSApp.mainMenu) ? bookmarkRoot : supermenu.item(at: idx)?.representedObject as? KMBookmark else { return }
  304. //
  305. // let bookmarks = bm.children
  306. // var i = menu.numberOfItems
  307. //
  308. // while i > 0 {
  309. // if let menuItem = menu.item(at: i - 1), menuItem.isSeparatorItem || menuItem.representedObject != nil {
  310. // menu.removeItem(menuItem)
  311. // }
  312. // i -= 1
  313. // }
  314. //
  315. // if supermenu == NSApp.mainMenu, let previousSession = previousSession {
  316. // menu.addItem(.separator())
  317. // addItemForBookmark(previousSession, toMenu: menu, isFolder: false, isAlternate: false)
  318. // addItemForBookmark(previousSession, toMenu: menu, isFolder: true, isAlternate: true)
  319. // }
  320. //
  321. // if menu.numberOfItems > 0, bookmarks.count > 0 {
  322. // menu.addItem(.separator())
  323. // }
  324. //
  325. // for bm in bookmarks {
  326. // switch bm.bookmarkType {
  327. // case .folder:
  328. // addItemForBookmark(bm, toMenu: menu, isFolder: true, isAlternate: false)
  329. // addItemForBookmark(bm, toMenu: menu, isFolder: false, isAlternate: true)
  330. // case .session:
  331. // addItemForBookmark(bm, toMenu: menu, isFolder: false, isAlternate: false)
  332. // addItemForBookmark(bm, toMenu: menu, isFolder: true, isAlternate: true)
  333. // case .separator:
  334. // menu.addItem(.separator())
  335. // default:
  336. // addItemForBookmark(bm, toMenu: menu, isFolder: false, isAlternate: false)
  337. // }
  338. // }
  339. // }
  340. // }
  341. //
  342. // // avoid rebuilding the bookmarks menu on every key event
  343. // func menuHasKeyEquivalent(_ menu: NSMenu, for event: NSEvent, target: AutoreleasingUnsafeMutablePointer<AnyObject?>?, action: UnsafeMutablePointer<Selector?>?) -> Bool { false }
  344. //
  345. // // MARK: - Toolbar
  346. //
  347. func setupToolbar() {
  348. // Create a new toolbar instance, and attach it to our document window
  349. let toolbar = NSToolbar(identifier: kBookmarksToolbarIdentifier)
  350. var dict = [String: NSToolbarItem]()
  351. // Set up toolbar properties: Allow customization, give a default display mode, and remember state in user defaults
  352. toolbar.allowsUserCustomization = true
  353. toolbar.autosavesConfiguration = true
  354. toolbar.displayMode = .default
  355. // We are the delegate
  356. toolbar.delegate = self
  357. // Add template toolbar items
  358. var item = NSToolbarItem(itemIdentifier: NSToolbarItem.Identifier(kBookmarksNewFolderToolbarItemIdentifier))
  359. item.label = NSLocalizedString("New Folder", comment: "Toolbar item label")
  360. item.paletteLabel = NSLocalizedString("New Folder", comment: "Toolbar item label")
  361. item.toolTip = NSLocalizedString("Add a New Folder", comment: "Tool tip message")
  362. // item.image = NSImage(named: "NewFolder")
  363. item.image = NSImage(named: NSImage.folderName)!
  364. item.target = self
  365. item.action = #selector(insertBookmarkFolder(_:))
  366. dict[kBookmarksNewFolderToolbarItemIdentifier] = item
  367. item = NSToolbarItem(itemIdentifier: NSToolbarItem.Identifier(kBookmarksNewSeparatorToolbarItemIdentifier))
  368. item.label = NSLocalizedString("New Separator", comment: "Toolbar item label")
  369. item.paletteLabel = NSLocalizedString("New Separator", comment: "Toolbar item label")
  370. item.toolTip = NSLocalizedString("Add a New Separator", comment: "Tool tip message")
  371. // item.image = NSImage(named: "NewSeparator")
  372. item.image = NSImage(named: NSImage.shareTemplateName)!
  373. item.target = self
  374. item.action = #selector(insertBookmarkSeparator(_:))
  375. dict[kBookmarksNewSeparatorToolbarItemIdentifier] = item
  376. item = NSToolbarItem(itemIdentifier: NSToolbarItem.Identifier(kBookmarksDeleteToolbarItemIdentifier))
  377. item.label = NSLocalizedString("Delete", comment: "Toolbar item label")
  378. item.paletteLabel = NSLocalizedString("Delete", comment: "Toolbar item label")
  379. item.toolTip = NSLocalizedString("Delete Selected Items", comment: "Tool tip message")
  380. item.image = NSWorkspace.shared.icon(forFileType: NSFileTypeForHFSTypeCode(OSType(kToolbarDeleteIcon)))
  381. item.target = self
  382. item.action = #selector(deleteBookmark(_:))
  383. dict[kBookmarksDeleteToolbarItemIdentifier] = item
  384. toolbarItems = dict
  385. // Attach the toolbar to the window
  386. self.window?.toolbar = toolbar
  387. }
  388. //
  389. // // MARK: - Quick Look Panel Support
  390. //
  391. // func acceptsPreviewPanelControl(_ panel: QLPreviewPanel) -> Bool {
  392. // return true
  393. // }
  394. //
  395. // func beginPreviewPanelControl(_ panel: QLPreviewPanel) {
  396. // panel.delegate = self
  397. // panel.dataSource = self
  398. // }
  399. //
  400. // func endPreviewPanelControl(_ panel: QLPreviewPanel) {
  401. // }
  402. //
  403. // func previewItems() -> [KMBookmark] {
  404. // var items = [KMBookmark]()
  405. //
  406. // outlineView.selectedRowIndexes.enumerated().forEach { (idx, _) in
  407. // if let item = outlineView.item(atRow: idx) as? KMBookmark {
  408. // if item.bookmarkType == .bookmark {
  409. // items.append(item)
  410. // } else if item.bookmarkType == .session {
  411. // items.append(contentsOf: item.children)
  412. // }
  413. // }
  414. // }
  415. // return items
  416. // }
  417. //
  418. // func numberOfPreviewItems(in panel: QLPreviewPanel) -> Int {
  419. // return previewItems().count
  420. // }
  421. //
  422. // func previewPanel(_ panel: QLPreviewPanel, previewItemAt anIndex: Int) -> QLPreviewItem {
  423. // return previewItems()[anIndex]
  424. // }
  425. //
  426. // func previewPanel(_ panel: QLPreviewPanel, sourceFrameOnScreenForPreviewItem item: QLPreviewItem) -> NSRect {
  427. // var item = item
  428. // if let parent = (item as? KMBookmark)?.parent, parent.bookmarkType == .session {
  429. // item = parent
  430. // }
  431. // let row = outlineView.row(forItem: item)
  432. // var iconRect = NSZeroRect
  433. // if let item = item as? KMBookmark, row != -1 {
  434. // let cell = outlineView.preparedCell(atColumn: 0, row: row) as? SKTextWithIconCell
  435. // iconRect = cell?.iconRect(forBounds: outlineView.frameOfCell(atColumn: 0, row: row)) ?? NSZeroRect
  436. // if outlineView.visibleRect.intersects(iconRect) {
  437. // iconRect = outlineView.convert(iconRect, to: nil)
  438. // } else {
  439. // iconRect = NSZeroRect
  440. // }
  441. // }
  442. // return iconRect
  443. // }
  444. //
  445. // func previewPanel(_ panel: QLPreviewPanel, transitionImageForPreviewItem item: QLPreviewItem, contentRect: UnsafeMutablePointer<NSRect>) -> NSImage? {
  446. // var item = item
  447. // if let parent = (item as? KMBookmark)?.parent, parent.bookmarkType == .session {
  448. // item = parent
  449. // }
  450. // return (item as? KMBookmark)?.icon
  451. // }
  452. //
  453. // func previewPanel(_ panel: QLPreviewPanel, handle event: NSEvent) -> Bool {
  454. // if event.type == .keyDown {
  455. // outlineView.keyDown(with: event)
  456. // return true
  457. // }
  458. // return false
  459. // }
  460. //
  461. }
  462. extension KMBookmarkController: NSOutlineViewDelegate, NSOutlineViewDataSource {
  463. //MARK: NSOutlineViewDataSource
  464. func minimumCoverForBookmarks(_ items: [KMBookmark]) -> [KMBookmark] {
  465. var lastBm: KMBookmark?
  466. var minimalCover = [KMBookmark]()
  467. for bm in items {
  468. if !(bm.isDescendant(of: lastBm)) {
  469. minimalCover.append(bm)
  470. lastBm = bm
  471. }
  472. }
  473. return minimalCover
  474. }
  475. func outlineView(_ ov: NSOutlineView, numberOfChildrenOfItem item: Any?) -> Int {
  476. let bookmark = item as? KMBookmark ?? bookmarkRoot
  477. return bookmark.bookmarkType == .folder ? bookmark.children.count : 0
  478. }
  479. func outlineView(_ ov: NSOutlineView, isItemExpandable item: Any) -> Bool {
  480. let bookmark = item as! KMBookmark
  481. return bookmark.bookmarkType == .folder
  482. }
  483. func outlineView(_ ov: NSOutlineView, child index: Int, ofItem item: Any?) -> Any {
  484. let bookmark = (item as? KMBookmark) ?? bookmarkRoot
  485. return bookmark.objectOfChidren(index: index)
  486. }
  487. func outlineView(_ ov: NSOutlineView, objectValueFor tableColumn: NSTableColumn?, byItem item: Any?) -> Any? {
  488. guard let column = tableColumn else { return nil }
  489. guard let bm = item as? KMBookmark else { return nil }
  490. let tcID = column.identifier
  491. switch tcID {
  492. case kLabelIdentifier:
  493. return [kTextWithIconStringKey: bm.label, kTextWithIconImageKey: bm.icon]
  494. case kFileIdentifier:
  495. if bm.bookmarkType == .folder || bm.bookmarkType == .session {
  496. let count = bm.children.count
  497. return count == 1 ? NSLocalizedString("1 item", comment: "Bookmark folder description") : String.localizedStringWithFormat(NSLocalizedString("%ld items", comment: "Bookmark folder description"), count)
  498. } else {
  499. return bm.fileURL?.path ?? ""
  500. }
  501. case kPageIdentifier:
  502. return bm.pageNumber
  503. default:
  504. return nil
  505. }
  506. }
  507. func outlineView(_ ov: NSOutlineView, setObjectValue object: Any?, for tableColumn: NSTableColumn?, byItem item: Any?) {
  508. guard let column = tableColumn else { return }
  509. guard let bm = item as? KMBookmark else { return }
  510. let tcID = column.identifier
  511. switch tcID {
  512. case kLabelIdentifier:
  513. if let newLabel = (object as? [String: Any])?[kTextWithIconStringKey] as? String, newLabel != bm.label {
  514. bm.label = newLabel
  515. }
  516. case kPageIdentifier:
  517. if let newPageNumber = object as? Int, newPageNumber != bm.pageNumber.intValue {
  518. bm.pageNumber = newPageNumber as NSNumber
  519. }
  520. default:
  521. break
  522. }
  523. }
  524. func outlineView(_ ov: NSOutlineView, writeItems items: [Any], to pboard: NSPasteboard) -> Bool {
  525. draggedBookmarks = minimumCoverForBookmarks(items as! [KMBookmark])
  526. pboard.clearContents()
  527. pboard.setData(Data(), forType: kPasteboardTypeBookmarkRows)
  528. return true
  529. }
  530. func outlineView(_ ov: NSOutlineView, validateDrop info: NSDraggingInfo, proposedItem item: Any?, proposedChildIndex index: Int) -> NSDragOperation {
  531. guard index != NSOutlineViewDropOnItemIndex else { return [] }
  532. let pboard = info.draggingPasteboard
  533. if pboard.canReadItem(withDataConformingToTypes: [kPasteboardTypeBookmarkRows.rawValue]) && info.draggingSource as? NSOutlineView == ov {
  534. return .move
  535. } else if NSURL.canReadFileURL(from: pboard) {
  536. return .every
  537. }
  538. return []
  539. }
  540. func outlineView(_ ov: NSOutlineView, acceptDrop info: NSDraggingInfo, item: Any?, childIndex index: Int) -> Bool {
  541. let pboard = info.draggingPasteboard
  542. if pboard.canReadItem(withDataConformingToTypes: [kPasteboardTypeBookmarkRows.rawValue]) && info.draggingSource as? NSOutlineView == ov {
  543. var movedBookmarks = [KMBookmark]()
  544. var indexes = IndexSet()
  545. var insertionIndex = index
  546. let targetItem = item as? KMBookmark ?? bookmarkRoot
  547. for bookmark in draggedBookmarks {
  548. guard let parent = bookmark.parent else { continue }
  549. guard let bookmarkIndex = parent.children.firstIndex(of: bookmark) else { continue }
  550. if targetItem == parent {
  551. if insertionIndex > bookmarkIndex {
  552. insertionIndex -= 1
  553. }
  554. if insertionIndex == bookmarkIndex {
  555. continue
  556. }
  557. }
  558. parent.removeObjectFromChildren(index: bookmarkIndex)
  559. targetItem.insert(child: bookmark, atIndex: insertionIndex)
  560. movedBookmarks.append(bookmark)
  561. insertionIndex += 1
  562. }
  563. for bookmark in movedBookmarks {
  564. let row = ov.row(forItem: bookmark)
  565. if row != -1 {
  566. indexes.insert(row)
  567. }
  568. }
  569. if !indexes.isEmpty {
  570. ov.selectRowIndexes(indexes, byExtendingSelection: false)
  571. }
  572. return true
  573. } else {
  574. let urls = NSURL.readFileURLs(from: pboard)
  575. let newBookmarks = KMBookmark.bookmarks(urls: urls)
  576. if !newBookmarks.isEmpty {
  577. var indexes = IndexSet(integersIn: index..<(index + newBookmarks.count))
  578. (item as? KMBookmark ?? bookmarkRoot).mutableArrayValue(forKey: "children").insert(newBookmarks, at: indexes)
  579. if (item as? KMBookmark ?? bookmarkRoot) === bookmarkRoot || ov.isItemExpanded(item) {
  580. if (item as? KMBookmark ?? bookmarkRoot) !== bookmarkRoot {
  581. indexes.shift(startingAt: 0, by: ov.row(forItem: item) + 1)
  582. }
  583. ov.selectRowIndexes(indexes, byExtendingSelection: false)
  584. }
  585. return true
  586. }
  587. return false
  588. }
  589. }
  590. func outlineView(_ ov: NSOutlineView, dragEndedWith operation: NSDragOperation) {
  591. draggedBookmarks.removeAll()
  592. }
  593. // MARK: NSOutlineViewDelegate
  594. func outlineView(_ ov: NSOutlineView, dataCellFor tableColumn: NSTableColumn?, item: Any) -> Any? {
  595. if tableColumn == nil {
  596. return (item as? KMBookmark)?.bookmarkType == .separator ? KMSeparatorCell() : nil
  597. }
  598. return tableColumn?.dataCell(forRow: ov.row(forItem: item))
  599. }
  600. func outlineView(_ ov: NSOutlineView, willDisplayCell cell: Any, for tableColumn: NSTableColumn?, item: Any) {
  601. guard let column = tableColumn else { return }
  602. guard let cell = cell as? NSCell else { return }
  603. if column.identifier == kFileIdentifier {
  604. if let bm = item as? KMBookmark {
  605. // if bm.bookmarkType == .folder || bm.bookmarkType == .session {
  606. // cell.textColor = .disabledControlTextColor
  607. // } else {
  608. // cell.textColor = .controlTextColor
  609. // }
  610. }
  611. }
  612. }
  613. func outlineView(_ ov: NSOutlineView, shouldEdit tableColumn: NSTableColumn?, item: Any) -> Bool {
  614. guard let column = tableColumn else { return false }
  615. guard let bm = item as? KMBookmark else { return false }
  616. let tcID = column.identifier
  617. switch tcID {
  618. case kLabelIdentifier:
  619. return bm.bookmarkType != .separator
  620. case kPageIdentifier:
  621. return bm.pageIndex != NSNotFound
  622. default:
  623. return false
  624. }
  625. }
  626. func outlineView(_ ov: NSOutlineView, toolTipFor cell: NSCell, rect: UnsafeMutablePointer<NSRect>, tableColumn tc: NSTableColumn?, item: Any, mouseLocation: NSPoint) -> String {
  627. guard let column = tc else { return "" }
  628. guard let bm = item as? KMBookmark else { return "" }
  629. let tcID = column.identifier
  630. switch tcID {
  631. case kLabelIdentifier:
  632. return bm.label
  633. case kFileIdentifier:
  634. if bm.bookmarkType == .session {
  635. return ""
  636. // return bm.children.map { $0.path ?? "" }.joined(separator: "\n")
  637. } else if bm.bookmarkType == .folder {
  638. let count = bm.children.count
  639. return count == 1 ? NSLocalizedString("1 item", comment: "Bookmark folder description") : String.localizedStringWithFormat(NSLocalizedString("%ld items", comment: "Bookmark folder description"), count)
  640. } else {
  641. return bm.fileURL?.path ?? ""
  642. }
  643. case kPageIdentifier:
  644. return bm.pageNumber.stringValue
  645. default:
  646. return ""
  647. }
  648. }
  649. func outlineViewSelectionDidChange(_ notification: Notification) {
  650. updateStatus()
  651. if QLPreviewPanel.sharedPreviewPanelExists(), let previewPanel = QLPreviewPanel.shared(), previewPanel.isVisible, previewPanel.dataSource === self {
  652. previewPanel.reloadData()
  653. }
  654. }
  655. func outlineView(_ ov: NSOutlineView, deleteItems items: [Any]) {
  656. for item in minimumCoverForBookmarks(items as! [KMBookmark]).reversed() {
  657. guard let parent = item.parent, let itemIndex = parent.children.firstIndex(of: item) else { continue }
  658. parent.removeObjectFromChildren(index: itemIndex)
  659. }
  660. }
  661. func outlineView(_ ov: NSOutlineView, canDeleteItems items: [Any]) -> Bool {
  662. return !items.isEmpty
  663. }
  664. func outlineView(_ ov: NSOutlineView, copyItems items: [Any]) {
  665. var urls = [URL]()
  666. addBookmarkURLsToArray(minimumCoverForBookmarks(items as! [KMBookmark]), &urls)
  667. if !urls.isEmpty {
  668. let pboard = NSPasteboard.general
  669. pboard.clearContents()
  670. pboard.writeObjects(urls as [NSPasteboardWriting])
  671. }
  672. }
  673. func outlineView(_ ov: NSOutlineView, canCopyItems items: [Any]) -> Bool {
  674. return !items.isEmpty
  675. }
  676. func outlineView(_ ov: NSOutlineView, pasteFromPasteboard pboard: NSPasteboard) {
  677. let urls = NSURL.readFileURLs(from: pboard)
  678. let newBookmarks = KMBookmark.bookmarks(urls: urls)
  679. if !newBookmarks.isEmpty {
  680. var item: KMBookmark?
  681. var anIndex = 0
  682. getInsertionFolder(&item, childIndex: &anIndex)
  683. var indexes = IndexSet(integersIn: anIndex..<(anIndex + newBookmarks.count))
  684. (item ?? bookmarkRoot).mutableArrayValue(forKey: "children").insert(newBookmarks, at: indexes)
  685. if item === bookmarkRoot || ov.isItemExpanded(item) {
  686. if item !== bookmarkRoot {
  687. indexes.shift(startingAt: 0, by: ov.row(forItem: item) + 1)
  688. }
  689. ov.selectRowIndexes(indexes, byExtendingSelection: false)
  690. }
  691. }
  692. }
  693. func outlineView(_ ov: NSOutlineView, canPasteFromPasteboard pboard: NSPasteboard) -> Bool {
  694. return NSURL.canReadFileURL(from: pboard)
  695. }
  696. func outlineView(_ ov: NSOutlineView, typeSelectHelperSelectionStrings typeSelectHelper: SKTypeSelectHelper) -> [String] {
  697. let count = ov.numberOfRows
  698. var labels = [String]()
  699. for i in 0..<count {
  700. if let label = ov.item(atRow: i) {
  701. labels.append(label as! String)
  702. }
  703. }
  704. return labels
  705. }
  706. // func outlineView(_ ov: NSOutlineView, typeSelectHelper typeSelectHelper: SKTypeSelectHelper, didFailToFindMatchForSearchString searchString: String) {
  707. // statusBar.setLeftStringValue(String.localizedStringWithFormat(NSLocalizedString("No match: \"%@\"", comment: "Status message"), searchString))
  708. // }
  709. //
  710. // func outlineView(_ ov: NSOutlineView, typeSelectHelper typeSelectHelper: SKTypeSelectHelper, updateSearchString searchString: String?) {
  711. // if let searchString = searchString {
  712. // statusBar.setLeftStringValue(String.localizedStringWithFormat(NSLocalizedString("Finding: \"%@\"", comment: "Status message"), searchString))
  713. // } else {
  714. // updateStatus()
  715. // }
  716. // }
  717. func addBookmarkURLsToArray(_ items: [KMBookmark], _ array: inout [URL]) {
  718. for bm in items {
  719. if bm.bookmarkType == .bookmark {
  720. if let url = bm.fileURL {
  721. array.append(url)
  722. }
  723. } else if bm.bookmarkType != .separator {
  724. addBookmarkURLsToArray(bm.children, &array)
  725. }
  726. }
  727. }
  728. }
  729. extension KMBookmarkController: NSToolbarDelegate, NSToolbarItemValidation {
  730. func toolbarDefaultItemIdentifiers(_ toolbar: NSToolbar) -> [NSToolbarItem.Identifier] {
  731. // return [.flexibleSpace, .yourItem1, .yourItem2, .yourItem3]
  732. return [
  733. NSToolbarItem.Identifier(kBookmarksNewFolderToolbarItemIdentifier),
  734. NSToolbarItem.Identifier(kBookmarksNewSeparatorToolbarItemIdentifier),
  735. NSToolbarItem.Identifier(kBookmarksDeleteToolbarItemIdentifier)
  736. ]
  737. }
  738. func toolbarAllowedItemIdentifiers(_ toolbar: NSToolbar) -> [NSToolbarItem.Identifier] {
  739. // return [.yourItem1, .yourItem2, .yourItem3, .flexibleSpace, .space]
  740. return [
  741. NSToolbarItem.Identifier(kBookmarksNewFolderToolbarItemIdentifier),
  742. NSToolbarItem.Identifier(kBookmarksNewSeparatorToolbarItemIdentifier),
  743. NSToolbarItem.Identifier(kBookmarksDeleteToolbarItemIdentifier),
  744. .flexibleSpace,
  745. .space
  746. ]
  747. }
  748. func toolbar(_ toolbar: NSToolbar, itemForItemIdentifier itemIdentifier: NSToolbarItem.Identifier, willBeInsertedIntoToolbar flag: Bool) -> NSToolbarItem? {
  749. return toolbarItems[itemIdentifier.rawValue]
  750. }
  751. func validateToolbarItem(_ item: NSToolbarItem) -> Bool {
  752. // guard let toolbar = self.window?.toolbar else { return false }
  753. //
  754. // if toolbar.customizationPaletteIsRunning {
  755. // return false
  756. // } else if toolbarItem.itemIdentifier == kBookmarksDeleteToolbarItemIdentifier {
  757. // return outlineView.canDelete
  758. // }
  759. return true
  760. }
  761. }
  762. extension KMBookmarkController: NSMenuDelegate, NSMenuItemValidation {
  763. func validateMenuItem(_ menuItem: NSMenuItem) -> Bool {
  764. // if menuItem.action == #selector(toggleStatusBar(_:)) {
  765. // if statusBar.isVisible {
  766. // menuItem.title = NSLocalizedString("Hide Status Bar", comment: "Menu item title")
  767. // } else {
  768. // menuItem.title = NSLocalizedString("Show Status Bar", comment: "Menu item title")
  769. // }
  770. // return true
  771. // } else if menuItem.action == #selector(addBookmark(_:)) {
  772. // return menuItem.tag == 0
  773. // }
  774. return true
  775. }
  776. }
  777. extension NSURL {
  778. static func canReadFileURL(from pboard: NSPasteboard) -> Bool {
  779. let canReadFileURLsOnly = [NSPasteboard.ReadingOptionKey.urlReadingFileURLsOnly: true]
  780. let canReadClasses = [NSURL.self]
  781. return pboard.canReadObject(forClasses: canReadClasses, options: canReadFileURLsOnly) ||
  782. pboard.canReadItem(withDataConformingToTypes: [NSPasteboard.PasteboardType.fileURL.rawValue])
  783. }
  784. static func readFileURLs(from pboard: NSPasteboard) -> [URL] {
  785. if let fileURLs = pboard.readObjects(forClasses: [NSURL.self], options: [.urlReadingFileURLsOnly: true]) as? [URL], !fileURLs.isEmpty {
  786. return fileURLs
  787. } else if ((pboard.types?.contains(.fileURL)) != nil) {
  788. if let filenames = pboard.propertyList(forType: .fileURL) as? [String] {
  789. return filenames.compactMap { URL(fileURLWithPath: $0) }
  790. }
  791. }
  792. return []
  793. }
  794. }