NCSectionHeaderMenu.swift 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442
  1. //
  2. // NCSectionHeaderFooter.swift
  3. // Nextcloud
  4. //
  5. // Created by Marino Faggiana on 09/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. import MarkdownKit
  25. class NCSectionHeaderMenu: UICollectionReusableView, UIGestureRecognizerDelegate {
  26. @IBOutlet weak var buttonSwitch: UIButton!
  27. @IBOutlet weak var buttonOrder: UIButton!
  28. @IBOutlet weak var buttonMore: UIButton!
  29. @IBOutlet weak var buttonTransfer: UIButton!
  30. @IBOutlet weak var imageButtonTransfer: UIImageView!
  31. @IBOutlet weak var labelTransfer: UILabel!
  32. @IBOutlet weak var progressTransfer: UIProgressView!
  33. @IBOutlet weak var transferSeparatorBottom: UIView!
  34. @IBOutlet weak var textViewRichWorkspace: UITextView!
  35. @IBOutlet weak var labelSection: UILabel!
  36. @IBOutlet weak var viewTransfer: UIView!
  37. @IBOutlet weak var viewButtonsView: UIView!
  38. @IBOutlet weak var viewSeparator: UIView!
  39. @IBOutlet weak var viewRichWorkspace: UIView!
  40. @IBOutlet weak var viewSection: UIView!
  41. @IBOutlet weak var viewTransferHeightConstraint: NSLayoutConstraint!
  42. @IBOutlet weak var viewButtonsViewHeightConstraint: NSLayoutConstraint!
  43. @IBOutlet weak var viewSeparatorHeightConstraint: NSLayoutConstraint!
  44. @IBOutlet weak var viewRichWorkspaceHeightConstraint: NSLayoutConstraint!
  45. @IBOutlet weak var viewSectionHeightConstraint: NSLayoutConstraint!
  46. @IBOutlet weak var transferSeparatorBottomHeightConstraint: NSLayoutConstraint!
  47. weak var delegate: NCSectionHeaderMenuDelegate?
  48. private var markdownParser = MarkdownParser()
  49. private var richWorkspaceText: String?
  50. private var textViewColor: UIColor?
  51. private let gradient: CAGradientLayer = CAGradientLayer()
  52. override func awakeFromNib() {
  53. super.awakeFromNib()
  54. backgroundColor = .clear
  55. buttonSwitch.setImage(UIImage(systemName: "list.bullet")!.image(color: .systemGray, size: 25), for: .normal)
  56. buttonOrder.setTitle("", for: .normal)
  57. buttonOrder.setTitleColor(.systemBlue, for: .normal)
  58. buttonMore.setImage(UIImage(named: "more")!.image(color: .systemGray, size: 25), for: .normal)
  59. // Gradient
  60. gradient.startPoint = CGPoint(x: 0, y: 0.50)
  61. gradient.endPoint = CGPoint(x: 0, y: 1)
  62. viewRichWorkspace.layer.addSublayer(gradient)
  63. let tap = UITapGestureRecognizer(target: self, action: #selector(touchUpInsideViewRichWorkspace(_:)))
  64. tap.delegate = self
  65. viewRichWorkspace?.addGestureRecognizer(tap)
  66. viewSeparatorHeightConstraint.constant = 0.5
  67. viewSeparator.backgroundColor = .separator
  68. markdownParser = MarkdownParser(font: UIFont.systemFont(ofSize: 15), color: .label)
  69. markdownParser.header.font = UIFont.systemFont(ofSize: 25)
  70. if let richWorkspaceText = richWorkspaceText {
  71. textViewRichWorkspace.attributedText = markdownParser.parse(richWorkspaceText)
  72. }
  73. textViewColor = .label
  74. labelSection.text = ""
  75. viewSectionHeightConstraint.constant = 0
  76. buttonTransfer.backgroundColor = .clear
  77. buttonTransfer.setImage(nil, for: .normal)
  78. buttonTransfer.layer.cornerRadius = 6
  79. buttonTransfer.layer.masksToBounds = true
  80. imageButtonTransfer.image = UIImage(systemName: "stop.circle")
  81. imageButtonTransfer.tintColor = .white
  82. labelTransfer.text = ""
  83. progressTransfer.progress = 0
  84. progressTransfer.tintColor = NCBrandColor.shared.brand
  85. progressTransfer.trackTintColor = NCBrandColor.shared.brand.withAlphaComponent(0.2)
  86. transferSeparatorBottom.backgroundColor = .separator
  87. transferSeparatorBottomHeightConstraint.constant = 0.5
  88. }
  89. override func layoutSublayers(of layer: CALayer) {
  90. super.layoutSublayers(of: layer)
  91. gradient.frame = viewRichWorkspace.bounds
  92. setInterfaceColor()
  93. }
  94. override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
  95. super.traitCollectionDidChange(previousTraitCollection)
  96. setInterfaceColor()
  97. }
  98. // MARK: - View
  99. func setStatusButtonsView(enable: Bool) {
  100. buttonSwitch.isEnabled = enable
  101. buttonOrder.isEnabled = enable
  102. buttonMore.isEnabled = enable
  103. }
  104. func buttonMoreIsHidden(_ isHidden: Bool) {
  105. buttonMore.isHidden = isHidden
  106. }
  107. func setImageSwitchList() {
  108. buttonSwitch.setImage(UIImage(systemName: "list.bullet")!.image(color: .systemGray, width: 20, height: 15), for: .normal)
  109. }
  110. func setImageSwitchGrid() {
  111. buttonSwitch.setImage(UIImage(systemName: "square.grid.2x2")!.image(color: .systemGray, size: 20), for: .normal)
  112. }
  113. func setButtonsView(height: CGFloat) {
  114. viewButtonsViewHeightConstraint.constant = height
  115. if height == 0 {
  116. viewButtonsView.isHidden = true
  117. } else {
  118. viewButtonsView.isHidden = false
  119. }
  120. }
  121. func setSortedTitle(_ title: String) {
  122. let title = NSLocalizedString(title, comment: "")
  123. buttonOrder.setTitle(title, for: .normal)
  124. }
  125. // MARK: - RichWorkspace
  126. func setRichWorkspaceHeight(_ size: CGFloat) {
  127. viewRichWorkspaceHeightConstraint.constant = size
  128. if size == 0 {
  129. viewRichWorkspace.isHidden = true
  130. } else {
  131. viewRichWorkspace.isHidden = false
  132. }
  133. }
  134. func setInterfaceColor() {
  135. if traitCollection.userInterfaceStyle == .dark {
  136. gradient.colors = [UIColor(white: 0, alpha: 0).cgColor, UIColor.black.cgColor]
  137. } else {
  138. gradient.colors = [UIColor(white: 1, alpha: 0).cgColor, UIColor.white.cgColor]
  139. }
  140. }
  141. func setRichWorkspaceText(_ text: String?) {
  142. guard let text = text else { return }
  143. if text != self.richWorkspaceText {
  144. textViewRichWorkspace.attributedText = markdownParser.parse(text)
  145. self.richWorkspaceText = text
  146. }
  147. }
  148. // MARK: - Transfer
  149. func setViewTransfer(isHidden: Bool, ocId: String? = nil, text: String? = nil) {
  150. labelTransfer.text = text
  151. viewTransfer.isHidden = isHidden
  152. progressTransfer.progress = 0
  153. if isHidden {
  154. viewTransferHeightConstraint.constant = 0
  155. } else {
  156. var image: UIImage?
  157. if let ocId = ocId,
  158. let metadata = NCManageDatabase.shared.getMetadataFromOcId(ocId) {
  159. image = NCUtility.shared.createFilePreviewImage(ocId: metadata.ocId, etag: metadata.etag, fileNameView: metadata.fileNameView, classFile: metadata.classFile, status: metadata.status, createPreviewMedia: true)?.darken()
  160. if image == nil {
  161. image = UIImage(named: metadata.iconName)
  162. buttonTransfer.backgroundColor = .lightGray
  163. } else {
  164. buttonTransfer.backgroundColor = .clear
  165. }
  166. buttonTransfer.setImage(image, for: .normal)
  167. }
  168. viewTransferHeightConstraint.constant = NCGlobal.shared.heightHeaderTransfer
  169. }
  170. }
  171. // MARK: - Section
  172. func setSectionHeight(_ size: CGFloat) {
  173. viewSectionHeightConstraint.constant = size
  174. if size == 0 {
  175. viewSection.isHidden = true
  176. } else {
  177. viewSection.isHidden = false
  178. }
  179. }
  180. // MARK: - Action
  181. @IBAction func touchUpInsideSwitch(_ sender: Any) {
  182. delegate?.tapButtonSwitch(sender)
  183. }
  184. @IBAction func touchUpInsideOrder(_ sender: Any) {
  185. delegate?.tapButtonOrder(sender)
  186. }
  187. @IBAction func touchUpInsideMore(_ sender: Any) {
  188. delegate?.tapButtonMore(sender)
  189. }
  190. @IBAction func touchUpTransfer(_ sender: Any) {
  191. delegate?.tapButtonTransfer(sender)
  192. }
  193. @objc func touchUpInsideViewRichWorkspace(_ sender: Any) {
  194. delegate?.tapRichWorkspace(sender)
  195. }
  196. }
  197. protocol NCSectionHeaderMenuDelegate: AnyObject {
  198. func tapButtonSwitch(_ sender: Any)
  199. func tapButtonOrder(_ sender: Any)
  200. func tapButtonMore(_ sender: Any)
  201. func tapButtonTransfer(_ sender: Any)
  202. func tapRichWorkspace(_ sender: Any)
  203. }
  204. // optional func
  205. extension NCSectionHeaderMenuDelegate {
  206. func tapButtonSwitch(_ sender: Any) {}
  207. func tapButtonOrder(_ sender: Any) {}
  208. func tapButtonMore(_ sender: Any) {}
  209. func tapButtonTransfer(_ sender: Any) {}
  210. func tapRichWorkspace(_ sender: Any) {}
  211. }
  212. class NCSectionHeader: UICollectionReusableView {
  213. @IBOutlet weak var labelSection: UILabel!
  214. override func awakeFromNib() {
  215. super.awakeFromNib()
  216. self.backgroundColor = UIColor.clear
  217. self.labelSection.text = ""
  218. }
  219. }
  220. class NCSectionFooter: UICollectionReusableView, NCSectionFooterDelegate {
  221. @IBOutlet weak var buttonSection: UIButton!
  222. @IBOutlet weak var activityIndicatorSection: UIActivityIndicatorView!
  223. @IBOutlet weak var labelSection: UILabel!
  224. @IBOutlet weak var separator: UIView!
  225. @IBOutlet weak var separatorHeightConstraint: NSLayoutConstraint!
  226. @IBOutlet weak var buttonSectionHeightConstraint: NSLayoutConstraint!
  227. weak var delegate: NCSectionFooterDelegate?
  228. var metadataForSection: NCMetadataForSection?
  229. override func awakeFromNib() {
  230. super.awakeFromNib()
  231. self.backgroundColor = UIColor.clear
  232. labelSection.textColor = UIColor.systemGray
  233. labelSection.text = ""
  234. separator.backgroundColor = .separator
  235. separatorHeightConstraint.constant = 0.5
  236. buttonIsHidden(true)
  237. activityIndicatorSection.isHidden = true
  238. activityIndicatorSection.color = .label
  239. }
  240. func setTitleLabel(directories: Int, files: Int, size: Int64) {
  241. var foldersText = ""
  242. var filesText = ""
  243. if directories > 1 {
  244. foldersText = "\(directories) " + NSLocalizedString("_folders_", comment: "")
  245. } else if directories == 1 {
  246. foldersText = "1 " + NSLocalizedString("_folder_", comment: "")
  247. }
  248. if files > 1 {
  249. filesText = "\(files) " + NSLocalizedString("_files_", comment: "") + " " + CCUtility.transformedSize(size)
  250. } else if files == 1 {
  251. filesText = "1 " + NSLocalizedString("_file_", comment: "") + " " + CCUtility.transformedSize(size)
  252. }
  253. if foldersText.isEmpty {
  254. labelSection.text = filesText
  255. } else if filesText.isEmpty {
  256. labelSection.text = foldersText
  257. } else {
  258. labelSection.text = foldersText + ", " + filesText
  259. }
  260. }
  261. func setTitleLabel(_ text: String) {
  262. labelSection.text = text
  263. }
  264. func setButtonText(_ text: String) {
  265. buttonSection.setTitle(text, for: .normal)
  266. }
  267. func separatorIsHidden(_ isHidden: Bool) {
  268. separator.isHidden = isHidden
  269. }
  270. func buttonIsHidden(_ isHidden: Bool) {
  271. buttonSection.isHidden = isHidden
  272. if isHidden {
  273. buttonSectionHeightConstraint.constant = 0
  274. } else {
  275. buttonSectionHeightConstraint.constant = NCGlobal.shared.heightFooterButton
  276. }
  277. }
  278. func showActivityIndicatorSection() {
  279. buttonSection.isHidden = true
  280. buttonSectionHeightConstraint.constant = NCGlobal.shared.heightFooterButton
  281. activityIndicatorSection.isHidden = false
  282. activityIndicatorSection.startAnimating()
  283. }
  284. func hideActivityIndicatorSection() {
  285. activityIndicatorSection.stopAnimating()
  286. activityIndicatorSection.isHidden = true
  287. }
  288. // MARK: - Action
  289. @IBAction func touchUpInsideButton(_ sender: Any) {
  290. delegate?.tapButtonSection(sender, metadataForSection: metadataForSection)
  291. }
  292. }
  293. protocol NCSectionFooterDelegate: AnyObject {
  294. func tapButtonSection(_ sender: Any, metadataForSection: NCMetadataForSection?)
  295. }
  296. // optional func
  297. extension NCSectionFooterDelegate {
  298. func tapButtonSection(_ sender: Any, metadataForSection: NCMetadataForSection?) {}
  299. }
  300. // https://stackoverflow.com/questions/16278463/darken-an-uiimage
  301. public extension UIImage {
  302. private enum BlendMode {
  303. case multiply // This results in colors that are at least as dark as either of the two contributing sample colors
  304. case screen // This results in colors that are at least as light as either of the two contributing sample colors
  305. }
  306. // A level of zero yeilds the original image, a level of 1 results in black
  307. func darken(level: CGFloat = 0.5) -> UIImage? {
  308. return blend(mode: .multiply, level: level)
  309. }
  310. // A level of zero yeilds the original image, a level of 1 results in white
  311. func lighten(level: CGFloat = 0.5) -> UIImage? {
  312. return blend(mode: .screen, level: level)
  313. }
  314. private func blend(mode: BlendMode, level: CGFloat) -> UIImage? {
  315. let context = CIContext(options: nil)
  316. var level = level
  317. if level < 0 {
  318. level = 0
  319. } else if level > 1 {
  320. level = 1
  321. }
  322. let filterName: String
  323. switch mode {
  324. case .multiply: // As the level increases we get less white
  325. level = abs(level - 1.0)
  326. filterName = "CIMultiplyBlendMode"
  327. case .screen: // As the level increases we get more white
  328. filterName = "CIScreenBlendMode"
  329. }
  330. let blender = CIFilter(name: filterName)!
  331. let backgroundColor = CIColor(color: UIColor(white: level, alpha: 1))
  332. guard let inputImage = CIImage(image: self) else { return nil }
  333. blender.setValue(inputImage, forKey: kCIInputImageKey)
  334. guard let backgroundImageGenerator = CIFilter(name: "CIConstantColorGenerator") else { return nil }
  335. backgroundImageGenerator.setValue(backgroundColor, forKey: kCIInputColorKey)
  336. guard let backgroundImage = backgroundImageGenerator.outputImage?.cropped(to: CGRect(origin: CGPoint.zero, size: self.size)) else { return nil }
  337. blender.setValue(backgroundImage, forKey: kCIInputBackgroundImageKey)
  338. guard let blendedImage = blender.outputImage else { return nil }
  339. guard let cgImage = context.createCGImage(blendedImage, from: blendedImage.extent) else { return nil }
  340. return UIImage(cgImage: cgImage)
  341. }
  342. }