123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266 |
- //
- // BFVideoThumbProgressView.swift
- // BFRecordScreenKit
- //
- // Created by 胡志强 on 2021/12/3.
- //
- import AVFoundation
- import BFCommonKit
- import BFUIKit
- import Foundation
- import SnapKit
- import UIKit
- class BFVideoThumbProgressView: UIView {
- var recordItem: BFRecordItemModel? {
- didSet {
- // 指针回归
- progress = 0
- if recordItem?.mediaType == .VIDEO {
- dealWithVideoThumb()
- } else if recordItem?.mediaType == .IMAGE {
- dealWithImageThumb()
- }
- }
- }
- var dragScrollProgressHandle: ((Bool, Float) -> Void)?
- var dragEndHandle: ((Float) -> Void)?
- var dragStartHandle: (() -> Void)?
- var isDrag = false
- let thumbImageWidth = 70.0
- let fetchThumbStrategy: BFVideoThumbProgressStrategyProtocol = BFVideoThumbProgressStrategy()
- var progress: Double = 0 {
- didSet {
- updateProgress(progress: progress)
- }
- }
- var thumbImgs = [UIImage]()
- lazy var progressView: BFAutolayoutScrollView = {
- let sv = BFAutolayoutScrollView()
- sv.bounces = false
- // sv.backgroundColor = .clear
- // sv.backgroundColor = UIColor.hexColor(hexadecimal: "#888888",alpha:0.3)
- sv.backgroundColor = UIColor(red: 0, green: 0, blue: 0, alpha: 0.3)
- sv.decelerationRate = .fast
- sv.showsHorizontalScrollIndicator = false
- sv.delegate = self
- return sv
- }()
- lazy var progessIndicateBackV: UIView = {
- let vv = UIView()
- return vv
- }()
- // MARK: - 生命周期
- override init(frame: CGRect) {
- super.init(frame: frame)
- addSubview(progressView)
- let line = UIView()
- line.backgroundColor = .white
- line.layer.borderColor = UIColor.black.cgColor
- line.layer.borderWidth = 0.3
- line.layer.cornerRadius = 1.5
- addSubview(line)
- line.snp.makeConstraints { make in
- make.width.equalTo(3)
- make.center.height.equalToSuperview()
- }
- }
- required init?(coder _: NSCoder) {
- fatalError("init(coder:) has not been implemented")
- }
- override func layoutSubviews() {
- super.layoutSubviews()
- progressView.snp.makeConstraints { make in
- make.edges.equalToSuperview()
- }
- }
- /// 处理视频缩略图
- func dealWithVideoThumb() {
- guard let videoAsset = recordItem?.videoAsset else {
- return
- }
- self.addThumbImages(images: recordItem!.thumbImgs)
- if recordItem!.thumbImgs.count > 0{
- return
- }
-
- let date = Date()
- let dur = videoAsset.duration.seconds
- if dur > 0 {
- let fps = Double(fetchThumbStrategy.frameNumberOfVideo(assetDuration: dur)) / dur
- splitVideoFileUrlFps(urlAsset: videoAsset, fps: fps, firstImagesCount: 4) { [weak self] hadGetAll, images in
- BFLog(1, message: "获取缩略图:\(hadGetAll), \(Date().timeIntervalSince(date))")
- self?.recordItem!.thumbImgs.removeAll()
- self?.recordItem!.thumbImgs.append(contentsOf: images)
- self?.addThumbImages(images: images)
- }
- }
- }
- /// 处理图片缩略图
- func dealWithImageThumb() {
- guard let image = recordItem?.coverImg else {
- return
- }
- addThumbImages(images: Array(repeating: image, count: 10))
- }
- /// 添加缩略图
- /// - Parameter images: <#images description#>
- func addThumbImages(images: [UIImage]) {
- DispatchQueue.main.async { [weak self] in
- self?.progressView.contentView.subviews.forEach { subview in
- subview.removeFromSuperview()
- }
- if images.count > 0 {
- self?.thumbImgs = images
- if let sself = self {
- var lastiv = UIImageView()
- for (i, img) in images.enumerated() {
- let iv = UIImageView(image: img)
- iv.contentMode = .scaleAspectFill
- iv.clipsToBounds = true
- sself.progressView.contentView.addSubview(iv)
- iv.snp.makeConstraints { make in
- make.left.equalTo(CGFloat(i) * CGFloat(sself.thumbImageWidth) + sself.width * 0.5)
- make.top.bottom.equalToSuperview()
- make.height.equalTo(50)
- make.width.equalTo(sself.thumbImageWidth)
- }
- lastiv = iv
- }
- lastiv.snp.makeConstraints { make in
- make.right.equalTo(sself.width * -0.5)
- }
- sself.progressView.contentView.addSubview(sself.progessIndicateBackV)
- sself.progessIndicateBackV.snp.makeConstraints { make in
- make.left.equalTo(sself.width * 0.5)
- make.right.equalTo(sself.width * -0.5)
- make.bottom.equalToSuperview()
- make.height.equalTo(6)
- }
- }
- }
- }
- }
- func appendThumb(progress: Double = 0) {
- let count: Int = Int(progress / 2)
- BFLog(message: "需要的图片个数:progress=\(progress),count=\(count)")
- if recordItem?.mediaType == .IMAGE, (thumbImgs.count - 5) < count {
- guard let image = recordItem?.coverImg else {
- return
- }
- var lastiv = UIImageView()
- let lastIndex = thumbImgs.count - 1
- for i in lastIndex ... lastIndex + 10 {
- let iv = UIImageView(image: image)
- iv.contentMode = .scaleAspectFill
- iv.clipsToBounds = true
- progressView.contentView.addSubview(iv)
- iv.snp.makeConstraints { make in
- make.left.equalTo(CGFloat(i) * CGFloat(thumbImageWidth) + width * 0.5)
- make.top.bottom.equalToSuperview()
- make.height.equalTo(50)
- make.width.equalTo(thumbImageWidth)
- }
- lastiv = iv
- thumbImgs.append(image)
- }
- lastiv.snp.makeConstraints { make in
- make.right.equalTo(width * -0.5)
- }
- }
- }
- /// 更新进度
- /// - Parameter progress: <#progress description#>
- func updateProgress(progress: Double = 0) {
- if progressView.contentSize.width <= 0 {
- return
- }
- if recordItem?.mediaType == .VIDEO {
- if let second = recordItem?.videoAsset?.duration.seconds, second > 0 {
- let w = progressView.contentSize.width - width
- BFLog(message: "录音进度--指示器:progress=\(progress),duration=\(second),w=\(w),perW=\(Double(w) / second),totalW:\(progress * Double(w) / second)")
- progressView.contentOffset = CGPoint(x: progress * Double(w) / second, y: 0)
- }
- } else if recordItem?.mediaType == .IMAGE {
- BFLog(message: "updateProgress:\(progress)")
- BFLog(message: "录音进度--指示器:progress=\(progress),duration=\(recordItem?.materialDuraion ?? 0),w=\(progressView.contentSize.width - width),perW=\(thumbImageWidth),totalW:\(progress * thumbImageWidth / 2.0)")
- progressView.contentOffset = CGPoint(x: progress * thumbImageWidth / 2.0, y: 0)
- appendThumb(progress: progress)
- }
- }
- }
- extension BFVideoThumbProgressView: UIScrollViewDelegate {
- func scrollViewDidScroll(_ scrollView: UIScrollView) {
- if recordItem?.mediaType == .VIDEO {
- if isDrag {
- let dur = scrollView.contentOffset.x / (scrollView.contentSize.width - width)
- dragScrollProgressHandle?(false, Float(dur))
- }
- } else if recordItem?.mediaType == .IMAGE {
- if isDrag {
- if scrollView.contentOffset.x > CGFloat(recordItem?.materialDuraion ?? 0) * thumbImageWidth / 2.0 {
- scrollView.contentOffset = CGPoint(x: CGFloat(recordItem?.materialDuraion ?? 0) * thumbImageWidth / 2.0, y: 0)
- }
- let dur = scrollView.contentOffset.x / (CGFloat(recordItem?.materialDuraion ?? 0) * thumbImageWidth / 2.0)
- dragScrollProgressHandle?(false, Float(dur))
- }
- }
- }
- func scrollViewWillBeginDragging(_ scrollView: UIScrollView) {
- isDrag = true
- var dur: CGFloat = 0
- if recordItem?.mediaType == .VIDEO {
- dur = scrollView.contentOffset.x / (scrollView.contentSize.width - width)
- } else {
- dur = scrollView.contentOffset.x / (CGFloat(recordItem?.materialDuraion ?? 0) * thumbImageWidth / 2.0)
- }
- dragStartHandle?()
- dragScrollProgressHandle?(true, Float(dur))
- }
- func scrollViewDidEndDragging(_ scrollView: UIScrollView, willDecelerate decelerate: Bool) {
- if !decelerate {
- var dur: CGFloat = 0
- if recordItem?.mediaType == .VIDEO {
- dur = scrollView.contentOffset.x / (scrollView.contentSize.width - width)
- } else {
- dur = scrollView.contentOffset.x / (CGFloat(recordItem?.materialDuraion ?? 0) * thumbImageWidth / 2.0)
- }
- isDrag = false
- dragEndHandle?(Float(dur))
- }
- }
- func scrollViewDidEndDecelerating(_ scrollView: UIScrollView) {
- var dur: CGFloat = 0
- if recordItem?.mediaType == .VIDEO {
- dur = scrollView.contentOffset.x / (scrollView.contentSize.width - width)
- } else {
- dur = scrollView.contentOffset.x / (CGFloat(recordItem?.materialDuraion ?? 0) * thumbImageWidth / 2.0)
- }
- isDrag = false
- dragEndHandle?(Float(dur))
- }
- }
|