KMPDFEditViewController.swift 41 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092
  1. //
  2. // KMPDFEditViewController.swift
  3. // PDF Master
  4. //
  5. // Created by tangchao on 2023/1/6.
  6. //
  7. class KMPDFEditViewController: KMPDFThumbViewBaseController {
  8. @IBOutlet weak var topBarBox: NSBox!
  9. @IBOutlet weak var contentBox: NSBox!
  10. var toolBar = KMPDFEditToolbar(frame: NSZeroRect)
  11. override var thumbnailView: KMPDFThumbnailView {
  12. get {
  13. if (super.thumbnailView is KMPageEditThumbnailView) {
  14. return super.thumbnailView
  15. }
  16. super.thumbnailView = KMPageEditThumbnailView()
  17. return super.thumbnailView
  18. }
  19. set {
  20. super.thumbnailView = newValue
  21. }
  22. }
  23. var pageEditThumbnailView: KMPageEditThumbnailView {
  24. get {
  25. return self.thumbnailView as! KMPageEditThumbnailView
  26. }
  27. }
  28. override var menuItems: [[KMPDFThumbViewMenu.Item]] {
  29. get {
  30. return [[.copy, .cut, .paste, .delete], [.insert, .extract, .replace, .split, .reverse],
  31. [.rightRotate, .leftRotate], [.showPageSize], [.print]]
  32. }
  33. }
  34. private var pdfDocument: CPDFDocument?
  35. var listView: CPDFView?
  36. var pageIsUpdate: Bool = true
  37. var selectedPages: [Int] = []
  38. private let defaultItemSize = NSMakeSize(218 + 12 * 2, 292 + 24)
  39. var itemSize: NSSize = NSMakeSize(218 + 12 * 2, 292 + 24)
  40. // index=1 双击退出;index=2 打印
  41. var itemClick: KMCommonClickBlock?
  42. var documentEditedCallback: KMCommonBlock?
  43. internal var viewFirstAppear = true
  44. private var _listViewCurrentIndex: Int = 0
  45. var listViewCurrentIndex: Int {
  46. get {
  47. return self._listViewCurrentIndex
  48. }
  49. }
  50. required init?(coder: NSCoder) {
  51. super.init(coder: coder)
  52. }
  53. init(_ document: CPDFDocument) {
  54. super.init(nibName: "KMPDFEditViewController", bundle: nil)
  55. self.pdfDocument = document
  56. self.kmUndoManager = UndoManager()
  57. }
  58. override func viewDidAppear() {
  59. super.viewDidAppear()
  60. if (self.pdfDocument?.isLocked == false && self.viewFirstAppear) {
  61. self.viewFirstAppear = false
  62. self.thumbnailView.document = self.pdfDocument!
  63. self.thumbnailView.reloadData()
  64. }
  65. if let view = self.listView { // 存储阅读视图的当前页面索引
  66. self._listViewCurrentIndex = view.currentPageIndex
  67. }
  68. if (self.selectedPages.isEmpty) {
  69. var index: Int = 0
  70. if (self.listView != nil) {
  71. index = self.listView!.currentPageIndex
  72. }
  73. self.selectPages(indexs: IndexSet(integer: index))
  74. self.toolBar.setPageRangeString("\(index+1)")
  75. } else {
  76. var indexs: IndexSet = []
  77. for pageIndex in self.selectedPages {
  78. if (pageIndex < 0 || pageIndex >= (self.listView?.document.pageCount)!) {
  79. continue
  80. }
  81. indexs.insert(pageIndex)
  82. }
  83. self.selectPages(indexs: indexs)
  84. self.toolBar.setPageRangeString(KMPageRangeTools.newParseSelectedIndexs(selectedIndex: indexs.sorted()))
  85. }
  86. self.view.window?.makeFirstResponder(self)
  87. }
  88. override func viewDidLoad() {
  89. super.viewDidLoad()
  90. self.thumbnailView.collectionView.backgroundColor(NSColor(hex: "#CED0D4"))
  91. var allowedFileTypes: [String] = []
  92. if let _types = KMConvertPDFManagerOC.supportFileType() as? [String] {
  93. allowedFileTypes = KMTools.pdfExtensions + _types
  94. } else {
  95. allowedFileTypes = KMTools.pdfExtensions + KMTools.imageExtensions
  96. }
  97. self.thumbnailView.kmAllowedFileTypes = allowedFileTypes
  98. let toolBar = self.toolBar
  99. self.topBarBox.contentView = toolBar
  100. self.topBarBox.fillColor = NSColor(hex: "#F7F8FA")
  101. toolBar.itemClick = { [unowned self] item, index in
  102. guard let type = KMPageEditType(rawValue: index) else {
  103. if (index == KMPageEditType.insert_files) {
  104. self.insertFileAction()
  105. } else if (index == KMPageEditType.insert_blank_page) {
  106. self.insertBlankPageAction()
  107. } else if (index == KMPageEditType.insert_custom_page) {
  108. self.item_insertCustomPage(sender: nil)
  109. }
  110. return
  111. }
  112. if (type == .leftRotate) {
  113. self.leftRotateAction()
  114. } else if (type == .rightRotate) {
  115. self.rightRotateAction()
  116. } else if (type == .zoomOut) {
  117. self.zoomOutButtonAction(sender: item?.zoomOutButton)
  118. } else if (type == .zoomIn) {
  119. self.zoomInButtonAction(sender: item?.zoomInButton)
  120. } else if (type == .extract) {
  121. self.extractAction()
  122. } else if (type == .replace) {
  123. self.replaceAction()
  124. } else if (type == .split) {
  125. self.splitAction()
  126. } else if (type == .reverse) {
  127. self.reverseAction()
  128. } else if (type == .delete) {
  129. self.deleteAction()
  130. } else if (type == .pageRange) {
  131. self.km_comboBoxSelectionDidChange((item?.pageRangeView)!)
  132. }
  133. }
  134. toolBar.pageRangeValueDidChange = { [unowned self] value, _ in
  135. self.km_controlTextDidEndEditing(self.toolBar.pageRangeView!)
  136. }
  137. let view = self.thumbnailView
  138. self.contentBox.contentView?.addSubview(view)
  139. view.frame = self.contentBox.contentView!.bounds
  140. view.autoresizingMask = [.width, .height]
  141. self.pageEditThumbnailView.delegate = self
  142. self.pageEditThumbnailView.selectionDidChange = { [unowned self] selectedIndexs in
  143. self.view.window?.makeFirstResponder(self)
  144. let enabled = (selectedIndexs.count > 0)
  145. for i in 0 ..< self.toolBar.toolBar.items.count {
  146. let item: KMToolBoxItem = self.toolBar.toolBar.items[i]
  147. if (item.itemIdentifier == KMToolbarPageEditExtractItemIdentifier ||
  148. item.itemIdentifier == KMToolbarPageEditDeleteItemIdentifier ||
  149. item.itemIdentifier == KMToolbarPageEditLeftRotateItemIdentifier ||
  150. item.itemIdentifier == KMToolbarPageEditRightRotateItemIdentifier ||
  151. item.itemIdentifier == KMToolbarPageEditReplaceItemIdentifier) {
  152. item.unEnabled = !enabled
  153. } else if (item.itemIdentifier == KMToolbarPageEditReverseItemIdentifier) {
  154. item.unEnabled = !(selectedIndexs.count > 1)
  155. }
  156. }
  157. if (self.pageIsUpdate == false) {
  158. self.pageIsUpdate = true
  159. return
  160. }
  161. var indexs: Array<Int> = []
  162. for indexPath in selectedIndexs {
  163. indexs.append(indexPath.item)
  164. }
  165. indexs.sort(){$0 < $1}
  166. self.toolBar.setPageRangeString(KMPageRangeTools.newParseSelectedIndexs(selectedIndex: indexs))
  167. }
  168. }
  169. // MARK: Private Methods
  170. private func getSelecteIndex() -> Int {
  171. let index = self.getPasteIndex()
  172. if (index != NSNotFound) {
  173. return index
  174. }
  175. return 0
  176. }
  177. private func getSelectedPage() -> CPDFPage {
  178. return (self.pdfDocument!.page(at: UInt(getSelecteIndex())))!
  179. }
  180. private func selectPages(indexs: IndexSet) {
  181. var indexpaths: Set<IndexPath> = []
  182. for index in indexs {
  183. indexpaths.insert(IndexPath(item: index, section: 0))
  184. }
  185. self.selectPages(indexpaths: indexpaths)
  186. }
  187. private func km_becomeFirstResponder() {
  188. DispatchQueue.main.async {
  189. self.view.window?.makeFirstResponder(self)
  190. }
  191. }
  192. fileprivate func dealPdfDocumentDidEditCallback() {
  193. // 提醒外面文档已编辑
  194. guard let callback = self.documentEditedCallback else {
  195. return
  196. }
  197. callback()
  198. }
  199. // MARK: - 清空选择
  200. private func clearSelectPages() {
  201. self.selectPages(indexs: IndexSet())
  202. }
  203. override func refreshUI(indexpaths: Set<IndexPath> = [], keepSelected: Bool = false) {
  204. super.refreshUI(indexpaths: indexpaths, keepSelected: keepSelected)
  205. self.listView?.layoutDocumentView()
  206. }
  207. override func cutAction() {
  208. if (self.thumbnailView.selectionIndexPaths.count == (self.listView?.document.pageCount)!) {
  209. let _ = CustomAlertView(message: NSLocalizedString("Unable to delete all pages", comment: ""), from: self.thumbnailView, with: .blue)
  210. return
  211. }
  212. super.cutAction()
  213. }
  214. override func deleteAction() {
  215. super.deleteAction()
  216. self.toolBar.unSelectItem(for: KMToolbarPageEditDeleteItemIdentifier)
  217. }
  218. override func deleteIndexPaths(indexpaths: Set<IndexPath>) {
  219. guard let pageCount = self.listView?.document.pageCount else {
  220. KMPrint("pageCount invalid.")
  221. return
  222. }
  223. if (indexpaths.count == pageCount) {
  224. let _ = CustomAlertView(message: NSLocalizedString("Unable to delete all pages", comment: ""), from: self.thumbnailView, with: .blue)
  225. return
  226. }
  227. self.dealPdfDocumentDidEditCallback()
  228. super.deleteIndexPaths(indexpaths: indexpaths)
  229. }
  230. // override func leftRotateAction() {
  231. // super.leftRotateAction()
  232. //
  233. // self.toolBar.unSelectItem(for: KMToolbarPageEditLeftRotateItemIdentifier)
  234. // }
  235. override func leftRotateIndexpaths(indexpaths: Set<IndexPath>) {
  236. if (!self.indexpathsIsValid(indexpaths: indexpaths) || !self.canRotate()) {
  237. return
  238. }
  239. super.leftRotateIndexpaths(indexpaths: indexpaths)
  240. self.dealPdfDocumentDidEditCallback()
  241. }
  242. // override func rightRotateAction() {
  243. // super.rightRotateAction()
  244. //
  245. // self.toolBar.unSelectItem(for: KMToolbarPageEditRightRotateItemIdentifier)
  246. // }
  247. override func rightRotateIndexpaths(indexpaths: Set<IndexPath>) {
  248. if (!self.indexpathsIsValid(indexpaths: indexpaths) || !self.canRotate()) {
  249. return
  250. }
  251. super.rightRotateIndexpaths(indexpaths: indexpaths)
  252. self.dealPdfDocumentDidEditCallback()
  253. }
  254. override func insertPages(pages: Array<CPDFPage>, at indexs: IndexSet) {
  255. super.insertPages(pages: pages, at: indexs)
  256. if (!indexs.isEmpty) {
  257. self.selectPages(indexs: indexs)
  258. self.dealPdfDocumentDidEditCallback()
  259. }
  260. }
  261. override func pasteAction() {
  262. if (self.canPaste()) {
  263. super.pasteAction()
  264. self.dealPdfDocumentDidEditCallback()
  265. }
  266. }
  267. override func extractAction() {
  268. super.extractAction()
  269. DispatchQueue.main.async {
  270. self.toolBar.unSelectItem(for: KMToolbarPageEditExtractItemIdentifier)
  271. }
  272. }
  273. override func replaceAction() {
  274. super.replaceAction()
  275. DispatchQueue.main.async {
  276. self.toolBar.unSelectItem(for: KMToolbarPageEditReplaceItemIdentifier)
  277. }
  278. }
  279. override func getInsertIndex() -> Int {
  280. var index = 0
  281. if let pageCount = self.thumbnailView.document?.pageCount, pageCount > 0 {
  282. // 默认为最后一页
  283. index = Int(pageCount-1)
  284. }
  285. // 在当前选中item之后插入
  286. // 多选时,则取最后一页
  287. for indexpath in self.thumbnailView.selectionIndexPaths.sorted() {
  288. index = indexpath.item
  289. }
  290. return index
  291. }
  292. override func splitAction() {
  293. Task { @MainActor in
  294. self.toolBar.unSelectItem(for: KMToolbarPageEditSplitItemIdentifier)
  295. if await (KMLightMemberManager.manager.canUseAdvanced() == false) {
  296. let _ = KMComparativeTableViewController.show(window: self.view.window!)
  297. return
  298. }
  299. self.splitMenuAction(sender: nil)
  300. }
  301. }
  302. override func reverseAction() {
  303. Task { @MainActor in
  304. self.toolBar.unSelectItem(for: KMToolbarPageEditReverseItemIdentifier)
  305. if await (KMLightMemberManager.manager.canUseAdvanced() == false) {
  306. let _ = KMComparativeTableViewController.show(window: self.view.window!)
  307. return
  308. }
  309. self.reverseMenuAction(sender: nil)
  310. }
  311. }
  312. // MARK: Undo / Redo
  313. @objc private func reversePages(indexs: IndexSet) {
  314. KMPageEditTools.reverse((self.listView?.document)!, indexs
  315. ) { result, error in
  316. if (result) {
  317. self.refreshUI(indexpaths: self.indexsToIndexpaths(indexs: indexs), keepSelected: true)
  318. self.kmUndoManager?.registerUndo(withTarget: self, selector: #selector(self.reversePages), object: indexs)
  319. }
  320. }
  321. }
  322. @objc private func dragPages(_ indexs: IndexSet, _ toIndex: Int, dp: NSCollectionView.DropOperation, reversed: Bool = false) {
  323. if (indexs.count <= 0) {
  324. return
  325. }
  326. self.listViewCurrentPageUpdateForDrag(indexs: indexs, toIndex: toIndex)
  327. // 插入位置偏移量
  328. var toIndexOffset: Int = 0
  329. // 需要移动的页面index数组
  330. var newPages: [CPDFPage] = []
  331. var currentPage: CPDFPage?
  332. var originalIndex = self.listViewCurrentIndex
  333. // 从后面往前删除
  334. for index in indexs.reversed() {
  335. let newPage = (self.listView?.document.page(at: UInt(index)))?.copy() as? CPDFPage
  336. if (newPage == nil) {
  337. continue
  338. }
  339. newPages.insert(newPage!, at: 0)
  340. self.listView?.document.removePage(at: UInt(index))
  341. if (index < toIndex) { /// 删除了插入位置前面的页面,需要改变 toIndex
  342. toIndexOffset += 1
  343. }
  344. if (currentPage == nil) && (originalIndex == index) {
  345. currentPage = newPage
  346. }
  347. }
  348. if (indexs.last! < toIndex && dp == .on) { // 往后拖拽
  349. toIndexOffset -= 1
  350. }
  351. let insertIndex = max(0, toIndex-toIndexOffset)
  352. var selectIndexs: IndexSet = []
  353. for (index, page) in newPages.enumerated() {
  354. self.listView?.document.insertPageObject(page, at: UInt(insertIndex+index))
  355. selectIndexs.insert(insertIndex+index)
  356. if (currentPage?.pageIndex() == page.pageIndex()) {
  357. self._listViewCurrentIndex = insertIndex+index
  358. }
  359. }
  360. // 新增 undo
  361. self.kmUndoManager?.registerUndo(withTarget: self) { target in
  362. target.clearSelectPages()
  363. target.dragPages_reversed(indexs, toIndex, newPages, dp: dp)
  364. }
  365. // 刷新UI
  366. self.refreshUI()
  367. if (reversed == false) {
  368. self.selectPages(indexs: selectIndexs)
  369. }
  370. }
  371. @objc private func dragPages_reversed(_ indexs: IndexSet, _ toIndex: Int, _ pages: [CPDFPage], dp: NSCollectionView.DropOperation) {
  372. if (indexs.count <= 0) {
  373. return
  374. }
  375. // 插入位置偏移量
  376. var toIndexOffset: Int = 0
  377. for index in indexs {
  378. if (index < toIndex) { /// 删除了插入位置前面的页面,需要改变 toIndex
  379. toIndexOffset += 1
  380. }
  381. }
  382. if (indexs.last! < toIndex && dp == .on) { // 往后拖拽
  383. toIndexOffset -= 1
  384. }
  385. let insertIndex = max(0, toIndex-toIndexOffset)
  386. // 需要移动的页面index数组
  387. var datas: [Int] = []
  388. for index in insertIndex ..< (insertIndex + indexs.count) {
  389. datas.append(index)
  390. }
  391. // 从后面开始删除
  392. for index in datas.reversed() {
  393. self.listView?.document.removePage(at: UInt(index))
  394. }
  395. // 再插入到对应的位置
  396. for (index,value) in indexs.enumerated() {
  397. var page: CPDFPage?
  398. if (index < pages.count) {
  399. page = pages[index]
  400. }
  401. if (page != nil) {
  402. self.listView?.document.insertPageObject(page!, at: UInt(value))
  403. }
  404. }
  405. // 新增 undo
  406. self.kmUndoManager?.registerUndo(withTarget: self) { target in
  407. target.dragPages(indexs, toIndex, dp: dp, reversed: true)
  408. }
  409. // 刷新UI
  410. self.refreshUI()
  411. }
  412. // MARK: topBar 事件
  413. @objc func zoomOutButtonAction(sender: NSButton?) {
  414. if (self.itemSize.width > self.defaultItemSize.width) {
  415. return
  416. }
  417. if (self.itemSize.width < self.defaultItemSize.width) {
  418. self.itemSize = self.defaultItemSize
  419. } else if (self.itemSize.width == self.defaultItemSize.width) {
  420. self.itemSize = NSSize(width: self.defaultItemSize.width*1.5, height: self.defaultItemSize.height*1.5)
  421. }
  422. self.toolBar.zoomInButton?.isEnabled = true
  423. if (self.itemSize.width > self.defaultItemSize.width) {
  424. sender?.isEnabled = false
  425. }
  426. self.pageEditThumbnailView.zoomOut()
  427. }
  428. @objc func zoomInButtonAction(sender: NSButton?) {
  429. if (self.itemSize.width < self.defaultItemSize.width) {
  430. return
  431. }
  432. if (self.itemSize.width > self.defaultItemSize.width) {
  433. self.itemSize = self.defaultItemSize
  434. } else if (self.itemSize.width == self.defaultItemSize.width) {
  435. self.itemSize = NSSize(width: self.defaultItemSize.width*0.5, height: self.defaultItemSize.height*0.5)
  436. }
  437. self.toolBar.zoomOutButton?.isEnabled = true
  438. if (self.itemSize.width < self.defaultItemSize.width) {
  439. sender?.isEnabled = false
  440. }
  441. self.pageEditThumbnailView.zoomIn()
  442. }
  443. // MARK: menu 菜单事件
  444. @objc func item_insertCustomPage(sender: NSMenuItem?) {
  445. Task { @MainActor in
  446. if await (KMLightMemberManager.manager.canUseAdvanced() == false) {
  447. let _ = KMComparativeTableViewController.show(window: self.view.window!)
  448. return
  449. }
  450. let windowController = KMPageEditInsertCustomPageWindowController()
  451. if (self.thumbnailView.selectionIndexPaths.count > 0) {
  452. let page = self.getSelectedPage()
  453. let width = page.bounds.size.width * 210 / 595
  454. let height = page.bounds.size.height * 297 / 842
  455. windowController.selectedPageSize = NSSize(width: width, height: height)
  456. // windowController.selectedPageSize = self.getSelectedPage().bounds.size
  457. }
  458. self.km_beginSheet(windowC: windowController)
  459. windowController.itemClick = { [weak self] index in
  460. if (index == 1) { /// 取消
  461. self?.km_endSheet()
  462. return
  463. }
  464. /// 插入
  465. guard let windowC = self?.currentWindowC as? KMPageEditInsertCustomPageWindowController else {
  466. self?.km_endSheet()
  467. return
  468. }
  469. guard let _insertIndex = self?.getInsertIndex() else {
  470. self?.km_endSheet()
  471. return
  472. }
  473. /// 样式
  474. // let type = windowC.typeIndex
  475. // mm 单位的大小
  476. let pageSize = windowC.pageSize
  477. let direction = windowC.direction
  478. let _pageSize = KMPageEditTools.sizeUnitToBoundSize(unit: .mm, pageSize: pageSize)
  479. /// 插入位置
  480. // if (type == 1) { /// 空白页
  481. let document = CPDFDocument()
  482. document?.insertPage(_pageSize, at: 0)
  483. if let page: CPDFPage = (document?.page(at: 0)) {
  484. if (direction == 0) { /// 纵向
  485. page.rotation = 90
  486. }
  487. self?.insertPages(pages: [page], at: IndexSet(integer: _insertIndex+1))
  488. }
  489. // } else {
  490. // let document = CPDFDocument()
  491. // var imageName = "plaid"
  492. // if (type == 2) {
  493. // imageName = "horizontal_line"
  494. // } else if (type == 3) {
  495. // imageName = "five_line_score"
  496. // }
  497. // let imagePath = Bundle.main.pathForImageResource(imageName)
  498. // document?.insertPage(pageSize, withImage: imagePath, at: 0)
  499. // let page: CPDFPage = (document?.page(at: 0))!
  500. // if (direction == 0) { /// 纵向
  501. // page.rotation = 90
  502. // }
  503. //
  504. // self?.insertPagesForDescSort(pages: [page], indexs: IndexSet(integer: _insertIndex+1))
  505. // }
  506. self?.km_endSheet()
  507. }
  508. }
  509. }
  510. // MARK: toolBar 菜单事件
  511. @objc func splitMenuAction(sender: AnyObject?) {
  512. let model = KMPageEditSplitSettingModel()
  513. model.documentURL = self.listView?.document.documentURL
  514. if let pagecnt = self.thumbnailView.document?.pageCount {
  515. model.pageCount = Int(pagecnt)
  516. }
  517. model.fileName = model.documentURL.lastPathComponent
  518. model.pathExtension = model.fileName.components(separatedBy: ".").last
  519. if (self.thumbnailView.selectionIndexPaths.count > 0) {
  520. model.type = 2
  521. model.pageRangeType = 2
  522. var selectedIndexs: Array<Int> = []
  523. for indexpath in self.thumbnailView.selectionIndexPaths {
  524. selectedIndexs.append(indexpath.item)
  525. }
  526. model.pageRangeString = KMPageRangeTools.newParseSelectedIndexs(selectedIndex: selectedIndexs)
  527. } else {
  528. model.type = 0
  529. }
  530. let windowController = KMPageEditSplitWindowController(model)
  531. self.km_beginSheet(windowC: windowController)
  532. windowController.itemClick = { [weak self] index, value in
  533. if (index == 1) { /// 取消
  534. self?.km_endSheet()
  535. return
  536. }
  537. guard let _windowC = self?.currentWindowC as? KMPageEditSplitWindowController else {
  538. self?.km_endSheet()
  539. return
  540. }
  541. /// 拆分
  542. let outputModel = _windowC.model! as! KMPageEditSplitSettingModel
  543. self?.km_endSheet()
  544. let panel = NSOpenPanel()
  545. panel.canChooseFiles = false
  546. panel.canChooseDirectories = true
  547. panel.canCreateDirectories = true
  548. panel.beginSheetModal(for: (self?.view.window)!) { response in
  549. if (response == .cancel) {
  550. return
  551. }
  552. let file_indexs = outputModel.getSplitIndexSets
  553. if (file_indexs == nil || file_indexs!.count <= 0) {
  554. return
  555. }
  556. DispatchQueue.main.async {
  557. self?.showProgressWindow(message: NSLocalizedString("Spliting...", comment: ""))
  558. self?.progressC?.maxValue = Double(file_indexs!.count)
  559. let _document: CPDFDocument = (self?.listView?.document)!
  560. let filePath = "\(panel.url!.path)/\(_document.documentURL.deletingPathExtension().lastPathComponent)"
  561. let uniqueFilePath = KMTools.getUniqueFilePath(filePath: filePath)
  562. // Swift.debugPrint(uniqueFilePath)
  563. if (!FileManager.default.fileExists(atPath: uniqueFilePath)) {
  564. try?FileManager.default.createDirectory(atPath: uniqueFilePath, withIntermediateDirectories: true)
  565. }
  566. var filepaths: [String] = []
  567. for i in 0 ..< file_indexs!.count {
  568. let indexs = file_indexs![i]
  569. if (indexs.isEmpty) {
  570. continue
  571. }
  572. let filepath = String(format: "%@/%@.pdf", uniqueFilePath, model.getNewOutputFileName(index: i+1))
  573. filepaths.append(filepath)
  574. let newDocument = CPDFDocument()
  575. newDocument?.importPages(indexs, from: _document, at: 0)
  576. self?.progressC?.increment(by: 1.0)
  577. newDocument?.write(to: URL(fileURLWithPath: filepath))
  578. }
  579. self?.hiddenProgressWindow()
  580. if let _urlString = filepaths.first?.deletingLastPathComponent {
  581. NSWorkspace.shared.activateFileViewerSelecting([URL(fileURLWithPath: _urlString)])
  582. }
  583. }
  584. }
  585. }
  586. }
  587. @objc func reverseMenuAction(sender: AnyObject?) {
  588. let selectedIndexs = self.thumbnailView.selectionIndexPaths
  589. if (selectedIndexs.count < 2 ) {
  590. let alertView = CustomAlertView(message: NSLocalizedString("Please select at least two pages for editing first", comment: ""), from: self.thumbnailView, with: .blue)
  591. alertView?.layer?.backgroundColor = NSColor(hex: "#36383B").cgColor
  592. alertView?.layer?.cornerRadius = 4
  593. return
  594. }
  595. var indexs: IndexSet = []
  596. for indexpath in selectedIndexs {
  597. indexs.insert(indexpath.item)
  598. }
  599. self.reversePages(indexs: indexs)
  600. self.dealPdfDocumentDidEditCallback()
  601. }
  602. override func addMenuItem(menu: NSMenu, item: KMPDFThumbViewMenu.Item) -> NSMenuItem? {
  603. var item_: NSMenuItem?
  604. if (item.sel == KMPDFThumbViewMenu.insertSelector) {
  605. if (self.thumbnailView.selectionIndexPaths.count == 1) {
  606. item_ = menu.addItem(withTitle: NSLocalizedString("Insert", comment: ""), action: nil, target: self)
  607. let subMenu = NSMenu()
  608. subMenu.addItem(withTitle: NSLocalizedString("Insert a Blank Page", comment: ""), action: #selector(insertPageItemAction), target: self, tag:0)
  609. subMenu.addItem(withTitle: NSLocalizedString("Insert Custom Page", comment: ""), action: #selector(insertPageItemAction), target: self, tag:1)
  610. subMenu.addItem(withTitle: NSLocalizedString("Insert File", comment: ""), action: #selector(insertPageItemAction), target: self, tag:2)
  611. item_?.submenu = subMenu
  612. }
  613. } else if (item.sel == KMPDFThumbViewMenu.splitSelector) {
  614. if (self.thumbnailView.selectionIndexPaths.count > 1) {
  615. item_ = menu.addItem(withTitle: NSLocalizedString("Split", comment: ""), action: KMPDFThumbViewMenu.splitSelector, target: self)
  616. }
  617. } else if (item.sel == KMPDFThumbViewMenu.reverseSelector) {
  618. if (self.thumbnailView.selectionIndexPaths.count > 1) {
  619. item_ = menu.addItem(withTitle: NSLocalizedString("Reverse", comment: ""), action: KMPDFThumbViewMenu.reverseSelector, target: self)
  620. }
  621. } else {
  622. return super.addMenuItem(menu: menu, item: item)
  623. }
  624. return item_
  625. }
  626. }
  627. extension KMPDFEditViewController {
  628. //MARK: menu Action
  629. @objc func insertPageItemAction(menu:NSMenuItem) {
  630. if (menu.tag == 0) {
  631. self.insertBlankPageAction()
  632. } else if (menu.tag == 1) {
  633. self.item_insertCustomPage(sender: nil)
  634. } else if (menu.tag == 2) {
  635. self.insertFileAction()
  636. }
  637. }
  638. @objc func sharePageItemAction(menu:NSMenuItem) {
  639. let item = menu.parent!
  640. let index = (item.representedObject as! IndexSet).first ?? -1
  641. if Int(index) >= 0 {
  642. let doucument = self.pdfDocument!
  643. let page = doucument.page(at: UInt(index))
  644. let filename : String = doucument.documentURL.lastPathComponent
  645. let folderPath = (NSSearchPathForDirectoriesInDomains(FileManager.SearchPathDirectory.applicationSupportDirectory, FileManager.SearchPathDomainMask.userDomainMask, true).last?.stringByAppendingPathComponent(filename))!
  646. try? FileManager.default.removeItem(atPath: folderPath)
  647. let pdfdocument = CPDFDocument()
  648. let ret = pdfdocument!.importPages((item.representedObject as! IndexSet), from: self.thumbnailView.document, at: 0)
  649. let url = URL(fileURLWithPath: folderPath)
  650. if ret {
  651. let success = pdfdocument!.write(to:url)
  652. if success {
  653. let workspace = NSWorkspace.shared
  654. workspace.activateFileViewerSelecting([url])
  655. }
  656. }
  657. let represent : NSSharingService = menu.representedObject as! NSSharingService
  658. represent.perform(withItems: [url])
  659. }
  660. }
  661. @objc func showPageSizeItemAction(menu: NSMenuItem) {
  662. self.thumbnailView.isShowPageSize = !self.thumbnailView.isShowPageSize
  663. self.refreshUIForKeepSelecteds()
  664. }
  665. @objc func printItemAction(menu:NSMenuItem) {
  666. var datas: [Int] = []
  667. for indexpath in self.thumbnailView.selectionIndexPaths {
  668. datas.append(indexpath.item)
  669. }
  670. guard let callback = self.itemClick else {
  671. return
  672. }
  673. callback(2, datas.sorted())
  674. }
  675. }
  676. extension KMPDFEditViewController {
  677. override func menuItemAction_print(_ sender: Any?) {
  678. self.printItemAction(menu: sender as! NSMenuItem)
  679. }
  680. override func menuItemAction_showPageSize(_ sender: Any?) {
  681. self.showPageSizeItemAction(menu: sender as! NSMenuItem)
  682. }
  683. }
  684. // MARK: - KMSelectPopButtonDelegate
  685. extension KMPDFEditViewController {
  686. func km_comboBoxSelectionDidChange(_ obj: KMDesignSelect) {
  687. self.pageIsUpdate = false
  688. let index: Int = obj.indexOfSelectedItem
  689. var indexpaths: Set<IndexPath> = []
  690. var type: KMPageRange?
  691. if (index <= 0) { /// 全部页面
  692. type = .all
  693. } else if (index == 1) { /// 奇数页
  694. type = .odd
  695. } else if (index == 2) { /// 偶数页
  696. type = .even
  697. } else if (index == 3) { /// 横向页
  698. type = .horizontal
  699. } else if (index == 4) { /// 纵向页
  700. type = .vertical
  701. } else { /// 自定义
  702. type = .custom
  703. }
  704. for i in 0 ..< (self.pdfDocument?.pageCount)! {
  705. if (type == .all) {
  706. indexpaths.insert(IndexPath(item: Int(i), section: 0))
  707. } else if (type == .odd) {
  708. if (i % 2 == 1) {
  709. continue
  710. }
  711. indexpaths.insert(IndexPath(item: Int(i), section: 0))
  712. } else if (type == .even) {
  713. if (i % 2 == 0) {
  714. continue
  715. }
  716. indexpaths.insert(IndexPath(item: Int(i), section: 0))
  717. } else if (type == .horizontal) {
  718. let page = self.pdfDocument?.page(at: i)
  719. guard let isVertical = page?.isVertical(), !isVertical else {
  720. continue
  721. }
  722. indexpaths.insert(IndexPath(item: Int(i), section: 0))
  723. } else if (type == .vertical) {
  724. let page = self.pdfDocument?.page(at: i)
  725. guard let isHorizontal = page?.isHorizontal(), !isHorizontal else {
  726. continue
  727. }
  728. indexpaths.insert(IndexPath(item: Int(i), section: 0))
  729. }
  730. }
  731. obj.editable = type == .custom
  732. if (type == .custom) {
  733. self.clearSelectPages()
  734. DispatchQueue.main.async {
  735. self.toolBar.pageRangeView?.stringValue = ""
  736. self.view.window?.makeFirstResponder(self.toolBar.pageRangeView?.textField)
  737. }
  738. } else {
  739. self.pageEditThumbnailView.selectPages(at: indexpaths)
  740. }
  741. }
  742. func km_controlTextDidEndEditing(_ obj: KMDesignSelect) {
  743. self.view.window?.makeFirstResponder(self)
  744. for index in 0 ..< obj.numberOfItems {
  745. if (obj.items[index] == obj.stringValue) {
  746. return
  747. }
  748. }
  749. let pages = KMPageRangeTools.findSelectPage(pageRangeString: self.toolBar.pageRangeView!.stringValue, pageCount: Int((self.listView?.document.pageCount)!))
  750. if (pages.isEmpty) {
  751. let alert = NSAlert()
  752. alert.messageText = NSLocalizedString("Invalid page range or the page number is out of range. Please try again.", comment: "")
  753. alert.runModal()
  754. self.toolBar.setPageRangeString("")
  755. DispatchQueue.main.async {
  756. self.view.window?.makeFirstResponder(self.toolBar.pageRangeView?.textField)
  757. }
  758. return
  759. }
  760. var indexpaths: Set<IndexPath> = []
  761. for page in pages {
  762. if (page >= 1) {
  763. indexpaths.insert(IndexPath(item: page-1, section: 0))
  764. }
  765. }
  766. self.pageIsUpdate = false
  767. self.pageEditThumbnailView.selectPages(at: indexpaths)
  768. }
  769. }
  770. // MARK: -
  771. // MARK: KMThumbnailViewDelegate
  772. extension KMPDFEditViewController: KMThumbnailViewDelegate {
  773. func thumbnailView(thumbanView: KMThumbnailView, didDragAddFiles files: [URL], indexpath: IndexPath) {
  774. self.km_add_file_multi(fileUrls: files) { index, params in
  775. if (self.fetchProgressBlockParamsIsPasswordFile(params: params)) { // 加密文档进度回调
  776. return
  777. }
  778. let _fileUrl = self.fetchProgressBlockParamsForFileUrl(params: params)
  779. if let exn = _fileUrl?.pathExtension, KMTools.isPDFType(exn) {
  780. if (_fileUrl!.path.isPDFValid() == false) {
  781. let alert = NSAlert()
  782. alert.alertStyle = .critical
  783. alert.messageText = NSLocalizedString("An error occurred while opening this document. The file is damaged and could not be repaired.", comment: "")
  784. alert.runModal()
  785. }
  786. }
  787. } completionBlock: { [unowned self] documents in
  788. var insertIndex: Int = indexpath.item
  789. var pages: Array<CPDFPage> = []
  790. var indexs = IndexSet()
  791. for document in documents {
  792. for i in 0 ..< document.pageCount {
  793. let page = document.page(at: i)
  794. pages.append(page!)
  795. indexs.insert(insertIndex)
  796. insertIndex += 1
  797. }
  798. }
  799. self.insertPages(pages: pages, at: indexs)
  800. }
  801. }
  802. func thumbnailView(thumbanView: KMThumbnailView, didDrag dragedIndexPaths: [IndexPath], indexpath: IndexPath, dragInfo: [KMThumbnailViewDragInfoKey.RawValue : Any]) {
  803. let toIndex = max(0, indexpath.item)
  804. var indexs = IndexSet()
  805. for indexpath in dragedIndexPaths {
  806. indexs.insert(indexpath.item)
  807. }
  808. var dp = NSCollectionView.DropOperation.on
  809. if let _dp = dragInfo[KMThumbnailViewDragInfoKey.dropOperation.rawValue] as? NSCollectionView.DropOperation {
  810. dp = _dp
  811. }
  812. self.dragPages(indexs, toIndex, dp: dp)
  813. if (dragedIndexPaths.count > 0) {
  814. self.dealPdfDocumentDidEditCallback()
  815. }
  816. }
  817. func thumbnailView(thumbanView: KMThumbnailView, insetForSectionAt section: Int) -> NSEdgeInsets {
  818. return NSEdgeInsets(top: 16, left: 16, bottom: 0, right: 32)
  819. }
  820. func thumbnailView(thumbanView: KMThumbnailView, sizeForItemAt indexpath: IndexPath) -> NSSize {
  821. self.thumbnailView.thumbnailSzie = CGSize(width: 120, height: 155)
  822. if (!self.thumbnailView.isShowPageSize) {
  823. return NSMakeSize(self.itemSize.width, self.itemSize.height+26)
  824. }
  825. return self.itemSize
  826. }
  827. func thumbnailView(thumbanView: KMThumbnailView, numberOfItemsInSection section: Int) -> Int {
  828. if let count = self.pdfDocument?.pageCount {
  829. return Int(count)
  830. }
  831. return 0
  832. }
  833. func thumbnailView(thumbanView: KMThumbnailView, itemForRepresentedObjectAt indexpath: IndexPath) -> NSCollectionViewItem {
  834. let cellView: KMPageEditThumbnailItem = thumbanView.collectionView.makeItem(withIdentifier: KMPageEditThumbnailItem.km_identifier, for: indexpath) as! KMPageEditThumbnailItem
  835. cellView.isShowPageSize = self.thumbnailView.isShowPageSize
  836. cellView.page = (self.pdfDocument?.page(at: UInt(indexpath.item)))!
  837. cellView.setPage(page: (self.pdfDocument?.page(at: UInt(indexpath.item)))!)
  838. cellView.doubleClickAction = { [unowned self] _ in
  839. guard let callback = self.itemClick else {
  840. return
  841. }
  842. callback(1, indexpath.item)
  843. }
  844. cellView.mouseDownAction = { [unowned self] _ in
  845. let selectionIndexPaths = self.thumbnailView.selectionIndexPaths
  846. if (selectionIndexPaths.count == 1 && selectionIndexPaths.first?.item == indexpath.item) {
  847. return
  848. }
  849. self.selectPages(indexs: IndexSet.init(integer: indexpath.item))
  850. }
  851. cellView.rightMouseDownAction = { [unowned self] _ in
  852. self.selectPages(indexs: IndexSet.init(integer: indexpath.item))
  853. }
  854. return cellView
  855. }
  856. }
  857. // MARK: - 系统菜单栏
  858. extension KMPDFEditViewController {
  859. override func undo(_ sender: Any?) {
  860. super.undo(sender)
  861. self.km_becomeFirstResponder()
  862. }
  863. override func redo(_ sender: Any?) {
  864. super.redo(sender)
  865. self.km_becomeFirstResponder()
  866. }
  867. }
  868. // MARK: - KMExtensions
  869. extension KMPDFEditViewController {
  870. override func deleteAfter(indexpaths: Set<IndexPath>) {
  871. self.clearSelectPages()
  872. self.km_becomeFirstResponder()
  873. }
  874. override func copyAfter(indexpaths: Set<IndexPath>) {
  875. self.km_becomeFirstResponder()
  876. }
  877. override func pasteAfter(at index: Int) {
  878. self.km_becomeFirstResponder()
  879. }
  880. override func cutAfter(indexpaths: Set<IndexPath>) {
  881. self.km_becomeFirstResponder()
  882. }
  883. override func leftRotateAfter(indexpaths: Set<IndexPath>) {
  884. NotificationCenter.default.post(name: NSNotification.Name(rawValue: "KMPDFViewRotatePage"), object: self.thumbnailView.document)
  885. self.km_becomeFirstResponder()
  886. }
  887. override func rightRotateAfter(indexpaths: Set<IndexPath>) {
  888. NotificationCenter.default.post(name: NSNotification.Name(rawValue: "KMPDFViewRotatePage"), object: self.thumbnailView.document)
  889. self.km_becomeFirstResponder()
  890. }
  891. override func replaceAfter(of targetIndexs: IndexSet, with pages: [CPDFPage], at index: Int) {
  892. var indexpaths = Set<IndexPath>()
  893. var insertIndex = index
  894. for _ in pages {
  895. indexpaths.insert(IndexPath(item: insertIndex, section: 0))
  896. insertIndex += 1
  897. }
  898. self.thumbnailView.selectionIndexPaths = indexpaths
  899. self.dealPdfDocumentDidEditCallback()
  900. }
  901. override func insertBlankPageAfter(size: NSSize, at index: Int) {
  902. self.selectPages(indexs: IndexSet(integer: index))
  903. self.dealPdfDocumentDidEditCallback()
  904. }
  905. func listViewCurrentPageUpdateForDrag(indexs: IndexSet, toIndex: Int) {
  906. // listView 当前页面 被拖拽
  907. if (indexs.contains(self._listViewCurrentIndex)) {
  908. // let idx = indexs.firstIndex(of: self._listViewCurrentIndex)
  909. var idx: Int = 0
  910. for index in indexs {
  911. if (index == self._listViewCurrentIndex) {
  912. break
  913. }
  914. idx += 1
  915. }
  916. // 插入位置偏移量
  917. var toIndexOffset: Int = 0
  918. // 从后面往前删除
  919. for index in indexs.reversed() {
  920. let newPage = (self.listView?.document.page(at: UInt(index)))?.copy() as? CPDFPage
  921. if (newPage == nil) {
  922. continue
  923. }
  924. if (index < toIndex) { /// 删除了插入位置前面的页面,需要改变 toIndex
  925. toIndexOffset += 1
  926. }
  927. }
  928. if (indexs.last! < toIndex) { // 往后拖拽
  929. toIndexOffset -= 1
  930. }
  931. let insertIndex = max(0, toIndex-toIndexOffset)
  932. // self._listViewCurrentIndex = insertIndex+idx
  933. }
  934. }
  935. }