NCShareAdvancePermission.swift 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. //
  2. // NCShareAdvancePermission.swift
  3. // Nextcloud
  4. //
  5. // Created by T-systems on 09/08/21.
  6. // Copyright © 2022 Henrik Storch. All rights reserved.
  7. //
  8. // Author Henrik Storch <henrik.storch@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 UIKit
  24. import NextcloudKit
  25. import SVGKit
  26. import CloudKit
  27. class NCShareAdvancePermission: UITableViewController, NCShareAdvanceFotterDelegate, NCShareDetail {
  28. func dismissShareAdvanceView(shouldSave: Bool) {
  29. guard shouldSave else {
  30. guard oldTableShare?.hasChanges(comparedTo: share) != false else {
  31. navigationController?.popViewController(animated: true)
  32. return
  33. }
  34. let alert = UIAlertController(
  35. title: NSLocalizedString("_cancel_request_", comment: ""),
  36. message: NSLocalizedString("_discard_changes_info_", comment: ""),
  37. preferredStyle: .alert)
  38. alert.addAction(UIAlertAction(
  39. title: NSLocalizedString("_discard_changes_", comment: ""),
  40. style: .destructive,
  41. handler: { _ in self.navigationController?.popViewController(animated: true) }))
  42. alert.addAction(UIAlertAction(title: NSLocalizedString("_continue_editing_", comment: ""), style: .default))
  43. self.present(alert, animated: true)
  44. return
  45. }
  46. Task {
  47. if isNewShare {
  48. let serverUrl = metadata.serverUrl + "/" + metadata.fileName
  49. if share.shareType != NCShareCommon().SHARE_TYPE_LINK, metadata.e2eEncrypted,
  50. NCCapabilities.shared.getCapabilities(account: metadata.account).capabilityE2EEApiVersion == NCGlobal.shared.e2eeVersionV20 {
  51. if NCNetworkingE2EE().isInUpload(account: metadata.account, serverUrl: serverUrl) {
  52. let error = NKError(errorCode: NCGlobal.shared.errorE2EEUploadInProgress, errorDescription: NSLocalizedString("_e2e_in_upload_", comment: ""))
  53. return NCContentPresenter().showInfo(error: error)
  54. }
  55. let error = await NCNetworkingE2EE().uploadMetadata(serverUrl: serverUrl, addUserId: share.shareWith, removeUserId: nil, account: metadata.account)
  56. if error != .success {
  57. return NCContentPresenter().showError(error: error)
  58. }
  59. }
  60. networking?.createShare(option: share)
  61. } else {
  62. networking?.updateShare(option: share)
  63. }
  64. }
  65. navigationController?.popViewController(animated: true)
  66. }
  67. var oldTableShare: tableShare?
  68. var share: NCTableShareable!
  69. var isNewShare: Bool { share is NCTableShareOptions }
  70. var metadata: tableMetadata!
  71. var shareConfig: NCShareConfig!
  72. var networking: NCShareNetworking?
  73. override func viewDidLoad() {
  74. super.viewDidLoad()
  75. self.shareConfig = NCShareConfig(parentMetadata: metadata, share: share)
  76. tableView.estimatedRowHeight = tableView.rowHeight
  77. tableView.rowHeight = UITableView.automaticDimension
  78. self.setNavigationTitle()
  79. self.navigationItem.hidesBackButton = true
  80. // disbale pull to dimiss
  81. isModalInPresentation = true
  82. }
  83. override func viewWillLayoutSubviews() {
  84. super.viewWillLayoutSubviews()
  85. guard tableView.tableHeaderView == nil, tableView.tableFooterView == nil else { return }
  86. setupHeaderView()
  87. setupFooterView()
  88. }
  89. func setupFooterView() {
  90. guard let footerView = (Bundle.main.loadNibNamed("NCShareAdvancePermissionFooter", owner: self, options: nil)?.first as? NCShareAdvancePermissionFooter) else { return }
  91. footerView.setupUI(delegate: self, account: metadata.account)
  92. // tableFooterView can't use auto layout directly
  93. let container = UIView(frame: CGRect(x: 0, y: 0, width: view.frame.width, height: 120))
  94. container.addSubview(footerView)
  95. tableView.tableFooterView = container
  96. footerView.translatesAutoresizingMaskIntoConstraints = false
  97. footerView.bottomAnchor.constraint(equalTo: container.bottomAnchor).isActive = true
  98. footerView.heightAnchor.constraint(equalTo: container.heightAnchor).isActive = true
  99. footerView.widthAnchor.constraint(equalTo: container.widthAnchor).isActive = true
  100. }
  101. func setupHeaderView() {
  102. guard let headerView = (Bundle.main.loadNibNamed("NCShareHeader", owner: self, options: nil)?.first as? NCShareHeader) else { return }
  103. headerView.setupUI(with: metadata)
  104. let container = UIView(frame: CGRect(x: 0, y: 0, width: view.frame.width, height: 220))
  105. container.addSubview(headerView)
  106. tableView.tableHeaderView = container
  107. headerView.translatesAutoresizingMaskIntoConstraints = false
  108. headerView.topAnchor.constraint(equalTo: container.topAnchor).isActive = true
  109. headerView.heightAnchor.constraint(equalTo: container.heightAnchor).isActive = true
  110. headerView.widthAnchor.constraint(equalTo: container.widthAnchor).isActive = true
  111. }
  112. override func tableView(_ tableView: UITableView, titleForHeaderInSection section: Int) -> String? {
  113. if section == 0 {
  114. return NSLocalizedString("_permissions_", comment: "")
  115. } else if section == 1 {
  116. return NSLocalizedString("_advanced_", comment: "")
  117. } else { return nil }
  118. }
  119. override func numberOfSections(in tableView: UITableView) -> Int {
  120. return 2
  121. }
  122. override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
  123. if section == 0 {
  124. // check reshare permission, if restricted add note
  125. let maxPermission = metadata.directory ? NCPermissions().permissionMaxFolderShare : NCPermissions().permissionMaxFileShare
  126. return shareConfig.resharePermission != maxPermission ? shareConfig.permissions.count + 1 : shareConfig.permissions.count
  127. } else if section == 1 {
  128. return shareConfig.advanced.count
  129. } else { return 0 }
  130. }
  131. override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
  132. guard let cell = shareConfig.cellFor(indexPath: indexPath) else {
  133. let noteCell = UITableViewCell(style: .subtitle, reuseIdentifier: "noteCell")
  134. noteCell.detailTextLabel?.text = NSLocalizedString("_share_reshare_restricted_", comment: "")
  135. noteCell.detailTextLabel?.isEnabled = false
  136. noteCell.isUserInteractionEnabled = false
  137. noteCell.detailTextLabel?.numberOfLines = 0
  138. return noteCell
  139. }
  140. if let cell = cell as? NCShareDateCell { cell.onReload = tableView.reloadData }
  141. return cell
  142. }
  143. override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
  144. tableView.deselectRow(at: indexPath, animated: true)
  145. guard let cellConfig = shareConfig.config(for: indexPath) else { return }
  146. guard let cellConfig = cellConfig as? NCShareDetails else {
  147. cellConfig.didSelect(for: share)
  148. tableView.reloadData()
  149. return
  150. }
  151. switch cellConfig {
  152. case .hideDownload:
  153. share.hideDownload.toggle()
  154. tableView.reloadData()
  155. case .expirationDate:
  156. let cell = tableView.cellForRow(at: indexPath) as? NCShareDateCell
  157. cell?.textField.becomeFirstResponder()
  158. cell?.checkMaximumDate(account: metadata.account)
  159. case .password:
  160. guard share.password.isEmpty else {
  161. share.password = ""
  162. tableView.reloadData()
  163. return
  164. }
  165. let alertController = UIAlertController.password(titleKey: "_share_password_") { password in
  166. self.share.password = password ?? ""
  167. tableView.reloadData()
  168. }
  169. self.present(alertController, animated: true)
  170. case .note:
  171. let storyboard = UIStoryboard(name: "NCShare", bundle: nil)
  172. guard let viewNewUserComment = storyboard.instantiateViewController(withIdentifier: "NCShareNewUserAddComment") as? NCShareNewUserAddComment else { return }
  173. viewNewUserComment.metadata = self.metadata
  174. viewNewUserComment.share = self.share
  175. viewNewUserComment.onDismiss = tableView.reloadData
  176. self.navigationController?.pushViewController(viewNewUserComment, animated: true)
  177. case .label:
  178. let alertController = UIAlertController.withTextField(titleKey: "_share_link_name_") { textField in
  179. textField.placeholder = cellConfig.title
  180. textField.text = self.share.label
  181. } completion: { newValue in
  182. self.share.label = newValue ?? ""
  183. self.setNavigationTitle()
  184. tableView.reloadData()
  185. }
  186. self.present(alertController, animated: true)
  187. }
  188. }
  189. }