KMEditImageController.swift 24 KB

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