LinearProgressBar.swift 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. //
  2. // LinearProgressBar.swift
  3. // Learn Genie
  4. //
  5. // Created by Suraj Kumar Mandal on 02/09/21.
  6. //
  7. import UIKit
  8. @IBDesignable
  9. public class DSGradientProgressView: UIView, CAAnimationDelegate {
  10. @IBInspectable public var barColor: UIColor = UIColor(hue: (29.0/360.0), saturation: 1.0, brightness: 1.0, alpha: 1.0) {
  11. didSet {
  12. initialize()
  13. }
  14. }
  15. private let serialIncrementQueue = DispatchQueue(label: "com.dholstudio.DSGradientProgressView.serialIncrementQueue")
  16. private var numberOfOperations: Int = 0
  17. override public class var layerClass: AnyClass {
  18. get {
  19. return CAGradientLayer.self
  20. }
  21. }
  22. // https://theswiftdev.com/2015/08/05/swift-init-patterns/
  23. override public init(frame: CGRect) {
  24. super.init(frame: frame)
  25. self.initialize()
  26. }
  27. required public init(coder aDecoder: NSCoder) {
  28. super.init(coder: aDecoder)!
  29. self.initialize()
  30. }
  31. override public func awakeFromNib() {
  32. super.awakeFromNib()
  33. self.initialize()
  34. }
  35. private func initialize() {
  36. let layer = self.layer as! CAGradientLayer
  37. // Use a horizontal gradient
  38. layer.startPoint = CGPoint(x: 0.0, y: 0.5)
  39. layer.endPoint = CGPoint(x: 1.0, y: 0.5)
  40. var colors: [CGColor] = []
  41. // http://stackoverflow.com/a/39126332/2607823
  42. // http://bjmiller.me/post/137624096422/on-c-style-for-loops-removed-from-swift-3
  43. // stride syntax changed from the blog's syntax
  44. /*
  45. // ==== Blue gradient with changing hues ====
  46. for hue in stride(from: 190, through: 230, by: 2) {
  47. let color = UIColor(hue: CGFloat(1.0 * Double(hue) / 360.0),
  48. saturation: 1.0,
  49. brightness: 1.0,
  50. alpha: 1.0)
  51. colors.append(color.cgColor)
  52. }
  53. // can be done with stride with -2 too. but just to explore another way..
  54. for hue in (190...230).reversed() where hue % 2 == 0 {
  55. let color = UIColor(hue: CGFloat(1.0 * Double(hue) / 360.0),
  56. saturation: 1.0,
  57. brightness: 1.0,
  58. alpha: 1.0)
  59. colors.append(color.cgColor)
  60. }
  61. */
  62. // ==== Constant hue with changing alpha ====
  63. /*
  64. for alpha in stride(from: 0, through: 40, by: 2) {
  65. let color = UIColor(hue: CGFloat(1.0 * Double(hue) / 360.0),
  66. saturation: 1.0,
  67. brightness: 1.0,
  68. alpha: CGFloat(1.0 * Double(alpha)/100.0))
  69. barColor.withAlphaComponent(CGFloat(alpha))
  70. colors.append(color.cgColor)
  71. }
  72. */
  73. // === constant color from storyboard or default with changing alpha ===
  74. for alpha in stride(from: 0, through: 40, by: 2) {
  75. let color = barColor.withAlphaComponent(CGFloat(Double(alpha)/100.0))
  76. colors.append(color.cgColor)
  77. }
  78. for alpha in stride(from: 40, through: 90, by: 10) {
  79. let color = barColor.withAlphaComponent(CGFloat(Double(alpha)/100.0))
  80. colors.append(color.cgColor)
  81. }
  82. for alpha in stride(from: 90, through: 100, by: 10) {
  83. let color = barColor.withAlphaComponent(CGFloat(Double(alpha)/100.0))
  84. colors.append(color.cgColor)
  85. colors.append(color.cgColor) // adding twice
  86. }
  87. for alpha in stride(from: 100, through: 0, by: -20) {
  88. let color = barColor.withAlphaComponent(CGFloat(Double(alpha)/100.0))
  89. colors.append(color.cgColor)
  90. }
  91. layer.colors = colors
  92. }
  93. private func performAnimation() {
  94. // Move the last color in the array to the front
  95. // shifting all the other colors.
  96. let layer = self.layer as! CAGradientLayer
  97. guard let color = layer.colors?.popLast() else {
  98. print("FATAL ERR: GradientProgressView : Layer should contain colors!")
  99. return
  100. }
  101. layer.colors?.insert(color, at: 0)
  102. let shiftedColors = layer.colors!
  103. let animation = CABasicAnimation(keyPath: "colors")
  104. animation.toValue = shiftedColors
  105. animation.duration = 0.03
  106. animation.isRemovedOnCompletion = true
  107. animation.fillMode = CAMediaTimingFillMode.forwards
  108. animation.delegate = self
  109. layer.add(animation, forKey: "animateGradient")
  110. }
  111. public func animationDidStop(_ anim: CAAnimation, finished flag: Bool) {
  112. // TODO: Make reads on serial queue too?
  113. if flag && numberOfOperations > 0 {
  114. performAnimation()
  115. }
  116. else {
  117. self.isHidden = true
  118. }
  119. }
  120. public func startProgress() {
  121. serialIncrementQueue.sync {
  122. numberOfOperations += 1
  123. }
  124. self.isHidden = false
  125. if numberOfOperations == 1 { // rest will be called from animationDidStop
  126. performAnimation()
  127. }
  128. }
  129. public func stopProgress() {
  130. if numberOfOperations == 0 {
  131. return
  132. }
  133. serialIncrementQueue.sync {
  134. numberOfOperations -= 1
  135. }
  136. }
  137. /*
  138. // Only override draw() if you perform custom drawing.
  139. // An empty implementation adversely affects performance during animation.
  140. override func draw(_ rect: CGRect) {
  141. // Drawing code
  142. }
  143. */
  144. }