BFVideoThumbProgressView.swift 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  1. //
  2. // BFVideoThumbProgressView.swift
  3. // BFRecordScreenKit
  4. //
  5. // Created by 胡志强 on 2021/12/3.
  6. //
  7. import AVFoundation
  8. import BFCommonKit
  9. import BFUIKit
  10. import Foundation
  11. import SnapKit
  12. import UIKit
  13. class BFVideoThumbProgressView: UIView {
  14. var recordItem: BFRecordItemModel? {
  15. didSet {
  16. // 指针回归
  17. BFLog(1, message: "new recorditem")
  18. progress = 0
  19. if recordItem?.mediaType == .VIDEO {
  20. dealWithVideoThumb()
  21. } else if recordItem?.mediaType == .IMAGE {
  22. dealWithImageThumb()
  23. }
  24. }
  25. }
  26. var dragScrollProgressHandle: ((Bool, Float) -> Void)?
  27. var dragEndHandle: ((Float) -> Void)?
  28. var dragStartHandle: (() -> Void)?
  29. var isDrag = false
  30. let thumbImageWidth : CGFloat = 70.0
  31. let fetchThumbStrategy: BFVideoThumbProgressStrategyProtocol = BFVideoThumbProgressStrategy()
  32. var progress: Double = 0 {
  33. didSet {
  34. updateProgress(progress: progress)
  35. }
  36. }
  37. var thumbImgs = [UIImage]()
  38. lazy var progressView: BFAutolayoutScrollView = {
  39. let sv = BFAutolayoutScrollView()
  40. sv.bounces = false
  41. // sv.backgroundColor = .clear
  42. // sv.backgroundColor = UIColor.hexColor(hexadecimal: "#888888",alpha:0.3)
  43. sv.backgroundColor = UIColor(red: 0, green: 0, blue: 0, alpha: 0.3)
  44. sv.decelerationRate = .fast
  45. sv.showsHorizontalScrollIndicator = false
  46. sv.delegate = self
  47. return sv
  48. }()
  49. lazy var progessIndicateBackV: UIView = {
  50. let vv = UIView()
  51. return vv
  52. }()
  53. // MARK: - 生命周期
  54. override init(frame: CGRect) {
  55. super.init(frame: frame)
  56. addSubview(progressView)
  57. let line = UIView()
  58. line.backgroundColor = .white
  59. line.layer.borderColor = UIColor.black.cgColor
  60. line.layer.borderWidth = 0.3
  61. line.layer.cornerRadius = 1.5
  62. addSubview(line)
  63. line.snp.makeConstraints { make in
  64. make.width.equalTo(3)
  65. make.center.height.equalToSuperview()
  66. }
  67. progressView.contentView.addSubview(progessIndicateBackV)
  68. progessIndicateBackV.snp.makeConstraints { make in
  69. make.left.equalTo(width * 0.5)
  70. make.right.equalTo(width * -0.5)
  71. make.bottom.equalToSuperview()
  72. make.height.equalTo(6)
  73. make.width.equalTo(1)
  74. }
  75. }
  76. required init?(coder _: NSCoder) {
  77. fatalError("init(coder:) has not been implemented")
  78. }
  79. override func layoutSubviews() {
  80. super.layoutSubviews()
  81. progressView.snp.makeConstraints { make in
  82. make.edges.equalToSuperview()
  83. }
  84. }
  85. /// 处理视频缩略图
  86. func dealWithVideoThumb() {
  87. guard let videoAsset = recordItem?.videoAsset else {
  88. return
  89. }
  90. self.addThumbImages(images: recordItem!.thumbImgs)
  91. if recordItem!.thumbImgs.count > 0{
  92. // 代表已经获取过了,不用在重新去获得
  93. return
  94. }
  95. let date = Date()
  96. let dur = videoAsset.duration.seconds
  97. if dur > 0 {
  98. let count = fetchThumbStrategy.frameNumberOfVideo(assetDuration: dur)
  99. progessIndicateBackV.snp.remakeConstraints { make in
  100. make.left.equalTo(width * 0.5)
  101. make.right.equalTo(width * -0.5)
  102. make.bottom.equalToSuperview()
  103. make.height.equalTo(6)
  104. make.width.equalTo(CGFloat(count) * thumbImageWidth)
  105. }
  106. let fps = Double(count) / dur
  107. recordItem!.splitVideoFileUrlFps(fps: fps, firstImagesCount: Int(ceil(width/2.0/thumbImageWidth))) { [weak self, weak recordItem] hadGetAll, images in
  108. guard let sself = self, let sitem = recordItem else {
  109. return
  110. }
  111. BFLog(1, message: "获取缩略图:\(hadGetAll), \(Date().timeIntervalSince(date)), \( sitem.localPath ?? "aa")")
  112. sitem.thumbImgs.removeAll()
  113. sitem.thumbImgs.append(contentsOf: images)
  114. // 不足数则补充足够帧数
  115. while (hadGetAll && sitem.thumbImgs.count < count && images.count > 0) {
  116. sitem.thumbImgs.append(images.last!)
  117. }
  118. if sitem.localPath == sself.recordItem!.localPath {
  119. sself.addThumbImages(images: sitem.thumbImgs)
  120. }else{
  121. BFLog(1, message: "thumbImgs.count:\(sitem.thumbImgs.count)")
  122. }
  123. }
  124. }
  125. }
  126. /// 处理图片缩略图
  127. func dealWithImageThumb() {
  128. guard let image = recordItem?.coverImg else {
  129. return
  130. }
  131. addThumbImages(images: Array(repeating: image, count: 10))
  132. }
  133. /// 添加缩略图
  134. /// - Parameter images: <#images description#>
  135. func addThumbImages(images: [UIImage]) {
  136. DispatchQueue.main.async { [weak self] in
  137. self?.progressView.contentView.subviews.forEach { subview in
  138. if subview is UIImageView {
  139. subview.removeFromSuperview()
  140. }
  141. }
  142. if images.count > 0 {
  143. // self?.thumbImgs = images
  144. if let sself = self {
  145. for (i, img) in images.enumerated() {
  146. let iv = UIImageView(image: img)
  147. iv.contentMode = .scaleAspectFill
  148. iv.clipsToBounds = true
  149. sself.progressView.contentView.insertSubview(iv, belowSubview: sself.progessIndicateBackV)
  150. iv.snp.makeConstraints { make in
  151. make.left.equalTo(CGFloat(i) * CGFloat(sself.thumbImageWidth) + sself.width * 0.5)
  152. make.top.bottom.equalToSuperview()
  153. make.height.equalTo(50)
  154. make.width.equalTo(sself.thumbImageWidth)
  155. }
  156. }
  157. }
  158. }
  159. }
  160. }
  161. func appendThumb(progress: Double = 0) {
  162. let count: Int = Int(progress / 2)
  163. BFLog(message: "需要的图片个数:progress=\(progress),count=\(count)")
  164. if recordItem?.mediaType == .IMAGE, (thumbImgs.count - 5) < count {
  165. guard let image = recordItem?.coverImg else {
  166. return
  167. }
  168. var lastiv = UIImageView()
  169. let lastIndex = thumbImgs.count - 1
  170. for i in lastIndex ... lastIndex + 10 {
  171. let iv = UIImageView(image: image)
  172. iv.contentMode = .scaleAspectFill
  173. iv.clipsToBounds = true
  174. progressView.contentView.addSubview(iv)
  175. iv.snp.makeConstraints { make in
  176. make.left.equalTo(CGFloat(i) * CGFloat(thumbImageWidth) + width * 0.5)
  177. make.top.bottom.equalToSuperview()
  178. make.height.equalTo(50)
  179. make.width.equalTo(thumbImageWidth)
  180. }
  181. lastiv = iv
  182. thumbImgs.append(image)
  183. }
  184. lastiv.snp.makeConstraints { make in
  185. make.right.equalTo(width * -0.5)
  186. }
  187. }
  188. }
  189. /// 更新进度
  190. /// - Parameter progress: <#progress description#>
  191. func updateProgress(progress: Double = 0) {
  192. if progressView.contentSize.width <= 0 {
  193. return
  194. }
  195. if recordItem?.mediaType == .VIDEO {
  196. if let second = recordItem?.videoAsset?.duration.seconds, second > 0 {
  197. let w = progressView.contentSize.width - width
  198. BFLog(message: "录音进度--指示器:progress=\(progress),duration=\(second),w=\(w),perW=\(Double(w) / second),totalW:\(progress * Double(w) / second)")
  199. progressView.contentOffset = CGPoint(x: progress * Double(w) / second, y: 0)
  200. }
  201. } else if recordItem?.mediaType == .IMAGE {
  202. // if (recordItem?.materialDuraion ?? 0) > progress {
  203. BFLog(message: "updateProgress:\(progress)")
  204. progressView.contentOffset = CGPoint(x: progress * thumbImageWidth / 2.0, y: 0)
  205. appendThumb(progress: progress)
  206. // }
  207. }
  208. }
  209. }
  210. extension BFVideoThumbProgressView: UIScrollViewDelegate {
  211. func scrollViewDidScroll(_ scrollView: UIScrollView) {
  212. if recordItem?.mediaType == .VIDEO {
  213. if isDrag {
  214. let dur = scrollView.contentOffset.x / (scrollView.contentSize.width - width)
  215. dragScrollProgressHandle?(false, Float(dur))
  216. }
  217. } else if recordItem?.mediaType == .IMAGE {
  218. if isDrag {
  219. if scrollView.contentOffset.x > CGFloat(recordItem?.materialDuraion ?? 0) * thumbImageWidth / 2.0 {
  220. scrollView.contentOffset = CGPoint(x: CGFloat(recordItem?.materialDuraion ?? 0) * thumbImageWidth / 2.0, y: 0)
  221. }
  222. let dur = scrollView.contentOffset.x / (CGFloat(recordItem?.materialDuraion ?? 0) * thumbImageWidth / 2.0)
  223. dragScrollProgressHandle?(false, Float(dur))
  224. }
  225. }
  226. }
  227. func scrollViewWillBeginDragging(_ scrollView: UIScrollView) {
  228. isDrag = true
  229. var dur: CGFloat = 0
  230. if recordItem?.mediaType == .VIDEO {
  231. dur = scrollView.contentOffset.x / (scrollView.contentSize.width - width)
  232. } else {
  233. dur = scrollView.contentOffset.x / (CGFloat(recordItem?.materialDuraion ?? 0) * thumbImageWidth / 2.0)
  234. }
  235. dragStartHandle?()
  236. dragScrollProgressHandle?(true, Float(dur))
  237. }
  238. func scrollViewDidEndDragging(_ scrollView: UIScrollView, willDecelerate decelerate: Bool) {
  239. if !decelerate {
  240. var dur: CGFloat = 0
  241. if recordItem?.mediaType == .VIDEO {
  242. dur = scrollView.contentOffset.x / (scrollView.contentSize.width - width)
  243. } else {
  244. dur = scrollView.contentOffset.x / (CGFloat(recordItem?.materialDuraion ?? 0) * thumbImageWidth / 2.0)
  245. }
  246. isDrag = false
  247. dragEndHandle?(Float(dur))
  248. }
  249. }
  250. func scrollViewDidEndDecelerating(_ scrollView: UIScrollView) {
  251. var dur: CGFloat = 0
  252. if recordItem?.mediaType == .VIDEO {
  253. dur = scrollView.contentOffset.x / (scrollView.contentSize.width - width)
  254. } else {
  255. dur = scrollView.contentOffset.x / (CGFloat(recordItem?.materialDuraion ?? 0) * thumbImageWidth / 2.0)
  256. }
  257. isDrag = false
  258. dragEndHandle?(Float(dur))
  259. }
  260. }