NCCreateFormUploadAssets.swift 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456
  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"] = UIColor.black
  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. row.cellConfig["textLabel.textColor"] = UIColor.black
  89. section.addFormRow(row)
  90. // User folder Autoupload
  91. row = XLFormRowDescriptor(tag: "useFolderAutoUpload", rowType: XLFormRowDescriptorTypeBooleanSwitch, title: NSLocalizedString("_use_folder_auto_upload_", comment: ""))
  92. row.value = 0
  93. row.cellConfig["textLabel.font"] = UIFont.systemFont(ofSize: 15.0)
  94. section.addFormRow(row)
  95. // Use Sub folder
  96. row = XLFormRowDescriptor(tag: "useSubFolder", rowType: XLFormRowDescriptorTypeBooleanSwitch, title: NSLocalizedString("_autoupload_create_subfolder_", comment: ""))
  97. let tableAccount = NCManageDatabase.sharedInstance.getAccountActive()
  98. if tableAccount?.autoUploadCreateSubfolder == true {
  99. row.value = 1
  100. } else {
  101. row.value = 0
  102. }
  103. row.hidden = "$\("useFolderAutoUpload") == 0"
  104. row.cellConfig["textLabel.font"] = UIFont.systemFont(ofSize: 15.0)
  105. section.addFormRow(row)
  106. // Section Mode filename
  107. section = XLFormSectionDescriptor.formSection(withTitle: NSLocalizedString("_mode_filename_", comment: ""))
  108. form.addFormSection(section)
  109. // Maintain the original fileName
  110. row = XLFormRowDescriptor(tag: "maintainOriginalFileName", rowType: XLFormRowDescriptorTypeBooleanSwitch, title: NSLocalizedString("_maintain_original_filename_", comment: ""))
  111. row.value = CCUtility.getOriginalFileName(k_keyFileNameOriginal)
  112. row.cellConfig["textLabel.font"] = UIFont.systemFont(ofSize: 15.0)
  113. section.addFormRow(row)
  114. // Add File Name Type
  115. row = XLFormRowDescriptor(tag: "addFileNameType", rowType: XLFormRowDescriptorTypeBooleanSwitch, title: NSLocalizedString("_add_filenametype_", comment: ""))
  116. row.value = CCUtility.getFileNameType(k_keyFileNameType)
  117. row.hidden = "$\("maintainOriginalFileName") == 1"
  118. row.cellConfig["textLabel.font"] = UIFont.systemFont(ofSize: 15.0)
  119. section.addFormRow(row)
  120. // Section: Rename File Name
  121. section = XLFormSectionDescriptor.formSection(withTitle: NSLocalizedString("_filename_", comment: ""))
  122. form.addFormSection(section)
  123. row = XLFormRowDescriptor(tag: "maskFileName", rowType: XLFormRowDescriptorTypeAccount, title: (NSLocalizedString("_filename_", comment: "")))
  124. let fileNameMask : String = CCUtility.getFileNameMask(k_keyFileNameMask)
  125. if fileNameMask.count > 0 {
  126. row.value = fileNameMask
  127. }
  128. row.hidden = "$\("maintainOriginalFileName") == 1"
  129. row.cellConfig["textLabel.font"] = UIFont.systemFont(ofSize: 15.0)
  130. row.cellConfig["textField.textAlignment"] = NSTextAlignment.right.rawValue
  131. row.cellConfig["textField.font"] = UIFont.systemFont(ofSize: 15.0)
  132. section.addFormRow(row)
  133. // Section: Preview File Name
  134. row = XLFormRowDescriptor(tag: "previewFileName", rowType: XLFormRowDescriptorTypeTextView, title: "")
  135. row.height = 180
  136. row.disabled = true
  137. row.cellConfig["textView.backgroundColor"] = NCBrandColor.sharedInstance.backgroundView
  138. row.cellConfig["textView.font"] = UIFont.systemFont(ofSize: 14.0)
  139. section.addFormRow(row)
  140. self.form = form
  141. }
  142. override func formRowDescriptorValueHasChanged(_ formRow: XLFormRowDescriptor!, oldValue: Any!, newValue: Any!) {
  143. super.formRowDescriptorValueHasChanged(formRow, oldValue: oldValue, newValue: newValue)
  144. if formRow.tag == "useFolderAutoUpload" {
  145. if (formRow.value! as AnyObject).boolValue == true {
  146. let buttonDestinationFolder : XLFormRowDescriptor = self.form.formRow(withTag: "ButtonDestinationFolder")!
  147. buttonDestinationFolder.hidden = true
  148. } else{
  149. let buttonDestinationFolder : XLFormRowDescriptor = self.form.formRow(withTag: "ButtonDestinationFolder")!
  150. buttonDestinationFolder.hidden = false
  151. }
  152. }
  153. else if formRow.tag == "useSubFolder" {
  154. if (formRow.value! as AnyObject).boolValue == true {
  155. } else{
  156. }
  157. }
  158. else if formRow.tag == "maintainOriginalFileName" {
  159. CCUtility.setOriginalFileName((formRow.value! as AnyObject).boolValue, key: k_keyFileNameOriginal)
  160. self.reloadForm()
  161. }
  162. else if formRow.tag == "addFileNameType" {
  163. CCUtility.setFileNameType((formRow.value! as AnyObject).boolValue, key: k_keyFileNameType)
  164. self.reloadForm()
  165. }
  166. else if formRow.tag == "maskFileName" {
  167. let fileName = formRow.value as? String
  168. self.form.delegate = nil
  169. if let fileName = fileName {
  170. formRow.value = CCUtility.removeForbiddenCharactersServer(fileName)
  171. }
  172. self.form.delegate = self
  173. let previewFileName : XLFormRowDescriptor = self.form.formRow(withTag: "previewFileName")!
  174. previewFileName.value = self.previewFileName(valueRename: formRow.value as? String)
  175. // reload cell
  176. if fileName != nil {
  177. if newValue as! String != formRow.value as! String {
  178. self.reloadFormRow(formRow)
  179. appDelegate.messageNotification("_info_", description: "_forbidden_characters_", visible: true, delay: TimeInterval(k_dismissAfterSecond), type: TWMessageBarMessageType.info, errorCode: 0)
  180. }
  181. }
  182. self.reloadFormRow(previewFileName)
  183. }
  184. }
  185. // MARK: - View Life Cycle
  186. override func viewDidLoad() {
  187. super.viewDidLoad()
  188. let cancelButton : UIBarButtonItem = UIBarButtonItem(title: NSLocalizedString("_cancel_", comment: ""), style: UIBarButtonItem.Style.plain, target: self, action: #selector(cancel))
  189. let saveButton : UIBarButtonItem = UIBarButtonItem(title: NSLocalizedString("_save_", comment: ""), style: UIBarButtonItem.Style.plain, target: self, action: #selector(save))
  190. self.navigationItem.leftBarButtonItem = cancelButton
  191. self.navigationItem.rightBarButtonItem = saveButton
  192. self.navigationController?.navigationBar.isTranslucent = false
  193. self.navigationController?.navigationBar.barTintColor = NCBrandColor.sharedInstance.brand
  194. self.navigationController?.navigationBar.tintColor = NCBrandColor.sharedInstance.brandText
  195. self.navigationController?.navigationBar.titleTextAttributes = [NSAttributedString.Key.foregroundColor: NCBrandColor.sharedInstance.brandText]
  196. self.tableView.separatorStyle = UITableViewCell.SeparatorStyle.none
  197. self.reloadForm()
  198. }
  199. override func viewWillDisappear(_ animated: Bool)
  200. {
  201. super.viewWillDisappear(animated)
  202. self.delegate?.dismissFormUploadAssets()
  203. }
  204. func reloadForm() {
  205. self.form.delegate = nil
  206. let buttonDestinationFolder : XLFormRowDescriptor = self.form.formRow(withTag: "ButtonDestinationFolder")!
  207. buttonDestinationFolder.title = self.titleServerUrl
  208. let maskFileName : XLFormRowDescriptor = self.form.formRow(withTag: "maskFileName")!
  209. let previewFileName : XLFormRowDescriptor = self.form.formRow(withTag: "previewFileName")!
  210. previewFileName.value = self.previewFileName(valueRename: maskFileName.value as? String)
  211. self.tableView.reloadData()
  212. self.form.delegate = self
  213. }
  214. // MARK: - Action
  215. func dismissSelect(serverUrl: String?, metadata: tableMetadata?, type: String) {
  216. if serverUrl != nil {
  217. self.serverUrl = serverUrl!
  218. if serverUrl == CCUtility.getHomeServerUrlActiveUrl(appDelegate.activeUrl) {
  219. self.titleServerUrl = "/"
  220. } else {
  221. self.titleServerUrl = (serverUrl! as NSString).lastPathComponent
  222. }
  223. // Update
  224. let row : XLFormRowDescriptor = self.form.formRow(withTag: "ButtonDestinationFolder")!
  225. row.title = self.titleServerUrl
  226. self.updateFormRow(row)
  227. }
  228. }
  229. @objc func save() {
  230. self.dismiss(animated: true, completion: {
  231. let useFolderPhotoRow : XLFormRowDescriptor = self.form.formRow(withTag: "useFolderAutoUpload")!
  232. let useSubFolderRow : XLFormRowDescriptor = self.form.formRow(withTag: "useSubFolder")!
  233. var useSubFolder : Bool = false
  234. if (useFolderPhotoRow.value! as AnyObject).boolValue == true {
  235. self.serverUrl = NCManageDatabase.sharedInstance.getAccountAutoUploadPath(self.appDelegate.activeUrl)
  236. useSubFolder = (useSubFolderRow.value! as AnyObject).boolValue
  237. }
  238. self.appDelegate.activeMain.uploadFileAsset(self.assets, serverUrl: self.serverUrl, useSubFolder: useSubFolder, session: self.session)
  239. })
  240. }
  241. @objc func cancel() {
  242. self.dismiss(animated: true, completion: nil)
  243. }
  244. // MARK: - Utility
  245. func previewFileName(valueRename : String?) -> String {
  246. var returnString: String = ""
  247. let asset = assets[0] as! PHAsset
  248. if (CCUtility.getOriginalFileName(k_keyFileNameOriginal)) {
  249. return (NSLocalizedString("_filename_", comment: "") + ": " + (asset.value(forKey: "filename") as! String))
  250. } else if let valueRename = valueRename {
  251. let valueRenameTrimming = valueRename.trimmingCharacters(in: CharacterSet.whitespacesAndNewlines)
  252. if valueRenameTrimming.count > 0 {
  253. self.form.delegate = nil
  254. CCUtility.setFileNameMask(valueRename, key: k_keyFileNameMask)
  255. self.form.delegate = self
  256. returnString = CCUtility.createFileName(asset.value(forKey: "filename") as! String?, fileDate: asset.creationDate, fileType: asset.mediaType, keyFileName: k_keyFileNameMask, keyFileNameType: k_keyFileNameType, keyFileNameOriginal: k_keyFileNameOriginal)
  257. } else {
  258. CCUtility.setFileNameMask("", key: k_keyFileNameMask)
  259. returnString = CCUtility.createFileName(asset.value(forKey: "filename") as! String?, fileDate: asset.creationDate, fileType: asset.mediaType, keyFileName: nil, keyFileNameType: k_keyFileNameType, keyFileNameOriginal: k_keyFileNameOriginal)
  260. }
  261. } else {
  262. CCUtility.setFileNameMask("", key: k_keyFileNameMask)
  263. returnString = CCUtility.createFileName(asset.value(forKey: "filename") as! String?, fileDate: asset.creationDate, fileType: asset.mediaType, keyFileName: nil, keyFileNameType: k_keyFileNameType, keyFileNameOriginal: k_keyFileNameOriginal)
  264. }
  265. return String(format: NSLocalizedString("_preview_filename_", comment: ""), "MM,MMM,DD,YY,YYYY and HH,hh,mm,ss,ampm") + ":" + "\n\n" + returnString
  266. }
  267. @objc func changeDestinationFolder(_ sender: XLFormRowDescriptor) {
  268. self.deselectFormRow(sender)
  269. let storyboard = UIStoryboard(name: "NCSelect", bundle: nil)
  270. let navigationController = storyboard.instantiateInitialViewController() as! UINavigationController
  271. let viewController = navigationController.topViewController as! NCSelect
  272. viewController.delegate = self
  273. viewController.hideButtonCreateFolder = false
  274. viewController.includeDirectoryE2EEncryption = true
  275. viewController.includeImages = false
  276. viewController.layoutViewSelect = k_layout_view_move
  277. viewController.selectFile = false
  278. viewController.titleButtonDone = NSLocalizedString("_select_", comment: "")
  279. viewController.type = ""
  280. navigationController.modalPresentationStyle = UIModalPresentationStyle.formSheet
  281. self.present(navigationController, animated: true, completion: nil)
  282. }
  283. @objc func photoEditor(_ sender: XLFormRowDescriptor) {
  284. self.deselectFormRow(sender)
  285. PHImageManager.default().requestImage(for: assets[0] as! PHAsset, targetSize: PHImageManagerMaximumSize, contentMode: PHImageContentMode.default, options: requestOptions, resultHandler: { (image, info) in
  286. let photoEditor = PhotoEditorViewController(nibName:"PhotoEditorViewController",bundle: Bundle(for: PhotoEditorViewController.self))
  287. photoEditor.image = image
  288. photoEditor.photoEditorDelegate = self
  289. photoEditor.hiddenControls = [.save, .share, .sticker]
  290. photoEditor.cancelButtonImage = CCGraphics.changeThemingColorImage(UIImage(named: "photoEditorCancel")!, multiplier:2, color: .white)
  291. photoEditor.cropButtonImage = CCGraphics.changeThemingColorImage(UIImage(named: "photoEditorCrop")!, multiplier:2, color: .white)
  292. photoEditor.drawButtonImage = CCGraphics.changeThemingColorImage(UIImage(named: "photoEditorDraw")!, multiplier:2, color: .white)
  293. photoEditor.textButtonImage = CCGraphics.changeThemingColorImage(UIImage(named: "photoEditorText")!, multiplier:2, color: .white)
  294. photoEditor.clearButtonImage = CCGraphics.changeThemingColorImage(UIImage(named: "photoEditorClear")!, multiplier:2, color: .white)
  295. photoEditor.continueButtonImage = CCGraphics.changeThemingColorImage(UIImage(named: "photoEditorDone")!, multiplier:2, color: .white)
  296. self.present(photoEditor, animated: true, completion: nil)
  297. })
  298. }
  299. // MARK: - Photo Editor Delegate
  300. func doneEditing(image: UIImage) {
  301. UIImageWriteToSavedPhotosAlbum(image, self, #selector(imageSaved(_:didFinishSavingWithError:contextInfo:)), nil)
  302. }
  303. @objc func imageSaved(_ image: UIImage, didFinishSavingWithError error: Error?, contextInfo: UnsafeRawPointer) {
  304. let fetchOptions = PHFetchOptions()
  305. fetchOptions.sortDescriptors = [NSSortDescriptor(key: "creationDate", ascending: false)]
  306. let fetchResult = PHAsset.fetchAssets(with: PHAssetMediaType.image, options: fetchOptions)
  307. if let asset = fetchResult.firstObject {
  308. self.assets = NSMutableArray(array: [asset])
  309. }
  310. // Preview
  311. PHImageManager.default().requestImage(for: assets[0] as! PHAsset, targetSize: targetSizeImagePreview, contentMode: PHImageContentMode.aspectFill, options: requestOptions, resultHandler: { (image, info) in
  312. let row : XLFormRowDescriptor = self.form.formRow(withTag: "ButtonPhotoEditor")!
  313. row.cellConfig["imageView.image"] = image
  314. self.updateFormRow(row)
  315. })
  316. }
  317. func canceledEditing() {
  318. print("Canceled")
  319. }
  320. }