NCListCell.swift 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358
  1. //
  2. // NCListCell.swift
  3. // Nextcloud
  4. //
  5. // Created by Marino Faggiana on 24/10/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 UIKit
  24. class NCListCell: UICollectionViewCell, UIGestureRecognizerDelegate, NCCellProtocol {
  25. @IBOutlet weak var imageItem: UIImageView!
  26. @IBOutlet weak var imageSelect: UIImageView!
  27. @IBOutlet weak var imageStatus: UIImageView!
  28. @IBOutlet weak var imageFavorite: UIImageView!
  29. @IBOutlet weak var imageLocal: UIImageView!
  30. @IBOutlet weak var labelTitle: UILabel!
  31. @IBOutlet weak var labelInfo: UILabel!
  32. @IBOutlet weak var imageShared: UIImageView!
  33. @IBOutlet weak var buttonShared: UIButton!
  34. @IBOutlet weak var imageMore: UIImageView!
  35. @IBOutlet weak var buttonMore: UIButton!
  36. @IBOutlet weak var progressView: UIProgressView!
  37. @IBOutlet weak var separator: UIView!
  38. @IBOutlet weak var tag0: UILabel!
  39. @IBOutlet weak var tag1: UILabel!
  40. @IBOutlet weak var imageItemLeftConstraint: NSLayoutConstraint!
  41. @IBOutlet weak var separatorHeightConstraint: NSLayoutConstraint!
  42. @IBOutlet weak var titleTrailingConstraint: NSLayoutConstraint!
  43. @IBOutlet weak var infoTrailingConstraint: NSLayoutConstraint!
  44. private var objectId = ""
  45. private var user = ""
  46. weak var delegate: NCListCellDelegate?
  47. var namedButtonMore = ""
  48. var fileAvatarImageView: UIImageView? {
  49. get { return imageShared }
  50. }
  51. var fileObjectId: String? {
  52. get { return objectId }
  53. set { objectId = newValue ?? "" }
  54. }
  55. var filePreviewImageView: UIImageView? {
  56. get { return imageItem }
  57. set { imageItem = newValue }
  58. }
  59. var fileUser: String? {
  60. get { return user }
  61. set { user = newValue ?? "" }
  62. }
  63. var fileTitleLabel: UILabel? {
  64. get { return labelTitle }
  65. set { labelTitle = newValue }
  66. }
  67. var fileInfoLabel: UILabel? {
  68. get { return labelInfo }
  69. set { labelInfo = newValue }
  70. }
  71. var fileProgressView: UIProgressView? {
  72. get { return progressView }
  73. set { progressView = newValue }
  74. }
  75. var fileSelectImage: UIImageView? {
  76. get { return imageSelect }
  77. set { imageSelect = newValue }
  78. }
  79. var fileStatusImage: UIImageView? {
  80. get { return imageStatus }
  81. set { imageStatus = newValue }
  82. }
  83. var fileLocalImage: UIImageView? {
  84. get { return imageLocal }
  85. set { imageLocal = newValue }
  86. }
  87. var fileFavoriteImage: UIImageView? {
  88. get { return imageFavorite }
  89. set { imageFavorite = newValue }
  90. }
  91. var fileSharedImage: UIImageView? {
  92. get { return imageShared }
  93. set { imageShared = newValue }
  94. }
  95. var fileMoreImage: UIImageView? {
  96. get { return imageMore }
  97. set { imageMore = newValue }
  98. }
  99. var cellSeparatorView: UIView? {
  100. get { return separator }
  101. set { separator = newValue }
  102. }
  103. override func awakeFromNib() {
  104. super.awakeFromNib()
  105. imageItem.layer.cornerRadius = 6
  106. imageItem.layer.masksToBounds = true
  107. // use entire cell as accessibility element
  108. accessibilityHint = nil
  109. accessibilityLabel = nil
  110. accessibilityValue = nil
  111. isAccessibilityElement = true
  112. progressView.tintColor = NCBrandColor.shared.brand
  113. progressView.transform = CGAffineTransform(scaleX: 1.0, y: 0.5)
  114. progressView.trackTintColor = .clear
  115. let longPressedGesture = UILongPressGestureRecognizer(target: self, action: #selector(longPress(gestureRecognizer:)))
  116. longPressedGesture.minimumPressDuration = 0.5
  117. longPressedGesture.delegate = self
  118. longPressedGesture.delaysTouchesBegan = true
  119. self.addGestureRecognizer(longPressedGesture)
  120. let longPressedGestureMore = UILongPressGestureRecognizer(target: self, action: #selector(longPressInsideMore(gestureRecognizer:)))
  121. longPressedGestureMore.minimumPressDuration = 0.5
  122. longPressedGestureMore.delegate = self
  123. longPressedGestureMore.delaysTouchesBegan = true
  124. buttonMore.addGestureRecognizer(longPressedGestureMore)
  125. separator.backgroundColor = .separator
  126. separatorHeightConstraint.constant = 0.5
  127. labelTitle.text = ""
  128. labelInfo.text = ""
  129. labelTitle.textColor = .label
  130. labelInfo.textColor = .systemGray
  131. }
  132. override func prepareForReuse() {
  133. super.prepareForReuse()
  134. imageItem.backgroundColor = nil
  135. accessibilityHint = nil
  136. accessibilityLabel = nil
  137. accessibilityValue = nil
  138. }
  139. override func snapshotView(afterScreenUpdates afterUpdates: Bool) -> UIView? {
  140. return nil
  141. }
  142. @IBAction func touchUpInsideShare(_ sender: Any) {
  143. delegate?.tapShareListItem(with: objectId, sender: sender)
  144. }
  145. @IBAction func touchUpInsideMore(_ sender: Any) {
  146. delegate?.tapMoreListItem(with: objectId, namedButtonMore: namedButtonMore, image: imageItem.image, sender: sender)
  147. }
  148. @objc func longPressInsideMore(gestureRecognizer: UILongPressGestureRecognizer) {
  149. delegate?.longPressMoreListItem(with: objectId, namedButtonMore: namedButtonMore, gestureRecognizer: gestureRecognizer)
  150. }
  151. @objc func longPress(gestureRecognizer: UILongPressGestureRecognizer) {
  152. delegate?.longPressListItem(with: objectId, gestureRecognizer: gestureRecognizer)
  153. }
  154. fileprivate func setA11yActions() {
  155. let moreName = namedButtonMore == NCGlobal.shared.buttonMoreStop ? "_cancel_" : "_more_"
  156. self.accessibilityCustomActions = [
  157. UIAccessibilityCustomAction(
  158. name: NSLocalizedString("_share_", comment: ""),
  159. target: self,
  160. selector: #selector(touchUpInsideShare)),
  161. UIAccessibilityCustomAction(
  162. name: NSLocalizedString(moreName, comment: ""),
  163. target: self,
  164. selector: #selector(touchUpInsideMore))
  165. ]
  166. }
  167. func titleInfoTrailingFull() {
  168. titleTrailingConstraint.constant = 10
  169. infoTrailingConstraint.constant = 10
  170. }
  171. func titleInfoTrailingDefault() {
  172. titleTrailingConstraint.constant = 90
  173. infoTrailingConstraint.constant = 90
  174. }
  175. func setButtonMore(named: String, image: UIImage) {
  176. namedButtonMore = named
  177. imageMore.image = image
  178. setA11yActions()
  179. }
  180. func hideButtonMore(_ status: Bool) {
  181. imageMore.isHidden = status
  182. buttonMore.isHidden = status
  183. }
  184. func hideButtonShare(_ status: Bool) {
  185. imageShared.isHidden = status
  186. buttonShared.isHidden = status
  187. }
  188. func hideSeparator(_ status: Bool) {
  189. separator.isHidden = status
  190. }
  191. func selectMode(_ status: Bool) {
  192. if status {
  193. imageItemLeftConstraint.constant = 45
  194. imageSelect.isHidden = false
  195. imageShared.isHidden = true
  196. imageMore.isHidden = true
  197. buttonShared.isHidden = true
  198. buttonMore.isHidden = true
  199. accessibilityCustomActions = nil
  200. } else {
  201. imageItemLeftConstraint.constant = 10
  202. imageSelect.isHidden = true
  203. imageShared.isHidden = false
  204. imageMore.isHidden = false
  205. buttonShared.isHidden = false
  206. buttonMore.isHidden = false
  207. backgroundView = nil
  208. setA11yActions()
  209. }
  210. }
  211. func selected(_ status: Bool) {
  212. guard let metadata = NCManageDatabase.shared.getMetadataFromOcId(objectId), !metadata.isInTransfer else {
  213. backgroundView = nil
  214. separator.isHidden = false
  215. return
  216. }
  217. if status {
  218. var blurEffect: UIVisualEffect?
  219. var blurEffectView: UIView?
  220. imageSelect.image = NCBrandColor.cacheImages.checkedYes
  221. if traitCollection.userInterfaceStyle == .dark {
  222. blurEffect = UIBlurEffect(style: .dark)
  223. blurEffectView = UIVisualEffectView(effect: blurEffect)
  224. blurEffectView?.backgroundColor = .black
  225. } else {
  226. blurEffect = UIBlurEffect(style: .extraLight)
  227. blurEffectView = UIVisualEffectView(effect: blurEffect)
  228. blurEffectView?.backgroundColor = .lightGray
  229. }
  230. blurEffectView?.frame = self.bounds
  231. blurEffectView?.autoresizingMask = [.flexibleWidth, .flexibleHeight]
  232. backgroundView = blurEffectView
  233. separator.isHidden = true
  234. } else {
  235. imageSelect.image = NCBrandColor.cacheImages.checkedNo
  236. backgroundView = nil
  237. separator.isHidden = false
  238. }
  239. }
  240. func writeInfoDateSize(date: NSDate, size: Int64) {
  241. labelInfo.text = CCUtility.dateDiff(date as Date) + " · " + CCUtility.transformedSize(size)
  242. }
  243. func setAccessibility(label: String, value: String) {
  244. accessibilityLabel = label
  245. accessibilityValue = value
  246. }
  247. func setTags(tags: [String]) {
  248. if tags.isEmpty {
  249. tag0.isHidden = true
  250. tag1.isHidden = true
  251. labelInfo.isHidden = false
  252. } else {
  253. tag0.isHidden = false
  254. tag1.isHidden = true
  255. labelInfo.isHidden = true
  256. if let tag = tags.first {
  257. tag0.text = tag
  258. if tags.count > 1 {
  259. tag1.isHidden = false
  260. tag1.text = "+\(tags.count-1)"
  261. }
  262. }
  263. }
  264. }
  265. }
  266. protocol NCListCellDelegate: AnyObject {
  267. func tapShareListItem(with objectId: String, sender: Any)
  268. func tapMoreListItem(with objectId: String, namedButtonMore: String, image: UIImage?, sender: Any)
  269. func longPressMoreListItem(with objectId: String, namedButtonMore: String, gestureRecognizer: UILongPressGestureRecognizer)
  270. func longPressListItem(with objectId: String, gestureRecognizer: UILongPressGestureRecognizer)
  271. }
  272. // optional func
  273. extension NCListCellDelegate {
  274. func tapShareListItem(with objectId: String, sender: Any) {}
  275. func tapMoreListItem(with objectId: String, namedButtonMore: String, image: UIImage?, sender: Any) {}
  276. func longPressMoreListItem(with objectId: String, namedButtonMore: String, gestureRecognizer: UILongPressGestureRecognizer) {}
  277. func longPressListItem(with objectId: String, gestureRecognizer: UILongPressGestureRecognizer) {}
  278. }
  279. // MARK: - List Layout
  280. class NCListLayout: UICollectionViewFlowLayout {
  281. var itemHeight: CGFloat = 60
  282. // MARK: - View Life Cycle
  283. override init() {
  284. super.init()
  285. sectionHeadersPinToVisibleBounds = false
  286. minimumInteritemSpacing = 0
  287. minimumLineSpacing = 1
  288. self.scrollDirection = .vertical
  289. self.sectionInset = UIEdgeInsets(top: 0, left: 0, bottom: 0, right: 0)
  290. }
  291. required init?(coder aDecoder: NSCoder) {
  292. fatalError("init(coder:) has not been implemented")
  293. }
  294. override var itemSize: CGSize {
  295. get {
  296. if let collectionView = collectionView {
  297. let itemWidth: CGFloat = collectionView.frame.width
  298. return CGSize(width: itemWidth, height: self.itemHeight)
  299. }
  300. // Default fallback
  301. return CGSize(width: 100, height: 100)
  302. }
  303. set {
  304. super.itemSize = newValue
  305. }
  306. }
  307. override func targetContentOffset(forProposedContentOffset proposedContentOffset: CGPoint) -> CGPoint {
  308. return proposedContentOffset
  309. }
  310. }