2021-12-19 17:17:04 +00:00
|
|
|
import AVKit
|
2021-10-25 08:25:41 +00:00
|
|
|
import Defaults
|
2021-10-05 20:20:09 +00:00
|
|
|
import Foundation
|
2021-10-16 22:48:58 +00:00
|
|
|
import Siesta
|
2021-10-05 20:20:09 +00:00
|
|
|
|
|
|
|
extension PlayerModel {
|
|
|
|
var currentVideo: Video? {
|
|
|
|
currentItem?.video
|
|
|
|
}
|
|
|
|
|
2022-01-02 18:59:57 +00:00
|
|
|
func play(_ videos: [Video], shuffling: Bool = false, inNavigationView: Bool = false) {
|
|
|
|
let videosToPlay = shuffling ? videos.shuffled() : videos
|
2021-10-05 20:20:09 +00:00
|
|
|
|
2022-01-02 18:59:57 +00:00
|
|
|
guard let first = videosToPlay.first else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
enqueueVideo(first, prepending: true) { _, item in
|
|
|
|
self.advanceToItem(item)
|
|
|
|
}
|
|
|
|
|
|
|
|
videosToPlay.dropFirst().reversed().forEach { video in
|
|
|
|
enqueueVideo(video, prepending: true) { _, item in
|
2021-10-05 20:20:09 +00:00
|
|
|
if item.video == first {
|
|
|
|
self.advanceToItem(item)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-01-02 18:59:57 +00:00
|
|
|
|
|
|
|
if inNavigationView {
|
|
|
|
playerNavigationLinkActive = true
|
|
|
|
} else {
|
|
|
|
show()
|
|
|
|
}
|
2021-10-05 20:20:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func playNext(_ video: Video) {
|
|
|
|
enqueueVideo(video, prepending: true) { _, item in
|
2021-10-16 22:48:58 +00:00
|
|
|
if self.currentItem.isNil {
|
2021-10-05 20:20:09 +00:00
|
|
|
self.advanceToItem(item)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-24 09:16:04 +00:00
|
|
|
func playNow(_ video: Video, at time: TimeInterval? = nil) {
|
2021-12-26 21:14:46 +00:00
|
|
|
if playingInPictureInPicture, closePiPOnNavigation {
|
2021-12-19 17:17:04 +00:00
|
|
|
closePiP()
|
|
|
|
}
|
|
|
|
|
2021-12-26 21:14:46 +00:00
|
|
|
prepareCurrentItemForHistory()
|
2021-10-05 20:20:09 +00:00
|
|
|
|
|
|
|
enqueueVideo(video, prepending: true) { _, item in
|
2021-10-24 09:16:04 +00:00
|
|
|
self.advanceToItem(item, at: time)
|
2021-10-05 20:20:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-24 09:16:04 +00:00
|
|
|
func playItem(_ item: PlayerQueueItem, video: Video? = nil, at time: TimeInterval? = nil) {
|
2021-12-19 17:17:04 +00:00
|
|
|
if !playingInPictureInPicture {
|
|
|
|
player.replaceCurrentItem(with: nil)
|
|
|
|
}
|
|
|
|
|
2021-12-05 17:14:49 +00:00
|
|
|
comments.reset()
|
2021-12-19 17:17:04 +00:00
|
|
|
stream = nil
|
2021-10-05 20:20:09 +00:00
|
|
|
currentItem = item
|
2021-10-24 09:16:04 +00:00
|
|
|
|
|
|
|
if !time.isNil {
|
2021-10-24 18:01:08 +00:00
|
|
|
currentItem.playbackTime = .secondsInDefaultTimescale(time!)
|
2021-10-24 09:16:04 +00:00
|
|
|
} else if currentItem.playbackTime.isNil {
|
2021-10-22 20:49:31 +00:00
|
|
|
currentItem.playbackTime = .zero
|
|
|
|
}
|
2021-10-05 20:20:09 +00:00
|
|
|
|
|
|
|
if video != nil {
|
|
|
|
currentItem.video = video!
|
|
|
|
}
|
|
|
|
|
2021-12-17 20:01:05 +00:00
|
|
|
preservedTime = currentItem.playbackTime
|
|
|
|
restoreLoadedChannel()
|
2021-10-24 18:01:08 +00:00
|
|
|
|
2021-12-29 18:55:41 +00:00
|
|
|
DispatchQueue.main.async { [weak self] in
|
|
|
|
guard let video = self?.currentVideo else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
self?.loadAvailableStreams(video)
|
|
|
|
}
|
2021-10-05 20:20:09 +00:00
|
|
|
}
|
|
|
|
|
2021-12-19 16:56:47 +00:00
|
|
|
func preferredStream(_ streams: [Stream]) -> Stream? {
|
2021-11-05 19:35:27 +00:00
|
|
|
let quality = Defaults[.quality]
|
|
|
|
var streams = streams
|
|
|
|
|
2021-11-03 23:40:01 +00:00
|
|
|
if let id = Defaults[.playerInstanceID] {
|
2021-11-05 19:35:27 +00:00
|
|
|
streams = streams.filter { $0.instance.id == id }
|
|
|
|
}
|
|
|
|
|
|
|
|
switch quality {
|
|
|
|
case .best:
|
|
|
|
return streams.first { $0.kind == .hls } ?? streams.first
|
|
|
|
default:
|
|
|
|
let sorted = streams.filter { $0.kind != .hls }.sorted { $0.resolution > $1.resolution }
|
|
|
|
return sorted.first(where: { $0.resolution.height <= quality.value.height })
|
2021-11-03 23:40:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-05 20:20:09 +00:00
|
|
|
func advanceToNextItem() {
|
2021-12-26 21:14:46 +00:00
|
|
|
prepareCurrentItemForHistory()
|
2021-10-05 20:20:09 +00:00
|
|
|
|
|
|
|
if let nextItem = queue.first {
|
|
|
|
advanceToItem(nextItem)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-24 09:16:04 +00:00
|
|
|
func advanceToItem(_ newItem: PlayerQueueItem, at time: TimeInterval? = nil) {
|
2021-12-26 21:14:46 +00:00
|
|
|
prepareCurrentItemForHistory()
|
2021-10-22 20:49:31 +00:00
|
|
|
|
2021-10-24 18:01:08 +00:00
|
|
|
remove(newItem)
|
|
|
|
|
2021-12-29 18:55:41 +00:00
|
|
|
currentItem = newItem
|
|
|
|
player.pause()
|
|
|
|
|
2021-10-24 18:01:08 +00:00
|
|
|
accounts.api.loadDetails(newItem) { newItem in
|
|
|
|
self.playItem(newItem, video: newItem.video, at: time)
|
2021-10-05 20:20:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@discardableResult func remove(_ item: PlayerQueueItem) -> PlayerQueueItem? {
|
2021-10-26 22:59:59 +00:00
|
|
|
if let index = queue.firstIndex(where: { $0.videoID == item.videoID }) {
|
2021-10-05 20:20:09 +00:00
|
|
|
return queue.remove(at: index)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func resetQueue() {
|
2021-10-24 18:01:08 +00:00
|
|
|
DispatchQueue.main.async { [weak self] in
|
|
|
|
guard let self = self else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-10-05 20:20:09 +00:00
|
|
|
self.currentItem = nil
|
|
|
|
self.stream = nil
|
|
|
|
self.removeQueueItems()
|
|
|
|
}
|
|
|
|
|
|
|
|
player.replaceCurrentItem(with: nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
func isAutoplaying(_ item: AVPlayerItem) -> Bool {
|
2021-12-19 17:17:04 +00:00
|
|
|
player.currentItem == item
|
2021-10-05 20:20:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@discardableResult func enqueueVideo(
|
|
|
|
_ video: Video,
|
|
|
|
play: Bool = false,
|
2021-10-22 20:49:31 +00:00
|
|
|
atTime: CMTime? = nil,
|
2021-10-05 20:20:09 +00:00
|
|
|
prepending: Bool = false,
|
|
|
|
videoDetailsLoadHandler: @escaping (Video, PlayerQueueItem) -> Void = { _, _ in }
|
|
|
|
) -> PlayerQueueItem? {
|
2021-10-22 20:49:31 +00:00
|
|
|
let item = PlayerQueueItem(video, playbackTime: atTime)
|
2021-10-05 20:20:09 +00:00
|
|
|
|
2021-12-29 18:55:41 +00:00
|
|
|
if play {
|
|
|
|
currentItem = item
|
|
|
|
// pause playing current video as it's going to be replaced with next one
|
|
|
|
player.pause()
|
|
|
|
}
|
|
|
|
|
2021-10-05 20:20:09 +00:00
|
|
|
queue.insert(item, at: prepending ? 0 : queue.endIndex)
|
|
|
|
|
2021-10-24 18:01:08 +00:00
|
|
|
accounts.api.loadDetails(item) { newItem in
|
|
|
|
videoDetailsLoadHandler(newItem.video, newItem)
|
2021-10-05 20:20:09 +00:00
|
|
|
|
|
|
|
if play {
|
2021-10-24 18:01:08 +00:00
|
|
|
self.playItem(newItem, video: video)
|
2021-10-05 20:20:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return item
|
|
|
|
}
|
|
|
|
|
2021-12-26 21:14:46 +00:00
|
|
|
func prepareCurrentItemForHistory(finished: Bool = false) {
|
|
|
|
if !currentItem.isNil, Defaults[.saveHistory] {
|
|
|
|
if let video = currentVideo, !historyVideos.contains(where: { $0 == video }) {
|
|
|
|
historyVideos.append(video)
|
|
|
|
}
|
|
|
|
updateWatch(finished: finished)
|
2021-10-05 20:20:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-13 22:05:19 +00:00
|
|
|
func playHistory(_ item: PlayerQueueItem) {
|
2021-10-22 20:49:31 +00:00
|
|
|
var time = item.playbackTime
|
|
|
|
|
|
|
|
if item.shouldRestartPlaying {
|
|
|
|
time = .zero
|
|
|
|
}
|
|
|
|
|
|
|
|
let newItem = enqueueVideo(item.video, atTime: time, prepending: true)
|
2021-10-13 22:05:19 +00:00
|
|
|
|
|
|
|
advanceToItem(newItem!)
|
2021-10-05 20:20:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func removeQueueItems() {
|
|
|
|
queue.removeAll()
|
|
|
|
}
|
2022-01-09 15:05:05 +00:00
|
|
|
|
|
|
|
func restoreQueue() {
|
|
|
|
guard !accounts.current.isNil else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
queue = ([Defaults[.lastPlayed]] + Defaults[.queue]).compactMap { $0 }
|
|
|
|
Defaults[.lastPlayed] = nil
|
|
|
|
|
|
|
|
queue.forEach { item in
|
|
|
|
accounts.api.loadDetails(item) { newItem in
|
|
|
|
if let index = self.queue.firstIndex(where: { $0.id == item.id }) {
|
|
|
|
self.queue[index] = newItem
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-10-05 20:20:09 +00:00
|
|
|
}
|