NCTransfers.swift 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349
  1. //
  2. // NCTransfers.swift
  3. // Nextcloud
  4. //
  5. // Created by Marino Faggiana on 17/09/2020.
  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. import NCCommunication
  25. class NCTransfers: NCCollectionViewCommon, NCTransferCellDelegate {
  26. var metadataTemp: tableMetadata?
  27. required init?(coder aDecoder: NSCoder) {
  28. super.init(coder: aDecoder)
  29. appDelegate.activeTransfers = self
  30. titleCurrentFolder = NSLocalizedString("_transfers_", comment: "")
  31. layoutKey = k_layout_view_transfers
  32. enableSearchBar = false
  33. emptyImage = CCGraphics.changeThemingColorImage(UIImage.init(named: "load"), width: 300, height: 300, color: .gray)
  34. emptyTitle = "_no_transfer_"
  35. emptyDescription = "_no_transfer_sub_"
  36. }
  37. override func viewDidLoad() {
  38. super.viewDidLoad()
  39. listLayout.itemHeight = 105
  40. }
  41. override func viewWillAppear(_ animated: Bool) {
  42. appDelegate.activeViewController = self
  43. collectionView?.collectionViewLayout = listLayout
  44. self.navigationItem.title = titleCurrentFolder
  45. setNavigationItem()
  46. reloadDataSource()
  47. }
  48. override func setNavigationItem() {
  49. self.navigationItem.rightBarButtonItem = nil
  50. self.navigationItem.leftBarButtonItem = nil
  51. }
  52. // MARK: - NotificationCenter
  53. override func downloadStartFile(_ notification: NSNotification) {
  54. if self.view?.window == nil { return }
  55. reloadDataSource()
  56. }
  57. override func downloadedFile(_ notification: NSNotification) {
  58. if self.view?.window == nil { return }
  59. reloadDataSource()
  60. }
  61. override func downloadCancelFile(_ notification: NSNotification) {
  62. if self.view?.window == nil { return }
  63. reloadDataSource()
  64. }
  65. override func uploadStartFile(_ notification: NSNotification) {
  66. if self.view?.window == nil { return }
  67. if let userInfo = notification.userInfo as NSDictionary? {
  68. if let ocId = userInfo["ocId"] as? String {
  69. if let metadata = NCManageDatabase.sharedInstance.getMetadataFromOcId(ocId) {
  70. if let row = dataSource.addMetadata(metadata) {
  71. let indexPath = IndexPath(row: row, section: 0)
  72. collectionView?.performBatchUpdates({
  73. collectionView?.insertItems(at: [indexPath])
  74. }, completion: { (_) in
  75. self.reloadDataSource()
  76. })
  77. }
  78. }
  79. }
  80. }
  81. }
  82. override func uploadedFile(_ notification: NSNotification) {
  83. if self.view?.window == nil { return }
  84. if let userInfo = notification.userInfo as NSDictionary? {
  85. if let ocId = userInfo["ocId"] as? String, let ocIdTemp = userInfo["ocIdTemp"] as? String, let errorCode = userInfo["errorCode"] as? Int {
  86. if let metadata = NCManageDatabase.sharedInstance.getMetadataFromOcId(ocId) {
  87. if errorCode == 0 {
  88. if let row = dataSource.deleteMetadata(ocId: metadata.ocId) {
  89. let indexPath = IndexPath(row: row, section: 0)
  90. collectionView?.performBatchUpdates({
  91. collectionView?.deleteItems(at: [indexPath])
  92. }, completion: { (_) in
  93. self.collectionView?.reloadData()
  94. })
  95. } else {
  96. reloadDataSource()
  97. }
  98. } else if errorCode != NSURLErrorCancelled {
  99. if let row = dataSource.reloadMetadata(ocId: metadata.ocId, ocIdTemp: ocIdTemp) {
  100. let indexPath = IndexPath(row: row, section: 0)
  101. collectionView?.performBatchUpdates({
  102. collectionView?.reloadItems(at: [indexPath])
  103. }, completion: { (_) in
  104. self.collectionView?.reloadData()
  105. })
  106. } else {
  107. reloadDataSource()
  108. }
  109. }
  110. }
  111. }
  112. }
  113. }
  114. override func uploadCancelFile(_ notification: NSNotification) {
  115. if self.view?.window == nil { return }
  116. if let userInfo = notification.userInfo as NSDictionary? {
  117. if let ocId = userInfo["ocId"] as? String {
  118. if let metadata = NCManageDatabase.sharedInstance.getMetadataFromOcId(ocId) {
  119. if let row = dataSource.deleteMetadata(ocId: metadata.ocId) {
  120. let indexPath = IndexPath(row: row, section: 0)
  121. collectionView?.performBatchUpdates({
  122. collectionView?.deleteItems(at: [indexPath])
  123. }, completion: { (_) in
  124. self.collectionView?.reloadData()
  125. })
  126. } else {
  127. self.reloadDataSource()
  128. }
  129. }
  130. }
  131. }
  132. }
  133. // MARK: TAP EVENT
  134. override func longPressMoreListItem(with objectId: String, namedButtonMore: String, gestureRecognizer: UILongPressGestureRecognizer) {
  135. if gestureRecognizer.state != .began { return }
  136. let alertController = UIAlertController(title: nil, message: nil, preferredStyle: .alert)
  137. alertController.addAction(UIAlertAction(title: NSLocalizedString("_cancel_", comment: ""), style: .cancel, handler: nil))
  138. alertController.addAction(UIAlertAction(title: NSLocalizedString("_cancel_all_task_", comment: ""), style: .default, handler: { action in
  139. NCNetworking.shared.cancelAllTransfer(account: self.appDelegate.account) {
  140. self.reloadDataSource()
  141. }
  142. }))
  143. self.present(alertController, animated: true, completion: nil)
  144. }
  145. override func longPressListItem(with objectId: String, gestureRecognizer: UILongPressGestureRecognizer) {
  146. if gestureRecognizer.state != .began { return }
  147. if let metadata = NCManageDatabase.sharedInstance.getMetadataFromOcId(objectId) {
  148. metadataTemp = metadata
  149. let touchPoint = gestureRecognizer.location(in: collectionView)
  150. becomeFirstResponder()
  151. let startTaskItem = UIMenuItem.init(title: NSLocalizedString("_force_start_", comment: ""), action: #selector(startTask(_:)))
  152. UIMenuController.shared.menuItems = [startTaskItem]
  153. UIMenuController.shared.setTargetRect(CGRect(x: touchPoint.x, y: touchPoint.y, width: 0, height: 0), in: collectionView)
  154. UIMenuController.shared.setMenuVisible(true, animated: true)
  155. }
  156. }
  157. override func longPressCollecationView(_ gestureRecognizer: UILongPressGestureRecognizer) { }
  158. @objc func startTask(_ notification: Any) {
  159. guard let metadata = metadataTemp else { return }
  160. metadata.status = Int(k_metadataStatusInUpload)
  161. metadata.session = NCCommunicationCommon.shared.sessionIdentifierUpload
  162. NCManageDatabase.sharedInstance.addMetadata(metadata)
  163. NCNetworking.shared.upload(metadata: metadata) { (_, _) in }
  164. }
  165. override func canPerformAction(_ action: Selector, withSender sender: Any?) -> Bool {
  166. if action != #selector(startTask(_:)) { return false }
  167. guard let metadata = metadataTemp else { return false }
  168. if metadata.e2eEncrypted { return false }
  169. if metadata.status == k_metadataStatusWaitUpload || metadata.status == k_metadataStatusInUpload || metadata.status == k_metadataStatusUploading {
  170. return true
  171. }
  172. return false
  173. }
  174. // MARK: - Collection View
  175. override func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
  176. // nothing
  177. }
  178. override func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, referenceSizeForHeaderInSection section: Int) -> CGSize {
  179. return CGSize(width: collectionView.frame.width, height: 0)
  180. }
  181. override func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
  182. guard let metadata = dataSource.cellForItemAt(indexPath: indexPath) else {
  183. return UICollectionViewCell()
  184. }
  185. let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "transferCell", for: indexPath) as! NCTransferCell
  186. cell.delegate = self
  187. cell.objectId = metadata.ocId
  188. cell.indexPath = indexPath
  189. cell.imageItem.image = nil
  190. cell.imageItem.backgroundColor = nil
  191. cell.labelTitle.text = metadata.fileNameView
  192. cell.labelTitle.textColor = NCBrandColor.sharedInstance.textView
  193. let serverUrlHome = NCUtility.shared.getHomeServer(urlBase: metadata.urlBase, account: metadata.account)
  194. var pathText = metadata.serverUrl.replacingOccurrences(of: serverUrlHome, with: "")
  195. if pathText == "" { pathText = "/" }
  196. cell.labelPath.text = pathText
  197. cell.setButtonMore(named: k_buttonMoreStop, image: NCCollectionCommon.images.cellButtonStop)
  198. cell.progressView.progress = 0.0
  199. cell.separator.backgroundColor = NCBrandColor.sharedInstance.separator
  200. if FileManager().fileExists(atPath: CCUtility.getDirectoryProviderStorageIconOcId(metadata.ocId, etag: metadata.etag)) {
  201. cell.imageItem.image = UIImage(contentsOfFile: CCUtility.getDirectoryProviderStorageIconOcId(metadata.ocId, etag: metadata.etag))
  202. } else {
  203. if metadata.hasPreview {
  204. cell.imageItem.backgroundColor = .lightGray
  205. } else {
  206. if metadata.iconName.count > 0 {
  207. cell.imageItem.image = UIImage.init(named: metadata.iconName)
  208. } else {
  209. cell.imageItem.image = NCCollectionCommon.images.cellFileImage
  210. }
  211. }
  212. }
  213. cell.labelInfo.text = CCUtility.dateDiff(metadata.date as Date) + " · " + CCUtility.transformedSize(metadata.size)
  214. // Transfer
  215. var progress: Float = 0.0
  216. var totalBytes: Double = 0.0
  217. let progressArray = appDelegate.listProgressMetadata.object(forKey: metadata.ocId) as? NSArray
  218. if progressArray != nil && progressArray?.count == 3 {
  219. progress = progressArray?.object(at: 0) as? Float ?? 0
  220. totalBytes = progressArray?.object(at: 1) as? Double ?? 0
  221. }
  222. if metadata.status == k_metadataStatusInDownload || metadata.status == k_metadataStatusDownloading || metadata.status >= k_metadataStatusTypeUpload {
  223. cell.progressView.isHidden = false
  224. } else {
  225. cell.progressView.isHidden = true
  226. cell.progressView.progress = progress
  227. }
  228. // Write status on Label Info
  229. switch metadata.status {
  230. case Int(k_metadataStatusWaitDownload):
  231. cell.labelStatus.text = NSLocalizedString("_status_wait_download_", comment: "")
  232. cell.labelInfo.text = CCUtility.transformedSize(metadata.size)
  233. break
  234. case Int(k_metadataStatusInDownload):
  235. cell.labelStatus.text = NSLocalizedString("_status_in_download_", comment: "")
  236. cell.labelInfo.text = CCUtility.transformedSize(metadata.size)
  237. break
  238. case Int(k_metadataStatusDownloading):
  239. cell.labelStatus.text = NSLocalizedString("_status_downloading_", comment: "")
  240. cell.labelInfo.text = CCUtility.transformedSize(metadata.size) + " - ↓ " + CCUtility.transformedSize(totalBytes)
  241. break
  242. case Int(k_metadataStatusWaitUpload):
  243. cell.labelStatus.text = NSLocalizedString("_status_wait_upload_", comment: "")
  244. cell.labelInfo.text = CCUtility.transformedSize(metadata.size)
  245. break
  246. case Int(k_metadataStatusInUpload):
  247. cell.labelStatus.text = NSLocalizedString("_status_in_upload_", comment: "")
  248. cell.labelInfo.text = CCUtility.transformedSize(metadata.size)
  249. break
  250. case Int(k_metadataStatusUploading):
  251. cell.labelStatus.text = NSLocalizedString("_status_uploading_", comment: "")
  252. cell.labelInfo.text = CCUtility.transformedSize(metadata.size) + " - ↑ " + CCUtility.transformedSize(totalBytes)
  253. break
  254. default:
  255. cell.labelStatus.text = ""
  256. cell.labelInfo.text = ""
  257. break
  258. }
  259. // Remove last separator
  260. if collectionView.numberOfItems(inSection: indexPath.section) == indexPath.row + 1 {
  261. cell.separator.isHidden = true
  262. } else {
  263. cell.separator.isHidden = false
  264. }
  265. return cell
  266. }
  267. // MARK: - DataSource + NC Endpoint
  268. override func reloadDataSource() {
  269. super.reloadDataSource()
  270. metadatasSource = NCManageDatabase.sharedInstance.getAdvancedMetadatas(predicate: NSPredicate(format: "(session CONTAINS 'upload') OR (session CONTAINS 'download')"), page: 1, limit: 100, sorted: "sessionTaskIdentifier", ascending: false)
  271. self.dataSource = NCDataSource.init(metadatasSource: metadatasSource)
  272. refreshControl.endRefreshing()
  273. collectionView.reloadData()
  274. }
  275. override func reloadDataSourceNetwork(forced: Bool = false) {
  276. super.reloadDataSourceNetwork(forced: forced)
  277. reloadDataSource()
  278. }
  279. }