mirror of
https://github.com/yattee/yattee.git
synced 2025-11-16 15:08:43 +00:00
Added comprehensive nil checks for stream resolution values across PlayerBackend, QualityProfile, and PlayerQueue to prevent crashes when streams have missing resolution metadata. Also added backend nil checks in PlayerQueue.
432 lines
13 KiB
Swift
432 lines
13 KiB
Swift
import AVKit
|
|
import Defaults
|
|
import Foundation
|
|
import Siesta
|
|
import SwiftUI
|
|
|
|
extension PlayerModel {
|
|
var currentVideo: Video? {
|
|
currentItem?.video
|
|
}
|
|
|
|
var videoForDisplay: Video? {
|
|
videoBeingOpened ?? currentVideo
|
|
}
|
|
|
|
func play(_ videos: [Video], shuffling: Bool = false) {
|
|
navigation.presentingChannelSheet = false
|
|
|
|
playbackMode = shuffling ? .shuffle : .queue
|
|
|
|
videos.forEach { enqueueVideo($0, loadDetails: false) }
|
|
|
|
#if os(iOS)
|
|
onPresentPlayer.append { [weak self] in self?.advanceToNextItem() }
|
|
#else
|
|
advanceToNextItem()
|
|
#endif
|
|
|
|
show()
|
|
}
|
|
|
|
func playNext(_ video: Video) {
|
|
enqueueVideo(video, play: currentItem.isNil, prepending: true)
|
|
}
|
|
|
|
func playNow(_ video: Video, at time: CMTime? = nil) {
|
|
navigation.presentingChannelSheet = false
|
|
|
|
if playingInPictureInPicture, closePiPOnNavigation {
|
|
closePiP()
|
|
}
|
|
|
|
videoBeingOpened = video
|
|
|
|
prepareCurrentItemForHistory()
|
|
|
|
enqueueVideo(video, play: true, atTime: time, prepending: true) { _, item in
|
|
self.advanceToItem(item, at: time)
|
|
}
|
|
}
|
|
|
|
func playItem(_ item: PlayerQueueItem, at time: CMTime? = nil) {
|
|
advancing = false
|
|
|
|
if !playingInPictureInPicture, !currentItem.isNil, backend != nil {
|
|
backend.closeItem()
|
|
}
|
|
|
|
comments.reset()
|
|
stream = nil
|
|
navigation.presentingChannelSheet = false
|
|
|
|
withAnimation {
|
|
aspectRatio = VideoPlayerView.defaultAspectRatio
|
|
currentItem = item
|
|
}
|
|
|
|
if !time.isNil {
|
|
currentItem.playbackTime = time
|
|
} else if currentItem.playbackTime.isNil {
|
|
currentItem.playbackTime = .zero
|
|
}
|
|
|
|
preservedTime = currentItem.playbackTime
|
|
|
|
DispatchQueue.main.async { [weak self] in
|
|
guard let self else { return }
|
|
guard let video = item.video else {
|
|
return
|
|
}
|
|
|
|
if video.isLocal {
|
|
self.videoBeingOpened = nil
|
|
self.availableStreams = video.streams
|
|
return
|
|
}
|
|
|
|
guard let playerInstance = self.playerInstance else { return }
|
|
let streamsInstance = video.streams.compactMap(\.instance).first
|
|
|
|
if video.streams.isEmpty || streamsInstance.isNil || streamsInstance!.apiURLString != playerInstance.apiURLString {
|
|
self.loadAvailableStreams(video) { [weak self] _ in
|
|
self?.videoBeingOpened = nil
|
|
}
|
|
} else {
|
|
self.videoBeingOpened = nil
|
|
self.streamsWithInstance(instance: playerInstance, streams: video.streams) { processedStreams in
|
|
self.availableStreams = processedStreams
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
var playerInstance: Instance? {
|
|
InstancesModel.shared.forPlayer ?? accounts.current?.instance ?? InstancesModel.shared.all.first
|
|
}
|
|
|
|
func playerAPI(_ video: Video) -> VideosAPI? {
|
|
guard let url = video.instanceURL else { return accounts.api }
|
|
if accounts.current?.url == url { return accounts.api }
|
|
switch video.app {
|
|
case .local:
|
|
return nil
|
|
case .peerTube:
|
|
return PeerTubeAPI.withAnonymousAccountForInstanceURL(url)
|
|
case .invidious:
|
|
return InvidiousAPI.withAnonymousAccountForInstanceURL(url)
|
|
case .piped:
|
|
return PipedAPI.withAnonymousAccountForInstanceURL(url)
|
|
}
|
|
}
|
|
|
|
var qualityProfile: QualityProfile? {
|
|
qualityProfileSelection ?? QualityProfilesModel.shared.automaticProfile
|
|
}
|
|
|
|
var streamByQualityProfile: Stream? {
|
|
// Safety check: Ensure backend is available
|
|
guard backend != nil else {
|
|
logger.error("Backend is nil when trying to select stream by quality profile")
|
|
return nil
|
|
}
|
|
|
|
let profile = qualityProfile ?? .defaultProfile
|
|
|
|
// First attempt: Filter by both `canPlay` and `isPreferred`
|
|
if let streamPreferredForProfile = backend.bestPlayable(
|
|
availableStreams.filter { backend.canPlay($0) && profile.isPreferred($0) },
|
|
maxResolution: profile.resolution, formatOrder: profile.formats
|
|
) {
|
|
return streamPreferredForProfile
|
|
}
|
|
|
|
// Fallback: Filter by `canPlay` only
|
|
let fallbackStream = backend.bestPlayable(
|
|
availableStreams.filter { backend.canPlay($0) },
|
|
maxResolution: profile.resolution, formatOrder: profile.formats
|
|
)
|
|
|
|
// If no stream is found, trigger the error handler
|
|
guard let finalStream = fallbackStream else {
|
|
let error = RequestError(
|
|
userMessage: "No supported streams available.",
|
|
cause: NSError(domain: "stream.yatte.app", code: -1, userInfo: [NSLocalizedDescriptionKey: "No supported streams available"])
|
|
)
|
|
videoLoadFailureHandler(error, video: currentVideo)
|
|
return nil
|
|
}
|
|
|
|
// Return the found stream
|
|
return finalStream
|
|
}
|
|
|
|
func advanceToNextItem() {
|
|
guard !advancing else {
|
|
return
|
|
}
|
|
advancing = true
|
|
prepareCurrentItemForHistory()
|
|
|
|
var nextItem: PlayerQueueItem?
|
|
switch playbackMode {
|
|
case .queue:
|
|
nextItem = queue.first
|
|
case .shuffle:
|
|
nextItem = queue.randomElement()
|
|
case .related:
|
|
nextItem = autoplayItem
|
|
case .loopOne:
|
|
nextItem = nil
|
|
}
|
|
|
|
resetAutoplay()
|
|
|
|
if let nextItem {
|
|
advanceToItem(nextItem)
|
|
} else {
|
|
advancing = false
|
|
}
|
|
}
|
|
|
|
var isAdvanceToNextItemAvailable: Bool {
|
|
switch playbackMode {
|
|
case .loopOne:
|
|
return false
|
|
case .queue, .shuffle:
|
|
return !queue.isEmpty
|
|
case .related:
|
|
return autoplayItem != nil
|
|
}
|
|
}
|
|
|
|
func advanceToItem(_ newItem: PlayerQueueItem, at time: CMTime? = nil) {
|
|
prepareCurrentItemForHistory()
|
|
|
|
remove(newItem)
|
|
|
|
navigation.presentingChannelSheet = false
|
|
currentItem = newItem
|
|
currentItem.playbackTime = time
|
|
|
|
let playTime = currentItem.shouldRestartPlaying ? CMTime.zero : time
|
|
guard let video = newItem.video else { return }
|
|
playerAPI(video)?.loadDetails(currentItem, failureHandler: { self.videoLoadFailureHandler($0, video: video) }) { newItem in
|
|
self.playItem(newItem, at: playTime)
|
|
}
|
|
}
|
|
|
|
@discardableResult func remove(_ item: PlayerQueueItem) -> PlayerQueueItem? {
|
|
if let index = queue.firstIndex(where: { $0.videoID == item.videoID }) {
|
|
return queue.remove(at: index)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func resetQueue() {
|
|
DispatchQueue.main.async { [weak self] in
|
|
guard let self else {
|
|
return
|
|
}
|
|
|
|
self.currentItem = nil
|
|
self.stream = nil
|
|
self.removeQueueItems()
|
|
}
|
|
|
|
if backend != nil {
|
|
backend.closeItem()
|
|
}
|
|
}
|
|
|
|
@discardableResult func enqueueVideo(
|
|
_ video: Video,
|
|
play: Bool = false,
|
|
atTime: CMTime? = nil,
|
|
prepending: Bool = false,
|
|
loadDetails: Bool = true,
|
|
videoDetailsLoadHandler: @escaping (Video, PlayerQueueItem) -> Void = { _, _ in }
|
|
) -> PlayerQueueItem? {
|
|
let item = PlayerQueueItem(video, playbackTime: atTime)
|
|
|
|
if play {
|
|
navigation.presentingChannelSheet = false
|
|
|
|
withAnimation {
|
|
aspectRatio = VideoPlayerView.defaultAspectRatio
|
|
navigation.presentingChannelSheet = false
|
|
currentItem = item
|
|
}
|
|
videoBeingOpened = video
|
|
}
|
|
|
|
if loadDetails {
|
|
playerAPI(item.video)?.loadDetails(item, failureHandler: { self.videoLoadFailureHandler($0, video: video) }) { [weak self] newItem in
|
|
guard let self else { return }
|
|
videoDetailsLoadHandler(newItem.video, newItem)
|
|
|
|
if play {
|
|
self.playItem(newItem)
|
|
} else {
|
|
self.queue.insert(newItem, at: prepending ? 0 : self.queue.endIndex)
|
|
}
|
|
}
|
|
} else {
|
|
videoDetailsLoadHandler(video, item)
|
|
queue.insert(item, at: prepending ? 0 : queue.endIndex)
|
|
}
|
|
|
|
return item
|
|
}
|
|
|
|
func prepareCurrentItemForHistory(finished: Bool = false) {
|
|
if let currentItem {
|
|
if Defaults[.saveHistory] {
|
|
if let video = currentVideo, !historyVideos.contains(where: { $0 == video }) {
|
|
historyVideos.append(video)
|
|
}
|
|
updateWatch(finished: finished, time: backend.currentTime)
|
|
}
|
|
|
|
if let video = currentItem.video,
|
|
video.isLocal,
|
|
video.localStreamIsFile,
|
|
let localURL = video.localStream?.localURL
|
|
{
|
|
logger.info("stopping security scoped resource access for \(localURL)")
|
|
localURL.stopAccessingSecurityScopedResource()
|
|
}
|
|
}
|
|
}
|
|
|
|
func playHistory(_ item: PlayerQueueItem, at time: CMTime? = nil) {
|
|
guard let video = item.video else { return }
|
|
|
|
var time = time ?? item.playbackTime
|
|
|
|
if item.shouldRestartPlaying {
|
|
time = .zero
|
|
}
|
|
|
|
let newItem = enqueueVideo(video, atTime: time, prepending: true)
|
|
|
|
advanceToItem(newItem!, at: time)
|
|
}
|
|
|
|
func removeQueueItems() {
|
|
queue.removeAll()
|
|
}
|
|
|
|
func restoreQueue() {
|
|
var restoredQueue = [PlayerQueueItem?]()
|
|
|
|
if let lastPlayed,
|
|
!Defaults[.queue].contains(where: { $0.videoID == lastPlayed.videoID })
|
|
{
|
|
restoredQueue.append(lastPlayed)
|
|
self.lastPlayed = nil
|
|
}
|
|
|
|
restoredQueue.append(contentsOf: Defaults[.queue])
|
|
queue = restoredQueue.compactMap(\.self)
|
|
queue.forEach { loadQueueVideoDetails($0) }
|
|
}
|
|
|
|
func loadQueueVideoDetails(_ item: PlayerQueueItem) {
|
|
guard !accounts.current.isNil, !item.hasDetailsLoaded else { return }
|
|
|
|
let videoID = item.video?.videoID ?? item.videoID
|
|
|
|
let video = item.video ?? Video(app: item.app ?? .local, instanceURL: item.instanceURL, videoID: videoID)
|
|
|
|
let replaceQueueItem: (PlayerQueueItem) -> Void = { newItem in
|
|
self.queue.filter { $0.videoID == videoID }.forEach { item in
|
|
if let index = self.queue.firstIndex(of: item) {
|
|
self.queue[index] = newItem
|
|
}
|
|
}
|
|
}
|
|
|
|
if let video = VideosCacheModel.shared.retrieveVideo(video.cacheKey) {
|
|
var item = item
|
|
item.id = UUID()
|
|
item.video = video
|
|
replaceQueueItem(item)
|
|
return
|
|
}
|
|
|
|
playerAPI(video)?
|
|
.loadDetails(item, failureHandler: nil) { [weak self] newItem in
|
|
guard let self else { return }
|
|
|
|
replaceQueueItem(newItem)
|
|
|
|
self.logger.info("LOADED queue details: \(videoID)")
|
|
}
|
|
}
|
|
|
|
private func videoLoadFailureHandler(_ error: RequestError, video: Video? = nil) {
|
|
guard let video else {
|
|
presentErrorAlert(error)
|
|
return
|
|
}
|
|
|
|
let videoID = video.videoID
|
|
let currentRetry = retryAttempts[videoID] ?? 0
|
|
|
|
if currentRetry < Defaults[.videoLoadingRetryCount] {
|
|
retryAttempts[videoID] = currentRetry + 1
|
|
|
|
logger.info("Retry attempt \(currentRetry + 1) for video \(videoID) due to error: \(error)")
|
|
|
|
DispatchQueue.main.asyncAfter(deadline: .now() + 0.5) { [weak self] in
|
|
guard let self else { return }
|
|
self.enqueueVideo(video, play: true, prepending: true, loadDetails: true)
|
|
}
|
|
return
|
|
}
|
|
|
|
retryAttempts[videoID] = 0
|
|
presentErrorAlert(error, video: video)
|
|
}
|
|
|
|
private func presentErrorAlert(_ error: RequestError, video: Video? = nil) {
|
|
var message = error.userMessage
|
|
if let errorDictionary = error.json.dictionaryObject,
|
|
let errorMessage = errorDictionary["message"] ?? errorDictionary["error"],
|
|
let errorString = errorMessage as? String
|
|
{
|
|
message += "\n"
|
|
message += errorString
|
|
}
|
|
|
|
var retryButton: Alert.Button?
|
|
|
|
if let video {
|
|
retryButton = Alert.Button.default(Text("Retry")) { [weak self] in
|
|
if let self {
|
|
self.enqueueVideo(video, play: true, prepending: true, loadDetails: true)
|
|
}
|
|
}
|
|
}
|
|
|
|
var alert: Alert
|
|
if let retryButton {
|
|
alert = Alert(
|
|
title: Text("Could not load video"),
|
|
message: Text(message),
|
|
primaryButton: .cancel { [weak self] in
|
|
guard let self else { return }
|
|
self.closeCurrentItem()
|
|
},
|
|
secondaryButton: retryButton
|
|
)
|
|
} else {
|
|
alert = Alert(title: Text("Could not load video"))
|
|
}
|
|
|
|
navigation.presentAlert(alert)
|
|
}
|
|
}
|