NCCreateFormUploadAssets.swift 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455
  1. //
  2. // NCCreateFormUploadAssets.swift
  3. // Nextcloud
  4. //
  5. // Created by Marino Faggiana on 14/11/2018.
  6. // Copyright © 2018 Marino Faggiana. All rights reserved.
  7. //
  8. // Author Marino Faggiana <marino.faggiana@nextcloud.com>
  9. //
  10. // This program is free software: you can redistribute it and/or modify
  11. // it under the terms of the GNU General Public License as published by
  12. // the Free Software Foundation, either version 3 of the License, or
  13. // (at your option) any later version.
  14. //
  15. // This program is distributed in the hope that it will be useful,
  16. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  18. // GNU General Public License for more details.
  19. //
  20. // You should have received a copy of the GNU General Public License
  21. // along with this program. If not, see <http://www.gnu.org/licenses/>.
  22. //
  23. import Foundation
  24. @objc protocol createFormUploadAssetsDelegate {
  25. func dismissFormUploadAssets()
  26. }
  27. class NCCreateFormUploadAssets: XLFormViewController, NCSelectDelegate, PhotoEditorDelegate {
  28. var serverUrl: String = ""
  29. var titleServerUrl: String?
  30. var assets = NSMutableArray()
  31. var cryptated: Bool = false
  32. var session: String = ""
  33. weak var delegate: createFormUploadAssetsDelegate?
  34. let requestOptions = PHImageRequestOptions()
  35. var imagePreview: UIImage?
  36. let targetSizeImagePreview = CGSize(width:100, height: 100)
  37. let appDelegate = UIApplication.shared.delegate as! AppDelegate
  38. @objc convenience init(serverUrl : String, assets : NSMutableArray, cryptated : Bool, session : String, delegate: createFormUploadAssetsDelegate) {
  39. self.init()
  40. if serverUrl == CCUtility.getHomeServerUrlActiveUrl(appDelegate.activeUrl) {
  41. titleServerUrl = "/"
  42. } else {
  43. titleServerUrl = (serverUrl as NSString).lastPathComponent
  44. }
  45. self.serverUrl = serverUrl
  46. self.assets = assets
  47. self.cryptated = cryptated
  48. self.session = session
  49. self.delegate = delegate
  50. requestOptions.resizeMode = PHImageRequestOptionsResizeMode.exact
  51. requestOptions.deliveryMode = PHImageRequestOptionsDeliveryMode.highQualityFormat
  52. requestOptions.isSynchronous = true
  53. if assets.count == 1 && (assets[0] as! PHAsset).mediaType == PHAssetMediaType.image {
  54. PHImageManager.default().requestImage(for: assets[0] as! PHAsset, targetSize: targetSizeImagePreview, contentMode: PHImageContentMode.aspectFill, options: requestOptions, resultHandler: { (image, info) in
  55. self.imagePreview = image
  56. self.initializeForm()
  57. })
  58. } else {
  59. self.initializeForm()
  60. }
  61. }
  62. //MARK: XLFormDescriptorDelegate
  63. func initializeForm() {
  64. let form : XLFormDescriptor = XLFormDescriptor(title: NSLocalizedString("_upload_photos_videos_", comment: "")) as XLFormDescriptor
  65. form.rowNavigationOptions = XLFormRowNavigationOptions.stopDisableRow
  66. var section : XLFormSectionDescriptor
  67. var row : XLFormRowDescriptor
  68. // Section Photo Editor only for one photo
  69. if assets.count == 1 && (assets[0] as! PHAsset).mediaType == PHAssetMediaType.image && self.imagePreview != nil {
  70. section = XLFormSectionDescriptor.formSection(withTitle: NSLocalizedString("_modify_photo_", comment: ""))
  71. form.addFormSection(section)
  72. row = XLFormRowDescriptor(tag: "ButtonPhotoEditor", rowType: XLFormRowDescriptorTypeButton, title: NSLocalizedString("_modify_photo_", comment: ""))
  73. row.action.formSelector = #selector(photoEditor(_:))
  74. row.cellConfig["imageView.image"] = self.imagePreview
  75. row.cellConfig["textLabel.textColor"] = NCBrandColor.sharedInstance.brandElement
  76. row.cellConfig["textLabel.textAlignment"] = NSTextAlignment.right.rawValue
  77. row.cellConfig["textLabel.font"] = UIFont.systemFont(ofSize: 15.0)
  78. section.addFormRow(row)
  79. }
  80. // Section: Destination Folder
  81. section = XLFormSectionDescriptor.formSection(withTitle: NSLocalizedString("_save_path_", comment: ""))
  82. form.addFormSection(section)
  83. row = XLFormRowDescriptor(tag: "ButtonDestinationFolder", rowType: XLFormRowDescriptorTypeButton, title: self.titleServerUrl)
  84. row.action.formSelector = #selector(changeDestinationFolder(_:))
  85. row.cellConfig["imageView.image"] = CCGraphics.changeThemingColorImage(UIImage(named: "folder")!, width: 50, height: 50, color: NCBrandColor.sharedInstance.brandElement) as UIImage
  86. row.cellConfig["textLabel.textAlignment"] = NSTextAlignment.right.rawValue
  87. row.cellConfig["textLabel.font"] = UIFont.systemFont(ofSize: 15.0)
  88. section.addFormRow(row)
  89. // User folder Autoupload
  90. row = XLFormRowDescriptor(tag: "useFolderAutoUpload", rowType: XLFormRowDescriptorTypeBooleanSwitch, title: NSLocalizedString("_use_folder_auto_upload_", comment: ""))
  91. row.value = 0
  92. row.cellConfig["textLabel.font"] = UIFont.systemFont(ofSize: 15.0)
  93. section.addFormRow(row)
  94. // Use Sub folder
  95. row = XLFormRowDescriptor(tag: "useSubFolder", rowType: XLFormRowDescriptorTypeBooleanSwitch, title: NSLocalizedString("_autoupload_create_subfolder_", comment: ""))
  96. let tableAccount = NCManageDatabase.sharedInstance.getAccountActive()
  97. if tableAccount?.autoUploadCreateSubfolder == true {
  98. row.value = 1
  99. } else {
  100. row.value = 0
  101. }
  102. row.hidden = "$\("useFolderAutoUpload") == 0"
  103. row.cellConfig["textLabel.font"] = UIFont.systemFont(ofSize: 15.0)
  104. section.addFormRow(row)
  105. // Section Mode filename
  106. section = XLFormSectionDescriptor.formSection(withTitle: NSLocalizedString("_mode_filename_", comment: ""))
  107. form.addFormSection(section)
  108. // Maintain the original fileName
  109. row = XLFormRowDescriptor(tag: "maintainOriginalFileName", rowType: XLFormRowDescriptorTypeBooleanSwitch, title: NSLocalizedString("_maintain_original_filename_", comment: ""))
  110. row.value = CCUtility.getOriginalFileName(k_keyFileNameOriginal)
  111. row.cellConfig["textLabel.font"] = UIFont.systemFont(ofSize: 15.0)
  112. section.addFormRow(row)
  113. // Add File Name Type
  114. row = XLFormRowDescriptor(tag: "addFileNameType", rowType: XLFormRowDescriptorTypeBooleanSwitch, title: NSLocalizedString("_add_filenametype_", comment: ""))
  115. row.value = CCUtility.getFileNameType(k_keyFileNameType)
  116. row.hidden = "$\("maintainOriginalFileName") == 1"
  117. row.cellConfig["textLabel.font"] = UIFont.systemFont(ofSize: 15.0)
  118. section.addFormRow(row)
  119. // Section: Rename File Name
  120. section = XLFormSectionDescriptor.formSection(withTitle: NSLocalizedString("_filename_", comment: ""))
  121. form.addFormSection(section)
  122. row = XLFormRowDescriptor(tag: "maskFileName", rowType: XLFormRowDescriptorTypeAccount, title: (NSLocalizedString("_filename_", comment: "")))
  123. let fileNameMask : String = CCUtility.getFileNameMask(k_keyFileNameMask)
  124. if fileNameMask.count > 0 {
  125. row.value = fileNameMask
  126. }
  127. row.hidden = "$\("maintainOriginalFileName") == 1"
  128. row.cellConfig["textLabel.font"] = UIFont.systemFont(ofSize: 15.0)
  129. row.cellConfig["textField.textAlignment"] = NSTextAlignment.right.rawValue
  130. row.cellConfig["textField.font"] = UIFont.systemFont(ofSize: 15.0)
  131. section.addFormRow(row)
  132. // Section: Preview File Name
  133. row = XLFormRowDescriptor(tag: "previewFileName", rowType: XLFormRowDescriptorTypeTextView, title: "")
  134. row.height = 180
  135. row.disabled = true
  136. row.cellConfig["textView.backgroundColor"] = NCBrandColor.sharedInstance.backgroundView
  137. row.cellConfig["textView.font"] = UIFont.systemFont(ofSize: 14.0)
  138. section.addFormRow(row)
  139. self.form = form
  140. }
  141. override func formRowDescriptorValueHasChanged(_ formRow: XLFormRowDescriptor!, oldValue: Any!, newValue: Any!) {
  142. super.formRowDescriptorValueHasChanged(formRow, oldValue: oldValue, newValue: newValue)
  143. if formRow.tag == "useFolderAutoUpload" {
  144. if (formRow.value! as AnyObject).boolValue == true {
  145. let buttonDestinationFolder : XLFormRowDescriptor = self.form.formRow(withTag: "ButtonDestinationFolder")!
  146. buttonDestinationFolder.hidden = true
  147. } else{
  148. let buttonDestinationFolder : XLFormRowDescriptor = self.form.formRow(withTag: "ButtonDestinationFolder")!
  149. buttonDestinationFolder.hidden = false
  150. }
  151. }
  152. else if formRow.tag == "useSubFolder" {
  153. if (formRow.value! as AnyObject).boolValue == true {
  154. } else{
  155. }
  156. }
  157. else if formRow.tag == "maintainOriginalFileName" {
  158. CCUtility.setOriginalFileName((formRow.value! as AnyObject).boolValue, key: k_keyFileNameOriginal)
  159. self.reloadForm()
  160. }
  161. else if formRow.tag == "addFileNameType" {
  162. CCUtility.setFileNameType((formRow.value! as AnyObject).boolValue, key: k_keyFileNameType)
  163. self.reloadForm()
  164. }
  165. else if formRow.tag == "maskFileName" {
  166. let fileName = formRow.value as? String
  167. self.form.delegate = nil
  168. if let fileName = fileName {
  169. formRow.value = CCUtility.removeForbiddenCharactersServer(fileName)
  170. }
  171. self.form.delegate = self
  172. let previewFileName : XLFormRowDescriptor = self.form.formRow(withTag: "previewFileName")!
  173. previewFileName.value = self.previewFileName(valueRename: formRow.value as? String)
  174. // reload cell
  175. if fileName != nil {
  176. if newValue as! String != formRow.value as! String {
  177. self.reloadFormRow(formRow)
  178. appDelegate.messageNotification("_info_", description: "_forbidden_characters_", visible: true, delay: TimeInterval(k_dismissAfterSecond), type: TWMessageBarMessageType.info, errorCode: 0)
  179. }
  180. }
  181. self.reloadFormRow(previewFileName)
  182. }
  183. }
  184. // MARK: - View Life Cycle
  185. override func viewDidLoad() {
  186. super.viewDidLoad()
  187. let cancelButton : UIBarButtonItem = UIBarButtonItem(title: NSLocalizedString("_cancel_", comment: ""), style: UIBarButtonItem.Style.plain, target: self, action: #selector(cancel))
  188. let saveButton : UIBarButtonItem = UIBarButtonItem(title: NSLocalizedString("_save_", comment: ""), style: UIBarButtonItem.Style.plain, target: self, action: #selector(save))
  189. self.navigationItem.leftBarButtonItem = cancelButton
  190. self.navigationItem.rightBarButtonItem = saveButton
  191. self.navigationController?.navigationBar.isTranslucent = false
  192. self.navigationController?.navigationBar.barTintColor = NCBrandColor.sharedInstance.brand
  193. self.navigationController?.navigationBar.tintColor = NCBrandColor.sharedInstance.brandText
  194. self.navigationController?.navigationBar.titleTextAttributes = [NSAttributedString.Key.foregroundColor: NCBrandColor.sharedInstance.brandText]
  195. self.tableView.separatorStyle = UITableViewCell.SeparatorStyle.none
  196. self.reloadForm()
  197. }
  198. override func viewWillDisappear(_ animated: Bool)
  199. {
  200. super.viewWillDisappear(animated)
  201. self.delegate?.dismissFormUploadAssets()
  202. }
  203. func reloadForm() {
  204. self.form.delegate = nil
  205. let buttonDestinationFolder : XLFormRowDescriptor = self.form.formRow(withTag: "ButtonDestinationFolder")!
  206. buttonDestinationFolder.title = self.titleServerUrl
  207. let maskFileName : XLFormRowDescriptor = self.form.formRow(withTag: "maskFileName")!
  208. let previewFileName : XLFormRowDescriptor = self.form.formRow(withTag: "previewFileName")!
  209. previewFileName.value = self.previewFileName(valueRename: maskFileName.value as? String)
  210. self.tableView.reloadData()
  211. self.form.delegate = self
  212. }
  213. // MARK: - Action
  214. func dismissSelect(serverUrl: String?, metadata: tableMetadata?, type: String) {
  215. if serverUrl != nil {
  216. self.serverUrl = serverUrl!
  217. if serverUrl == CCUtility.getHomeServerUrlActiveUrl(appDelegate.activeUrl) {
  218. self.titleServerUrl = "/"
  219. } else {
  220. self.titleServerUrl = (serverUrl! as NSString).lastPathComponent
  221. }
  222. // Update
  223. let row : XLFormRowDescriptor = self.form.formRow(withTag: "ButtonDestinationFolder")!
  224. row.title = self.titleServerUrl
  225. self.updateFormRow(row)
  226. }
  227. }
  228. @objc func save() {
  229. self.dismiss(animated: true, completion: {
  230. let useFolderPhotoRow : XLFormRowDescriptor = self.form.formRow(withTag: "useFolderAutoUpload")!
  231. let useSubFolderRow : XLFormRowDescriptor = self.form.formRow(withTag: "useSubFolder")!
  232. var useSubFolder : Bool = false
  233. if (useFolderPhotoRow.value! as AnyObject).boolValue == true {
  234. self.serverUrl = NCManageDatabase.sharedInstance.getAccountAutoUploadPath(self.appDelegate.activeUrl)
  235. useSubFolder = (useSubFolderRow.value! as AnyObject).boolValue
  236. }
  237. self.appDelegate.activeMain.uploadFileAsset(self.assets, serverUrl: self.serverUrl, useSubFolder: useSubFolder, session: self.session)
  238. })
  239. }
  240. @objc func cancel() {
  241. self.dismiss(animated: true, completion: nil)
  242. }
  243. // MARK: - Utility
  244. func previewFileName(valueRename : String?) -> String {
  245. var returnString: String = ""
  246. let asset = assets[0] as! PHAsset
  247. if (CCUtility.getOriginalFileName(k_keyFileNameOriginal)) {
  248. return (NSLocalizedString("_filename_", comment: "") + ": " + (asset.value(forKey: "filename") as! String))
  249. } else if let valueRename = valueRename {
  250. let valueRenameTrimming = valueRename.trimmingCharacters(in: CharacterSet.whitespacesAndNewlines)
  251. if valueRenameTrimming.count > 0 {
  252. self.form.delegate = nil
  253. CCUtility.setFileNameMask(valueRename, key: k_keyFileNameMask)
  254. self.form.delegate = self
  255. returnString = CCUtility.createFileName(asset.value(forKey: "filename") as! String?, fileDate: asset.creationDate, fileType: asset.mediaType, keyFileName: k_keyFileNameMask, keyFileNameType: k_keyFileNameType, keyFileNameOriginal: k_keyFileNameOriginal)
  256. } else {
  257. CCUtility.setFileNameMask("", key: k_keyFileNameMask)
  258. returnString = CCUtility.createFileName(asset.value(forKey: "filename") as! String?, fileDate: asset.creationDate, fileType: asset.mediaType, keyFileName: nil, keyFileNameType: k_keyFileNameType, keyFileNameOriginal: k_keyFileNameOriginal)
  259. }
  260. } else {
  261. CCUtility.setFileNameMask("", key: k_keyFileNameMask)
  262. returnString = CCUtility.createFileName(asset.value(forKey: "filename") as! String?, fileDate: asset.creationDate, fileType: asset.mediaType, keyFileName: nil, keyFileNameType: k_keyFileNameType, keyFileNameOriginal: k_keyFileNameOriginal)
  263. }
  264. return String(format: NSLocalizedString("_preview_filename_", comment: ""), "MM,MMM,DD,YY,YYYY and HH,hh,mm,ss,ampm") + ":" + "\n\n" + returnString
  265. }
  266. @objc func changeDestinationFolder(_ sender: XLFormRowDescriptor) {
  267. self.deselectFormRow(sender)
  268. let storyboard = UIStoryboard(name: "NCSelect", bundle: nil)
  269. let navigationController = storyboard.instantiateInitialViewController() as! UINavigationController
  270. let viewController = navigationController.topViewController as! NCSelect
  271. viewController.delegate = self
  272. viewController.hideButtonCreateFolder = false
  273. viewController.includeDirectoryE2EEncryption = true
  274. viewController.includeImages = false
  275. viewController.layoutViewSelect = k_layout_view_move
  276. viewController.selectFile = false
  277. viewController.titleButtonDone = NSLocalizedString("_select_", comment: "")
  278. viewController.type = ""
  279. navigationController.modalPresentationStyle = UIModalPresentationStyle.formSheet
  280. self.present(navigationController, animated: true, completion: nil)
  281. }
  282. @objc func photoEditor(_ sender: XLFormRowDescriptor) {
  283. self.deselectFormRow(sender)
  284. PHImageManager.default().requestImage(for: assets[0] as! PHAsset, targetSize: PHImageManagerMaximumSize, contentMode: PHImageContentMode.default, options: requestOptions, resultHandler: { (image, info) in
  285. let photoEditor = PhotoEditorViewController(nibName:"PhotoEditorViewController",bundle: Bundle(for: PhotoEditorViewController.self))
  286. photoEditor.image = image
  287. photoEditor.photoEditorDelegate = self
  288. photoEditor.hiddenControls = [.save, .share, .sticker]
  289. photoEditor.cancelButtonImage = CCGraphics.changeThemingColorImage(UIImage(named: "photoEditorCancel")!, multiplier:2, color: .white)
  290. photoEditor.cropButtonImage = CCGraphics.changeThemingColorImage(UIImage(named: "photoEditorCrop")!, multiplier:2, color: .white)
  291. photoEditor.drawButtonImage = CCGraphics.changeThemingColorImage(UIImage(named: "photoEditorDraw")!, multiplier:2, color: .white)
  292. photoEditor.textButtonImage = CCGraphics.changeThemingColorImage(UIImage(named: "photoEditorText")!, multiplier:2, color: .white)
  293. photoEditor.clearButtonImage = CCGraphics.changeThemingColorImage(UIImage(named: "photoEditorClear")!, multiplier:2, color: .white)
  294. photoEditor.continueButtonImage = CCGraphics.changeThemingColorImage(UIImage(named: "photoEditorDone")!, multiplier:2, color: .white)
  295. self.present(photoEditor, animated: true, completion: nil)
  296. })
  297. }
  298. // MARK: - Photo Editor Delegate
  299. func doneEditing(image: UIImage) {
  300. UIImageWriteToSavedPhotosAlbum(image, self, #selector(imageSaved(_:didFinishSavingWithError:contextInfo:)), nil)
  301. }
  302. @objc func imageSaved(_ image: UIImage, didFinishSavingWithError error: Error?, contextInfo: UnsafeRawPointer) {
  303. let fetchOptions = PHFetchOptions()
  304. fetchOptions.sortDescriptors = [NSSortDescriptor(key: "creationDate", ascending: false)]
  305. let fetchResult = PHAsset.fetchAssets(with: PHAssetMediaType.image, options: fetchOptions)
  306. if let asset = fetchResult.firstObject {
  307. self.assets = NSMutableArray(array: [asset])
  308. }
  309. // Preview
  310. PHImageManager.default().requestImage(for: assets[0] as! PHAsset, targetSize: targetSizeImagePreview, contentMode: PHImageContentMode.aspectFill, options: requestOptions, resultHandler: { (image, info) in
  311. let row : XLFormRowDescriptor = self.form.formRow(withTag: "ButtonPhotoEditor")!
  312. row.cellConfig["imageView.image"] = image
  313. self.updateFormRow(row)
  314. })
  315. }
  316. func canceledEditing() {
  317. print("Canceled")
  318. }
  319. }