|  |  |  | // Copyright © 2022 Rangeproof Pty Ltd. All rights reserved. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import UIKit | 
					
						
							|  |  |  | import AVFoundation | 
					
						
							|  |  |  | import SessionUIKit | 
					
						
							|  |  |  | import SignalCoreKit | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @objc | 
					
						
							|  |  |  | public class VideoPlayerView: UIView { | 
					
						
							|  |  |  |     @objc | 
					
						
							|  |  |  |     public var player: AVPlayer? { | 
					
						
							|  |  |  |         get { | 
					
						
							|  |  |  |             return playerLayer.player | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         set { | 
					
						
							|  |  |  |             playerLayer.player = newValue | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     var playerLayer: AVPlayerLayer { | 
					
						
							|  |  |  |         return layer as! AVPlayerLayer | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Override UIView property | 
					
						
							|  |  |  |     override public static var layerClass: AnyClass { | 
					
						
							|  |  |  |         return AVPlayerLayer.self | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @objc | 
					
						
							|  |  |  | public protocol PlayerProgressBarDelegate { | 
					
						
							|  |  |  |     func playerProgressBarDidStartScrubbing(_ playerProgressBar: PlayerProgressBar) | 
					
						
							|  |  |  |     func playerProgressBar(_ playerProgressBar: PlayerProgressBar, scrubbedToTime time: CMTime) | 
					
						
							|  |  |  |     func playerProgressBar(_ playerProgressBar: PlayerProgressBar, didFinishScrubbingAtTime time: CMTime, shouldResumePlayback: Bool) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Allows the user to tap anywhere on the slider to set it's position, | 
					
						
							|  |  |  | // without first having to grab the thumb. | 
					
						
							|  |  |  | class TrackingSlider: UISlider { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     override init(frame: CGRect) { | 
					
						
							|  |  |  |         super.init(frame: frame) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     override func beginTracking(_ touch: UITouch, with event: UIEvent?) -> Bool { | 
					
						
							|  |  |  |         return true | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     required init?(coder aDecoder: NSCoder) { | 
					
						
							|  |  |  |         notImplemented() | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @objc | 
					
						
							|  |  |  | public class PlayerProgressBar: UIView { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @objc | 
					
						
							|  |  |  |     public weak var delegate: PlayerProgressBarDelegate? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private lazy var formatter: DateComponentsFormatter = { | 
					
						
							|  |  |  |         let formatter = DateComponentsFormatter() | 
					
						
							|  |  |  |         formatter.unitsStyle = .positional | 
					
						
							|  |  |  |         formatter.allowedUnits = [.minute, .second ] | 
					
						
							|  |  |  |         formatter.zeroFormattingBehavior = [ .pad ] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return formatter | 
					
						
							|  |  |  |     }() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // MARK: Subviews | 
					
						
							|  |  |  |     private let positionLabel = UILabel() | 
					
						
							|  |  |  |     private let remainingLabel = UILabel() | 
					
						
							|  |  |  |     private let slider = TrackingSlider() | 
					
						
							|  |  |  |     private let blurView = UIVisualEffectView() | 
					
						
							|  |  |  |     weak private var progressObserver: AnyObject? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private let kPreferredTimeScale: CMTimeScale = 100 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @objc | 
					
						
							|  |  |  |     public var player: AVPlayer? { | 
					
						
							|  |  |  |         didSet { | 
					
						
							|  |  |  |             guard let item = player?.currentItem else { | 
					
						
							|  |  |  |                 owsFailDebug("No player item") | 
					
						
							|  |  |  |                 return | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             slider.minimumValue = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             let duration: CMTime = item.asset.duration | 
					
						
							|  |  |  |             slider.maximumValue = Float(CMTimeGetSeconds(duration)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             updateState() | 
					
						
							|  |  |  |              | 
					
						
							|  |  |  |             // OPTIMIZE We need a high frequency observer for smooth slider updates while playing, | 
					
						
							|  |  |  |             // but could use a much less frequent observer for label updates | 
					
						
							|  |  |  |             progressObserver = player?.addPeriodicTimeObserver(forInterval: CMTime(seconds: 0.1, preferredTimescale: kPreferredTimeScale), queue: nil, using: { [weak self] _ in | 
					
						
							|  |  |  |                 // If it is playing update the time | 
					
						
							|  |  |  |                 if self?.player?.rate != 0 && self?.player?.error == nil { | 
					
						
							|  |  |  |                     self?.updateState() | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             }) as AnyObject | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     required public init?(coder aDecoder: NSCoder) { | 
					
						
							|  |  |  |         notImplemented() | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     override public init(frame: CGRect) { | 
					
						
							|  |  |  |         super.init(frame: frame) | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         // Background & blur | 
					
						
							|  |  |  |         let backgroundView = UIView() | 
					
						
							|  |  |  |         backgroundView.themeBackgroundColor = .backgroundSecondary | 
					
						
							|  |  |  |         backgroundView.alpha = Values.lowOpacity | 
					
						
							|  |  |  |         addSubview(backgroundView) | 
					
						
							|  |  |  |         backgroundView.pin(to: self) | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         if !UIAccessibility.isReduceTransparencyEnabled { | 
					
						
							|  |  |  |             addSubview(blurView) | 
					
						
							|  |  |  |             blurView.pin(to: self) | 
					
						
							|  |  |  |              | 
					
						
							|  |  |  |             ThemeManager.onThemeChange(observer: blurView) { [weak blurView] theme, _ in | 
					
						
							|  |  |  |                 switch theme.interfaceStyle { | 
					
						
							|  |  |  |                     case .light: blurView?.effect = UIBlurEffect(style: .light) | 
					
						
							|  |  |  |                     default: blurView?.effect = UIBlurEffect(style: .dark) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Configure controls | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let kLabelFont = UIFont.monospacedDigitSystemFont(ofSize: 12, weight: UIFont.Weight.regular) | 
					
						
							|  |  |  |         positionLabel.font = kLabelFont | 
					
						
							|  |  |  |         remainingLabel.font = kLabelFont | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // We use a smaller thumb for the progress slider. | 
					
						
							|  |  |  |         slider.setThumbImage(#imageLiteral(resourceName: "sliderProgressThumb"), for: .normal) | 
					
						
							|  |  |  |         slider.themeMinimumTrackTintColor = .backgroundPrimary | 
					
						
							|  |  |  |         slider.themeMaximumTrackTintColor = .backgroundPrimary | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         slider.addTarget(self, action: #selector(handleSliderTouchDown), for: .touchDown) | 
					
						
							|  |  |  |         slider.addTarget(self, action: #selector(handleSliderTouchUp), for: .touchUpInside) | 
					
						
							|  |  |  |         slider.addTarget(self, action: #selector(handleSliderTouchUp), for: .touchUpOutside) | 
					
						
							|  |  |  |         slider.addTarget(self, action: #selector(handleSliderValueChanged), for: .valueChanged) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Panning is a no-op. We just absorb pan gesture's originating in the video controls | 
					
						
							|  |  |  |         // from propogating so we don't inadvertently change pages while trying to scrub in | 
					
						
							|  |  |  |         // the MediaPageView. | 
					
						
							|  |  |  |         let panAbsorber = UIPanGestureRecognizer(target: self, action: nil) | 
					
						
							|  |  |  |         self.addGestureRecognizer(panAbsorber) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Layout Subviews | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         addSubview(positionLabel) | 
					
						
							|  |  |  |         addSubview(remainingLabel) | 
					
						
							|  |  |  |         addSubview(slider) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         positionLabel.autoPinEdge(toSuperviewMargin: .leading) | 
					
						
							|  |  |  |         positionLabel.autoVCenterInSuperview() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let kSliderMargin: CGFloat = 8 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         slider.autoPinEdge(.leading, to: .trailing, of: positionLabel, withOffset: kSliderMargin) | 
					
						
							|  |  |  |         slider.autoVCenterInSuperview() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         remainingLabel.autoPinEdge(.leading, to: .trailing, of: slider, withOffset: kSliderMargin) | 
					
						
							|  |  |  |         remainingLabel.autoPinEdge(toSuperviewMargin: .trailing) | 
					
						
							|  |  |  |         remainingLabel.autoVCenterInSuperview() | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // MARK: Gesture handling | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     var wasPlayingWhenScrubbingStarted: Bool = false | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @objc | 
					
						
							|  |  |  |     private func handleSliderTouchDown(_ slider: UISlider) { | 
					
						
							|  |  |  |         guard let player = self.player else { | 
					
						
							|  |  |  |             owsFailDebug("player was nil") | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.wasPlayingWhenScrubbingStarted = (player.rate != 0) && (player.error == nil) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.delegate?.playerProgressBarDidStartScrubbing(self) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @objc | 
					
						
							|  |  |  |     private func handleSliderTouchUp(_ slider: UISlider) { | 
					
						
							|  |  |  |         let sliderTime = time(slider: slider) | 
					
						
							|  |  |  |         self.delegate?.playerProgressBar(self, didFinishScrubbingAtTime: sliderTime, shouldResumePlayback: wasPlayingWhenScrubbingStarted) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @objc | 
					
						
							|  |  |  |     private func handleSliderValueChanged(_ slider: UISlider) { | 
					
						
							|  |  |  |         let sliderTime = time(slider: slider) | 
					
						
							|  |  |  |         self.delegate?.playerProgressBar(self, scrubbedToTime: sliderTime) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // MARK: Render cycle | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public func updateState() { | 
					
						
							|  |  |  |         guard let player = player else { | 
					
						
							|  |  |  |             owsFailDebug("player isn't set.") | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         guard let item = player.currentItem else { | 
					
						
							|  |  |  |             owsFailDebug("player has no item.") | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let position = player.currentTime() | 
					
						
							|  |  |  |         let positionSeconds: Float64 = CMTimeGetSeconds(position) | 
					
						
							|  |  |  |         positionLabel.text = formatter.string(from: positionSeconds) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let duration: CMTime = item.asset.duration | 
					
						
							|  |  |  |         let remainingTime = duration - position | 
					
						
							|  |  |  |         let remainingSeconds = CMTimeGetSeconds(remainingTime) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         guard let remainingString = formatter.string(from: remainingSeconds) else { | 
					
						
							|  |  |  |             owsFailDebug("unable to format time remaining") | 
					
						
							|  |  |  |             remainingLabel.text = "0:00" | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // show remaining time as negative | 
					
						
							|  |  |  |         remainingLabel.text = "-\(remainingString)" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         slider.setValue(Float(positionSeconds), animated: false) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // MARK: Util | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private func time(slider: UISlider) -> CMTime { | 
					
						
							|  |  |  |         let seconds: Double = Double(slider.value) | 
					
						
							|  |  |  |         return CMTime(seconds: seconds, preferredTimescale: kPreferredTimeScale) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     // MARK: - Functions | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     public func manuallySetValue(_ positionSeconds: CGFloat, durationSeconds: CGFloat) { | 
					
						
							|  |  |  |         let remainingSeconds = (durationSeconds - positionSeconds) | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         slider.minimumValue = 0 | 
					
						
							|  |  |  |         slider.maximumValue = Float(durationSeconds) | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         positionLabel.text = formatter.string(from: positionSeconds) | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         guard let remainingString = formatter.string(from: remainingSeconds) else { | 
					
						
							|  |  |  |             owsFailDebug("unable to format time remaining") | 
					
						
							|  |  |  |             remainingLabel.text = "0:00" | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         // show remaining time as negative | 
					
						
							|  |  |  |         remainingLabel.text = "-\(remainingString)" | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         slider.setValue(Float(positionSeconds), animated: false) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |