NCSectionFirstHeader.swift 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. //
  2. // NCSectionFirstHeader.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. protocol NCSectionFirstHeaderDelegate: AnyObject {
  26. func tapButtonTransfer(_ sender: Any)
  27. func tapRichWorkspace(_ sender: Any)
  28. }
  29. class NCSectionFirstHeader: UICollectionReusableView, UIGestureRecognizerDelegate {
  30. @IBOutlet weak var buttonTransfer: UIButton!
  31. @IBOutlet weak var imageButtonTransfer: UIImageView!
  32. @IBOutlet weak var labelTransfer: UILabel!
  33. @IBOutlet weak var progressTransfer: UIProgressView!
  34. @IBOutlet weak var transferSeparatorBottom: UIView!
  35. @IBOutlet weak var textViewRichWorkspace: UITextView!
  36. @IBOutlet weak var labelSection: UILabel!
  37. @IBOutlet weak var viewTransfer: UIView!
  38. @IBOutlet weak var viewRichWorkspace: UIView!
  39. @IBOutlet weak var viewSection: UIView!
  40. @IBOutlet weak var viewTransferHeightConstraint: NSLayoutConstraint!
  41. @IBOutlet weak var viewRichWorkspaceHeightConstraint: NSLayoutConstraint!
  42. @IBOutlet weak var viewSectionHeightConstraint: NSLayoutConstraint!
  43. @IBOutlet weak var transferSeparatorBottomHeightConstraint: NSLayoutConstraint!
  44. weak var delegate: NCSectionFirstHeaderDelegate?
  45. let utility = NCUtility()
  46. private var markdownParser = MarkdownParser()
  47. private var richWorkspaceText: String?
  48. private var textViewColor: UIColor?
  49. private let gradient: CAGradientLayer = CAGradientLayer()
  50. override func awakeFromNib() {
  51. super.awakeFromNib()
  52. backgroundColor = .clear
  53. // Gradient
  54. gradient.startPoint = CGPoint(x: 0, y: 0.8)
  55. gradient.endPoint = CGPoint(x: 0, y: 0.9)
  56. viewRichWorkspace.layer.addSublayer(gradient)
  57. let tap = UITapGestureRecognizer(target: self, action: #selector(touchUpInsideViewRichWorkspace(_:)))
  58. tap.delegate = self
  59. viewRichWorkspace?.addGestureRecognizer(tap)
  60. markdownParser = MarkdownParser(font: UIFont.systemFont(ofSize: 15), color: NCBrandColor.shared.textColor)
  61. markdownParser.header.font = UIFont.systemFont(ofSize: 25)
  62. if let richWorkspaceText = richWorkspaceText {
  63. textViewRichWorkspace.attributedText = markdownParser.parse(richWorkspaceText)
  64. }
  65. textViewColor = NCBrandColor.shared.textColor
  66. labelSection.text = ""
  67. viewSectionHeightConstraint.constant = 0
  68. buttonTransfer.backgroundColor = .clear
  69. buttonTransfer.setImage(nil, for: .normal)
  70. buttonTransfer.layer.cornerRadius = 6
  71. buttonTransfer.layer.masksToBounds = true
  72. imageButtonTransfer.image = NCUtility().loadImage(named: "stop.circle")
  73. imageButtonTransfer.tintColor = .white
  74. labelTransfer.text = ""
  75. progressTransfer.progress = 0
  76. progressTransfer.tintColor = NCBrandColor.shared.brandElement
  77. progressTransfer.trackTintColor = NCBrandColor.shared.brandElement.withAlphaComponent(0.2)
  78. transferSeparatorBottom.backgroundColor = .separator
  79. transferSeparatorBottomHeightConstraint.constant = 0.5
  80. }
  81. override func layoutSublayers(of layer: CALayer) {
  82. super.layoutSublayers(of: layer)
  83. gradient.frame = viewRichWorkspace.bounds
  84. setInterfaceColor()
  85. }
  86. override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
  87. super.traitCollectionDidChange(previousTraitCollection)
  88. setInterfaceColor()
  89. }
  90. // MARK: - RichWorkspace
  91. func setRichWorkspaceHeight(_ size: CGFloat) {
  92. viewRichWorkspaceHeightConstraint.constant = size
  93. if size == 0 {
  94. viewRichWorkspace.isHidden = true
  95. } else {
  96. viewRichWorkspace.isHidden = false
  97. }
  98. }
  99. func setInterfaceColor() {
  100. if traitCollection.userInterfaceStyle == .dark {
  101. gradient.colors = [UIColor(white: 0, alpha: 0).cgColor, UIColor.black.cgColor]
  102. } else {
  103. gradient.colors = [UIColor(white: 1, alpha: 0).cgColor, UIColor.white.cgColor]
  104. }
  105. }
  106. func setRichWorkspaceText(_ text: String?) {
  107. guard let text = text else { return }
  108. if text != self.richWorkspaceText {
  109. textViewRichWorkspace.attributedText = markdownParser.parse(text)
  110. self.richWorkspaceText = text
  111. }
  112. }
  113. // MARK: - Transfer
  114. func setViewTransfer(isHidden: Bool, ocId: String? = nil, text: String? = nil, progress: Float? = nil) {
  115. labelTransfer.text = text
  116. viewTransfer.isHidden = isHidden
  117. progressTransfer.progress = 0
  118. if isHidden {
  119. viewTransferHeightConstraint.constant = 0
  120. } else {
  121. var image: UIImage?
  122. if let ocId,
  123. let metadata = NCManageDatabase.shared.getMetadataFromOcId(ocId) {
  124. image = utility.getIcon(metadata: metadata)?.darken()
  125. if image == nil {
  126. image = utility.loadImage(named: metadata.iconName, useTypeIconFile: true)
  127. buttonTransfer.backgroundColor = .lightGray
  128. } else {
  129. buttonTransfer.backgroundColor = .clear
  130. }
  131. }
  132. viewTransferHeightConstraint.constant = NCGlobal.shared.heightHeaderTransfer
  133. if let progress {
  134. progressTransfer.progress = progress
  135. }
  136. }
  137. }
  138. // MARK: - Section
  139. func setSectionHeight(_ size: CGFloat) {
  140. viewSectionHeightConstraint.constant = size
  141. if size == 0 {
  142. viewSection.isHidden = true
  143. } else {
  144. viewSection.isHidden = false
  145. }
  146. }
  147. // MARK: - Action
  148. @IBAction func touchUpTransfer(_ sender: Any) {
  149. delegate?.tapButtonTransfer(sender)
  150. }
  151. @objc func touchUpInsideViewRichWorkspace(_ sender: Any) {
  152. delegate?.tapRichWorkspace(sender)
  153. }
  154. }
  155. // https://stackoverflow.com/questions/16278463/darken-an-uiimage
  156. public extension UIImage {
  157. private enum BlendMode {
  158. case multiply // This results in colors that are at least as dark as either of the two contributing sample colors
  159. case screen // This results in colors that are at least as light as either of the two contributing sample colors
  160. }
  161. // A level of zero yeilds the original image, a level of 1 results in black
  162. func darken(level: CGFloat = 0.5) -> UIImage? {
  163. return blend(mode: .multiply, level: level)
  164. }
  165. // A level of zero yeilds the original image, a level of 1 results in white
  166. func lighten(level: CGFloat = 0.5) -> UIImage? {
  167. return blend(mode: .screen, level: level)
  168. }
  169. private func blend(mode: BlendMode, level: CGFloat) -> UIImage? {
  170. let context = CIContext(options: nil)
  171. var level = level
  172. if level < 0 {
  173. level = 0
  174. } else if level > 1 {
  175. level = 1
  176. }
  177. let filterName: String
  178. switch mode {
  179. case .multiply: // As the level increases we get less white
  180. level = abs(level - 1.0)
  181. filterName = "CIMultiplyBlendMode"
  182. case .screen: // As the level increases we get more white
  183. filterName = "CIScreenBlendMode"
  184. }
  185. let blender = CIFilter(name: filterName)!
  186. let backgroundColor = CIColor(color: UIColor(white: level, alpha: 1))
  187. guard let inputImage = CIImage(image: self) else { return nil }
  188. blender.setValue(inputImage, forKey: kCIInputImageKey)
  189. guard let backgroundImageGenerator = CIFilter(name: "CIConstantColorGenerator") else { return nil }
  190. backgroundImageGenerator.setValue(backgroundColor, forKey: kCIInputColorKey)
  191. guard let backgroundImage = backgroundImageGenerator.outputImage?.cropped(to: CGRect(origin: CGPoint.zero, size: self.size)) else { return nil }
  192. blender.setValue(backgroundImage, forKey: kCIInputBackgroundImageKey)
  193. guard let blendedImage = blender.outputImage else { return nil }
  194. guard let cgImage = context.createCGImage(blendedImage, from: blendedImage.extent) else { return nil }
  195. return UIImage(cgImage: cgImage)
  196. }
  197. }