KMAddBackgroundView.swift 39 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000
  1. //
  2. // KMAddBackgroundView.swift
  3. // PDF Reader Pro
  4. //
  5. // Created by lizhe on 2023/11/14.
  6. //
  7. import Cocoa
  8. enum KMBackgroundManagerType: Int {
  9. case add = 0
  10. case edit
  11. case use
  12. }
  13. typealias KMAddBackgroundViewOperateCallBack = (_ background: KMBackgroundModel, _ countType: Int) -> ()
  14. typealias KMAddBackgroundViewBatchAction = (_ view: KMAddBackgroundView, _ files: [KMFileAttribute]) -> Void
  15. typealias KMAddBackgroundViewCancelAction = (_ view: KMAddBackgroundView) -> Void
  16. class KMAddBackgroundView: KMBaseXibView, NSComboBoxDelegate {
  17. @IBOutlet weak var pdfView: KMWatermarkPDFView!
  18. @IBOutlet weak var previousButton: NSButton!
  19. @IBOutlet weak var nextButton: NSButton!
  20. @IBOutlet weak var currentPageIndexTextField: NSTextField!
  21. @IBOutlet weak var totalPageCountlabel: NSTextField!
  22. @IBOutlet weak var typeBox: NSBox!
  23. @IBOutlet weak var colorButton: NSButton!
  24. @IBOutlet weak var colorWell: NSColorWell!
  25. @IBOutlet weak var fileButton: NSButton!
  26. @IBOutlet weak var filePathLabel: NSTextField!
  27. @IBOutlet weak var browseButton: NSButton!
  28. @IBOutlet weak var ratioLabel: NSTextField!
  29. @IBOutlet weak var ratioTextField: NSTextField!
  30. @IBOutlet weak var ratioStepper: NSStepper!
  31. @IBOutlet weak var appearanceBox: NSBox!
  32. @IBOutlet weak var angleLabel: NSTextField!
  33. @IBOutlet weak var angleTextField: NSTextField!
  34. @IBOutlet weak var angleStepper: NSStepper!
  35. @IBOutlet weak var left45IndicateView: KMAngleIndicateView!
  36. @IBOutlet weak var horizontalIndicateView: KMAngleIndicateView!
  37. @IBOutlet weak var right45IndicateView: KMAngleIndicateView!
  38. @IBOutlet weak var alphaLabel: NSTextField!
  39. @IBOutlet weak var alphaSlider: NSSlider!
  40. @IBOutlet weak var alphaTextField: NSTextField!
  41. @IBOutlet weak var alphaStepper: NSStepper!
  42. @IBOutlet weak var postionView: KMPostionIndicateView!
  43. @IBOutlet weak var verticalGapLabel: NSTextField!
  44. @IBOutlet weak var verticalGapTextField: NSTextField!
  45. @IBOutlet weak var verticalStepper: NSStepper!
  46. @IBOutlet weak var horizontalGapLabel: NSTextField!
  47. @IBOutlet weak var horizontalGapTextField: NSTextField!
  48. @IBOutlet weak var horizontalStepper: NSStepper!
  49. @IBOutlet weak var pageRangeComboBox: NSComboBox!
  50. @IBOutlet weak var pageRangeLabel: NSTextField!
  51. @IBOutlet weak var saveToTemplateButton: NSButton!
  52. @IBOutlet weak var templateNameLabel: NSTextField!
  53. @IBOutlet weak var templateNameTextField: NSTextField!
  54. @IBOutlet weak var doneButton: NSButton!
  55. @IBOutlet weak var cancelButton: NSButton!
  56. @IBOutlet weak var batchButton: NSButton!
  57. var isHiddenBatchBtn: Bool = false
  58. private var workItem: DispatchWorkItem?
  59. var isAllowReloadDocument = true
  60. lazy var background: KMBackgroundModel = {
  61. var bg = KMBackgroundModel()
  62. bg.type = .color
  63. bg.color = .red
  64. bg.opacity = 1.0
  65. bg.scale = 1.0
  66. bg.verticalMode = 1
  67. bg.horizontalMode = 1
  68. bg.backgroundID = KMBackgroundManager.defaultManager.fetchAvailableName()
  69. self.initialID = bg.backgroundID
  70. bg.pagesString = ""
  71. bg.pageRangeType = .all
  72. return bg
  73. }()
  74. var originalBackground: KMBackgroundModel = KMBackgroundModel()
  75. var filePath: String = Bundle.main.path(forResource: "Quick Start Guide", ofType: "pdf") ?? ""
  76. var password: String = ""
  77. var type: KMBackgroundManagerType = .use
  78. var pdfDocument: CPDFDocument? {
  79. didSet {
  80. self._fileAttri = KMFileAttribute()
  81. self._fileAttri?.password = self.pdfDocument?.password ?? ""
  82. self._fileAttri?.filePath = self.pdfDocument?.documentURL.path ?? ""
  83. self.password = pdfDocument?.password ?? ""
  84. self.pdfView.document = pdfDocument
  85. self.reloadData()
  86. }
  87. }
  88. var backgroundType: KMBackgroundType = .color
  89. var initialID: String!
  90. var currentType: Int = 0
  91. var cancelAction: KMAddBackgroundViewCancelAction?
  92. var batchAction: KMAddBackgroundViewCancelAction?
  93. var operateCallBack: KMAddBackgroundViewOperateCallBack?
  94. var onlyManagerTemplate: Bool = true
  95. private var _fileAttri: KMFileAttribute?
  96. // MARK: - Dealloc
  97. deinit {
  98. NotificationCenter.default.removeObserver(self)
  99. }
  100. // MARK: - Init Methods
  101. convenience init?(baseFile filePath: String, background backgroundObject: KMBackgroundModel, password: String, type: KMBackgroundManagerType, fileType countType: Int) {
  102. self.init()
  103. self.filePath = filePath
  104. self.password = password
  105. self.background = backgroundObject
  106. self.originalBackground = backgroundObject
  107. self.initialID = backgroundObject.backgroundID
  108. self.type = type
  109. self.pdfDocument = CPDFDocument(url: URL(fileURLWithPath: self.filePath))
  110. if pdfDocument!.isLocked {
  111. pdfDocument!.unlock(withPassword: password)
  112. }
  113. if pdfDocument!.isLocked {
  114. return nil
  115. }
  116. }
  117. override func setup() {
  118. // pdfView.background = background
  119. // pdfView.document = pdfDocument
  120. pdfView.autoScales = true
  121. pdfView.setDisplay(.singlePage)
  122. // pdfView.documentView?.enclosingScrollView?.hasVerticalScroller = false
  123. // pdfView.documentView?.enclosingScrollView?.hasHorizontalScroller = false
  124. for i in 0..<3 {
  125. for j in 0..<3 {
  126. if i == Int(background.horizontalMode) && j == Int(background.verticalMode) {
  127. postionView.style = KMPositionIndicateViewStyle(rawValue: i + 3 * j)!
  128. }
  129. }
  130. }
  131. postionView.styleChangedCallBack = { [weak self] in
  132. guard let self = self else { return }
  133. self.background.horizontalMode = self.postionView.style.rawValue % 3
  134. self.background.verticalMode = self.postionView.style.rawValue / 3
  135. self.self.updatePDFView()
  136. if self.filePathLabel.stringValue.count > 0 {
  137. self.doneButton.isEnabled = true
  138. }
  139. }
  140. currentPageIndexTextField.stringValue = "1"
  141. // let numberFormatter = currentPageIndexTextField.formatter as! NumberFormatter
  142. // numberFormatter.maximum = NSNumber(value: pdfDocument.pageCount)
  143. left45IndicateView.style = .left45
  144. left45IndicateView.touchCallBack = { [weak self] in
  145. guard let self = self else { return }
  146. self.background.rotation = 45
  147. self.angleStepper.doubleValue = 45
  148. self.angleTextField.stringValue = "\(45)"
  149. self.checkAngle()
  150. if self.filePathLabel.stringValue.count > 0 {
  151. self.doneButton.isEnabled = true
  152. }
  153. self.updatePDFView()
  154. }
  155. horizontalIndicateView.style = .horizontal
  156. horizontalIndicateView.touchCallBack = { [weak self] in
  157. guard let self = self else { return }
  158. self.background.rotation = 0
  159. self.angleStepper.doubleValue = 0
  160. self.angleTextField.stringValue = "\(0)"
  161. self.checkAngle()
  162. if self.filePathLabel.stringValue.count > 0 {
  163. self.doneButton.isEnabled = true
  164. }
  165. self.updatePDFView()
  166. }
  167. right45IndicateView.style = .right45
  168. right45IndicateView.touchCallBack = { [weak self] in
  169. guard let self = self else { return }
  170. self.background.rotation = -45
  171. self.angleStepper.doubleValue = -45
  172. self.angleTextField.stringValue = "\(-45)"
  173. self.checkAngle()
  174. if self.filePathLabel.stringValue.count > 0 {
  175. self.doneButton.isEnabled = true
  176. }
  177. self.updatePDFView()
  178. }
  179. checkAngle()
  180. typeBox.titleFont = NSFont.systemFont(ofSize: 13)
  181. colorWell.color = background.color ?? NSColor.red
  182. templateNameTextField.stringValue = background.backgroundID
  183. appearanceBox.titleFont = NSFont.systemFont(ofSize: 13)
  184. saveToTemplateButton.isEnabled = onlyManagerTemplate
  185. // if type == .use {
  186. // saveToTemplateButton.isHidden = true
  187. // saveToTemplateButton.state = .off
  188. // } else {
  189. // saveToTemplateButton.isHidden = false
  190. // saveToTemplateButton.state = .on
  191. // }
  192. pageRangeComboBox.removeAllItems()
  193. pageRangeComboBox.addItems(withObjectValues: [
  194. NSLocalizedString("All Pages", comment: ""),
  195. NSLocalizedString("Odd Pages Only", comment: ""),
  196. NSLocalizedString("Even Pages Only", comment: ""),
  197. NSLocalizedString("e.g. 1,3-5,10", comment: "")
  198. ])
  199. pageRangeComboBox.placeholderString = NSLocalizedString("e.g. 1,3-5,10", comment: "")
  200. pageRangeComboBox.delegate = nil
  201. pageRangeComboBox.selectItem(at: 0)
  202. pageRangeComboBox.isEditable = false
  203. pageRangeComboBox.delegate = self
  204. colorWell.target = self
  205. colorWell.action = #selector(colorWellDidChange)
  206. }
  207. override func addNotification() {
  208. NotificationCenter.default.addObserver(self, selector: #selector(pageChangeNotification), name: NSNotification.Name.CPDFViewPageChanged, object: self.pdfView)
  209. }
  210. override func updateLanguage() {
  211. typeBox.title = NSLocalizedString("Source", comment: "")
  212. self.templateNameLabel.stringValue = NSLocalizedString("Name:", comment: "")
  213. batchButton.title = NSLocalizedString("Batch", comment: "");
  214. cancelButton.title = NSLocalizedString("Cancel", comment: "");
  215. colorButton.title = NSLocalizedString("Color", comment: "")
  216. fileButton.title = NSLocalizedString("File", comment: "")
  217. browseButton.title = NSLocalizedString("Choose...", comment: "")
  218. ratioLabel.stringValue = "\(NSLocalizedString("Ratio", comment: "")):"
  219. appearanceBox.title = NSLocalizedString("Appearance", comment: "")
  220. angleLabel.stringValue = "\(NSLocalizedString("Rotation", comment: "")):"
  221. alphaLabel.stringValue = "\(NSLocalizedString("Opacity", comment: "")):"
  222. pageRangeLabel.stringValue = "\(NSLocalizedString("Page Range", comment: "")):"
  223. horizontalGapLabel.stringValue = "X:"
  224. verticalGapLabel.stringValue = "Y:"
  225. templateNameLabel.stringValue = NSLocalizedString("Label", comment: "")
  226. saveToTemplateButton.title = NSLocalizedString("Add to Template", comment: "")
  227. if (self.type == .add) {
  228. self.doneButton.title = NSLocalizedString("Apply", comment: "");
  229. self.batchButton.isHidden = true
  230. } else if (self.type == .edit) {
  231. self.doneButton.title = NSLocalizedString("Apply", comment: "");
  232. self.batchButton.isHidden = true
  233. } else if (self.type == .use) {
  234. self.doneButton.title = NSLocalizedString("Save & Apply", comment: "");
  235. }
  236. }
  237. private func checkAngle() {
  238. left45IndicateView.isSelcted = false
  239. horizontalIndicateView.isSelcted = false
  240. right45IndicateView.isSelcted = false
  241. if background.rotation == -45 {
  242. right45IndicateView.isSelcted = true
  243. } else if background.rotation == 0 {
  244. horizontalIndicateView.isSelcted = true
  245. } else if background.rotation == 45 {
  246. left45IndicateView.isSelcted = true
  247. }
  248. }
  249. override func reloadData() {
  250. guard let pdfDocument = pdfDocument else { return }
  251. totalPageCountlabel.stringValue = "/ \(pdfDocument.pageCount)"
  252. templateNameTextField.stringValue = background.backgroundID
  253. colorWell.color = background.color ?? NSColor.red
  254. filePathLabel.stringValue = background.imagePath
  255. filePathLabel.placeholderString = NSLocalizedString("Select a File", comment: "")
  256. angleStepper.doubleValue = Double(-background.rotation)
  257. angleTextField.stringValue = "\(angleStepper.doubleValue)"
  258. alphaSlider.doubleValue = background.opacity
  259. alphaStepper.doubleValue = background.opacity
  260. let opacity = round(background.opacity * 100) / 100
  261. alphaTextField.stringValue = "\(Int(opacity * 100))%"
  262. ratioStepper.doubleValue = background.scale
  263. ratioTextField.stringValue = "\(Int(background.scale * 100))%"
  264. if currentType == 0 {
  265. changeTypeBoxState(true)
  266. } else {
  267. changeTypeBoxState(false)
  268. }
  269. pageRangeComboBox.delegate = nil
  270. switch background.pageRangeType {
  271. case .all:
  272. pageRangeComboBox.isEditable = false
  273. pageRangeComboBox.selectItem(at: 0)
  274. case .odd:
  275. pageRangeComboBox.isEditable = false
  276. pageRangeComboBox.selectItem(at: 1)
  277. case .even:
  278. pageRangeComboBox.isEditable = false
  279. pageRangeComboBox.selectItem(at: 2)
  280. case .other:
  281. pageRangeComboBox.isEditable = true
  282. pageRangeComboBox.selectItem(at: 3)
  283. pageRangeComboBox.stringValue = background.pagesString
  284. window?.makeFirstResponder(pageRangeComboBox)
  285. }
  286. pageRangeComboBox.delegate = self
  287. verticalStepper.doubleValue = background.verticalSpace
  288. verticalGapTextField.stringValue = "\(verticalStepper.doubleValue)"
  289. horizontalStepper.doubleValue = background.horizontalSpace
  290. horizontalGapTextField.stringValue = "\(horizontalStepper.doubleValue)"
  291. batchButton.isHidden = isHiddenBatchBtn
  292. if !isHiddenBatchBtn {
  293. self.batchButton.isHidden = type != .use
  294. }
  295. }
  296. private func changeTypeBoxState(_ isColor: Bool) {
  297. if isColor {
  298. currentType = 0
  299. colorButton.state = .on
  300. colorWell.isEnabled = true
  301. fileButton.state = .off
  302. browseButton.isEnabled = false
  303. ratioTextField.isEnabled = false
  304. ratioStepper.isEnabled = false
  305. background.color = colorWell.color
  306. background.type = .color
  307. background.scale = 1
  308. doneButton.isEnabled = true
  309. } else {
  310. currentType = 1
  311. colorButton.state = .off
  312. colorWell.isEnabled = false
  313. fileButton.state = .on
  314. browseButton.isEnabled = true
  315. ratioTextField.isEnabled = true
  316. ratioStepper.isEnabled = true
  317. background.color = NSColor.white
  318. background.type = .file
  319. background.scale = ratioStepper.doubleValue
  320. doneButton.isEnabled = filePathLabel.stringValue.count > 0
  321. }
  322. self.updatePDFView()
  323. }
  324. func updatePDFView() {
  325. // pdfView.needsDisplay = true
  326. // pdfView.layoutDocumentView()
  327. // pdfView.setNeedsDisplayForVisiblePages()
  328. // // Save to temporary path
  329. // let documentPath = NSTemporaryDirectory()
  330. // let tempPath = (documentPath as NSString).appendingPathComponent((path as NSString).lastPathComponent)
  331. // try? FileManager.default.removeItem(atPath: tempPath)
  332. //
  333. // let result = document.write(to: URL(fileURLWithPath: tempPath))
  334. // if result {
  335. // if FileManager.default.fileExists(atPath: path) {
  336. // try? FileManager.default.removeItem(atPath: path)
  337. // }
  338. // try? FileManager.default.moveItem(atPath: tempPath, toPath: path)
  339. // } else {
  340. // try? FileManager.default.removeItem(atPath: tempPath)
  341. // }
  342. // 如果前一次调用还在执行,则取消它
  343. workItem?.cancel()
  344. // 创建一个新的 DispatchWorkItem
  345. let newWorkItem = DispatchWorkItem {
  346. // 在这里执行你的方法的实际逻辑
  347. self.reloadDocument()
  348. }
  349. // 将新的 DispatchWorkItem 分配给实例变量
  350. workItem = newWorkItem
  351. // 在队列中执行 DispatchWorkItem
  352. DispatchQueue.main.async(execute: newWorkItem)
  353. }
  354. func reloadDocument() {
  355. if !self.isAllowReloadDocument {
  356. return
  357. }
  358. self.isAllowReloadDocument = false
  359. let document = CPDFDocument(url: self.pdfDocument?.documentURL)
  360. guard let document = document else { return }
  361. document.unlock(withPassword: self.pdfDocument?.password)
  362. let tBackground: CPDFBackground = document.background()
  363. tBackground.opacity = background.opacity
  364. tBackground.scale = background.scale
  365. tBackground.rotation = CGFloat(-background.rotation)
  366. tBackground.horizontalAlignment = UInt(background.horizontalMode)
  367. tBackground.verticalAlignment = UInt(background.verticalMode)
  368. tBackground.xOffset = background.horizontalSpace
  369. tBackground.yOffset = background.verticalSpace
  370. if let color = background.color {
  371. tBackground.color = color
  372. tBackground.type = .color
  373. }
  374. if background.imagePath.count != 0 {
  375. if let data = background.image {
  376. tBackground.setImage(data)
  377. } else {
  378. if let image = NSImage(contentsOfFile: background.imagePath) {
  379. tBackground.setImage(image)
  380. }
  381. }
  382. tBackground.type = .image
  383. }
  384. if background.pagesString.count != 0 {
  385. tBackground.pageString = background.pagesString
  386. } else {
  387. let pageString = "0-\(document.pageCount - 1)"
  388. tBackground.pageString = pageString
  389. }
  390. tBackground.update()
  391. self.pdfView.document = document
  392. DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + 0.1) {
  393. self.isAllowReloadDocument = true
  394. }
  395. }
  396. // Other methods...
  397. @objc func controlTextDidEndEditing(_ notification: Notification) {
  398. guard let textField = notification.object as? NSTextField else { return }
  399. switch textField {
  400. case ratioTextField:
  401. let formatter = textField.formatter as? NumberFormatter
  402. if let floatValue = formatter?.number(from: textField.stringValue)?.floatValue {
  403. ratioStepper.doubleValue = Double(floatValue)
  404. background.scale = Double(floatValue)
  405. self.updatePDFView()
  406. }
  407. case angleTextField:
  408. if let integerValue = Int(textField.stringValue) {
  409. background.rotation = CGFloat(-integerValue)
  410. angleStepper.doubleValue = Double(-background.rotation)
  411. checkAngle()
  412. self.updatePDFView()
  413. }
  414. case alphaTextField:
  415. let formatter = textField.formatter as? NumberFormatter
  416. if let floatValue = formatter?.number(from: textField.stringValue)?.floatValue {
  417. alphaSlider.doubleValue = Double(floatValue)
  418. alphaStepper.doubleValue = Double(floatValue)
  419. background.opacity = Double(floatValue)
  420. self.updatePDFView()
  421. }
  422. case pageRangeComboBox:
  423. if pageRangeComboBox.indexOfSelectedItem == -1 {
  424. if !checkPageRangeValidate(pageRangeComboBox.stringValue) {
  425. let alert = NSAlert()
  426. alert.alertStyle = .critical
  427. alert.messageText = "\(pdfDocument!.documentURL.lastPathComponent) \(NSLocalizedString("Invalid page range or the page number is out of range. Please try again.", comment: ""))"
  428. alert.runModal()
  429. window?.makeFirstResponder(pageRangeComboBox)
  430. return
  431. } else {
  432. background.pagesString = pageRangeComboBox.stringValue
  433. self.updatePDFView()
  434. }
  435. }
  436. case verticalGapTextField:
  437. if let integerValue = Int(verticalGapTextField.stringValue) {
  438. background.verticalSpace = CGFloat(integerValue)
  439. verticalStepper.doubleValue = background.verticalSpace
  440. self.updatePDFView()
  441. }
  442. case horizontalGapTextField:
  443. if let integerValue = Int(horizontalGapTextField.stringValue) {
  444. background.horizontalSpace = CGFloat(integerValue)
  445. horizontalStepper.doubleValue = background.horizontalSpace
  446. self.updatePDFView()
  447. }
  448. case currentPageIndexTextField:
  449. if let pageIndex = Int(currentPageIndexTextField.stringValue), let page = pdfDocument!.page(at: UInt(pageIndex - 1)) {
  450. pdfView.go(to: page)
  451. }
  452. default:
  453. break
  454. }
  455. if filePathLabel.stringValue.count > 0 {
  456. doneButton.isEnabled = true
  457. }
  458. }
  459. @objc func comboBoxSelectionDidChange(_ notification: Notification) {
  460. guard notification.object as? NSComboBox == pageRangeComboBox else { return }
  461. pageRangeComboBox.isEditable = false
  462. switch pageRangeComboBox.indexOfSelectedItem {
  463. case 0:
  464. background.pageRangeType = .all
  465. case 1:
  466. background.pageRangeType = .odd
  467. case 2:
  468. background.pageRangeType = .even
  469. default:
  470. background.pageRangeType = .other
  471. pageRangeComboBox.stringValue = ""
  472. pageRangeComboBox.isEditable = true
  473. window?.makeFirstResponder(pageRangeComboBox)
  474. }
  475. if filePathLabel.stringValue.count > 0 {
  476. doneButton.isEnabled = true
  477. }
  478. }
  479. func checkPageRangeValidate(_ pageRangeString: String) -> Bool {
  480. var fileAttribute = self._fileAttri
  481. if fileAttribute == nil {
  482. fileAttribute = KMFileAttribute()
  483. fileAttribute?.password = self.pdfDocument?.password ?? ""
  484. fileAttribute?.filePath = self.pdfDocument?.documentURL.path ?? ""
  485. self._fileAttri = fileAttribute
  486. }
  487. // fileAttribute?.filePath = pdfDocument!.documentURL.path
  488. fileAttribute?.bAllPage = false
  489. fileAttribute?.pagesString = pageRangeComboBox.stringValue
  490. var pageRange: KMPageRange = .all
  491. let pageRangeType: KMWatermarkeModelPageRangeType = background.pageRangeType
  492. if pageRangeType == .all {
  493. pageRange = .all
  494. } else if pageRangeType == .even {
  495. pageRange = .even
  496. } else if pageRangeType == .odd {
  497. pageRange = .odd
  498. } else if pageRangeType == .other {
  499. pageRange = .custom
  500. }
  501. fileAttribute?.pagesType = pageRange
  502. return fileAttribute!.fetchSelectPages().count != 0
  503. }
  504. func saveAsPDF(with background: KMBackgroundModel, to path: String, autoOpen: Bool) {
  505. DispatchQueue.global(qos: .default).async { [unowned self] in
  506. var filePath = self.pdfDocument!.documentURL?.path
  507. let password = self.password
  508. if filePath == nil {
  509. let writeSuccess = self.pdfDocument!.write(to: URL(fileURLWithPath: kNewDocumentTempSavePath(NSLocalizedString("Untitled", comment: ""))))
  510. if writeSuccess {
  511. self.pdfDocument = CPDFDocument(url: URL(fileURLWithPath: kNewDocumentTempSavePath(NSLocalizedString("Untitled", comment: ""))))!
  512. filePath = self.pdfDocument!.documentURL?.path
  513. }
  514. }
  515. let document = CPDFDocument(url: URL(fileURLWithPath: filePath!))!
  516. document.unlock(withPassword: password)
  517. let tBackground: CPDFBackground = document.background()
  518. tBackground.opacity = background.opacity
  519. tBackground.scale = background.scale
  520. tBackground.rotation = CGFloat(-background.rotation)
  521. tBackground.horizontalAlignment = UInt(background.horizontalMode)
  522. tBackground.verticalAlignment = UInt(background.verticalMode)
  523. tBackground.xOffset = background.horizontalSpace
  524. tBackground.yOffset = background.verticalSpace
  525. if let color = background.color {
  526. tBackground.color = color
  527. tBackground.type = .color
  528. } else if background.imagePath.count != 0 {
  529. let image = NSImage(contentsOfFile: background.imagePath)!
  530. tBackground.setImage(image)
  531. tBackground.type = .image
  532. }
  533. if background.pagesString.count != 0 {
  534. tBackground.pageString = background.pagesString
  535. } else {
  536. let pageString = "0-\(document.pageCount - 1)"
  537. tBackground.pageString = pageString
  538. }
  539. tBackground.update()
  540. // Save to temporary path
  541. let documentPath = NSTemporaryDirectory()
  542. let tempPath = (documentPath as NSString).appendingPathComponent((path as NSString).lastPathComponent)
  543. try? FileManager.default.removeItem(atPath: tempPath)
  544. let result = document.write(to: URL(fileURLWithPath: tempPath))
  545. if result {
  546. if FileManager.default.fileExists(atPath: path) {
  547. try? FileManager.default.removeItem(atPath: path)
  548. }
  549. try? FileManager.default.moveItem(atPath: tempPath, toPath: path)
  550. } else {
  551. try? FileManager.default.removeItem(atPath: tempPath)
  552. }
  553. if result {
  554. DispatchQueue.main.async {
  555. let needSave = self.saveToTemplateButton.state == .on
  556. if needSave {
  557. if self.checkPageRangeValidate(self.pageRangeComboBox.stringValue) && self.pageRangeComboBox.indexOfSelectedItem == -1 {
  558. self.background.pagesString = self.pageRangeComboBox.stringValue
  559. }
  560. KMBackgroundManager.defaultManager.addTemplate(model: self.background)
  561. NotificationCenter.default.post(name: NSNotification.Name("KMBatchOperateWatermarksNotification"), object: self)
  562. }
  563. self.cancelAction?(self)
  564. if autoOpen {
  565. NSDocumentController.shared.openDocument(withContentsOf: URL(fileURLWithPath: path), display: true) { _, _, _ in }
  566. } else {
  567. NSWorkspace.shared.selectFile(path, inFileViewerRootedAtPath: "")
  568. }
  569. }
  570. }
  571. }
  572. }
  573. static func saveAsPDFRemoveAllBackground(PDFDocument: CPDFDocument, password: String?, toPath path: String, completionHandler handler: ((Int) -> Void)?) {
  574. DispatchQueue.global(qos: .default).async {
  575. guard let filePath = PDFDocument.documentURL?.path else {
  576. return
  577. }
  578. let document = CPDFDocument(url: URL(fileURLWithPath: filePath))
  579. if let password = password {
  580. document?.unlock(withPassword: password)
  581. }
  582. let tBackground: CPDFBackground = document!.background()
  583. tBackground.clear()
  584. // Save to a temporary path
  585. let documentPath = NSTemporaryDirectory()
  586. let tempPath = (documentPath as NSString).appendingPathComponent((path as NSString).lastPathComponent)
  587. if FileManager.default.fileExists(atPath: tempPath) {
  588. try? FileManager.default.removeItem(atPath: tempPath)
  589. }
  590. if let result = document?.write(to: URL(fileURLWithPath: tempPath)) {
  591. if FileManager.default.fileExists(atPath: path) {
  592. try? FileManager.default.removeItem(atPath: path)
  593. }
  594. try? FileManager.default.moveItem(atPath: tempPath, toPath: path)
  595. DispatchQueue.main.async {
  596. handler?(1)
  597. }
  598. } else {
  599. try? FileManager.default.removeItem(atPath: tempPath)
  600. DispatchQueue.main.async {
  601. handler?(0)
  602. }
  603. }
  604. }
  605. }
  606. func kNewDocumentTempSavePath(_ fileName: String) -> String {
  607. let searchPath = NSSearchPathForDirectoriesInDomains(.applicationSupportDirectory, .userDomainMask, true).last
  608. let append1 = searchPath?.stringByAppendingPathComponent(Bundle.main.bundleIdentifier!)
  609. let append2 = append1!.stringByAppendingPathComponent(String(format: "%@", fileName))
  610. return append2
  611. }
  612. func isDamageImage(_ image: NSImage, imagePath path: String) -> Bool {
  613. let addImageAnnotation = (NSSearchPathForDirectoriesInDomains(.applicationSupportDirectory, .userDomainMask, true).last! as NSString).appendingPathComponent(Bundle.main.bundleIdentifier!).stringByAppendingPathComponent("addImageAnnotation")
  614. if !FileManager.default.fileExists(atPath: addImageAnnotation) {
  615. try? FileManager.default.createDirectory(atPath: addImageAnnotation, withIntermediateDirectories: false, attributes: nil)
  616. }
  617. let data = image.tiffRepresentation!
  618. let imageRep = NSBitmapImageRep(data: data)!
  619. imageRep.size = image.size
  620. var imageData: Data?
  621. if path.lowercased() == "png" {
  622. imageData = imageRep.representation(using: .png, properties: [:])
  623. } else {
  624. imageData = imageRep.representation(using: .jpeg, properties: [:])
  625. }
  626. let rPath = (addImageAnnotation as NSString).appendingPathComponent(tagString() + ".png")
  627. return ((try? imageData?.write(to: URL(fileURLWithPath: rPath), options: .atomic)) == nil)
  628. }
  629. func tagString() -> String {
  630. let dateFormatter = DateFormatter()
  631. dateFormatter.dateFormat = "yyMMddHHmmss"
  632. return String(format: "%@%04d", dateFormatter.string(from: Date()), arc4random() % 10000)
  633. }
  634. // ... (remaining methods)
  635. @objc func pageChangeNotification(notification: NSNotification) {
  636. currentPageIndexTextField.stringValue = "\(pdfView.currentPage().pageIndex() + 1)"
  637. }
  638. }
  639. extension KMAddBackgroundView {
  640. @IBAction func cancelButtonAction(_ sender: Any) {
  641. guard let callBack = cancelAction else { return }
  642. callBack(self)
  643. }
  644. @IBAction func buttonClicked_SwitchBackgroundType(_ sender: Any) {
  645. changeTypeBoxState(sender as? NSObject == colorButton)
  646. }
  647. // @IBAction func colorWellChanged(_ sender: NSColorWell) {
  648. // background.color = sender.color
  649. // self.updatePDFView()
  650. // }
  651. @objc func colorWellDidChange(_ sender: NSColorWell) {
  652. background.color = sender.color
  653. self.updatePDFView()
  654. }
  655. @IBAction func radioStepperAction(_ sender: NSStepper) {
  656. ratioTextField.stringValue = "\(Int(sender.doubleValue * 100))%"
  657. background.scale = sender.doubleValue
  658. self.updatePDFView()
  659. if filePathLabel.stringValue.count > 0 {
  660. doneButton.isEnabled = true
  661. }
  662. }
  663. @IBAction func angleStepperAction(_ sender: NSStepper) {
  664. angleTextField.stringValue = "\(sender.doubleValue)"
  665. background.rotation = CGFloat(Int(sender.doubleValue))
  666. checkAngle()
  667. self.updatePDFView()
  668. if filePathLabel.stringValue.count > 0 {
  669. doneButton.isEnabled = true
  670. }
  671. }
  672. @IBAction func alphaSliderAction(_ sender: NSSlider) {
  673. background.opacity = sender.doubleValue
  674. alphaStepper.doubleValue = sender.doubleValue
  675. alphaTextField.stringValue = "\(Int(sender.doubleValue * 100))%"
  676. if filePathLabel.stringValue.count > 0 {
  677. doneButton.isEnabled = true
  678. }
  679. self.updatePDFView()
  680. }
  681. @IBAction func verticalStepperAction(_ sender: NSStepper) {
  682. verticalGapTextField.stringValue = "\(sender.doubleValue)"
  683. background.verticalSpace = sender.doubleValue
  684. self.updatePDFView()
  685. if filePathLabel.stringValue.count > 0 {
  686. doneButton.isEnabled = true
  687. }
  688. }
  689. @IBAction func horizentalStepperAction(_ sender: NSStepper) {
  690. horizontalGapTextField.stringValue = "\(sender.doubleValue)"
  691. background.horizontalSpace = sender.doubleValue
  692. self.updatePDFView()
  693. if filePathLabel.stringValue.count > 0 {
  694. doneButton.isEnabled = true
  695. }
  696. }
  697. @IBAction func alphaSteperAction(_ sender: NSStepper) {
  698. alphaTextField.stringValue = "\(Int(sender.doubleValue * 100))%"
  699. background.opacity = sender.doubleValue
  700. alphaSlider.doubleValue = sender.doubleValue
  701. self.updatePDFView()
  702. if filePathLabel.stringValue.count > 0 {
  703. doneButton.isEnabled = true
  704. }
  705. }
  706. @IBAction func goPrevious(_ sender: Any) {
  707. if pdfView.canGoToPreviousPage() {
  708. pdfView.goToPreviousPage(nil)
  709. }
  710. // let index = pdfDocument!.index(for: pdfView.currentPage())
  711. // currentPageIndexTextField.stringValue = "\(index + 1)"
  712. }
  713. @IBAction func goNext(_ sender: Any) {
  714. if pdfView.canGoToNextPage() {
  715. pdfView.goToNextPage(nil)
  716. }
  717. // let index = pdfDocument!.index(for: pdfView.currentPage())
  718. // currentPageIndexTextField.stringValue = "\(index + 1)"
  719. }
  720. @IBAction func buttonClicked_BrowserFile(_ sender: Any) {
  721. let openPanel = NSOpenPanel()
  722. openPanel.canChooseDirectories = false
  723. openPanel.canChooseFiles = true
  724. openPanel.allowsMultipleSelection = false
  725. openPanel.allowedFileTypes = ["jpg", "cur", "bmp", "jpeg", "gif", "png", "tiff", "tif", "ico", "icns", "tga", "psd", "eps", "hdr", "jp2", "jpc", "pict", "sgi", "pdf"]
  726. openPanel.beginSheetModal(for: window!) { (result) in
  727. if result == .OK {
  728. guard let url = openPanel.url else { return }
  729. let filePath: NSString = url.path as NSString
  730. if filePath.pathExtension.lowercased() == "pdf" {
  731. let pdf = CPDFDocument(url: url)
  732. if pdf?.isEncrypted == true {
  733. return
  734. }
  735. }
  736. if let image = NSImage(contentsOfFile: url.path), !self.isDamageImage(image, imagePath: url.path) {
  737. self.filePathLabel.stringValue = url.path
  738. self.doneButton.isEnabled = true
  739. self.background.imagePath = url.path
  740. self.background.image = image
  741. self.background.backgroundID = url.path.lastPathComponent.deletingPathExtension
  742. self.templateNameTextField.stringValue = url.path.lastPathComponent.deletingPathExtension
  743. self.self.updatePDFView()
  744. }
  745. }
  746. }
  747. }
  748. // ... (remaining IBActions)
  749. @IBAction func buttonClicked_Batch(_ sender: Any) {
  750. if background.type == .color {
  751. background.imagePath = ""
  752. } else {
  753. background.color = nil
  754. if background.image == nil {
  755. return
  756. }
  757. }
  758. if templateNameTextField.stringValue.count < 1 {
  759. background.backgroundID = initialID
  760. } else {
  761. background.backgroundID = templateNameTextField.stringValue
  762. }
  763. let needSave = saveToTemplateButton.state == .on
  764. if needSave {
  765. KMBackgroundManager.defaultManager.addTemplate(model: background)
  766. }
  767. operateCallBack?(background, currentType)
  768. }
  769. @IBAction func buttonClicked_Done(_ sender: Any) {
  770. guard let pdfDocument = pdfDocument else { return }
  771. if background.type == .color {
  772. background.imagePath = ""
  773. } else {
  774. background.color = nil
  775. if background.image == nil {
  776. return
  777. }
  778. }
  779. if templateNameTextField.stringValue.isEmpty {
  780. background.backgroundID = initialID
  781. } else {
  782. background.backgroundID = templateNameTextField.stringValue
  783. }
  784. // Avoid showing the page range alert twice
  785. if checkPageRangeValidate(pageRangeComboBox.stringValue) {
  786. background.pagesString = pageRangeComboBox.stringValue
  787. self.updatePDFView()
  788. window?.makeFirstResponder(self)
  789. }
  790. let needSave = saveToTemplateButton.state == .on
  791. var pages = [Int]()
  792. switch pageRangeComboBox.indexOfSelectedItem {
  793. case 0:
  794. pages = Array(0..<Int(pdfDocument.pageCount))
  795. case 1:
  796. pages = Array(stride(from: 0, to: Int(pdfDocument.pageCount), by: 2))
  797. case 2:
  798. pages = Array(stride(from: 1, to: Int(pdfDocument.pageCount), by: 2))
  799. default:
  800. var fileAttribute = self._fileAttri
  801. if fileAttribute == nil {
  802. fileAttribute = KMFileAttribute()
  803. fileAttribute?.password = self.pdfDocument?.password ?? ""
  804. fileAttribute?.filePath = self.pdfDocument?.documentURL.path ?? ""
  805. self._fileAttri = fileAttribute
  806. }
  807. // fileAttribute.filePath = pdfDocument.documentURL?.path ?? ""
  808. fileAttribute?.bAllPage = false
  809. fileAttribute?.pagesType = .custom
  810. // fileAttribute.password = pdfDocument.password ?? ""
  811. fileAttribute?.pagesString = pageRangeComboBox.stringValue
  812. let selectPages = fileAttribute?.fetchSelectPages() ?? []
  813. if selectPages.count != 0 {
  814. pages = selectPages.map { $0 - 1 }
  815. } else {
  816. let alert = NSAlert()
  817. alert.alertStyle = .critical
  818. alert.messageText = "\(fileAttribute?.filePath.lastPathComponent ?? "") \(NSLocalizedString("Invalid page range or the page number is out of range. Please try again.", comment: ""))"
  819. alert.runModal()
  820. return
  821. }
  822. }
  823. background.pagesString = pages.isEmpty ? "" : pages.map { "\($0)" }.joined(separator: ",")
  824. switch type {
  825. case .add:
  826. if needSave {
  827. if checkPageRangeValidate(pageRangeComboBox.stringValue) && pageRangeComboBox.indexOfSelectedItem == -1 {
  828. background.pagesString = pageRangeComboBox.stringValue
  829. }
  830. KMBackgroundManager.defaultManager.addTemplate(model: background)
  831. }
  832. operateCallBack?(background, currentType)
  833. case .edit:
  834. if needSave {
  835. if checkPageRangeValidate(pageRangeComboBox.stringValue) && pageRangeComboBox.indexOfSelectedItem == -1 {
  836. background.pagesString = pageRangeComboBox.stringValue
  837. }
  838. originalBackground = (background.copy() as? KMBackgroundModel)!
  839. // KMBackgroundManager.defaultManager.store()
  840. }
  841. operateCallBack?(originalBackground, currentType)
  842. case .use:
  843. let fileName = "\(pdfDocument.documentURL?.deletingPathExtension().lastPathComponent ?? NSLocalizedString("Untitled", comment: ""))_Background"
  844. let savePanelAccessoryViewController = KMSavePanelAccessoryController()
  845. let savePanel = NSSavePanel()
  846. savePanel.nameFieldStringValue = fileName
  847. savePanel.allowedFileTypes = ["pdf"]
  848. savePanel.accessoryView = savePanelAccessoryViewController.view
  849. savePanel.beginSheetModal(for: window!) { result in
  850. if result == .OK {
  851. self.saveAsPDF(with: self.background, to: savePanel.url?.path ?? "", autoOpen: savePanelAccessoryViewController.openAutomaticButton.state == .on ? true : false)
  852. }
  853. }
  854. }
  855. }
  856. }