123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432 |
- //
- // NCUploadAssets.swift
- // Nextcloud
- //
- // Created by Marino Faggiana on 04/01/23.
- // Copyright © 2023 Marino Faggiana. All rights reserved.
- //
- // Author Marino Faggiana <marino.faggiana@nextcloud.com>
- //
- // This program is free software: you can redistribute it and/or modify
- // it under the terms of the GNU General Public License as published by
- // the Free Software Foundation, either version 3 of the License, or
- // (at your option) any later version.
- //
- // This program is distributed in the hope that it will be useful,
- // but WITHOUT ANY WARRANTY; without even the implied warranty of
- // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- // GNU General Public License for more details.
- //
- // You should have received a copy of the GNU General Public License
- // along with this program. If not, see <http://www.gnu.org/licenses/>.
- //
- import SwiftUI
- import NextcloudKit
- import TLPhotoPicker
- import Mantis
- class NCHostingUploadAssetsView: NSObject {
- func makeShipDetailsUI(assets: [TLPHAsset], serverUrl: String, userBaseUrl: NCUserBaseUrl) -> UIViewController {
- let uploadAssets = NCUploadAssets(assets: assets, serverUrl: serverUrl, userBaseUrl: userBaseUrl )
- let details = UploadAssetsView(uploadAssets: uploadAssets)
- return UIHostingController(rootView: details)
- }
- }
- // MARK: - Class
- struct PreviewStore {
- var image: UIImage
- var localIdentifier: String
- var modify: Bool
- }
- class NCUploadAssets: NSObject, ObservableObject, NCCreateFormUploadConflictDelegate {
- @Published var serverUrl: String
- @Published var assets: [TLPHAsset]
- @Published var userBaseUrl: NCUserBaseUrl
- @Published var dismiss = false
- @Published var previewStore: [PreviewStore] = []
- var metadatasNOConflict: [tableMetadata] = []
- var metadatasUploadInConflict: [tableMetadata] = []
- init(assets: [TLPHAsset], serverUrl: String, userBaseUrl: NCUserBaseUrl) {
- self.assets = assets
- self.serverUrl = serverUrl
- self.userBaseUrl = userBaseUrl
- }
- func loadImages() {
- DispatchQueue.global().async {
- for asset in self.assets {
- guard asset.type == .photo, let image = asset.fullResolutionImage, let localIdentifier = asset.phAsset?.localIdentifier else { continue }
- self.previewStore.append(PreviewStore(image: image, localIdentifier: localIdentifier, modify: false))
- }
- }
- }
- func dismissCreateFormUploadConflict(metadatas: [tableMetadata]?) {
- if let metadatas = metadatas {
- NCNetworkingProcessUpload.shared.createProcessUploads(metadatas: metadatas, completion: { _ in
- self.dismiss = true
- })
- } else {
- self.dismiss = true
- }
- }
- }
- // MARK: - View
- struct UploadAssetsView: View {
- @State private var fileName: String = CCUtility.getFileNameMask(NCGlobal.shared.keyFileNameMask)
- @State private var isMaintainOriginalFilename: Bool = CCUtility.getOriginalFileName(NCGlobal.shared.keyFileNameOriginal)
- @State private var isAddFilenametype: Bool = CCUtility.getFileNameType(NCGlobal.shared.keyFileNameType)
- @State private var isPresentedSelect = false
- @State private var isPresentedUploadConflict = false
- // Crop
- @State private var isPresentedCrop = false
- @State private var index: Int = 0
- @State private var imageForCrop: UIImage = UIImage()
- @State private var cropShapeType: Mantis.CropShapeType = .rect
- @State private var presetFixedRatioType: Mantis.PresetFixedRatioType = .canUseMultiplePresetFixedRatio()
- var gridItems: [GridItem] = [GridItem()]
- @ObservedObject var uploadAssets: NCUploadAssets
- @Environment(\.presentationMode) var presentationMode
- init(uploadAssets: NCUploadAssets) {
- self.uploadAssets = uploadAssets
- uploadAssets.loadImages()
- }
- func getOriginalFilename() -> String {
- CCUtility.setOriginalFileName(isMaintainOriginalFilename, key: NCGlobal.shared.keyFileNameOriginal)
- if let asset = uploadAssets.assets.first?.phAsset, let name = (asset.value(forKey: "filename") as? String) {
- return name
- } else {
- return ""
- }
- }
- func setFileNameMask(fileName: String?) -> String {
- guard let asset = uploadAssets.assets.first?.phAsset else { return "" }
- var preview: String = ""
- let creationDate = asset.creationDate ?? Date()
- CCUtility.setOriginalFileName(isMaintainOriginalFilename, key: NCGlobal.shared.keyFileNameOriginal)
- CCUtility.setFileNameType(isAddFilenametype, key: NCGlobal.shared.keyFileNameType)
- if let fileName = fileName {
- let fileName = fileName.trimmingCharacters(in: CharacterSet.whitespacesAndNewlines)
- if !fileName.isEmpty {
- CCUtility.setFileNameMask(fileName, key: NCGlobal.shared.keyFileNameMask)
- preview = CCUtility.createFileName(asset.value(forKey: "filename") as? String,
- fileDate: creationDate, fileType: asset.mediaType,
- keyFileName: NCGlobal.shared.keyFileNameMask,
- keyFileNameType: NCGlobal.shared.keyFileNameType,
- keyFileNameOriginal: NCGlobal.shared.keyFileNameOriginal,
- forcedNewFileName: false)
- } else {
- CCUtility.setFileNameMask("", key: NCGlobal.shared.keyFileNameMask)
- preview = CCUtility.createFileName(asset.value(forKey: "filename") as? String,
- fileDate: creationDate,
- fileType: asset.mediaType,
- keyFileName: nil,
- keyFileNameType: NCGlobal.shared.keyFileNameType,
- keyFileNameOriginal: NCGlobal.shared.keyFileNameOriginal,
- forcedNewFileName: false)
- }
- } else {
- CCUtility.setFileNameMask("", key: NCGlobal.shared.keyFileNameMask)
- preview = CCUtility.createFileName(asset.value(forKey: "filename") as? String,
- fileDate: creationDate,
- fileType: asset.mediaType,
- keyFileName: nil,
- keyFileNameType: NCGlobal.shared.keyFileNameType,
- keyFileNameOriginal: NCGlobal.shared.keyFileNameOriginal,
- forcedNewFileName: false)
- }
- return String(format: NSLocalizedString("_preview_filename_", comment: ""), "MM, MMM, DD, YY, YYYY, HH, hh, mm, ss, ampm") + ":" + "\n\n" + preview
- }
- func save(completion: @escaping (_ metadatasNOConflict: [tableMetadata], _ metadatasUploadInConflict: [tableMetadata]) -> Void) {
- var metadatasNOConflict: [tableMetadata] = []
- var metadatasUploadInConflict: [tableMetadata] = []
- for asset in uploadAssets.assets {
- guard let asset = asset.phAsset else { continue }
- let serverUrl = uploadAssets.serverUrl
- var livePhoto: Bool = false
- let creationDate = asset.creationDate ?? Date()
- let fileName = CCUtility.createFileName(asset.value(forKey: "filename") as? String,
- fileDate: creationDate,
- fileType: asset.mediaType,
- keyFileName: NCGlobal.shared.keyFileNameMask,
- keyFileNameType: NCGlobal.shared.keyFileNameType,
- keyFileNameOriginal: NCGlobal.shared.keyFileNameOriginal,
- forcedNewFileName: false)!
- if asset.mediaSubtypes.contains(.photoLive) && CCUtility.getLivePhoto() {
- livePhoto = true
- }
- // Check if is in upload
- let isRecordInSessions = NCManageDatabase.shared.getAdvancedMetadatas(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@ AND fileName == %@ AND session != ''", uploadAssets.userBaseUrl.account, serverUrl, fileName), sorted: "fileName", ascending: false)
- if !isRecordInSessions.isEmpty { continue }
- let metadata = NCManageDatabase.shared.createMetadata(account: uploadAssets.userBaseUrl.account, user: uploadAssets.userBaseUrl.user, userId: uploadAssets.userBaseUrl.userId, fileName: fileName, fileNameView: fileName, ocId: NSUUID().uuidString, serverUrl: serverUrl, urlBase: uploadAssets.userBaseUrl.urlBase, url: "", contentType: "", isLivePhoto: livePhoto)
- metadata.assetLocalIdentifier = asset.localIdentifier
- metadata.session = NCNetworking.shared.sessionIdentifierBackground
- metadata.sessionSelector = NCGlobal.shared.selectorUploadFile
- metadata.status = NCGlobal.shared.metadataStatusWaitUpload
- // Modified
- if let previewStore = uploadAssets.previewStore.first(where: {$0.localIdentifier == asset.localIdentifier && $0.modify == true }), let data = previewStore.image.jpegData(compressionQuality: 1) {
- if metadata.contentType == "image/heic" {
- let fileNameNoExtension = (fileName as NSString).deletingPathExtension
- metadata.contentType = "image/jpeg"
- metadata.fileName = fileNameNoExtension + ".jpg"
- metadata.fileNameView = fileNameNoExtension + ".jpg"
- }
- let fileNamePath = CCUtility.getDirectoryProviderStorageOcId(metadata.ocId, fileNameView: metadata.fileNameView)!
- do {
- try data.write(to: URL(fileURLWithPath: fileNamePath))
- metadata.isExtractFile = true
- metadata.size = NCUtilityFileSystem.shared.getFileSize(filePath: fileNamePath)
- metadata.creationDate = asset.creationDate as? NSDate ?? (Date() as NSDate)
- metadata.date = asset.modificationDate as? NSDate ?? (Date() as NSDate)
- } catch { }
- }
- if let result = NCManageDatabase.shared.getMetadataConflict(account: uploadAssets.userBaseUrl.account, serverUrl: serverUrl, fileNameView: fileName) {
- metadata.fileName = result.fileName
- metadatasUploadInConflict.append(metadata)
- } else {
- metadatasNOConflict.append(metadata)
- }
- }
- // Verify if file(s) exists
- if !metadatasUploadInConflict.isEmpty {
- completion(metadatasNOConflict, metadatasUploadInConflict)
- } else {
- NCNetworkingProcessUpload.shared.createProcessUploads(metadatas: metadatasNOConflict, completion: { _ in })
- completion(metadatasNOConflict, metadatasUploadInConflict)
- }
- }
- var body: some View {
- NavigationView {
- List {
- if !uploadAssets.previewStore.isEmpty {
- Section(header: Text(NSLocalizedString("_crop_", comment: "")), footer: Text(NSLocalizedString("_modify_crop_desc_", comment: ""))) {
- ScrollView(.horizontal) {
- LazyHGrid(rows: gridItems, alignment: .center, spacing: 10) {
- ForEach(0..<uploadAssets.previewStore.count, id: \.self) { index in
- VStack {
- Image(uiImage: uploadAssets.previewStore[index].image)
- .resizable()
- .frame(width: 100, height: 100, alignment: .center)
- .cornerRadius(10)
- .scaledToFit()
- .onTapGesture {
- self.index = index
- isPresentedCrop = true
- }
- }
- }
- }
- }
- }
- }
- Section(header: Text(NSLocalizedString("_mode_filename_", comment: ""))) {
- Toggle(NSLocalizedString("_maintain_original_filename_", comment: ""), isOn: $isMaintainOriginalFilename)
- .toggleStyle(SwitchToggleStyle(tint: Color(NCBrandColor.shared.brand)))
- if !isMaintainOriginalFilename {
- Toggle(NSLocalizedString("_add_filenametype_", comment: ""), isOn: $isAddFilenametype)
- .toggleStyle(SwitchToggleStyle(tint: Color(NCBrandColor.shared.brand)))
- }
- }
- Section {
- HStack {
- Label {
- if NCUtilityFileSystem.shared.getHomeServer(urlBase: uploadAssets.userBaseUrl.urlBase, userId: uploadAssets.userBaseUrl.userId) == uploadAssets.serverUrl {
- Text("/")
- .frame(maxWidth: .infinity, alignment: .trailing)
- } else {
- Text((uploadAssets.serverUrl as NSString).lastPathComponent)
- .frame(maxWidth: .infinity, alignment: .trailing)
- }
- } icon: {
- Image("folder")
- .renderingMode(.template)
- .resizable()
- .scaledToFit()
- .foregroundColor(Color(NCBrandColor.shared.brand))
- }
- }
- .contentShape(Rectangle())
- .onTapGesture {
- isPresentedSelect = true
- }
- HStack {
- Text(NSLocalizedString("_filename_", comment: ""))
- if isMaintainOriginalFilename {
- Text(getOriginalFilename())
- .frame(maxWidth: .infinity, alignment: .trailing)
- } else {
- TextField(NSLocalizedString("_enter_filename_", comment: ""), text: $fileName)
- .modifier(TextFieldClearButton(text: $fileName))
- .multilineTextAlignment(.trailing)
- }
- }
- if !isMaintainOriginalFilename {
- Text(setFileNameMask(fileName: fileName))
- .font(.system(size: 12))
- .foregroundColor(Color.gray)
- }
- }
- .complexModifier { view in
- if #available(iOS 15, *) {
- view.listRowSeparator(.hidden)
- }
- }
- Button(NSLocalizedString("_save_", comment: "")) {
- save { metadatasNOConflict, metadatasUploadInConflict in
- if metadatasUploadInConflict.isEmpty {
- uploadAssets.dismiss = true
- } else {
- uploadAssets.metadatasNOConflict = metadatasNOConflict
- uploadAssets.metadatasUploadInConflict = metadatasUploadInConflict
- isPresentedUploadConflict = true
- }
- }
- }
- .frame(maxWidth: .infinity)
- .buttonStyle(ButtonRounded(disabled: false))
- .listRowBackground(Color(UIColor.systemGroupedBackground))
- }
- .navigationTitle(NSLocalizedString("_upload_photos_videos_", comment: ""))
- .navigationBarTitleDisplayMode(.inline)
- }
- .sheet(isPresented: $isPresentedCrop) {
- ImageCropper(previewStore: $uploadAssets.previewStore, index: $index, cropShapeType: $cropShapeType, presetFixedRatioType: $presetFixedRatioType)
- .ignoresSafeArea()
- }
- .sheet(isPresented: $isPresentedSelect) {
- SelectView(serverUrl: $uploadAssets.serverUrl)
- }
- .sheet(isPresented: $isPresentedUploadConflict) {
- UploadConflictView(delegate: uploadAssets, serverUrl: uploadAssets.serverUrl, metadatasUploadInConflict: uploadAssets.metadatasUploadInConflict, metadatasNOConflict: uploadAssets.metadatasNOConflict)
- }
- .onReceive(uploadAssets.$dismiss) { newValue in
- if newValue {
- presentationMode.wrappedValue.dismiss()
- }
- }
- }
- }
- // MARK: - ImageCropper
- struct ImageCropper: UIViewControllerRepresentable {
- @Binding var previewStore: [PreviewStore]
- @Binding var index: Int
- @Binding var cropShapeType: Mantis.CropShapeType
- @Binding var presetFixedRatioType: Mantis.PresetFixedRatioType
- @Environment(\.presentationMode) var presentationMode
- class Coordinator: CropViewControllerDelegate {
- var parent: ImageCropper
- init(_ parent: ImageCropper) {
- self.parent = parent
- }
- func cropViewControllerDidCrop(_ cropViewController: CropViewController, cropped: UIImage, transformation: Transformation, cropInfo: CropInfo) {
- parent.previewStore[parent.index].image = cropped
- parent.previewStore[parent.index].modify = true
- parent.presentationMode.wrappedValue.dismiss()
- }
- func cropViewControllerDidCancel(_ cropViewController: CropViewController, original: UIImage) {
- parent.presentationMode.wrappedValue.dismiss()
- }
- func cropViewControllerDidFailToCrop(_ cropViewController: CropViewController, original: UIImage) { }
- func cropViewControllerDidBeginResize(_ cropViewController: CropViewController) { }
- func cropViewControllerDidEndResize(_ cropViewController: CropViewController, original: UIImage, cropInfo: CropInfo) { }
- func cropViewControllerDidImageTransformed(_ cropViewController: Mantis.CropViewController) { }
- }
- func makeCoordinator() -> Coordinator {
- Coordinator(self)
- }
- func makeUIViewController(context: Context) -> CropViewController {
- var config = Mantis.Config()
- if let bundleIdentifier = Bundle.main.bundleIdentifier {
- config.localizationConfig.bundle = Bundle(identifier: bundleIdentifier)
- config.localizationConfig.tableName = "Localizable"
- }
- config.cropViewConfig.cropShapeType = cropShapeType
- config.presetFixedRatioType = presetFixedRatioType
- let image = previewStore[index].image
- let cropViewController = Mantis.cropViewController(image: image, config: config)
- cropViewController.delegate = context.coordinator
- return cropViewController
- }
- func updateUIViewController(_ uiViewController: CropViewController, context: Context) { }
- }
- // MARK: - Preview
- struct UploadAssetsView_Previews: PreviewProvider {
- static var previews: some View {
- if let appDelegate = UIApplication.shared.delegate as? AppDelegate {
- let uploadAssets = NCUploadAssets(assets: [], serverUrl: "/", userBaseUrl: appDelegate)
- UploadAssetsView(uploadAssets: uploadAssets)
- }
- }
- }
|