KMEditImageController.swift 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606
  1. //
  2. // KMEditImageController.swift
  3. // PDF Reader Pro
  4. //
  5. // Created by Niehaoyu on 2024/11/14.
  6. //
  7. import Cocoa
  8. import KMComponentLibrary
  9. class KMEditImageController: NSViewController {
  10. @IBOutlet var scrollView: NSScrollView!
  11. @IBOutlet var contendView: NSView!
  12. @IBOutlet var sizeBGView: NSView!
  13. @IBOutlet var sizeLabel: NSTextField!
  14. @IBOutlet var sizeSyncButton: ComponentButton!
  15. @IBOutlet var sizeWidthInput: ComponentInputNumber!
  16. @IBOutlet var sizeHeightInput: ComponentInputNumber!
  17. @IBOutlet var rotateBGView: NSView!
  18. @IBOutlet var rotateLabel: NSTextField!
  19. @IBOutlet var rotateSelect: ComponentSelect!
  20. @IBOutlet var rotateLeftButton: ComponentButton!
  21. @IBOutlet var rotateRightButton: ComponentButton!
  22. @IBOutlet var flipVerticalButton: ComponentButton!
  23. @IBOutlet var flipHorizontalButton: ComponentButton!
  24. @IBOutlet var opacityBGView: NSView!
  25. @IBOutlet var opacityLabel: NSTextField!
  26. @IBOutlet var opacitySlider: ComponentSlider!
  27. @IBOutlet var opacitySelect: ComponentSelect!
  28. @IBOutlet var cropButton: ComponentButton!
  29. @IBOutlet var replaceButton: ComponentButton!
  30. @IBOutlet var extractButton: ComponentButton!
  31. @IBOutlet var extractBtnTopConst: NSLayoutConstraint!
  32. @IBOutlet var alignmentBGView: NSView!
  33. @IBOutlet var emptyView: ComponentEmpty!
  34. private var syncChangeBounds: Bool = true //同步修改宽高
  35. private var groupView: ComponentGroup!
  36. private var alignmentController: KMNAlignmentController?
  37. private var areas: [CPDFEditImageArea] = []
  38. var pdfView: CPDFListView?
  39. //MARK: - func
  40. override func viewDidAppear() {
  41. super.viewDidAppear()
  42. opacitySlider.reloadData()
  43. }
  44. override func viewDidLoad() {
  45. super.viewDidLoad()
  46. setupProperty()
  47. }
  48. func setupProperty() {
  49. //Size
  50. sizeLabel.stringValue = KMLocalizedString("Size")
  51. sizeLabel.textColor = ComponentLibrary.shared.getComponentColorFromKey("colorText/2")
  52. sizeLabel.font = ComponentLibrary.shared.getFontFromKey("mac/body-s-medium")
  53. sizeSyncButton.properties = ComponentButtonProperty(type: .text_gray, size: .xxs, onlyIcon: true, icon: NSImage(named: "sync_Change_unlock"), keepPressState: true)
  54. sizeSyncButton.properties.propertyInfo.leftIcon_press = NSImage(named: "sync_Change_lock")
  55. sizeSyncButton.toolTip = KMLocalizedString("Lock")
  56. sizeSyncButton.setTarget(self, action: #selector(sizeSyncButtonClicked(_:)))
  57. sizeSyncButton.reloadData()
  58. sizeWidthInput.properties = ComponentInputNumberProperty(alignment: .center,
  59. size: .s,
  60. state: .normal,
  61. showPrefix: true,
  62. leftIcon: NSImage(named: "w_icon"),
  63. showSuffix: false,
  64. minSize: 1,
  65. maxSize: 1000,
  66. text:"100",
  67. valueType: .floatType)
  68. sizeWidthInput.delegate = self
  69. sizeHeightInput.properties = ComponentInputNumberProperty(alignment: .center,
  70. size: .s,
  71. state: .normal,
  72. showPrefix: true,
  73. leftIcon: NSImage(named: "h_icon"),
  74. showSuffix: false,
  75. minSize: 1,
  76. maxSize: 1000,
  77. text:"100",
  78. valueType: .floatType)
  79. sizeHeightInput.delegate = self
  80. //Rotate
  81. rotateLabel.stringValue = KMLocalizedString("Rotate & Flip")
  82. rotateLabel.textColor = ComponentLibrary.shared.getComponentColorFromKey("colorText/2")
  83. rotateLabel.font = ComponentLibrary.shared.getFontFromKey("mac/body-s-medium")
  84. rotateSelect.properties = ComponentSelectProperties(size: .s,
  85. state: .normal,
  86. creatable: true,
  87. text: "100",
  88. textUnit: "°",
  89. regexString: "0123456789-")
  90. if true {
  91. var opacityItems: [ComponentMenuitemProperty] = []
  92. for string in ["0°", "45°", "-45°", "90°", "-90°"] {
  93. let item = ComponentMenuitemProperty(type: .normal, text: string)
  94. opacityItems.append(item)
  95. }
  96. rotateSelect.updateMenuItemsArr(opacityItems)
  97. }
  98. rotateSelect.delegate = self
  99. rotateLeftButton.properties = ComponentButtonProperty(type: .text_gray, size: .s, onlyIcon: true, icon: NSImage(named: "pageEdit_rotateRight"), keepPressState: false)
  100. rotateLeftButton.setTarget(self, action: #selector(buttonClicked(_:)))
  101. rotateRightButton.properties = ComponentButtonProperty(type: .text_gray, size: .s, onlyIcon: true, icon: NSImage(named: "pageEdit_rotateLeft"), keepPressState: false)
  102. rotateRightButton.setTarget(self, action: #selector(buttonClicked(_:)))
  103. flipVerticalButton.properties = ComponentButtonProperty(type: .text_gray, size: .s, onlyIcon: true, icon: NSImage(named: "flipVertical"), keepPressState: false)
  104. flipVerticalButton.setTarget(self, action: #selector(buttonClicked(_:)))
  105. flipHorizontalButton.properties = ComponentButtonProperty(type: .text_gray, size: .s, onlyIcon: true, icon: NSImage(named: "flipHorizontal"), keepPressState: false)
  106. flipHorizontalButton.setTarget(self, action: #selector(buttonClicked(_:)))
  107. //Opacity
  108. opacityLabel.stringValue = KMLocalizedString("Opacity")
  109. opacityLabel.textColor = ComponentLibrary.shared.getComponentColorFromKey("colorText/2")
  110. opacityLabel.font = ComponentLibrary.shared.getFontFromKey("mac/body-s-medium")
  111. opacitySlider.properties = ComponentSliderProperty(size: .m, percent: 1)
  112. opacitySlider.delegate = self
  113. opacitySelect.properties = ComponentSelectProperties(size: .s,
  114. state: .normal,
  115. creatable: true,
  116. text: "100",
  117. textUnit: "%",
  118. regexString: "0123456789%")
  119. if true {
  120. var opacityItems: [ComponentMenuitemProperty] = []
  121. for string in ["25%", "50%", "75%", "100%"] {
  122. let item = ComponentMenuitemProperty(type: .normal, text: string)
  123. opacityItems.append(item)
  124. }
  125. opacitySelect.updateMenuItemsArr(opacityItems)
  126. }
  127. opacitySelect.delegate = self
  128. cropButton.properties = ComponentButtonProperty(type: .default_tertiary, size: .s, buttonText: KMLocalizedString("Crop"), keepPressState: false)
  129. cropButton.setTarget(self, action: #selector(buttonClicked(_:)))
  130. replaceButton.properties = ComponentButtonProperty(type: .default_tertiary, size: .s, buttonText: KMLocalizedString("Replace"), keepPressState: false)
  131. replaceButton.setTarget(self, action: #selector(buttonClicked(_:)))
  132. extractButton.properties = ComponentButtonProperty(type: .default_tertiary, size: .s, buttonText: KMLocalizedString("Extract"), keepPressState: true)
  133. extractButton.setTarget(self, action: #selector(buttonClicked(_:)))
  134. if alignmentController == nil {
  135. alignmentController = KMNAlignmentController.init()
  136. }
  137. alignmentController?.view.frame = alignmentBGView.bounds
  138. alignmentController?.view.autoresizingMask = [.width, .height]
  139. alignmentController?.delegate = self
  140. alignmentBGView.addSubview(alignmentController!.view)
  141. emptyView.properties = ComponentEmptyProperty(emptyType: .addImage,
  142. text: KMLocalizedString("Add Image"),
  143. subText: KMLocalizedString("Click or Select an area on a page for a new image to be inserted."))
  144. }
  145. func reloadData() {
  146. areas = pdfView?.km_editingImageAreas() ?? []
  147. emptyView.isHidden = true
  148. contendView.isHidden = false
  149. if areas.count == 0 {
  150. emptyView.isHidden = false
  151. contendView.isHidden = true
  152. } else if areas.count == 1 {
  153. cropButton.isHidden = false
  154. replaceButton.isHidden = false
  155. extractBtnTopConst.constant = 96
  156. alignmentBGView.isHidden = true
  157. } else if areas.count > 1 {
  158. cropButton.isHidden = true
  159. replaceButton.isHidden = true
  160. extractBtnTopConst.constant = 16
  161. alignmentBGView.isHidden = false
  162. if areas.count == 2 {
  163. alignmentController?.updateMulti(false)
  164. } else {
  165. alignmentController?.updateMulti(true)
  166. }
  167. }
  168. var first_area: CPDFEditImageArea?
  169. if areas.count > 0 {
  170. first_area = areas.first
  171. }
  172. //refreshData
  173. var boundsWidth: CGFloat?
  174. var boundsHeight: CGFloat?
  175. var rotateAngle: CGFloat?
  176. var opacity_value: CGFloat?
  177. if areas.count == 0 {
  178. } else if areas.count == 1, let area = first_area {
  179. boundsWidth = area.bounds.size.width
  180. boundsHeight = area.bounds.size.height
  181. if let rotates = pdfView?.km_editAreasRotates([area]) {
  182. rotateAngle = rotates.first ?? 1
  183. }
  184. if let opacitys = pdfView?.km_editAreasOpacitys([area]) {
  185. var opacity = opacitys.first ?? 1
  186. opacity = min(1, opacity)
  187. opacity = max(0, opacity)
  188. opacity_value = opacity
  189. }
  190. } else if areas.count > 1, let area = first_area {
  191. if CPDFEditArea.isMultiBoundsWidth(areas) == true {
  192. boundsWidth = nil
  193. } else {
  194. boundsWidth = area.bounds.size.width
  195. }
  196. if CPDFEditArea.isMultiBoundsHeight(areas) == true {
  197. boundsHeight = nil
  198. } else {
  199. boundsHeight = area.bounds.size.height
  200. }
  201. if CPDFEditArea.isMultiAngle(areas, inPDFView: pdfView) == true {
  202. rotateAngle = nil
  203. } else {
  204. rotateAngle = pdfView?.getRotateWith(area)
  205. }
  206. if CPDFEditArea.isMultiOpacity(areas, inPDFView: pdfView) == true {
  207. opacity_value = nil
  208. } else {
  209. var firstValue = pdfView?.opacityByRange(for: area) ?? 0
  210. if firstValue < 0 {
  211. firstValue = pdfView?.opacity(for: area) ?? 0
  212. }
  213. opacity_value = firstValue
  214. }
  215. }
  216. if let value = boundsWidth {
  217. sizeWidthInput.properties.text = String(format: "%.1f", value)
  218. sizeWidthInput.properties.multiState = false
  219. } else {
  220. sizeWidthInput.properties.text = "-"
  221. sizeWidthInput.properties.multiState = true
  222. }
  223. if let value = boundsHeight {
  224. sizeHeightInput.properties.text = String(format: "%.1f", value)
  225. sizeHeightInput.properties.multiState = false
  226. } else {
  227. sizeHeightInput.properties.text = "-"
  228. sizeHeightInput.properties.multiState = true
  229. }
  230. sizeWidthInput.properties.isDisabled = pdfView?.isEditImage == true
  231. sizeWidthInput.reloadData()
  232. sizeHeightInput.properties.isDisabled = pdfView?.isEditImage == true
  233. sizeHeightInput.reloadData()
  234. sizeSyncButton.properties.isDisabled = pdfView?.isEditImage == true
  235. if syncChangeBounds {
  236. sizeSyncButton.properties.state = .pressed
  237. } else {
  238. sizeSyncButton.properties.state = .normal
  239. }
  240. sizeSyncButton.reloadData()
  241. rotateSelect.properties.isDisabled = pdfView?.isEditImage == true
  242. rotateLeftButton.properties.isDisabled = pdfView?.isEditImage == true
  243. rotateLeftButton.reloadData()
  244. rotateRightButton.properties.isDisabled = pdfView?.isEditImage == true
  245. rotateRightButton.reloadData()
  246. flipVerticalButton.properties.isDisabled = pdfView?.isEditImage == true
  247. flipVerticalButton.reloadData()
  248. flipHorizontalButton.properties.isDisabled = pdfView?.isEditImage == true
  249. flipHorizontalButton.reloadData()
  250. opacitySlider.properties.isDisabled = pdfView?.isEditImage == true
  251. opacitySlider.reloadData()
  252. opacitySelect.properties.isDisabled = pdfView?.isEditImage == true
  253. replaceButton.properties.isDisabled = pdfView?.isEditImage == true
  254. replaceButton.reloadData()
  255. extractButton.properties.isDisabled = pdfView?.isEditImage == true
  256. extractButton.reloadData()
  257. if let value = rotateAngle {
  258. rotateSelect.properties.text = String(format: "%.0f", value)
  259. } else {
  260. rotateSelect.properties.text = "-"
  261. }
  262. rotateSelect.reloadData()
  263. if let value = opacity_value {
  264. opacitySelect.properties.text = String(format: "%.0f", value*100)
  265. opacitySlider.properties.percent = value
  266. } else {
  267. opacitySelect.properties.text = "-"
  268. opacitySlider.properties.percent = 0
  269. }
  270. opacitySelect.reloadData()
  271. opacitySlider.reloadData()
  272. if pdfView?.isEditImage == true {
  273. cropButton.properties.buttonText = KMLocalizedString("Cancel Crop")
  274. } else {
  275. cropButton.properties.buttonText = KMLocalizedString("Crop")
  276. }
  277. cropButton.reloadData()
  278. }
  279. //MARK: - Action
  280. @objc func sizeSyncButtonClicked(_ sender: ComponentButton) {
  281. syncChangeBounds = !syncChangeBounds
  282. if syncChangeBounds {
  283. sizeSyncButton.properties.state = .pressed
  284. } else {
  285. sizeSyncButton.properties.state = .normal
  286. }
  287. sizeSyncButton.reloadData()
  288. }
  289. @objc func buttonClicked(_ sender: ComponentButton) {
  290. if sender == rotateLeftButton {
  291. self.pdfView?.leftRotateAction()
  292. } else if sender == rotateRightButton {
  293. self.pdfView?.rightRotateAction()
  294. } else if sender == flipVerticalButton {
  295. self.pdfView?.reverseYAction()
  296. } else if sender == flipHorizontalButton {
  297. self.pdfView?.reverseXAction()
  298. } else if sender == cropButton {
  299. if pdfView?.isEditImage == true {
  300. pdfView?.cropCancelAction()
  301. } else {
  302. pdfView?.cropAction()
  303. }
  304. } else if sender == replaceButton {
  305. if let areas = self.pdfView?.km_editingImageAreas() {
  306. let panel = NSOpenPanel()
  307. panel.allowsMultipleSelection = false
  308. panel.allowedFileTypes = ["png","jpg"]
  309. panel.beginSheetModal(for: NSApp.mainWindow!) { response in
  310. if response == .OK {
  311. let openPath = panel.url?.path
  312. for area in areas {
  313. self.pdfView?.replace(area, imagePath: openPath!)
  314. }
  315. }
  316. }
  317. }
  318. } else if sender == extractButton {
  319. extractAction()
  320. }
  321. reloadData()
  322. }
  323. func extractAction() {
  324. var menuItemArr: [ComponentMenuitemProperty] = []
  325. let items: [(String, String)] = [("jpg", "export_jpg_Key"),
  326. ("png", "export_png_Key"),
  327. ("pdf", "export_pdf_Key")]
  328. for (i, value) in items {
  329. let properties_Menuitem: ComponentMenuitemProperty = ComponentMenuitemProperty(multipleSelect: false,
  330. itemSelected: false,
  331. keyEquivalent: nil,
  332. text: KMLocalizedString(i),
  333. identifier: value)
  334. menuItemArr.append(properties_Menuitem)
  335. }
  336. if groupView == nil {
  337. groupView = ComponentGroup.createFromNib(in: ComponentLibrary.shared.componentBundle())
  338. }
  339. groupView.groupDelegate = self
  340. groupView?.frame = CGRectMake(0, 0, extractButton.frame.size.width, 36*3+8)
  341. groupView.updateGroupInfo(menuItemArr)
  342. if let point: CGPoint = extractButton.superview?.convert(extractButton.frame.origin, to: self.view.window?.contentView) {
  343. groupView.showWithPoint(CGPoint(x: point.x, y: point.y - CGRectGetHeight(groupView.frame)-4), relativeTo: nil)
  344. }
  345. }
  346. //MARK: - MouseEvent
  347. override func mouseDown(with event: NSEvent) {
  348. super.mouseDown(with: event)
  349. view.window?.makeFirstResponder(nil)
  350. }
  351. }
  352. //MARK: - ComponentInputNumberDelegate
  353. extension KMEditImageController: ComponentInputNumberDelegate {
  354. func componentInputNumberDidFinishEditing(inputNumber: ComponentInputNumber?) {
  355. if areas.count == 0 {
  356. return
  357. }
  358. var textValue = inputNumber?.properties.text.stringToCGFloat() ?? 0
  359. if inputNumber == sizeWidthInput {
  360. let areas = self.pdfView?.km_editingImageAreas() ?? []
  361. for area in areas {
  362. pdfView?.updateArea(area, newWidth: textValue, syncChangeBounds)
  363. }
  364. } else if inputNumber == sizeHeightInput {
  365. let areas = self.pdfView?.km_editingImageAreas() ?? []
  366. for area in areas {
  367. pdfView?.updateArea(area, newHeight: textValue, syncChangeBounds)
  368. }
  369. }
  370. reloadData()
  371. }
  372. func componentInputNumberDidIncrease(inputNumber: ComponentInputNumber?) {
  373. if areas.count == 0 {
  374. return
  375. }
  376. //只处理多选状态
  377. var textValue = inputNumber?.properties.text.stringToCGFloat() ?? 0
  378. if inputNumber == sizeWidthInput {
  379. let areas = self.pdfView?.km_editingImageAreas() ?? []
  380. for area in areas {
  381. textValue = area.bounds.size.width + 1
  382. pdfView?.updateArea(area, newWidth: textValue, syncChangeBounds)
  383. }
  384. } else if inputNumber == sizeHeightInput {
  385. let areas = self.pdfView?.km_editingImageAreas() ?? []
  386. for area in areas {
  387. textValue = area.bounds.size.height + 1
  388. pdfView?.updateArea(area, newHeight: textValue, syncChangeBounds)
  389. }
  390. }
  391. reloadData()
  392. }
  393. func componentInputNumberDidDecrease(inputNumber: ComponentInputNumber?) {
  394. if areas.count == 0 {
  395. return
  396. }
  397. //只处理多选状态
  398. var textValue = inputNumber?.properties.text.stringToCGFloat() ?? 0
  399. if inputNumber == sizeWidthInput {
  400. let areas = self.pdfView?.km_editingImageAreas() ?? []
  401. for area in areas {
  402. textValue = area.bounds.size.width - 1
  403. pdfView?.updateArea(area, newWidth: textValue, syncChangeBounds)
  404. }
  405. } else if inputNumber == sizeHeightInput {
  406. let areas = self.pdfView?.km_editingImageAreas() ?? []
  407. for area in areas {
  408. textValue = area.bounds.size.height - 1
  409. pdfView?.updateArea(area, newHeight: textValue, syncChangeBounds)
  410. }
  411. }
  412. reloadData()
  413. }
  414. }
  415. //MARK: - ComponentSelectDelegate
  416. extension KMEditImageController: ComponentSelectDelegate {
  417. func componentSelectDidSelect(view: ComponentSelect?, menuItemProperty: ComponentMenuitemProperty?) {
  418. if view == opacitySelect {
  419. if let text = opacitySelect.properties.text {
  420. let result = text.stringByDeleteCharString("%")
  421. let opacity = result.stringToCGFloat()/100
  422. pdfView?.setEditingAreasOpacity(opacity)
  423. reloadData()
  424. }
  425. } else if view == rotateSelect {
  426. if let text = rotateSelect.properties.text {
  427. let result = text.stringByDeleteCharString("°")
  428. let value = result.stringToCGFloat()
  429. pdfView?.rotateEditingAreas(-value)
  430. reloadData()
  431. }
  432. }
  433. }
  434. func componentSelectTextDidEndEditing(_ view: ComponentSelect) {
  435. if view == opacitySelect {
  436. if let text = opacitySelect.properties.text {
  437. let result = text.stringByDeleteCharString("%")
  438. if result != "-" {
  439. let opacity = result.stringToCGFloat()/100
  440. pdfView?.setEditingAreasOpacity(opacity)
  441. }
  442. reloadData()
  443. }
  444. } else if view == rotateSelect {
  445. if let text = rotateSelect.properties.text {
  446. let result = text.stringByDeleteCharString("°")
  447. if result != "-" {
  448. let value = result.stringToCGFloat()
  449. pdfView?.rotateEditingAreas(-value)
  450. }
  451. reloadData()
  452. }
  453. }
  454. }
  455. }
  456. //MARK: - ComponentCColorDelegate
  457. extension KMEditImageController: ComponentCColorDelegate {
  458. }
  459. //MARK: - ComponentSliderDelegate
  460. extension KMEditImageController: ComponentSliderDelegate {
  461. func componentSliderDidUpdate(_ view: ComponentSlider) {
  462. let percent = view.properties.percent
  463. self.pdfView?.setEditingAreasOpacity(percent)
  464. reloadData()
  465. }
  466. }
  467. //MARK: - ComponentGroupDelegate
  468. extension KMEditImageController: ComponentGroupDelegate {
  469. func componentGroupDidSelect(group: ComponentGroup?, menuItemProperty: ComponentMenuitemProperty?) {
  470. if menuItemProperty?.identifier == "export_jpg_Key" {
  471. pdfView?.exportEditingImageAreasAction(format: "jpg")
  472. } else if menuItemProperty?.identifier == "export_png_Key" {
  473. pdfView?.exportEditingImageAreasAction(format: "png")
  474. } else if menuItemProperty?.identifier == "export_pdf_Key" {
  475. pdfView?.exportEditingImageAreasAction(format: "pdf")
  476. }
  477. }
  478. func componentGroupDidDismiss(group: ComponentGroup?) {
  479. extractButton.properties.state = .normal
  480. extractButton.reloadData()
  481. }
  482. }
  483. //MARK: - KMNAlignmentControllerDelegate
  484. extension KMEditImageController: KMNAlignmentControllerDelegate {
  485. func alignmentControllerDidClick(_ controller: KMNAlignmentController, _ alignmentType: KMPDFActiveFormsAlignType) {
  486. pdfView?.changeEditingAreas(alignmentType)
  487. }
  488. }