KMLeftSideViewController+Note.swift 59 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361
  1. //
  2. // KMLeftSideViewController+Note.swift
  3. // PDF Reader Pro
  4. //
  5. // Created by tangchao on 2023/12/23.
  6. //
  7. import Foundation
  8. extension KMLeftSideViewController.Key {
  9. static let noteAscendSortKey = "KMLeftSideViewAscendSortBoolKey"
  10. static let noteSortTypeKey = "KMLeftSideViewNoteSortTypeKey"
  11. static let noteTableColumn = "KMNoteOutlineViewTableColumnKey"
  12. static let noteFilterPage = "kKMNoteFilterAnnotationPageKey"
  13. static let noteFilterTime = "kKMNoteFilterAnnotationTimeKey"
  14. static let noteFilterAuther = "kKMNoteFilterAnnotationAutherKey"
  15. }
  16. // MARK: - Action
  17. extension KMLeftSideViewController {
  18. func note_initSubViews() {
  19. self.noteSearchField.backgroundColor = KMAppearance.Layout.l_1Color()
  20. self.noteSearchField.wantsLayer = true
  21. self.noteSearchField.layer?.backgroundColor = KMAppearance.Layout.l_1Color().cgColor
  22. self.noteSearchField.layer?.borderWidth = 1.0
  23. self.noteMoreButton.target = self
  24. self.noteMoreButton.tag = 304
  25. self.noteMoreButton.action = #selector(leftSideViewMoreButtonAction)
  26. self.moreButtonLayer = KMButtonLayer()
  27. self.noteMoreButton.layer?.addSublayer(self.moreButtonLayer!)
  28. self.moreButtonLayer?.frame = NSMakeRect(0, 0, NSWidth(self.noteMoreButton.bounds), NSHeight(self.noteMoreButton.bounds))
  29. self.noteFilterButton.target = self
  30. self.noteFilterButton.action = #selector(noteFilterAction)
  31. self.filterButtonLayer = NSView()
  32. self.noteFilterButton.addSubview(self.filterButtonLayer!)
  33. self.filterButtonLayer?.frame = NSMakeRect(14, 2, 8, 8)
  34. self.noteDoneButton.action = #selector(leftSideViewDoneButtonAction)
  35. self.noteDoneButton.target = self
  36. self.noteDoneButton.tag = 311
  37. self.noteDoneButton.isHidden = true
  38. self.noteSearchField.delegate = self
  39. self.noteSearchField.isHidden = true
  40. self.noteSearchField.endEditCallBack = { [weak self] isEndEdit in
  41. // if (isEndEdit) {
  42. // self.noteSearchField.isHidden = true
  43. // self.noteSearchButton.isHidden = false
  44. // self.noteTitleLabel.isHidden = false
  45. // }
  46. }
  47. self.sortTypeBox.downCallback = { [unowned self] downEntered, mouseBox, _ in
  48. if (downEntered) {
  49. let menu = NSMenu()
  50. let timeItem = menu.addItem(title: KMLocalizedString("Time", nil), action: #selector(sortTypeAction), target: self)
  51. timeItem?.representedObject = self
  52. timeItem?.tag = 0
  53. let pageItem = menu.addItem(title: KMLocalizedString("Page", nil), action: #selector(sortTypeAction), target: self)
  54. pageItem?.representedObject = self
  55. timeItem?.tag = 1
  56. if (self.noteSortType == .time) {
  57. timeItem?.state = .on
  58. pageItem?.state = .off
  59. } else if (self.noteSortType == .page) {
  60. timeItem?.state = .off
  61. pageItem?.state = .on
  62. }
  63. menu.popUp(positioning: nil, at: NSMakePoint(-10, 0), in: self.sortTypeBox)
  64. }
  65. }
  66. self.noteOutlineView.delegate = self
  67. self.noteOutlineView.dataSource = self
  68. self.noteOutlineView.botaDelegate = self
  69. self.noteOutlineView.botaDataSource = self
  70. self.noteOutlineView.noteDelegate = self
  71. self.noteOutlineView.menu = NSMenu()
  72. self.noteOutlineView.menu?.delegate = self
  73. self.noteOutlineView.typeSelectHelper = SKTypeSelectHelper(matchOption: .SKSubstringMatch)
  74. self.noteOutlineView.enclosingScrollView?.scrollerStyle = .legacy
  75. self.noteOutlineView.enclosingScrollView?.autohidesScrollers = true
  76. self.noteOutlineView.registerForDraggedTypes(NSColor.readableTypes(for: NSPasteboard(name: .drag)))
  77. self.noteOutlineView.target = self
  78. self.noteOutlineView.doubleAction = #selector(selectSelectedNote)
  79. }
  80. func note_initDefalutValue() {
  81. self.noteView.wantsLayer = true
  82. self.noteView.layer?.backgroundColor = KMAppearance.Layout.l0Color().cgColor
  83. let sud = UserDefaults.standard
  84. if let dict = sud.dictionary(forKey: Self.Key.noteTableColumn) {
  85. self.noteTypeDict = dict
  86. } else {
  87. self.noteTypeDict = [Self.Key.noteFilterPage : false,
  88. Self.Key.noteFilterTime : false,
  89. Self.Key.noteFilterAuther : false]
  90. sud.sync_setValue(self.noteTypeDict, forKey: Self.Key.noteTableColumn)
  91. }
  92. self.caseInsensitiveNoteSearch = sud.bool(forKey: SKCaseInsensitiveNoteSearchKey)
  93. self.isAscendSort = KMDataManager.ud_bool(forKey: Self.Key.noteAscendSortKey)
  94. self.noteTitleLabel.stringValue = KMLocalizedString("Notes", nil);
  95. self.noteTitleLabel.textColor = KMAppearance.Layout.h0Color()
  96. self.noteSearchField.layer?.borderColor = KMAppearance.Interactive.a0Color().cgColor
  97. self.noteMoreButton.wantsLayer = true
  98. self.moreButtonLayer?.layerType = .none
  99. self.moreButtonLayer?.isHidden = true
  100. self.noteFilterButton.toolTip = KMLocalizedString("Sort", nil)
  101. self.noteFilterButton.wantsLayer = true
  102. self.filterButtonLayer?.isHidden = true
  103. self.filterButtonLayer?.wantsLayer = true
  104. self.filterButtonLayer?.layer?.backgroundColor = KMAppearance.Interactive.a0Color().cgColor
  105. self.filterButtonLayer?.layer?.cornerRadius = 4.0
  106. if (self.isAscendSort) {
  107. self.noteSortButton.image = NSImage(named: KMImageNameBtnSidebarRankReverse)
  108. self.noteSortButton.toolTip = KMLocalizedString("ascending sort", nil)
  109. } else {
  110. self.noteSortButton.image = NSImage(named: KMImageNameBtnSidebarRankPositive)
  111. self.noteSortButton.toolTip = KMLocalizedString("descending sort", nil)
  112. }
  113. self.noteSearchButton.toolTip = KMLocalizedString("Search", nil)
  114. self.noteDoneButton.title = KMLocalizedString("Done", nil)
  115. self.noteDoneButton.toolTip = KMLocalizedString("Done", nil)
  116. self.noteDoneButton.setTitleColor(KMAppearance.Layout.w0Color())
  117. self.noteDoneButton.wantsLayer = true
  118. self.noteDoneButton.layer?.backgroundColor = KMAppearance.Interactive.a0Color().cgColor
  119. self.noteDoneButton.layer?.cornerRadius = 4.0
  120. // self.noteHeaderView.wantsLayer = true
  121. // self.noteHeaderView.layer?.backgroundColor = KMAppearance.Else.textTagColor().cgColor
  122. // self.noteHeaderView.layer?.cornerRadius = 1.0
  123. let sortType = KMDataManager.ud_integer(forKey: Self.Key.noteSortTypeKey)
  124. if (sortType == 1) {
  125. self.noteSortType = KMNoteSortType(rawValue: sortType) ?? .none
  126. if (self.noteSortType == .time) {
  127. self.sortTypeLabel.stringValue = KMLocalizedString("Time", nil)
  128. self.sortTypeBox.toolTip = KMLocalizedString("Time", nil)
  129. } else if (self.noteSortType == .page) {
  130. self.sortTypeLabel.stringValue = KMLocalizedString("Page", nil)
  131. self.sortTypeBox.toolTip = KMLocalizedString("Page", nil)
  132. }
  133. } else {
  134. self.noteSortType = .page
  135. self.sortTypeLabel.stringValue = KMLocalizedString("Page", nil)
  136. }
  137. self.sortTypeLabel.textColor = KMAppearance.Layout.h1Color()
  138. self.noteOutlineView.backgroundColor = KMAppearance.Layout.l0Color()
  139. self.noteOutlineView.autoresizesOutlineColumn = false
  140. self.noteOutlineView.indentationPerLevel = 0
  141. }
  142. func annoListIsShowPage() -> Bool {
  143. return !(self.noteTypeDict[Self.Key.noteFilterPage] as? Bool ?? false)
  144. }
  145. func annoListIsShowTime() -> Bool {
  146. return !(self.noteTypeDict[Self.Key.noteFilterTime] as? Bool ?? false)
  147. }
  148. func annoListIsShowAnther() -> Bool {
  149. return !(self.noteTypeDict[Self.Key.noteFilterAuther] as? Bool ?? false)
  150. }
  151. }
  152. // MARK: - Menu
  153. extension KMLeftSideViewController {
  154. func annoListMenu(_ menu: NSMenu) {
  155. var item: NSMenuItem?
  156. var items: NSArray?
  157. var rowIndexes = self.noteOutlineView.selectedRowIndexes
  158. let row = self.noteOutlineView.clickedRow
  159. if row == -1 {
  160. // _ = self._addExportPDFMenu(menu)
  161. _ = self._addDeleteAllAnnoItem(menu)
  162. return
  163. }
  164. if rowIndexes.contains(row) == false {
  165. rowIndexes = IndexSet(integer: row)
  166. }
  167. items = self.noteOutlineView.itemsAtRowIndexes(rowIndexes) as NSArray
  168. guard let model = self.fetchAnnoModel(for: row) else {
  169. return
  170. }
  171. let isFold = model.isFold()
  172. item = menu.addItem(title: KMLocalizedString("Expand", nil), action: #selector(unfoldNoteAction), target: self)
  173. item?.state = isFold ? .off : .on
  174. item?.representedObject = items
  175. item = menu.addItem(title: KMLocalizedString("Collapse", nil), action: #selector(foldNoteAction), target: self)
  176. item?.state = isFold ? .on : .off
  177. item?.representedObject = items
  178. menu.addItem(.separator())
  179. let hideNotes = self.hideNotes()
  180. // var theItems: [KMBotaAnnotationModel] = []
  181. // for data in items ?? [] {
  182. // if let annoModel = data as? KMBotaAnnotationModel {
  183. // theItems.append(annoModel)
  184. // }
  185. // }
  186. if hideNotes == false && (items?.count ?? 0) == 1 {
  187. // let annotation = self.noteItems(items!).lastObject as? CPDFAnnotation
  188. // if let data = annotation?.isEditable(), data {
  189. // if annotation?.type == nil {
  190. // let isNote = annotation?.isNote() ?? false
  191. // if isNote {
  192. // // [NSLocalizedString(@"Edit", @"Menu item title") stringByAppendingEllipsis]
  193. // item = menu.addItem(title: KMLocalizedString("Edit", "Menu item title"), action: #selector(editNoteTextFromTable), target: self)
  194. // item?.representedObject = annotation
  195. // }
  196. // } else if let data = self.noteOutlineView.tableColumn(withIdentifier: NSUserInterfaceItemIdentifier("note"))?.isHidden, data {
  197. // // [NSLocalizedString(@"Edit", @"Menu item title") stringByAppendingEllipsis]
  198. // item = menu.addItem(title: KMLocalizedString("Edit", "Menu item title"), action: #selector(editThisAnnotation), target: self)
  199. // item?.representedObject = annotation
  200. // } else {
  201. // item = menu.addItem(title: KMLocalizedString("Edit", "Menu item title"), action: #selector(editNoteFromTable), target: self)
  202. // item?.representedObject = annotation
  203. // item = menu.addItem(title: KMLocalizedString("Edit", "Menu item title"), action: #selector(editThisAnnotation), target: self)
  204. // item?.representedObject = annotation
  205. // item?.keyEquivalentModifierMask = [.option]
  206. // item?.isAlternate = true
  207. // }
  208. // }
  209. }
  210. if menu.numberOfItems > 0 {
  211. // _ = self._addExportPDFMenu(menu)
  212. // menu.addItem(.separator())
  213. if self.outlineView(self.noteOutlineView, canDeleteItems: items as? [Any] ?? []) {
  214. item = menu.addItem(title: KMLocalizedString("Delete", "Menu item title"), action: #selector(deleteNotes), target: self)
  215. item?.representedObject = items
  216. }
  217. _ = self._addDeleteAllAnnoItem(menu)
  218. }
  219. }
  220. private func _addExportPDFMenu(_ menu: NSMenu) -> NSMenu {
  221. var item = menu.addItem(title: NSLocalizedString("Export Annotations…", tableName: "", comment: ""), action: nil, target: self)
  222. let subMenu = NSMenu()
  223. item?.submenu = subMenu
  224. item = subMenu.addItem(title: NSLocalizedString("PDF", tableName: "", comment: ""), action: #selector(exportAnnotationNotes), target: self)
  225. item?.tag = 0
  226. item = subMenu.addItem(title: NSLocalizedString("PDF Bundle", tableName: "", comment: ""), action: #selector(exportAnnotationNotes), target: self)
  227. item?.tag = 1
  228. item = subMenu.addItem(title: NSLocalizedString("PDF Reader Pro Edition Notes", tableName: "", comment: ""), action: #selector(exportAnnotationNotes), target: self)
  229. item?.tag = 2
  230. item = subMenu.addItem(title: NSLocalizedString("Notes as Text", tableName: "", comment: ""), action: #selector(exportAnnotationNotes), target: self)
  231. item?.tag = 3
  232. item = subMenu.addItem(title: NSLocalizedString("Notes as RTF", tableName: "", comment: ""), action: #selector(exportAnnotationNotes), target: self)
  233. item?.tag = 4
  234. item = subMenu.addItem(title: NSLocalizedString("Notes as RTFD", tableName: "", comment: ""), action: #selector(exportAnnotationNotes), target: self)
  235. item?.tag = 5
  236. item = subMenu.addItem(title: NSLocalizedString("Notes as FDF", tableName: "", comment: ""), action: #selector(exportAnnotationNotes), target: self)
  237. item?.tag = 6
  238. return menu
  239. }
  240. private func _addDeleteAllAnnoItem(_ menu: NSMenu) -> NSMenuItem? {
  241. return menu.addItem(title: NSLocalizedString("Remove All Annotations", tableName: "", comment: ""), action: #selector(removeAllAnnotations), target: self)
  242. }
  243. private func _addDeleteAllReplyAnnoItem(_ menu: NSMenu) -> NSMenuItem? {
  244. return menu.addItem(title: NSLocalizedString("Delete All Reply", tableName: "", comment: ""), action: #selector(removeAllReplyAnnotations), target: self)
  245. }
  246. func annoListMoreMenu(_ view: NSButton) {
  247. let menu = NSMenu()
  248. let object = KMPopupMenuObject()
  249. object.menuTag = 1001
  250. menu.delegate = object
  251. object.enterControllerCallback = { [weak self] isEnter in
  252. if (isEnter) {
  253. self?.moreButtonLayer?.isHidden = false
  254. } else {
  255. self?.moreButtonLayer?.isHidden = true
  256. }
  257. }
  258. let expandAllItem = menu.addItem(title: KMLocalizedString("Expand All", nil), action: #selector(note_expandAllComments), target: self)
  259. expandAllItem?.representedObject = self.noteOutlineView
  260. let foldAllItem = menu.addItem(title: KMLocalizedString("Collapse All", nil), action: #selector(note_foldAllComments), target: self)
  261. foldAllItem?.representedObject = self.noteOutlineView
  262. let type = self.annoListModel?.foldType ?? .none
  263. expandAllItem?.state = type == .unfold ? .on : .off
  264. foldAllItem?.state = type == .fold ? .on : .off
  265. let showItem = menu.addItem(title: KMLocalizedString("Show Note", nil), action: nil, target: self)
  266. let subMenu = NSMenu()
  267. let pageItem = subMenu.addItem(title: KMLocalizedString("Page", nil), action: #selector(noteShowNoteAction), target: self)
  268. pageItem?.state = self.annoListIsShowPage() ? .on : .off
  269. pageItem?.representedObject = self.noteOutlineView
  270. pageItem?.tag = 101
  271. let timeItem = subMenu.addItem(title: KMLocalizedString("Time", nil) , action: #selector(noteShowNoteAction), target: self)
  272. timeItem?.state = self.annoListIsShowTime() ? .on : .off
  273. timeItem?.representedObject = self.noteOutlineView
  274. timeItem?.tag = 102
  275. let authorItem = subMenu.addItem(title: KMLocalizedString("Author", nil) , action: #selector(noteShowNoteAction), target: self)
  276. authorItem?.state = self.annoListIsShowAnther() ? .on : .off
  277. authorItem?.representedObject = self.noteOutlineView
  278. authorItem?.tag = 103
  279. showItem?.submenu = subMenu
  280. menu.addItem(.separator())
  281. // _ = self._addExportPDFMenu(menu)
  282. // menu.addItem(.separator())
  283. _ = self._addDeleteAllAnnoItem(menu)
  284. _ = self._addDeleteAllReplyAnnoItem(menu)
  285. menu.addItem(.separator())
  286. let importItem = NSMenuItem(title: NSLocalizedString("Import Annotations", comment: ""), action: #selector(importNotes), keyEquivalent: "")
  287. importItem.target = self
  288. menu.addItem(importItem)
  289. let exportItem = NSMenuItem(title: NSLocalizedString("Export Annotations", comment: ""), action: #selector(exportNotes), keyEquivalent: "")
  290. exportItem.target = self
  291. menu.addItem(exportItem)
  292. if let data = NSApp.currentEvent {
  293. NSMenu.popUpContextMenu(menu, with: data, for: view, with: nil)
  294. }
  295. }
  296. func annoListValidateMenuItem(_ menuItem: NSMenuItem) -> Bool {
  297. let action = menuItem.action
  298. if (action == #selector(note_expandAllComments) ||
  299. action == #selector(note_foldAllComments) ||
  300. action == #selector(exportAnnotationNotes) ||
  301. action == #selector(removeAllAnnotations)) {
  302. let cnt = self.annoListModel?.datas.count ?? 0
  303. return cnt > 0
  304. } else if (action == #selector(unfoldNoteAction) ||
  305. action == #selector(foldNoteAction)) {
  306. let row = self.noteOutlineView.clickedRow
  307. let foldNote = self.fetchNote(for: row)
  308. // SKNPDFAnnotationNote
  309. if foldNote is CPDFMarkupAnnotation || foldNote is CPDFTextAnnotation {
  310. return true
  311. } else {
  312. return false
  313. }
  314. } else if (action == #selector(editNoteFromTable)) {
  315. let row = self.noteOutlineView.clickedRow
  316. let foldNote = self.fetchNote(for: row)
  317. // if (@available(macOS 10.13, *)) {
  318. // if ([foldNote.widgetFieldType isEqualToString:PDFAnnotationWidgetSubtypeSignature]) {
  319. // return NO;
  320. // }
  321. // }
  322. if foldNote is CPDFStampAnnotation || foldNote is KMAnnotationStamp || foldNote is CPDFListStampAnnotation {
  323. return false
  324. } else {
  325. return true
  326. }
  327. }
  328. return true
  329. }
  330. @IBAction func note_expandAllComments(_ sender: AnyObject?) {
  331. guard let model = self.annoListModel else {
  332. return
  333. }
  334. if (model.foldType == .unfold) { // 已全部展开
  335. return
  336. }
  337. model.foldType = .unfold
  338. for secM in self.annoListModel?.datas ?? [] {
  339. for model in secM.items {
  340. if let data = model as? KMBotaAnnotationFooterModel {
  341. data.isExpand = true
  342. }
  343. }
  344. }
  345. self.noteOutlineView.reloadData()
  346. }
  347. @IBAction func note_foldAllComments(_ sender: AnyObject?) {
  348. guard let model = self.annoListModel else {
  349. return
  350. }
  351. if (model.foldType == .fold) {
  352. return
  353. }
  354. model.foldType = .fold
  355. for secM in self.annoListModel?.datas ?? [] {
  356. for model in secM.items {
  357. if let data = model as? KMBotaAnnotationFooterModel {
  358. data.isExpand = false
  359. }
  360. }
  361. }
  362. self.noteOutlineView.reloadData()
  363. }
  364. @IBAction func noteShowNoteAction(_ sender: AnyObject?) {
  365. let item = sender as? NSMenuItem
  366. let tag = item?.tag ?? 0
  367. if (tag == 100) {
  368. } else if (tag == 101) {
  369. let isPage = !self.annoListIsShowPage()
  370. self.noteTypeDict[Self.Key.noteFilterPage] = !isPage
  371. } else if (tag == 102) {
  372. let isTime = !self.annoListIsShowTime()
  373. self.noteTypeDict[Self.Key.noteFilterTime] = !isTime
  374. } else if (tag == 103) {
  375. let isAuther = !self.annoListIsShowAnther()
  376. self.noteTypeDict[Self.Key.noteFilterAuther] = !isAuther
  377. }
  378. UserDefaults.standard.sync_setValue(self.noteTypeDict, forKey: Self.Key.noteTableColumn)
  379. // 更新数据
  380. var models: [KMBotaAnnotationModel] = []
  381. if self.noteSearchMode {
  382. models = self.noteSearchArray
  383. } else {
  384. let selModels = self.annoListModel?.datas ?? []
  385. for selModel in selModels {
  386. for item in selModel.items {
  387. if let data = item as? KMBotaAnnotationModel {
  388. models.append(data)
  389. }
  390. }
  391. }
  392. }
  393. for model in models {
  394. model.showPage = self.annoListIsShowPage()
  395. model.showTime = self.annoListIsShowTime()
  396. model.showAuthor = self.annoListIsShowAnther()
  397. }
  398. let selectRow = self.noteOutlineView.selectedRow
  399. self.noteOutlineView.reloadData()
  400. self.noteOutlineView.selectRowIndexes(IndexSet(integer: selectRow), byExtendingSelection: false)
  401. }
  402. @objc func importNotes(_ sender: NSMenuItem) {
  403. let panel = NSOpenPanel()
  404. panel.allowedFileTypes = ["xfdf"]
  405. panel.allowsMultipleSelection = false
  406. panel.beginSheetModal(for: self.view.window!) { resp in
  407. if resp != .OK {
  408. return
  409. }
  410. if let result = self.pdfDocument()?.importAnnotation(fromXFDFPath: panel.url?.path), result {
  411. self.mainViewController?.convertNotesUsingPDFDocument(self.pdfDocument()!, callback: { [weak self] in
  412. self?.reloadAnnotation()
  413. self?.listView?.setNeedsDisplayForVisiblePages()
  414. })
  415. }
  416. }
  417. }
  418. @objc func exportNotes(_ sender: NSMenuItem?) {
  419. guard let cnt = self.listView?.notes?.count, cnt > 0 else {
  420. NSSound.beep()
  421. return
  422. }
  423. let fileName = "\(self.pdfDocument()?.documentURL.deletingPathExtension().lastPathComponent ?? "")" + "_xfdf"
  424. let panel = NSSavePanel()
  425. panel.directoryURL = self.pdfDocument()?.documentURL.deletingLastPathComponent()
  426. panel.allowedFileTypes = ["xfdf"]
  427. panel.nameFieldStringValue = fileName
  428. panel.beginSheetModal(for: self.view.window!) { resp in
  429. if resp != .OK {
  430. return
  431. }
  432. let filePath = panel.url?.path
  433. if let success = self.pdfDocument()?.exportAnnotation(toXFDFPath: filePath), success {
  434. NSWorkspace.shared.selectFile(filePath, inFileViewerRootedAtPath: "")
  435. } else {
  436. Task {
  437. _ = await KMAlertTool.runModel(message: NSLocalizedString("Export Failure!", comment: ""), buttons: ["OK"])
  438. }
  439. }
  440. }
  441. }
  442. @objc func exportAnnotationNotes(_ sender: AnyObject?) {
  443. let doc = self.view.window?.windowController?.document as? NSDocument
  444. doc?.saveTo(sender)
  445. }
  446. // 展开
  447. @objc func unfoldNoteAction(_ sender: NSMenuItem) {
  448. if sender.state == .on {
  449. return
  450. }
  451. let row = self.noteOutlineView.clickedRow
  452. guard let model = self.fetchAnnoModel(for: row) else {
  453. return
  454. }
  455. model.foldType = .unfold
  456. let viewS = self.noteOutlineView.view(atColumn: 0, row: row, makeIfNecessary: true)
  457. (viewS as? KMNoteTableViewCell)?.isFold = false
  458. model.footerModel?.isExpand = true
  459. self.noteOutlineView.reloadItem(model.footerModel)
  460. }
  461. @objc func foldNoteAction(_ sender: NSMenuItem) {
  462. if sender.state == .on {
  463. return
  464. }
  465. let row = self.noteOutlineView.clickedRow
  466. guard let model = self.fetchAnnoModel(for: row) else {
  467. return
  468. }
  469. model.foldType = .fold
  470. let viewS = self.noteOutlineView.view(atColumn: 0, row: row, makeIfNecessary: true)
  471. (viewS as? KMNoteTableViewCell)?.isFold = true
  472. model.footerModel?.isExpand = false
  473. self.noteOutlineView.reloadItem(model.footerModel)
  474. }
  475. @objc func deleteNotes(_ sender: NSMenuItem) {
  476. self.outlineView(self.noteOutlineView, deleteItems: sender.representedObject as? [Any] ?? [])
  477. }
  478. @objc func removeAllAnnotations(_ sender: AnyObject?) {
  479. guard let doc = self.pdfDocument() else {
  480. return
  481. }
  482. Task {
  483. let response = await KMAlertTool.runModel(message: KMLocalizedString("This will permanently remove all annotations. Are you sure to continue?", nil), buttons: [KMLocalizedString("Yes", nil), KMLocalizedString("No", nil)])
  484. if response == .alertFirstButtonReturn {
  485. // var annos: [CPDFAnnotation] = []
  486. // for i in 0 ..< doc.pageCount {
  487. // let page = self.pdfDocument()?.page(at: i)
  488. // for anno in page?.annotations ?? [] {
  489. // if anno is CPDFTextWidgetAnnotation || anno is CPDFButtonWidgetAnnotation || anno is CPDFChoiceWidgetAnnotation {
  490. // continue
  491. // }
  492. // // if ([annotation.widgetFieldType isEqualToString:PDFAnnotationWidgetSubtypeSignature]) {
  493. // // continue;
  494. // // }
  495. // if anno is CPDFLinkAnnotation {
  496. // continue
  497. // }
  498. // annos.append(anno)
  499. // }
  500. // }
  501. self.dataUpdating = true
  502. // for anno in annos {
  503. // self.listView?.remove(anno)
  504. // }
  505. for model in self.annoListModel?.datas ?? [] {
  506. for item in model.items {
  507. if let anno = item.anno {
  508. self.listView?.remove(anno)
  509. }
  510. }
  511. }
  512. self.annoListModel?.datas.removeAll()
  513. self.dataUpdating = false
  514. self.note_refrshUIIfNeed()
  515. }
  516. }
  517. }
  518. @objc func removeAllReplyAnnotations(_ sender: NSMenuItem?) {
  519. Task {
  520. let response = await KMAlertTool.runModel(message: KMLocalizedString("Are you sure to delete all comment replies?", nil), buttons: [KMLocalizedString("Yes", nil), KMLocalizedString("No", nil)])
  521. if response == .alertFirstButtonReturn {
  522. self.dataUpdating = true
  523. var hasAnno = false
  524. for model in self.annoListModel?.datas ?? [] {
  525. for item in model.items {
  526. // if let anno = item.anno {
  527. // self.listView?.remove(anno)
  528. // }
  529. guard let annoM = item as? KMBotaAnnotationModel else {
  530. continue
  531. }
  532. for replyM in annoM.replyAnnos {
  533. // self.listView?.remove(replyM.replyAnno)
  534. replyM.replyAnno?.page.removeAnnotation(replyM.replyAnno)
  535. hasAnno = true
  536. }
  537. annoM.replyAnnos.removeAll()
  538. }
  539. }
  540. self.dataUpdating = false
  541. self.note_refrshUIIfNeed()
  542. if let data = self.view.window, hasAnno {
  543. KMTools.setDocumentEditedState(window: data)
  544. }
  545. }
  546. }
  547. }
  548. @objc func editNoteTextFromTable(_ sender: NSMenuItem) {
  549. // PDFAnnotation *annotation = [sender representedObject];
  550. guard let annotation = sender.representedObject as? CPDFAnnotation else {
  551. return
  552. }
  553. self.listView?.scrollAnnotationToVisible(annotation)
  554. // self.listView.activeAnnotation = annotation
  555. // [self showNote:annotation];
  556. // SKNoteWindowController *noteController = (SKNoteWindowController *)[self windowControllerForNote:annotation];
  557. // [[noteController window] makeFirstResponder:[noteController textView]];
  558. // [[noteController textView] selectAll:nil];
  559. }
  560. @objc func editThisAnnotation(_ sender: AnyObject?) {
  561. guard let annotation = (sender as? NSMenuItem)?.representedObject as? CPDFAnnotation else {
  562. NSSound.beep()
  563. return
  564. }
  565. self.listView?.edit(annotation)
  566. }
  567. @objc func editNoteFromTable(_ sender: AnyObject?) {
  568. guard let annotation = (sender as? NSMenuItem)?.representedObject as? CPDFAnnotation else {
  569. NSSound.beep()
  570. return
  571. }
  572. let model = fetchAnnoModel(for: annotation)
  573. let row = self.noteOutlineView.row(forItem: model)
  574. self.noteOutlineView.km_safe_selectRowIndexes(.init(integer: row), byExtendingSelection: false)
  575. let noteIndex = self.noteOutlineView.column(withIdentifier: .init("note"))
  576. if (noteIndex >= 0) {
  577. self.noteOutlineView.scrollColumnToVisible(noteIndex)
  578. //
  579. self.isRenameNoteOutline = true
  580. // self.renamePDFOutline = [rightSideController.noteOutlineView itemAtRow:rightSideController.noteOutlineView.clickedRow];
  581. let viewS = self.noteOutlineView.view(atColumn: 0, row: row, makeIfNecessary: true) as? KMNoteTableViewCell
  582. viewS!.isFold = false
  583. let targrtTextField = viewS?.noteContentLabel
  584. self.editNoteTextField = targrtTextField
  585. self.editNote = annotation
  586. targrtTextField?.delegate = self
  587. targrtTextField?.isEditable = true
  588. targrtTextField?.becomeFirstResponder()
  589. }
  590. }
  591. @IBAction func noteSortAction(_ sender: AnyObject?) {
  592. if (self.isAscendSort) {
  593. self.isAscendSort = false
  594. self.noteSortButton.image = NSImage(named: KMImageNameBtnSidebarRankPositive)
  595. self.noteSortButton.toolTip = KMLocalizedString("descending sort", nil)
  596. } else {
  597. self.isAscendSort = true
  598. self.noteSortButton.image = NSImage(named: KMImageNameBtnSidebarRankReverse)
  599. self.noteSortButton.toolTip = KMLocalizedString("ascending sort", nil)
  600. }
  601. KMDataManager.ud_set(self.isAscendSort, forKey: Self.Key.noteAscendSortKey)
  602. if self.noteSearchMode {
  603. self.reloadNoteForSearchMode()
  604. } else {
  605. self.reloadAnnotation()
  606. }
  607. }
  608. @IBAction func noteSearchAction(_ sender: NSButton) {
  609. self.noteSearchField.isHidden = false
  610. self.noteTitleLabel.isHidden = true
  611. self.noteSearchButton.isHidden = true
  612. self.noteDoneButton.isHidden = false
  613. self.noteFilterButton.isHidden = true
  614. self.noteMoreButton.isHidden = true
  615. self.noteSearchField.becomeFirstResponder()
  616. }
  617. @IBAction func noteFilterAction(_ sender: AnyObject?) {
  618. let button = sender as? NSButton
  619. let menu = NSMenu()
  620. let filterViewController = KMNoteOutlineFilterViewController()
  621. filterViewController.listView = self.listView
  622. filterViewController.view.layer?.backgroundColor = .clear
  623. var states: [CPDFAnnotationState] = [.none, .unMarked]
  624. for anno in self.allAnnotations {
  625. if let reviewS = self.noteReplyHanddler.fetchReviewState(anno) {
  626. if states.contains(reviewS) == false {
  627. states.append(reviewS)
  628. }
  629. }
  630. if let markS = self.noteReplyHanddler.fetchAnnoState(anno) {
  631. if states.contains(markS) == false {
  632. states.append(markS)
  633. }
  634. }
  635. }
  636. filterViewController.updateStates(states: states)
  637. filterViewController.setNotesArray(self.allAnnotations as NSArray)
  638. filterViewController.applyFilterCallback = { [weak self] typeArr, colorArr, authorArr, isEmpty in
  639. menu.cancelTracking()
  640. if (isEmpty) {
  641. self?.filterButtonLayer?.isHidden = true
  642. } else {
  643. self?.filterButtonLayer?.isHidden = false
  644. }
  645. self?.reloadAnnotation()
  646. }
  647. filterViewController.cancelCallback = { isCancel in
  648. if (isCancel) {
  649. menu.cancelTracking()
  650. }
  651. }
  652. let item = menu.addItem(withTitle: "", action: nil, keyEquivalent: "")
  653. item.target = self
  654. item.representedObject = filterViewController
  655. item.view = filterViewController.view
  656. menu.popUp(positioning: nil, at: NSMakePoint(-130, 30), in: button)
  657. // let win = KMNoteOutlineFilterViewController_window(contentRect: .zero, styleMask: .borderless, backing: .buffered, defer: false)
  658. // win.center()
  659. // win.contentViewController = filterViewController
  660. // win.orderFront(nil)
  661. }
  662. func fetchNote(for index: Int) -> CPDFAnnotation? {
  663. return self.fetchAnnoModel(for: index)?.anno
  664. }
  665. func fetchAnnoModel(for index: Int) -> KMBotaAnnotationModel? {
  666. if self.noteSearchMode { // 搜索模式
  667. return self.noteSearchArray.safe_element(for: index) as? KMBotaAnnotationModel
  668. } else { // 常规模式(非搜索)
  669. // let model = self.annoListModel?.datas.safe_element(for: index)
  670. let model = self.noteOutlineView.item(atRow: index)
  671. if let data = model as? KMBotaAnnotationSectionModel {
  672. return nil
  673. }
  674. if let data = model as? KMBotaAnnotationModel {
  675. return data
  676. }
  677. if let data = model as? KMBotaAnnotationFooterModel {
  678. // return data.annoModel
  679. }
  680. if let data = model as? KMBotaAnnotationReplyModel {
  681. // return data.annoModel
  682. }
  683. // return self.annoListModel?.datas.safe_element(for: index) as? KMBotaAnnotationModel
  684. return nil
  685. }
  686. }
  687. func fetchAnnoModel(for anno: CPDFAnnotation) -> KMBotaAnnotationModel? {
  688. if self.noteSearchMode { // 搜索模式
  689. for model in self.noteSearchArray {
  690. if anno.isEqual(to: model.anno) {
  691. return model
  692. }
  693. }
  694. } else { // 常规模式(非搜索)
  695. for model in self.annoListModel?.datas ?? [] {
  696. for item in model.items {
  697. if let data = item as? KMBotaAnnotationModel {
  698. if anno.isEqual(to: data.anno) {
  699. return data
  700. }
  701. }
  702. }
  703. }
  704. }
  705. return nil
  706. }
  707. @IBAction @objc func sortTypeAction(_ sender: NSMenuItem) {
  708. let item = sender
  709. let tag = item.tag
  710. if (item.state == .on) {
  711. item.state = .off
  712. } else {
  713. item.state = .on
  714. }
  715. if (tag == 0) {
  716. self.noteSortType = .page
  717. self.sortTypeLabel.stringValue = KMLocalizedString("Page", nil)
  718. self.sortTypeBox.toolTip = KMLocalizedString("Page", nil)
  719. } else if (tag == 1) {
  720. self.noteSortType = .time
  721. self.sortTypeLabel.stringValue = KMLocalizedString("Time", nil)
  722. self.sortTypeBox.toolTip = KMLocalizedString("Time", nil)
  723. }
  724. KMDataManager.ud_set(self.noteSortType.rawValue, forKey: Self.Key.noteSortTypeKey)
  725. if self.noteSearchMode {
  726. self.reloadNoteForSearchMode()
  727. } else {
  728. self.reloadAnnotation()
  729. }
  730. }
  731. func showNoteEmptyView() {
  732. let view = self.noteOutlineView.enclosingScrollView?.documentView
  733. let viewFrame = view?.frame ?? .zero
  734. let emptyVcSize = self.leftSideEmptyVC.emptyAnnotationView.frame.size
  735. self.leftSideEmptyVC.emptyAnnotationView.frame = NSMakeRect((viewFrame.size.width-emptyVcSize.width)/2.0,(viewFrame.size.height-emptyVcSize.height)/2.0, emptyVcSize.width, emptyVcSize.height)
  736. self.leftSideEmptyVC.emptyAnnotationView.autoresizingMask = [.minXMargin, .maxXMargin, .minYMargin, .maxYMargin]
  737. self.noteOutlineView.enclosingScrollView?.documentView?.addSubview(self.leftSideEmptyVC.emptyAnnotationView)
  738. self.leftSideEmptyVC.exportAnnotationBtn.isEnabled = false
  739. self.leftSideEmptyVC.deleteAnnotationBtn.isEnabled = false
  740. if (self.leftView.segmentedControl.selectedSegment == KMSelectedSegmentType.annotation.rawValue) {
  741. self.noteHeaderView.isHidden = true
  742. self.toolButtonBoxLayoutConstraint.constant = 40.0
  743. }
  744. }
  745. func hideNoteEmptyView() {
  746. self.leftSideEmptyVC.emptyAnnotationView.removeFromSuperview()
  747. self.leftSideEmptyVC.exportAnnotationBtn.isEnabled = true
  748. self.leftSideEmptyVC.deleteAnnotationBtn.isEnabled = true
  749. if (self.leftView.segmentedControl.selectedSegment == 3) {
  750. self.noteHeaderView.isHidden = false
  751. self.toolButtonBoxLayoutConstraint.constant = 64.0
  752. }
  753. }
  754. }
  755. // MARK: - Note
  756. extension KMLeftSideViewController {
  757. public func refreshUIForAddAnnotation(annos: [CPDFAnnotation]?, page: CPDFPage?) {
  758. let need = self._annoList_needRefreshUI(annos: annos)
  759. if need == false {
  760. return
  761. }
  762. self.updateThumbnail(at: Int(page?.pageIndex() ?? 0))
  763. self.note_reloadDataIfNeed()
  764. }
  765. public func refreshUIForAnnoAttributeDidChange(_ anno: CPDFAnnotation?, attributes: [String : Any]?) {
  766. self.updateThumbnail(at: Int(anno?.page?.pageIndex() ?? 0))
  767. let need = self._annoList_needRefreshUI(annos: anno != nil ? [anno!] : [])
  768. if need == false {
  769. return
  770. }
  771. if let data = anno {
  772. self.note_reloadDataForAnnoIfNeed(anno: data)
  773. }
  774. }
  775. public func annoList_refreshUIForDeleteAnnotations(annos: [CPDFAnnotation]?, page: CPDFPage?) {
  776. self.updateThumbnail(at: Int(page?.pageIndex() ?? 0))
  777. if self.type.methodType != .Annotation {
  778. return
  779. }
  780. let need = self._annoList_needRefreshUI(annos: annos)
  781. if need == false {
  782. return
  783. }
  784. for anno in annos ?? [] {
  785. if let model = self.fetchAnnoModel(for: anno) {
  786. self.noteSearchArray.removeObject(model)
  787. // self.annoListModel?.datas.removeObject(model)
  788. model.sectionModel?.items.removeObject(model)
  789. if let footer = model.footerModel {
  790. model.sectionModel?.items.removeObject(footer)
  791. }
  792. if self.allAnnotations.contains(anno) {
  793. self.allAnnotations.removeObject(anno)
  794. }
  795. }
  796. }
  797. if self.dataUpdating == false {
  798. self.note_refrshUIIfNeed()
  799. }
  800. }
  801. func note_refrshUIIfNeed() {
  802. if self.type.methodType != .Annotation {
  803. return
  804. }
  805. Task { @MainActor in
  806. self.noteOutlineView.reloadData()
  807. }
  808. }
  809. func note_reloadDataIfNeed() {
  810. if self.type.methodType != .Annotation {
  811. return
  812. }
  813. self.reloadAnnotation()
  814. }
  815. func note_reloadDataForAnnoIfNeed(anno: CPDFAnnotation) {
  816. if self.type.methodType != .Annotation {
  817. return
  818. }
  819. if anno is CPDFLineAnnotation || anno is CPDFSquareAnnotation || anno is CPDFCircleAnnotation || anno is CPDFInkAnnotation {
  820. // 形状注释 + Ink 需要显示框住的内容【刷新】
  821. for item in self.annoListModel?.datas ?? [] {
  822. for itemM in item.items {
  823. if anno.isEqual(to: itemM.anno) {
  824. self.noteOutlineView.reloadItem(itemM)
  825. break
  826. }
  827. }
  828. }
  829. } else {
  830. for item in self.annoListModel?.datas ?? [] {
  831. for itemM in item.items {
  832. if anno.isEqual(to: itemM.anno) {
  833. self.noteOutlineView.reloadItem(itemM)
  834. break
  835. }
  836. }
  837. }
  838. }
  839. }
  840. func reloadAnnotation() {
  841. if self.listView != nil {
  842. let filterKey = self.pdfDocument()?.documentURL.path ?? ""
  843. let typeArr: [String] = KMBotaTools.noteFilterAnnoTypes(key: filterKey)
  844. let colorArr: [Any] = KMBotaTools.noteFilterColors(key: filterKey)
  845. let authorArr: [Any] = KMBotaTools.noteFilterAuthors(key: filterKey)
  846. var stateArr: [NSNumber] = KMBotaTools.noteFilterStates(key: filterKey)
  847. if typeArr.count == 0 && colorArr.count == 0 && authorArr.count == 0 && stateArr.count == 0 {
  848. // self.filtrateButton.image = NSImage(named: "KMImageNameAnnotationsFiltrate")
  849. self.filterButtonLayer?.isHidden = true
  850. } else {
  851. // self.filtrateButton.image = NSImage(named: "icon_annotation_screening_select")self.filterButtonLayer?.isHidden = true
  852. self.filterButtonLayer?.isHidden = false
  853. }
  854. var hasMark = false
  855. var hasReview = false
  856. for data in stateArr {
  857. let state = data.intValue
  858. if state == CPDFAnnotationState.marked.rawValue || state == CPDFAnnotationState.unMarked.rawValue {
  859. hasMark = true
  860. }
  861. if state == CPDFAnnotationState.none.rawValue || state == CPDFAnnotationState.accepted.rawValue || state == CPDFAnnotationState.rejected.rawValue || state == CPDFAnnotationState.canceled.rawValue || state == CPDFAnnotationState.completed.rawValue {
  862. hasReview = true
  863. }
  864. }
  865. if hasMark == false {
  866. stateArr.append(NSNumber(value: CPDFAnnotationState.marked.rawValue))
  867. stateArr.append(NSNumber(value: CPDFAnnotationState.unMarked.rawValue))
  868. }
  869. if hasReview == false {
  870. stateArr.append(NSNumber(value: CPDFAnnotationState.none.rawValue))
  871. stateArr.append(NSNumber(value: CPDFAnnotationState.accepted.rawValue))
  872. stateArr.append(NSNumber(value: CPDFAnnotationState.rejected.rawValue))
  873. stateArr.append(NSNumber(value: CPDFAnnotationState.canceled.rawValue))
  874. stateArr.append(NSNumber(value: CPDFAnnotationState.completed.rawValue))
  875. }
  876. var annotationArray: [CPDFAnnotation] = []
  877. var allAnnotation: [CPDFAnnotation] = []
  878. for i in 0 ..< self.pageCount() {
  879. let page = self.pdfDocument()?.page(at: UInt(i))
  880. var annos: [CPDFAnnotation] = []
  881. // 处理过滤
  882. let types = ["Highlight","Underline","Strikeout","Squiggly","Freehand","FreeText","Note","Square","Circle","Line","Stamp","Arrow","Image","Redact","Sign"/*, "table"*/,"Polyline","Polygon"]
  883. if typeArr.count == 0 && colorArr.count == 0 && authorArr.count == 0 && stateArr.count == 0 {
  884. annos = KMOCToolClass.filterAnnotation(annotations: page?.annotations ?? [],types: types) as? [CPDFAnnotation] ?? []
  885. annotationArray += annos
  886. } else {
  887. var filterAnnos: [CPDFAnnotation] = page?.annotations ?? []
  888. let allAnnos = KMOCToolClass.filterAnnotation(annotations: filterAnnos,types: types) as? [CPDFAnnotation] ?? []
  889. annotationArray += allAnnos
  890. if typeArr.count > 0 {
  891. var theTypes = typeArr
  892. if typeArr.contains(CPDFAnnotation.kType.measureArrow) && typeArr.contains(CPDFAnnotation.kType.arrow) == false {
  893. theTypes.append(CPDFAnnotation.kType.arrow)
  894. }
  895. filterAnnos = (KMOCToolClass.filterAnnotation(annotations: filterAnnos, types: theTypes) as? [CPDFAnnotation]) ?? []
  896. }
  897. if (colorArr.count > 0) {
  898. filterAnnos = (KMOCToolClass.filterAnnotation(annotations: filterAnnos,colors: colorArr) as? [CPDFAnnotation]) ?? []
  899. }
  900. if (authorArr.count > 0) {
  901. filterAnnos = (KMOCToolClass.filterAnnotation(annotations: filterAnnos,authors: authorArr) as? [CPDFAnnotation]) ?? []
  902. }
  903. if typeArr.contains(CPDFAnnotation.kType.measureArrow) {
  904. if typeArr.contains(CPDFAnnotation.kType.arrow) == false {
  905. for anno in filterAnnos {
  906. if let data = anno as? CPDFLineAnnotation, data.type == CPDFAnnotation.kType.arrow && data.isMeasure == false {
  907. filterAnnos.removeObject(anno)
  908. }
  909. }
  910. }
  911. } else {
  912. for anno in filterAnnos {
  913. if let data = anno as? CPDFLineAnnotation, data.isMeasure {
  914. filterAnnos.removeObject(anno)
  915. }
  916. }
  917. }
  918. if stateArr.count > 0 {
  919. for anno in filterAnnos {
  920. let markState = self.noteReplyHanddler.fetchAnnoState(anno) ?? .unMarked
  921. let reviewState = self.noteReplyHanddler.fetchReviewState(anno) ?? .none
  922. if stateArr.contains(NSNumber(value: markState.rawValue)) == false || stateArr.contains(NSNumber(value: reviewState.rawValue)) == false {
  923. filterAnnos.removeObject(anno)
  924. }
  925. }
  926. }
  927. annos = filterAnnos
  928. }
  929. //添加签名注释
  930. for annotation in page?.annotations ?? [] {
  931. if annotation.isKind(of: CPDFSignatureAnnotation.self) {
  932. annos.append(annotation)
  933. annotationArray.append(annotation)
  934. }
  935. }
  936. for annotation in annos {
  937. if annotation.isKind(of: KMTableAnnotation.self) {
  938. annos.removeObject(annotation)
  939. if annotationArray.contains(annotation) {
  940. annotationArray.removeObject(annotation)
  941. }
  942. } else if annotation.annotationShouldDisplay() == false {
  943. annos.removeObject(annotation)
  944. if annotationArray.contains(annotation) {
  945. annotationArray.removeObject(annotation)
  946. }
  947. } else if annotation.isKind(of: CPDFLinkAnnotation.self) {
  948. annos.removeObject(annotation)
  949. if annotationArray.contains(annotation) {
  950. annotationArray.removeObject(annotation)
  951. }
  952. }
  953. }
  954. // 添加刷选后的注释
  955. allAnnotation += annos
  956. //添加所有annotation 用于筛选
  957. // annotationArray += (page?.annotations ?? [])
  958. // annotationArray += annos
  959. }
  960. // 处理排序
  961. if self.noteSortType == .page {
  962. /// 排序(升序)
  963. if self.isAscendSort {
  964. allAnnotation.sort {
  965. let idx0 = $0.page?.pageIndex() ?? 0
  966. let idx1 = $1.page?.pageIndex() ?? 0
  967. return idx0 <= idx1
  968. }
  969. } else {
  970. allAnnotation.sort {
  971. let idx0 = $0.page?.pageIndex() ?? 0
  972. let idx1 = $1.page?.pageIndex() ?? 0
  973. return idx0 > idx1
  974. }
  975. }
  976. } else if self.noteSortType == .time {
  977. /// 排序(升序)
  978. if self.isAscendSort {
  979. allAnnotation.sort {
  980. if $0.modificationDate() == nil {
  981. return false
  982. }
  983. if $1.modificationDate() == nil {
  984. return false
  985. }
  986. return $0.modificationDate() <= $1.modificationDate()
  987. }
  988. } else {
  989. allAnnotation.sort {
  990. if $0.modificationDate() == nil {
  991. return false
  992. }
  993. if $1.modificationDate() == nil {
  994. return false
  995. }
  996. return $0.modificationDate() > $1.modificationDate()
  997. }
  998. }
  999. }
  1000. // 数据模型\化
  1001. let model = KMAnnotationListModel()
  1002. var datas: [KMBotaAnnotationModel] = []
  1003. var prePageIdx: Int = NSNotFound
  1004. var preDate: Date?
  1005. var secM: KMBotaAnnotationSectionModel?
  1006. for anno in allAnnotation {
  1007. let item = KMBotaAnnotationModel()
  1008. item.anno = anno
  1009. item.showPage = self.annoListIsShowPage()
  1010. item.showTime = self.annoListIsShowTime()
  1011. item.showAuthor = self.annoListIsShowAnther()
  1012. // datas.append(item)
  1013. if self.noteSortType == .page {
  1014. let pageIdx = Int(anno.pageIndex())
  1015. if pageIdx != prePageIdx { // 不是同一个页面
  1016. secM = KMBotaAnnotationSectionModel()
  1017. model.datas.append(secM!)
  1018. }
  1019. secM?.items.append(item)
  1020. prePageIdx = Int(anno.pageIndex())
  1021. } else { // time
  1022. let date = anno.modificationDate()
  1023. if let same = date?.isSameDay(other: preDate), same == false { // 不是同一天
  1024. secM = KMBotaAnnotationSectionModel()
  1025. model.datas.append(secM!)
  1026. }
  1027. secM?.items.append(item)
  1028. preDate = date
  1029. }
  1030. item.sectionModel = secM
  1031. let replyAnnos = self.noteReplyHanddler.fetchReplyAnnotations(anno) ?? []
  1032. for replyAnno in replyAnnos {
  1033. let replyM = KMBotaAnnotationReplyModel()
  1034. replyM.anno = anno
  1035. replyM.replyAnno = replyAnno
  1036. // secM?.items.append(replyM)
  1037. replyM.annoModel = item
  1038. item.replyAnnos.append(replyM)
  1039. }
  1040. let footerI = KMBotaAnnotationFooterModel()
  1041. footerI.anno = anno
  1042. secM?.items.append(footerI)
  1043. item.footerModel = footerI
  1044. footerI.annoModel = item
  1045. }
  1046. // model.datas = datas
  1047. self.annoListModel = model
  1048. // 转换对象,用于数据显示
  1049. self.allAnnotations = annotationArray
  1050. self.noteFilterButton.isEnabled = self.allAnnotations.count >= 1
  1051. }
  1052. self.note_refrshUIIfNeed()
  1053. }
  1054. func reloadNoteForSearchMode() {
  1055. if self.noteSearchMode == false {
  1056. return
  1057. }
  1058. // 处理排序
  1059. if self.noteSortType == .page {
  1060. if self.isAscendSort { /// 排序(升序)
  1061. self.noteSearchArray.sort {
  1062. let idx0 = $0.anno?.page?.pageIndex() ?? 0
  1063. let idx1 = $1.anno?.page?.pageIndex() ?? 0
  1064. return idx0 <= idx1
  1065. }
  1066. } else {
  1067. self.noteSearchArray.sort {
  1068. let idx0 = $0.anno?.page?.pageIndex() ?? 0
  1069. let idx1 = $1.anno?.page?.pageIndex() ?? 0
  1070. return idx0 > idx1
  1071. }
  1072. }
  1073. } else if self.noteSortType == .time {
  1074. if self.isAscendSort { /// 排序(升序)
  1075. self.noteSearchArray.sort {
  1076. if $0.anno?.modificationDate() == nil {
  1077. return false
  1078. }
  1079. if $1.anno?.modificationDate() == nil {
  1080. return false
  1081. }
  1082. return $0.anno!.modificationDate() <= $1.anno!.modificationDate()
  1083. }
  1084. } else {
  1085. self.noteSearchArray.sort {
  1086. if $0.anno?.modificationDate() == nil {
  1087. return false
  1088. }
  1089. if $1.anno?.modificationDate() == nil {
  1090. return false
  1091. }
  1092. return $0.anno!.modificationDate() > $1.anno!.modificationDate()
  1093. }
  1094. }
  1095. }
  1096. self.note_refrshUIIfNeed()
  1097. }
  1098. // 搜索 Action
  1099. func updateNoteFilterPredicate() {
  1100. var stringValue = self.noteSearchField.stringValue
  1101. // 清空数据
  1102. self.noteSearchArray.removeAll()
  1103. if stringValue.isEmpty {
  1104. for model in self.annoListModel?.datas ?? [] {
  1105. for item in model.items {
  1106. guard let _ = item.anno else {
  1107. continue
  1108. }
  1109. guard let data = item as? KMBotaAnnotationModel else {
  1110. continue
  1111. }
  1112. self.noteSearchArray.append(data)
  1113. }
  1114. }
  1115. } else {
  1116. // 忽略大小写
  1117. let caseInsensite = self.caseInsensitiveNoteSearch
  1118. if caseInsensite {
  1119. stringValue = stringValue.lowercased()
  1120. }
  1121. for model in self.annoListModel?.datas ?? [] {
  1122. for item in model.items {
  1123. guard let note = item.anno else {
  1124. continue
  1125. }
  1126. var noteString = ""
  1127. if let anno = note as? CPDFMarkupAnnotation {
  1128. noteString = anno.markupContent()
  1129. } else {
  1130. noteString = KMBOTAAnnotationTool.fetchContentLabelString(annotation: note)
  1131. }
  1132. if caseInsensite {
  1133. noteString = noteString.lowercased()
  1134. }
  1135. guard let data = item as? KMBotaAnnotationModel else {
  1136. continue
  1137. }
  1138. if noteString.contains(stringValue) {
  1139. self.noteSearchArray.append(data)
  1140. }
  1141. }
  1142. }
  1143. }
  1144. self.note_refrshUIIfNeed()
  1145. }
  1146. @objc func selectSelectedNote(_ sender: AnyObject?) {
  1147. if self.hideNotes() == false {
  1148. let selectedNotes = self.selectedNotes()
  1149. if selectedNotes.count == 1 {
  1150. let annotation = selectedNotes.last!
  1151. self.listView?.go(to: annotation.bounds, on: annotation.page, animated: true)
  1152. // [pdfView scrollAnnotationToVisible:annotation];
  1153. // [pdfView setActiveAnnotation:annotation];
  1154. self.listView?.updateActiveAnnotations([annotation])
  1155. self.listView?.setNeedsDisplayAnnotationViewForVisiblePages()
  1156. if annotation is CPDFPolygonAnnotation || annotation is CPDFPolylineAnnotation {
  1157. self.listView?.pdfListViewDelegate.pdfListViewAnnotationMeasureInfoChange?(self.listView, with: annotation)
  1158. } else if let anno = annotation as? CPDFLineAnnotation {
  1159. if anno.isMeasure {
  1160. self.listView?.pdfListViewDelegate.pdfListViewAnnotationMeasureInfoChange?(self.listView, with: annotation)
  1161. }
  1162. }
  1163. // }
  1164. }
  1165. // NSInteger column = [sender clickedColumn];
  1166. // if (column != -1) {
  1167. // NSString *colID = [[[sender tableColumns] objectAtIndex:column] identifier];
  1168. //
  1169. // if ([colID isEqualToString:@"color"]){
  1170. // for (PDFAnnotation *annotation in self.pdfView.activeAnnotations) {
  1171. // if (![annotation isKindOfClass:[PDFAnnotationChoiceWidget class]] &&
  1172. // ![annotation isKindOfClass:[PDFAnnotationButtonWidget class]] &&
  1173. // ![annotation isKindOfClass:[PDFAnnotationTextWidget class]]) {
  1174. // [[NSColorPanel sharedColorPanel] orderFront:nil];
  1175. // break;
  1176. // }
  1177. //
  1178. // }
  1179. // }
  1180. // }
  1181. }
  1182. }
  1183. func selectedNotes() -> [CPDFAnnotation] {
  1184. var selectedNotes: [CPDFAnnotation] = []
  1185. let rowIndexes = self.noteOutlineView.selectedRowIndexes
  1186. for row in rowIndexes {
  1187. let item = self.noteOutlineView.item(atRow: row)
  1188. if item is KMBotaAnnotationModel {
  1189. if let anno = (item as! KMBotaAnnotationModel).anno {
  1190. // if anno.type == nil {
  1191. // item = [(SKNoteText *)item note];
  1192. // }
  1193. if selectedNotes.contains(anno) == false {
  1194. selectedNotes.append(anno)
  1195. }
  1196. }
  1197. }
  1198. }
  1199. return selectedNotes
  1200. }
  1201. func clearAnnotationFilterData() {
  1202. if let _key = self.pdfDocument()?.documentURL?.path {
  1203. let userDefaults = UserDefaults.standard
  1204. let typeData = try?NSKeyedArchiver.archivedData(withRootObject: [Any](), requiringSecureCoding: false)
  1205. userDefaults.set(typeData, forKey: NoteFilterVC.filterSelectTypeKey + _key)
  1206. let colorData = try?NSKeyedArchiver.archivedData(withRootObject: [Any](), requiringSecureCoding: false)
  1207. userDefaults.set(colorData, forKey: NoteFilterVC.filterSelectColorKey + _key)
  1208. let authorData = try?NSKeyedArchiver.archivedData(withRootObject: [Any](), requiringSecureCoding: false)
  1209. userDefaults.set(authorData, forKey: NoteFilterVC.filterSelectAuthorKey + _key)
  1210. userDefaults.synchronize()
  1211. }
  1212. }
  1213. private func _annoList_needRefreshUI(annos: [CPDFAnnotation]?) -> Bool {
  1214. guard let data = annos else {
  1215. return false
  1216. }
  1217. // for anno in data {
  1218. // if anno.isKind(of: KMTableAnnotation.self) == false {
  1219. // return true
  1220. // }
  1221. // }
  1222. return true
  1223. }
  1224. }