TM-SGNL-iOS/SignalUI/AV/VideoPlayer.swift
TeleMessage developers dde0620daf initial commit
2025-05-03 12:28:28 -07:00

210 lines
6.4 KiB
Swift

//
// Copyright 2018 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
//
public import AVFoundation
public import SignalServiceKit
public protocol VideoPlayerDelegate: AnyObject {
func videoPlayerDidPlayToCompletion(_ videoPlayer: VideoPlayer)
}
public class VideoPlayer {
public let avPlayer: AVPlayer
private let audioActivity: AudioActivity
private let shouldLoop: Bool
public var isMuted = false {
didSet {
avPlayer.volume = isMuted ? 0 : 1
}
}
weak public var delegate: VideoPlayerDelegate?
public convenience init(decryptedFileUrl: URL) {
self.init(decryptedFileUrl: decryptedFileUrl, shouldLoop: false)
}
public convenience init(decryptedFileUrl: URL, shouldLoop: Bool, shouldMixAudioWithOthers: Bool = false) {
let avPlayer = AVPlayer(url: decryptedFileUrl)
self.init(
avPlayer: avPlayer,
shouldLoop: shouldLoop,
shouldMixAudioWithOthers: shouldMixAudioWithOthers,
audioDescription: "[VideoPlayer] url:\(decryptedFileUrl)"
)
}
public convenience init(
attachment: ReferencedAttachmentStream,
shouldMixAudioWithOthers: Bool = false
) throws {
try self.init(
attachment: attachment.attachmentStream,
shouldLoop: attachment.reference.renderingFlag == .shouldLoop,
shouldMixAudioWithOthers: shouldMixAudioWithOthers,
audioDescription: attachment.reference.sourceFilename.map { "[VideoPlayer] \($0)" }
)
}
public convenience init(
attachment: AttachmentStream,
shouldLoop: Bool,
shouldMixAudioWithOthers: Bool = false
) throws {
try self.init(
attachment: attachment,
shouldLoop: shouldLoop,
shouldMixAudioWithOthers: shouldMixAudioWithOthers,
audioDescription: nil
)
}
private convenience init(
attachment: AttachmentStream,
shouldLoop: Bool,
shouldMixAudioWithOthers: Bool,
audioDescription: String?
) throws {
let asset = try attachment.decryptedAVAsset()
let playerItem = AVPlayerItem(asset: asset)
let avPlayer = AVPlayer(playerItem: playerItem)
self.init(
avPlayer: avPlayer,
shouldLoop: shouldLoop,
shouldMixAudioWithOthers: shouldMixAudioWithOthers,
audioDescription: "[VideoPlayer]"
)
}
public init(
avPlayer: AVPlayer,
shouldLoop: Bool,
shouldMixAudioWithOthers: Bool = false,
audioDescription: String = "[VideoPlayer]"
) {
self.avPlayer = avPlayer
audioActivity = AudioActivity(
audioDescription: audioDescription,
behavior: shouldMixAudioWithOthers ? .playbackMixWithOthers : .playback
)
self.shouldLoop = shouldLoop
NotificationCenter.default.addObserver(
self,
selector: #selector(playerItemDidPlayToCompletion(_:)),
name: .AVPlayerItemDidPlayToEndTime,
object: avPlayer.currentItem
)
}
deinit {
endAudioActivity()
}
// MARK: Playback Controls
public func endAudioActivity() {
SUIEnvironment.shared.audioSessionRef.endAudioActivity(audioActivity)
}
public func pause() {
avPlayer.pause()
endAudioActivity()
}
public func play() {
let success = SUIEnvironment.shared.audioSessionRef.startAudioActivity(audioActivity)
assert(success)
guard let item = avPlayer.currentItem else {
owsFailDebug("video player item was unexpectedly nil")
return
}
if item.currentTime() == item.duration {
// Rewind for repeated plays, but only if it previously played to end.
avPlayer.seek(to: CMTime.zero)
}
avPlayer.play()
}
public func stop() {
avPlayer.pause()
avPlayer.seek(to: CMTime.zero)
endAudioActivity()
}
public func seek(to time: CMTime) {
owsAssertBeta(avPlayer.rate == 0 || avPlayer.rate == 1)
// Seek with a tolerance (or precision) of a hundredth of a second.
let tolerance = CMTime(seconds: 0.01, preferredTimescale: Self.preferredTimescale)
// Bound the time
var boundedTime = max(time, CMTime(seconds: 0, preferredTimescale: Self.preferredTimescale))
boundedTime = min(boundedTime, avPlayer.currentItem?.asset.duration ?? CMTime(seconds: 0, preferredTimescale: Self.preferredTimescale))
avPlayer.seek(to: boundedTime, toleranceBefore: tolerance, toleranceAfter: tolerance)
}
public func rewind(_ seconds: TimeInterval) {
let newTime = avPlayer.currentTime() - CMTime(seconds: seconds, preferredTimescale: Self.preferredTimescale)
seek(to: newTime)
}
public func fastForward(_ seconds: TimeInterval) {
let newTime = avPlayer.currentTime() + CMTime(seconds: seconds, preferredTimescale: Self.preferredTimescale)
seek(to: newTime)
}
private var playbackRateToRestore: Float?
// Specify negative rate to rewind.
public func changePlaybackRate(to rate: Float) {
owsAssertBeta(abs(rate) > 1)
playbackRateToRestore = avPlayer.rate
if rate < 0 {
avPlayer.isMuted = true
}
avPlayer.rate = rate
}
public func restorePlaybackRate() {
avPlayer.isMuted = false
if let playbackRateToRestore {
avPlayer.rate = playbackRateToRestore
self.playbackRateToRestore = nil
}
}
public var isPlaying: Bool {
if let playbackRateToRestore {
// For UI consistency's sake, if currently playing at non-default speed
// return `true` if rewind/fast forward started when player was playing.
return playbackRateToRestore == 1
}
return avPlayer.timeControlStatus == .playing
}
public var currentTimeSeconds: Double {
return avPlayer.currentTime().seconds
}
private static var preferredTimescale: CMTimeScale { 1000 }
// MARK: private
@objc
private func playerItemDidPlayToCompletion(_ notification: Notification) {
playbackRateToRestore = nil
delegate?.videoPlayerDidPlayToCompletion(self)
if shouldLoop {
avPlayer.seek(to: CMTime.zero)
avPlayer.play()
} else {
endAudioActivity()
}
}
}