ViewerQuickLook.swift 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. //
  2. // ViewerQuickLook.swift
  3. // Nextcloud
  4. //
  5. // Created by Marino Faggiana on 12/01/23.
  6. // Copyright © 2023 Marino Faggiana. All rights reserved.
  7. //
  8. import SwiftUI
  9. import QuickLook
  10. import Mantis
  11. import NextcloudKit
  12. struct ViewerQuickLook: UIViewControllerRepresentable {
  13. let url: URL
  14. @Binding var index: Int
  15. @Binding var isPresentedQuickLook: Bool
  16. @ObservedObject var uploadAssets: NCUploadAssets
  17. func makeUIViewController(context: Context) -> UINavigationController {
  18. let controller = QLPreviewController()
  19. controller.dataSource = context.coordinator
  20. controller.delegate = context.coordinator
  21. context.coordinator.viewController = controller
  22. let buttonDone = UIBarButtonItem(barButtonSystemItem: .done, target: context.coordinator, action: #selector(context.coordinator.dismiss))
  23. let buttonCrop = UIBarButtonItem(title: NSLocalizedString("_crop_", comment: ""), style: UIBarButtonItem.Style.plain, target: context.coordinator,
  24. action: #selector(context.coordinator.crop))
  25. controller.navigationItem.leftBarButtonItems = [buttonDone, buttonCrop]
  26. uploadAssets.startTimer(navigationItem: controller.navigationItem)
  27. DispatchQueue.main.asyncAfter(deadline: .now() + 1) {
  28. if uploadAssets.previewStore[index].asset.type == .livePhoto {
  29. let error = NKError(errorCode: NCGlobal.shared.errorCharactersForbidden, errorDescription: "_message_disable_livephoto_")
  30. NCContentPresenter.shared.showInfo(error: error)
  31. }
  32. }
  33. let navigationController = UINavigationController(rootViewController: controller)
  34. return navigationController
  35. }
  36. func updateUIViewController(_ uiViewController: UINavigationController, context: Context) { }
  37. func makeCoordinator() -> Coordinator {
  38. return Coordinator(parent: self)
  39. }
  40. class Coordinator: NSObject, QLPreviewControllerDataSource, QLPreviewControllerDelegate, CropViewControllerDelegate {
  41. weak var viewController: QLPreviewController?
  42. let parent: ViewerQuickLook
  43. var image: UIImage?
  44. var hasChange = false
  45. init(parent: ViewerQuickLook) {
  46. self.parent = parent
  47. }
  48. @objc func dismiss() {
  49. parent.uploadAssets.stopTimer()
  50. parent.isPresentedQuickLook = false
  51. if let image = image {
  52. parent.uploadAssets.previewStore[parent.index].image = image
  53. parent.uploadAssets.previewStore[parent.index].hasChanges = hasChange
  54. }
  55. }
  56. // MARK: -
  57. func numberOfPreviewItems(in controller: QLPreviewController) -> Int {
  58. return 1
  59. }
  60. func previewController(_ controller: QLPreviewController, editingModeFor previewItem: QLPreviewItem) -> QLPreviewItemEditingMode {
  61. return .createCopy
  62. }
  63. func previewController(_ controller: QLPreviewController, didSaveEditedCopyOf previewItem: QLPreviewItem, at modifiedContentsURL: URL) {
  64. guard NCUtilityFileSystem.shared.moveFile(atPath: modifiedContentsURL.path, toPath: parent.url.path) else { return }
  65. if let image = UIImage(contentsOfFile: parent.url.path) {
  66. self.image = image
  67. self.hasChange = true
  68. }
  69. }
  70. func previewController(_ controller: QLPreviewController, previewItemAt index: Int) -> QLPreviewItem {
  71. return parent.url as NSURL
  72. }
  73. // MARK: -
  74. func cropViewControllerDidCrop(_ cropViewController: Mantis.CropViewController, cropped: UIImage, transformation: Mantis.Transformation, cropInfo: Mantis.CropInfo) {
  75. cropViewController.dismiss(animated: true)
  76. guard let data = cropped.jpegData(compressionQuality: 1) else { return }
  77. do {
  78. try data.write(to: parent.url)
  79. self.image = cropped
  80. self.hasChange = true
  81. viewController?.reloadData()
  82. } catch { }
  83. }
  84. func cropViewControllerDidCancel(_ cropViewController: Mantis.CropViewController, original: UIImage) {
  85. cropViewController.dismiss(animated: true)
  86. }
  87. func cropViewControllerDidFailToCrop(_ cropViewController: Mantis.CropViewController, original: UIImage) {}
  88. func cropViewControllerDidBeginResize(_ cropViewController: Mantis.CropViewController) {}
  89. func cropViewControllerDidEndResize(_ cropViewController: Mantis.CropViewController, original: UIImage, cropInfo: Mantis.CropInfo) {}
  90. func cropViewControllerDidImageTransformed(_ cropViewController: Mantis.CropViewController) { }
  91. @objc func crop() {
  92. guard let image = UIImage(contentsOfFile: parent.url.path) else { return }
  93. var toolbarConfig = CropToolbarConfig()
  94. toolbarConfig.heightForVerticalOrientation = 90
  95. toolbarConfig.widthForHorizontalOrientation = 130
  96. toolbarConfig.optionButtonFontSize = 16
  97. toolbarConfig.optionButtonFontSizeForPad = 21
  98. toolbarConfig.backgroundColor = .systemGray6
  99. toolbarConfig.foregroundColor = .systemBlue
  100. var viewConfig = CropViewConfig()
  101. viewConfig.cropMaskVisualEffectType = .none
  102. viewConfig.cropBorderColor = .red
  103. var config = Mantis.Config()
  104. if let bundleIdentifier = Bundle.main.bundleIdentifier {
  105. config.localizationConfig.bundle = Bundle(identifier: bundleIdentifier)
  106. config.localizationConfig.tableName = "Localizable"
  107. }
  108. config.cropToolbarConfig = toolbarConfig
  109. config.cropViewConfig = viewConfig
  110. let cropViewController = Mantis.cropViewController(image: image, config: config)
  111. cropViewController.delegate = self
  112. cropViewController.backgroundColor = .systemBackground
  113. cropViewController.modalPresentationStyle = .fullScreen
  114. viewController?.present(cropViewController, animated: true)
  115. }
  116. }
  117. }