NCShare.swift 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364
  1. //
  2. // NCShare.swift
  3. // Nextcloud
  4. //
  5. // Created by Marino Faggiana on 17/07/2019.
  6. // Copyright © 2019 Marino Faggiana. All rights reserved.
  7. // Copyright © 2022 Henrik Storch. All rights reserved.
  8. //
  9. // Author Marino Faggiana <marino.faggiana@nextcloud.com>
  10. // Author Henrik Storch <henrik.storch@nextcloud.com>
  11. //
  12. // This program is free software: you can redistribute it and/or modify
  13. // it under the terms of the GNU General Public License as published by
  14. // the Free Software Foundation, either version 3 of the License, or
  15. // (at your option) any later version.
  16. //
  17. // This program is distributed in the hope that it will be useful,
  18. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  19. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  20. // GNU General Public License for more details.
  21. //
  22. // You should have received a copy of the GNU General Public License
  23. // along with this program. If not, see <http://www.gnu.org/licenses/>.
  24. //
  25. import UIKit
  26. import Parchment
  27. import DropDown
  28. import NextcloudKit
  29. import MarqueeLabel
  30. class NCShare: UIViewController, NCShareNetworkingDelegate, NCSharePagingContent {
  31. @IBOutlet weak var viewContainerConstraint: NSLayoutConstraint!
  32. @IBOutlet weak var sharedWithYouByView: UIView!
  33. @IBOutlet weak var sharedWithYouByImage: UIImageView!
  34. @IBOutlet weak var sharedWithYouByLabel: UILabel!
  35. @IBOutlet weak var sharedWithYouByNoteImage: UIImageView!
  36. @IBOutlet weak var sharedWithYouByNote: MarqueeLabel!
  37. @IBOutlet weak var searchFieldTopConstraint: NSLayoutConstraint!
  38. @IBOutlet weak var searchField: UITextField!
  39. var textField: UITextField? { searchField }
  40. @IBOutlet weak var tableView: UITableView!
  41. weak var appDelegate = UIApplication.shared.delegate as? AppDelegate
  42. public var metadata: tableMetadata?
  43. public var sharingEnabled = true
  44. public var height: CGFloat = 0
  45. var canReshare: Bool {
  46. guard let metadata = metadata else { return true }
  47. return ((metadata.sharePermissionsCollaborationServices & NCGlobal.shared.permissionShareShare) != 0)
  48. }
  49. var shares: (firstShareLink: tableShare?, share: [tableShare]?) = (nil, nil)
  50. private var dropDown = DropDown()
  51. var networking: NCShareNetworking?
  52. // MARK: - View Life Cycle
  53. override func viewDidLoad() {
  54. super.viewDidLoad()
  55. view.backgroundColor = .systemBackground
  56. viewContainerConstraint.constant = height
  57. searchFieldTopConstraint.constant = 10
  58. searchField.placeholder = NSLocalizedString("_shareLinksearch_placeholder_", comment: "")
  59. searchField.autocorrectionType = .no
  60. tableView.dataSource = self
  61. tableView.delegate = self
  62. tableView.allowsSelection = false
  63. tableView.backgroundColor = .systemBackground
  64. tableView.register(UINib(nibName: "NCShareLinkCell", bundle: nil), forCellReuseIdentifier: "cellLink")
  65. tableView.register(UINib(nibName: "NCShareUserCell", bundle: nil), forCellReuseIdentifier: "cellUser")
  66. NotificationCenter.default.addObserver(self, selector: #selector(reloadData), name: NSNotification.Name(rawValue: NCGlobal.shared.notificationCenterReloadDataNCShare), object: nil)
  67. guard let metadata = metadata else { return }
  68. if metadata.e2eEncrypted {
  69. let direcrory = NCManageDatabase.shared.getTableDirectory(account: metadata.account, serverUrl: metadata.serverUrl)
  70. if NCGlobal.shared.capabilityE2EEApiVersion == NCGlobal.shared.e2eeVersionV12 ||
  71. (NCGlobal.shared.capabilityE2EEApiVersion == NCGlobal.shared.e2eeVersionV20 && direcrory?.e2eEncrypted ?? false) {
  72. // searchField.isEnabled = false
  73. searchFieldTopConstraint.constant = -50
  74. searchField.isHidden = true
  75. }
  76. } else {
  77. checkSharedWithYou()
  78. }
  79. reloadData()
  80. networking = NCShareNetworking(metadata: metadata, view: self.view, delegate: self)
  81. if sharingEnabled {
  82. let isVisible = (self.navigationController?.topViewController as? NCSharePaging)?.page == .sharing
  83. networking?.readShare(showLoadingIndicator: isVisible)
  84. }
  85. }
  86. func makeNewLinkShare() {
  87. guard
  88. let advancePermission = UIStoryboard(name: "NCShare", bundle: nil).instantiateViewController(withIdentifier: "NCShareAdvancePermission") as? NCShareAdvancePermission,
  89. let navigationController = self.navigationController,
  90. let metadata = self.metadata else { return }
  91. self.checkEnforcedPassword(shareType: NCShareCommon.shared.SHARE_TYPE_LINK) { password in
  92. advancePermission.networking = self.networking
  93. advancePermission.share = NCTableShareOptions.shareLink(metadata: metadata, password: password)
  94. advancePermission.metadata = self.metadata
  95. navigationController.pushViewController(advancePermission, animated: true)
  96. }
  97. }
  98. // Shared with you by ...
  99. func checkSharedWithYou() {
  100. guard let appDelegate = self.appDelegate, let metadata = metadata, !metadata.ownerId.isEmpty, metadata.ownerId != appDelegate.userId else { return }
  101. if !canReshare {
  102. searchField.isEnabled = false
  103. searchField.placeholder = NSLocalizedString("_share_reshare_disabled_", comment: "")
  104. }
  105. searchFieldTopConstraint.constant = 65
  106. sharedWithYouByView.isHidden = false
  107. sharedWithYouByLabel.text = NSLocalizedString("_shared_with_you_by_", comment: "") + " " + metadata.ownerDisplayName
  108. sharedWithYouByImage.image = NCUtility.shared.loadUserImage(
  109. for: metadata.ownerId,
  110. displayName: metadata.ownerDisplayName,
  111. userBaseUrl: appDelegate)
  112. sharedWithYouByLabel.accessibilityHint = NSLocalizedString("_show_profile_", comment: "")
  113. let shareAction = UITapGestureRecognizer(target: self, action: #selector(openShareProfile))
  114. sharedWithYouByImage.addGestureRecognizer(shareAction)
  115. let shareLabelAction = UITapGestureRecognizer(target: self, action: #selector(openShareProfile))
  116. sharedWithYouByLabel.addGestureRecognizer(shareLabelAction)
  117. if !metadata.note.isEmpty {
  118. searchFieldTopConstraint.constant = 95
  119. sharedWithYouByNoteImage.isHidden = false
  120. sharedWithYouByNoteImage.image = NCUtility.shared.loadImage(named: "note.text", color: .gray)
  121. sharedWithYouByNote.isHidden = false
  122. sharedWithYouByNote.text = metadata.note
  123. sharedWithYouByNote.textColor = .label
  124. sharedWithYouByNote.trailingBuffer = sharedWithYouByNote.frame.width
  125. } else {
  126. sharedWithYouByNoteImage.isHidden = true
  127. sharedWithYouByNote.isHidden = true
  128. }
  129. let fileName = appDelegate.userBaseUrl + "-" + metadata.ownerId + ".png"
  130. if NCManageDatabase.shared.getImageAvatarLoaded(fileName: fileName) == nil {
  131. let fileNameLocalPath = String(CCUtility.getDirectoryUserData()) + "/" + fileName
  132. let etag = NCManageDatabase.shared.getTableAvatar(fileName: fileName)?.etag
  133. NextcloudKit.shared.downloadAvatar(
  134. user: metadata.ownerId,
  135. fileNameLocalPath: fileNameLocalPath,
  136. sizeImage: NCGlobal.shared.avatarSize,
  137. avatarSizeRounded: NCGlobal.shared.avatarSizeRounded,
  138. etag: etag) { _, imageAvatar, _, etag, error in
  139. if error == .success, let etag = etag, let imageAvatar = imageAvatar {
  140. NCManageDatabase.shared.addAvatar(fileName: fileName, etag: etag)
  141. self.sharedWithYouByImage.image = imageAvatar
  142. } else if error.errorCode == NCGlobal.shared.errorNotModified, let imageAvatar = NCManageDatabase.shared.setAvatarLoaded(fileName: fileName) {
  143. self.sharedWithYouByImage.image = imageAvatar
  144. }
  145. }
  146. }
  147. }
  148. // MARK: - Notification Center
  149. @objc func openShareProfile() {
  150. guard let metadata = metadata else { return }
  151. self.showProfileMenu(userId: metadata.ownerId)
  152. }
  153. // MARK: -
  154. @objc func reloadData() {
  155. if let metadata = metadata {
  156. shares = NCManageDatabase.shared.getTableShares(metadata: metadata)
  157. }
  158. tableView.reloadData()
  159. }
  160. // MARK: - IBAction
  161. @IBAction func searchFieldDidEndOnExit(textField: UITextField) {
  162. guard let searchString = textField.text, !searchString.isEmpty else { return }
  163. if searchString.contains("@"), !NCUtility.shared.isValidEmail(searchString) { return }
  164. networking?.getSharees(searchString: searchString)
  165. }
  166. func checkEnforcedPassword(shareType: Int, completion: @escaping (String?) -> Void) {
  167. guard NCGlobal.shared.capabilityFileSharingPubPasswdEnforced,
  168. shareType == NCShareCommon.shared.SHARE_TYPE_LINK || shareType == NCShareCommon.shared.SHARE_TYPE_EMAIL
  169. else { return completion(nil) }
  170. self.present(UIAlertController.password(titleKey: "_enforce_password_protection_", completion: completion), animated: true)
  171. }
  172. // MARK: - NCShareNetworkingDelegate
  173. func readShareCompleted() {
  174. NotificationCenter.default.postOnMainThread(name: NCGlobal.shared.notificationCenterReloadDataNCShare)
  175. }
  176. func shareCompleted() {
  177. NotificationCenter.default.postOnMainThread(name: NCGlobal.shared.notificationCenterReloadDataNCShare)
  178. }
  179. func unShareCompleted() {
  180. self.reloadData()
  181. }
  182. func updateShareWithError(idShare: Int) {
  183. self.reloadData()
  184. }
  185. func getSharees(sharees: [NKSharee]?) {
  186. guard let sharees = sharees, let appDelegate = appDelegate else { return }
  187. dropDown = DropDown()
  188. let appearance = DropDown.appearance()
  189. appearance.backgroundColor = .systemBackground
  190. appearance.cornerRadius = 10
  191. appearance.shadowColor = UIColor(white: 0.5, alpha: 1)
  192. appearance.shadowOpacity = 0.9
  193. appearance.shadowRadius = 25
  194. appearance.animationduration = 0.25
  195. appearance.textColor = .darkGray
  196. appearance.setupMaskedCorners([.layerMaxXMaxYCorner, .layerMinXMaxYCorner])
  197. for sharee in sharees {
  198. var label = sharee.label
  199. if sharee.shareType == NCShareCommon.shared.SHARE_TYPE_CIRCLE {
  200. label += " (\(sharee.circleInfo), \(sharee.circleOwner))"
  201. }
  202. dropDown.dataSource.append(label)
  203. }
  204. dropDown.anchorView = searchField
  205. dropDown.bottomOffset = CGPoint(x: 0, y: searchField.bounds.height)
  206. dropDown.width = searchField.bounds.width
  207. dropDown.direction = .bottom
  208. dropDown.cellNib = UINib(nibName: "NCSearchUserDropDownCell", bundle: nil)
  209. dropDown.customCellConfiguration = { (index: Index, _, cell: DropDownCell) -> Void in
  210. guard let cell = cell as? NCSearchUserDropDownCell else { return }
  211. let sharee = sharees[index]
  212. cell.setupCell(sharee: sharee, baseUrl: appDelegate)
  213. }
  214. dropDown.selectionAction = { index, _ in
  215. let sharee = sharees[index]
  216. guard
  217. let advancePermission = UIStoryboard(name: "NCShare", bundle: nil).instantiateViewController(withIdentifier: "NCShareAdvancePermission") as? NCShareAdvancePermission,
  218. let navigationController = self.navigationController,
  219. let metadata = self.metadata else { return }
  220. self.checkEnforcedPassword(shareType: sharee.shareType) { password in
  221. let shareOptions = NCTableShareOptions(sharee: sharee, metadata: metadata, password: password)
  222. advancePermission.share = shareOptions
  223. advancePermission.networking = self.networking
  224. advancePermission.metadata = metadata
  225. navigationController.pushViewController(advancePermission, animated: true)
  226. }
  227. }
  228. dropDown.show()
  229. }
  230. }
  231. // MARK: - UITableViewDelegate
  232. extension NCShare: UITableViewDelegate {
  233. func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {
  234. if indexPath.section == 0, indexPath.row == 0 {
  235. // internal cell has description
  236. return 90
  237. }
  238. return 70
  239. }
  240. }
  241. // MARK: - UITableViewDataSource
  242. extension NCShare: UITableViewDataSource {
  243. func numberOfSections(in tableView: UITableView) -> Int {
  244. return 2
  245. }
  246. func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
  247. guard let metadata = self.metadata else { return 0}
  248. var numRows = shares.share?.count ?? 0
  249. if section == 0 {
  250. if metadata.e2eEncrypted && NCGlobal.shared.capabilityE2EEApiVersion == NCGlobal.shared.e2eeVersionV12 {
  251. numRows = 1
  252. } else {
  253. // don't allow link creation if reshare is disabled
  254. numRows = shares.firstShareLink != nil || canReshare ? 2 : 1
  255. }
  256. }
  257. return numRows
  258. }
  259. func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
  260. // Setup default share cells
  261. guard indexPath.section != 0 else {
  262. guard let cell = tableView.dequeueReusableCell(withIdentifier: "cellLink", for: indexPath) as? NCShareLinkCell, let metadata = self.metadata
  263. else { return UITableViewCell() }
  264. cell.delegate = self
  265. if metadata.e2eEncrypted && NCGlobal.shared.capabilityE2EEApiVersion == NCGlobal.shared.e2eeVersionV12 {
  266. cell.tableShare = shares.firstShareLink
  267. } else {
  268. if indexPath.row == 0 {
  269. cell.isInternalLink = true
  270. } else if shares.firstShareLink?.isInvalidated != true {
  271. cell.tableShare = shares.firstShareLink
  272. }
  273. }
  274. cell.setupCellUI()
  275. return cell
  276. }
  277. guard let appDelegate = appDelegate, let tableShare = shares.share?[indexPath.row] else { return UITableViewCell() }
  278. // LINK
  279. if tableShare.shareType == NCShareCommon.shared.SHARE_TYPE_LINK {
  280. if let cell = tableView.dequeueReusableCell(withIdentifier: "cellLink", for: indexPath) as? NCShareLinkCell {
  281. cell.indexPath = indexPath
  282. cell.tableShare = tableShare
  283. cell.delegate = self
  284. cell.setupCellUI()
  285. return cell
  286. }
  287. } else {
  288. // USER / GROUP etc.
  289. if let cell = tableView.dequeueReusableCell(withIdentifier: "cellUser", for: indexPath) as? NCShareUserCell {
  290. cell.indexPath = indexPath
  291. cell.tableShare = tableShare
  292. cell.delegate = self
  293. cell.setupCellUI(userId: appDelegate.userId)
  294. let fileName = appDelegate.userBaseUrl + "-" + tableShare.shareWith + ".png"
  295. NCOperationQueue.shared.downloadAvatar(user: tableShare.shareWith, dispalyName: tableShare.shareWithDisplayname, fileName: fileName, cell: cell, view: tableView, cellImageView: cell.fileAvatarImageView)
  296. return cell
  297. }
  298. }
  299. return UITableViewCell()
  300. }
  301. }