KMBatchOperateBaseViewController.swift 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251
  1. //
  2. // KMBatchOperateBaseViewController.swift
  3. // PDF Master
  4. //
  5. // Created by kdanmobile on 2023/10/30.
  6. //
  7. import Cocoa
  8. var kBatchQueueOperationsChanged = "kBatchQueueOperationsChanged"
  9. class KMBatchOperateBaseViewController: NSViewController, KMBatchOperateProtocol{
  10. var operateType: KMBatchOperationType?
  11. var convertType: KMConvertWithPDFType?
  12. var files: [KMBatchOperateFile]?
  13. lazy var queue: KMOperationQueue? = {
  14. var queue: KMOperationQueue?
  15. if !(self.view.window as! KMBatchWindow).isBatch! {
  16. queue = KMOperationQueue()
  17. } else {
  18. if self.operateType == .Convert {
  19. // queue = KMConvertOperationQueue.sharedQueue()
  20. } else if self.operateType == .Compress {
  21. // queue = KMCompressOperationQueue.sharedQueue()
  22. } else if self.operateType == .AddPassword {
  23. // queue = KMAddPasswordOperationQueue.sharedQueue()
  24. } else if self.operateType == .RemovePassword {
  25. // queue = KMRemovePasswordOperationQueue.sharedQueue()
  26. } else if self.operateType == .AddWatermark {
  27. // queue = KMAddWatermarkOprationQueue.sharedQueue()
  28. } else if self.operateType == .RemoveWatermark {
  29. // queue = KMRemoveWatermarkOperationQueue.sharedQueue()
  30. } else if self.operateType == .AddBackground {
  31. // queue = KMAddBackgroundOperationQueue.sharedQueue()
  32. } else if self.operateType == .RemoveBackground {
  33. // queue = KMRemoveBackgroundOperationQueue.sharedQueue()
  34. } else if self.operateType == .AddHeaderFooter {
  35. // queue = KMAddHeaderFooterOperationQueue.sharedQueue()
  36. } else if self.operateType == .RemoveHeaderFooter {
  37. // queue = KMRemoveHeaderFooterQueue.sharedQueue()
  38. } else if self.operateType == .AddBates {
  39. // queue = KMAddBatesOperationQueue.sharedQueue()
  40. } else if self.operateType == .RemoveBates {
  41. // queue = KMRemoveBatesOperationQueue.sharedQueue()
  42. }
  43. }
  44. return queue
  45. }()
  46. lazy var successFilePathURLArray: [URL]? = {
  47. let arr = NSMutableArray()
  48. return (arr as! [URL])
  49. }()
  50. var name: String = ""
  51. var interfaceStatus: KMBatchOperateInterfaceStatus?
  52. var choosePath: String = ""
  53. init(files: NSMutableArray) {
  54. super.init(nibName: String(describing: type(of: self)), bundle: nil)
  55. self.files = (files as! [KMBatchOperateFile])
  56. }
  57. required init?(coder: NSCoder) {
  58. super.init(coder: coder)
  59. }
  60. deinit {
  61. queue?.removeObserver(self, forKeyPath: "operations")
  62. }
  63. override func viewDidLoad() {
  64. super.viewDidLoad()
  65. self.queue!.addObserver(self, forKeyPath: "operations", options: .new, context: &kBatchQueueOperationsChanged)
  66. }
  67. func operateCompleted() {
  68. showWindowCloseButton()
  69. if successFilePathURLArray!.count > 0 {
  70. NSWorkspace.shared.activateFileViewerSelecting(successFilePathURLArray!)
  71. }
  72. self.interfaceStatus = .PrepareProcess
  73. }
  74. func showWindowCloseButton() {
  75. let btn = self.view.window?.standardWindowButton(NSWindow.ButtonType.closeButton)
  76. btn?.isHidden = false
  77. }
  78. func beginBatchOperation() {
  79. for i in 0..<self.files!.count {
  80. let file = self.files?[i]
  81. file?.currentOperateInfo?.resetState()
  82. }
  83. if self.files!.count < 1 {
  84. NSSound.beep()
  85. return
  86. }
  87. self.fetchFileListViewController()!.reloadData()
  88. let openPanel = NSOpenPanel()
  89. openPanel.canChooseFiles = false
  90. openPanel.canChooseDirectories = true
  91. openPanel.canCreateDirectories = true
  92. openPanel.beginSheetModal(for: self.view.window!) { (result) in
  93. if result == NSApplication.ModalResponse.OK {
  94. for fileURL in openPanel.urls {
  95. self.choosePath = fileURL.path
  96. self.beginQueueOperation()
  97. }
  98. }
  99. }
  100. }
  101. func fetchFileListViewController() -> KMBatchOperateLeftViewController? {
  102. guard let vc = self.view.window?.contentViewController as? KMBatchOperateSplitViewController,
  103. let viewController = vc.splitViewItems.first?.viewController as? KMBatchOperateLeftViewController
  104. else {
  105. return nil
  106. }
  107. return viewController
  108. }
  109. func beginQueueOperation() {//业务逻辑转移到数据处理层
  110. // hiddenWindowCloseButtonIfNeeded()
  111. self.successFilePathURLArray?.removeAll()
  112. for i in 0..<self.files!.count {
  113. let file = self.files![i]
  114. if file.fileType == .PDF {
  115. if self.operateType == .Convert {
  116. file.convertType = self.convertType
  117. }
  118. file.currentOperateInfo?.savePath = self.choosePath
  119. let operation = operationWithFile(file)
  120. operation!.delegate = self
  121. self.queue?.addOperation(operation!)
  122. }
  123. }
  124. if (self.queue?.operations.count)! > 0 {
  125. self.interfaceStatus = .Processing
  126. }
  127. }
  128. func operationWithFile(_ file: KMBatchOperateFile) -> KMBatchOperation? {
  129. let operation: KMBatchOperation? = nil
  130. if self.operateType == .Convert {
  131. // operation = KMBatchConvertOperation(file: file, convertType: self.convertType)
  132. } else if self.operateType == .Compress {
  133. // operation = KMCompressOperation(file: file)
  134. } else if self.operateType == .AddPassword {
  135. // handle AddPassword operation
  136. } else if self.operateType == .RemovePassword {
  137. // operation = KMBatchRemovePasswordOperation(file: file)
  138. } else if self.operateType == .AddWatermark {
  139. // handle AddWatermark operation
  140. } else if self.operateType == .RemoveWatermark {
  141. // operation = KMBatchRemoveWatermarkOperation(file: file)
  142. } else if self.operateType == .AddBackground {
  143. // handle AddBackground operation
  144. } else if self.operateType == .RemoveBackground {
  145. // operation = KMBatchRemoveWatermarkOperation(file: file)
  146. } else if self.operateType == .AddHeaderFooter {
  147. // handle AddHeaderFooter operation
  148. } else if self.operateType == .RemoveHeaderFooter {
  149. // operation = KMBatchRemoveHeaderFooterOperation(file: file)
  150. } else if self.operateType == .AddBates {
  151. // operation = KMBatchRemoveHeaderFooterOperation(file: file)
  152. } else if self.operateType == .RemoveBates {
  153. // operation = KMBatchRemoveHeaderFooterOperation(file: file)
  154. // operation?.setValue(true, forKey: "isBates")
  155. }
  156. return operation
  157. }
  158. func cancelBatchOperation() {
  159. for i in 0..<(self.queue?.operations.count)! {
  160. let operation = self.queue?.operations[i]
  161. if !operation!.isExecuting {
  162. operation?.cancel()
  163. } else {
  164. // Do nothing
  165. }
  166. }
  167. self.interfaceStatus = .PrepareProcess
  168. }
  169. func hiddenWindowCloseButtonIfNeeded() {
  170. let btn = self.view.window?.standardWindowButton(NSWindow.ButtonType.closeButton)
  171. btn?.isHidden = true
  172. }
  173. func allPageNumbers(_ pageString: String) -> [NSNumber] {
  174. let array = pageString.components(separatedBy: ",")
  175. var pageNumbers = NSMutableArray()
  176. var isInvalid = false
  177. for s in array {
  178. if s == "" {
  179. isInvalid = true
  180. break
  181. } else {
  182. let pages = s.components(separatedBy: "-")
  183. if pages.count > 2 {
  184. isInvalid = true
  185. break
  186. } else if pages.count == 1 {
  187. let p = pages[0]
  188. if p == "" || Int(p) == 0 {
  189. isInvalid = true
  190. break
  191. } else {
  192. var isEqual = false
  193. for pageNumber in pageNumbers {
  194. if (pageNumber as AnyObject).intValue == Int(p) {
  195. isEqual = true
  196. isInvalid = true
  197. break
  198. }
  199. }
  200. if !isEqual {
  201. pageNumbers.add(NSNumber(value: Int(p)!))
  202. }
  203. }
  204. } else if pages.count == 2 {
  205. let p1 = pages[0]
  206. let p2 = pages[1]
  207. if p1 == "" || p2 == "" || Int(p1)! >= Int(p2)! || Int(p1) == 0 {
  208. isInvalid = true
  209. break
  210. } else {
  211. var isEqual = false
  212. for i in Int(p1)!...Int(p2)! {
  213. for pageNumber in pageNumbers {
  214. if (pageNumber as AnyObject).intValue == i {
  215. isEqual = true
  216. isInvalid = true
  217. break
  218. }
  219. }
  220. }
  221. if !isEqual {
  222. for i in Int(p1)!...Int(p2)! {
  223. pageNumbers.add(NSNumber(value: i))
  224. }
  225. }
  226. }
  227. }
  228. }
  229. }
  230. return pageNumbers as! [NSNumber]
  231. }
  232. override func observeValue(forKeyPath keyPath: String?, of object: Any?, change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) {
  233. if keyPath == "operations" && context == &kBatchQueueOperationsChanged {
  234. if self.queue?.operations.count == 0 {
  235. DispatchQueue.main.async {
  236. self.operateCompleted()
  237. }
  238. }
  239. } else {
  240. super.observeValue(forKeyPath: keyPath, of: object, change: change, context: context)
  241. }
  242. }
  243. }