564 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			Swift
		
	
	
	
	
	
			
		
		
	
	
			564 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			Swift
		
	
	
	
	
	
| //
 | |
| //  PlayerViewController.swift
 | |
| //  JellyfinVideoPlayerDev
 | |
| //
 | |
| //  Created by Ethan Pippin on 11/12/21.
 | |
| //
 | |
| 
 | |
| import AVKit
 | |
| import AVFoundation
 | |
| import Combine
 | |
| import Defaults
 | |
| import JellyfinAPI
 | |
| import MediaPlayer
 | |
| import MobileVLCKit
 | |
| import SwiftUI
 | |
| import UIKit
 | |
| 
 | |
| // TODO: Make the VLC player layer a view
 | |
| // This will allow changing media and putting the view somewhere else
 | |
| // in a compact state, like a small viewer while navigating the app
 | |
| 
 | |
| // TODO: Look at making overlays handle timer and all gesture events
 | |
| 
 | |
| class VLCPlayerViewController: UIViewController {
 | |
|     
 | |
|     // MARK: variables
 | |
|     
 | |
|     private var viewModel: VideoPlayerViewModel
 | |
|     private var vlcMediaPlayer = VLCMediaPlayer()
 | |
|     private var lastPlayerTicks: Int64 = 0
 | |
|     private var lastProgressReportTicks: Int64 = 0
 | |
|     private var viewModelReactCancellables = Set<AnyCancellable>()
 | |
|     private var overlayDismissTimer: Timer?
 | |
|     
 | |
|     private var currentPlayerTicks: Int64 {
 | |
|         return Int64(vlcMediaPlayer.time.intValue) * 100_000
 | |
|     }
 | |
|     
 | |
|     private var displayingOverlay: Bool {
 | |
|         return currentOverlayHostingController?.view.alpha ??  0 > 0
 | |
|     }
 | |
|     
 | |
|     private var jumpForwardLength: VideoPlayerJumpLength {
 | |
|         return Defaults[.videoPlayerJumpForward]
 | |
|     }
 | |
| 
 | |
|     private var jumpBackwardLength: VideoPlayerJumpLength {
 | |
|         return Defaults[.videoPlayerJumpBackward]
 | |
|     }
 | |
|     
 | |
|     private lazy var videoContentView = makeVideoContentView()
 | |
|     private lazy var tapGestureView = makeTapGestureView()
 | |
|     private var currentOverlayHostingController: UIHostingController<VLCPlayerCompactOverlayView>?
 | |
|     
 | |
|     // MARK: init
 | |
|     
 | |
|     init(viewModel: VideoPlayerViewModel) {
 | |
|         
 | |
|         self.viewModel = viewModel
 | |
|         
 | |
|         super.init(nibName: nil, bundle: nil)
 | |
|         
 | |
|         viewModel.playerOverlayDelegate = self
 | |
|     }
 | |
|     
 | |
|     required init?(coder: NSCoder) {
 | |
|         fatalError("init(coder:) has not been implemented")
 | |
|     }
 | |
|     
 | |
|     private func setupSubviews() {
 | |
|         view.addSubview(videoContentView)
 | |
|         view.addSubview(tapGestureView)
 | |
|     }
 | |
|     
 | |
|     private func setupConstraints() {
 | |
|         NSLayoutConstraint.activate([
 | |
|             videoContentView.topAnchor.constraint(equalTo: view.topAnchor),
 | |
|             videoContentView.bottomAnchor.constraint(equalTo: view.bottomAnchor),
 | |
|             videoContentView.leftAnchor.constraint(equalTo: view.leftAnchor),
 | |
|             videoContentView.rightAnchor.constraint(equalTo: view.rightAnchor)
 | |
|         ])
 | |
|         NSLayoutConstraint.activate([
 | |
|             tapGestureView.topAnchor.constraint(equalTo: videoContentView.topAnchor),
 | |
|             tapGestureView.bottomAnchor.constraint(equalTo: videoContentView.bottomAnchor),
 | |
|             tapGestureView.leftAnchor.constraint(equalTo: videoContentView.leftAnchor),
 | |
|             tapGestureView.rightAnchor.constraint(equalTo: videoContentView.rightAnchor)
 | |
|         ])
 | |
|     }
 | |
|     
 | |
|     // MARK: viewWillDisappear
 | |
|     
 | |
|     override func viewWillDisappear(_ animated: Bool) {
 | |
|         super.viewWillDisappear(animated)
 | |
|         
 | |
|         let defaultNotificationCenter = NotificationCenter.default
 | |
|         defaultNotificationCenter.removeObserver(self, name: UIApplication.willTerminateNotification, object: nil)
 | |
|         defaultNotificationCenter.removeObserver(self, name: UIApplication.willResignActiveNotification, object: nil)
 | |
|         defaultNotificationCenter.removeObserver(self, name: UIApplication.didEnterBackgroundNotification, object: nil)
 | |
|     }
 | |
|     
 | |
|     // MARK: viewDidLoad
 | |
|     
 | |
|     override func viewDidLoad() {
 | |
|         super.viewDidLoad()
 | |
|         
 | |
|         setupSubviews()
 | |
|         setupConstraints()
 | |
|         
 | |
|         view.backgroundColor = .black
 | |
|         
 | |
|         // These are kept outside of 'setupMediaPlayer' such that
 | |
|         // they aren't unnecessarily set more than once
 | |
|         vlcMediaPlayer.delegate = self
 | |
|         vlcMediaPlayer.drawable = videoContentView
 | |
|         vlcMediaPlayer.perform(Selector(("setTextRendererFontSize:")), with: 14)
 | |
|         
 | |
|         setupMediaPlayer(newViewModel: viewModel)
 | |
|         
 | |
|         let defaultNotificationCenter = NotificationCenter.default
 | |
|         defaultNotificationCenter.addObserver(self, selector: #selector(appWillTerminate), name: UIApplication.willTerminateNotification, object: nil)
 | |
|         defaultNotificationCenter.addObserver(self, selector: #selector(appWillResignActive), name: UIApplication.willResignActiveNotification, object: nil)
 | |
|         defaultNotificationCenter.addObserver(self, selector: #selector(appWillResignActive), name: UIApplication.didEnterBackgroundNotification, object: nil)
 | |
|     }
 | |
|     
 | |
|     @objc private func appWillTerminate() {
 | |
|         viewModel.sendStopReport()
 | |
|     }
 | |
|     
 | |
|     @objc private func appWillResignActive() {
 | |
|         showOverlay()
 | |
|         
 | |
|         stopOverlayDismissTimer()
 | |
|         
 | |
|         vlcMediaPlayer.pause()
 | |
|         
 | |
|         viewModel.sendPauseReport(paused: true)
 | |
|     }
 | |
|     
 | |
|     override func viewDidAppear(_ animated: Bool) {
 | |
|         super.viewDidAppear(animated)
 | |
|         
 | |
|         startPlayback()
 | |
|     }
 | |
|     
 | |
|     // MARK: subviews
 | |
|     
 | |
|     private func makeVideoContentView() -> UIView {
 | |
|         let view = UIView()
 | |
|         view.translatesAutoresizingMaskIntoConstraints = false
 | |
|         view.backgroundColor = .black
 | |
|         
 | |
|         return view
 | |
|     }
 | |
|     
 | |
|     private func makeTapGestureView() -> UIView {
 | |
|         let view = UIView()
 | |
|         view.translatesAutoresizingMaskIntoConstraints = false
 | |
|         
 | |
|         let singleTapGesture = UITapGestureRecognizer(target: self, action: #selector(didTap))
 | |
|         let rightSwipeGesture = UISwipeGestureRecognizer(target: self, action: #selector(didRightSwipe))
 | |
|         rightSwipeGesture.direction = .right
 | |
|         let leftSwipeGesture = UISwipeGestureRecognizer(target: self, action: #selector(didLeftSwipe))
 | |
|         leftSwipeGesture.direction = .left
 | |
|         
 | |
|         view.addGestureRecognizer(singleTapGesture)
 | |
|         view.addGestureRecognizer(rightSwipeGesture)
 | |
|         view.addGestureRecognizer(leftSwipeGesture)
 | |
|         
 | |
|         return view
 | |
|     }
 | |
|     
 | |
|     @objc private func didTap() {
 | |
|         self.didGenerallyTap()
 | |
|     }
 | |
|     
 | |
|     @objc private func didRightSwipe() {
 | |
|         self.didSelectForward()
 | |
|     }
 | |
|     
 | |
|     @objc private func didLeftSwipe() {
 | |
|         self.didSelectBackward()
 | |
|     }
 | |
|     
 | |
|     // MARK: setupOverlayHostingController
 | |
|     private func setupOverlayHostingController(viewModel: VideoPlayerViewModel) {
 | |
| 
 | |
|         // TODO: Look at injecting viewModel into the environment so it updates the current overlay
 | |
|         if let currentOverlayHostingController = currentOverlayHostingController {
 | |
|             // UX fade-out
 | |
|             UIView.animate(withDuration: 0.5) {
 | |
|                 currentOverlayHostingController.view.alpha = 0
 | |
|             } completion: { _ in
 | |
|                 currentOverlayHostingController.view.isHidden = true
 | |
|                 
 | |
|                 currentOverlayHostingController.view.removeFromSuperview()
 | |
|                 currentOverlayHostingController.removeFromParent()
 | |
| //                self.currentOverlayHostingController = nil
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         let newOverlayView = VLCPlayerCompactOverlayView(viewModel: viewModel)
 | |
|         let newOverlayHostingController = UIHostingController(rootView: newOverlayView)
 | |
|         
 | |
|         newOverlayHostingController.view.translatesAutoresizingMaskIntoConstraints = false
 | |
|         newOverlayHostingController.view.backgroundColor = UIColor.clear
 | |
|         
 | |
|         // UX fade-in
 | |
|         newOverlayHostingController.view.alpha = 0
 | |
|         
 | |
|         addChild(newOverlayHostingController)
 | |
|         view.addSubview(newOverlayHostingController.view)
 | |
|         newOverlayHostingController.didMove(toParent: self)
 | |
|         
 | |
|         NSLayoutConstraint.activate([
 | |
|             newOverlayHostingController.view.topAnchor.constraint(equalTo: videoContentView.topAnchor),
 | |
|             newOverlayHostingController.view.bottomAnchor.constraint(equalTo: videoContentView.bottomAnchor),
 | |
|             newOverlayHostingController.view.leftAnchor.constraint(equalTo: videoContentView.leftAnchor),
 | |
|             newOverlayHostingController.view.rightAnchor.constraint(equalTo: videoContentView.rightAnchor)
 | |
|         ])
 | |
|         
 | |
|         // UX fade-in
 | |
|         UIView.animate(withDuration: 0.5) {
 | |
|             newOverlayHostingController.view.alpha = 1
 | |
|         }
 | |
|         
 | |
|         self.currentOverlayHostingController = newOverlayHostingController
 | |
|         
 | |
|         // There is a behavior when setting this that the navigation bar
 | |
|         // on the current navigation controller pops up, re-hide it
 | |
|         self.navigationController?.isNavigationBarHidden = true
 | |
|     }
 | |
| }
 | |
| 
 | |
| // MARK: setupMediaPlayer
 | |
| extension VLCPlayerViewController {
 | |
|     
 | |
|     /// Main function that handles setting up the media player with the current VideoPlayerViewModel
 | |
|     /// and also takes the role of setting the 'viewModel' property with the given viewModel
 | |
|     ///
 | |
|     /// Use case for this is setting new media within the same VLCPlayerViewController
 | |
|     func setupMediaPlayer(newViewModel: VideoPlayerViewModel) {
 | |
|         
 | |
|         stopOverlayDismissTimer()
 | |
|         
 | |
|         // Stop current media if there is one
 | |
|         if vlcMediaPlayer.media != nil {
 | |
|             viewModelReactCancellables.forEach({ $0.cancel() })
 | |
|             
 | |
|             vlcMediaPlayer.stop()
 | |
|             viewModel.sendStopReport()
 | |
|             viewModel.playerOverlayDelegate = nil
 | |
|         }
 | |
|         
 | |
|         lastPlayerTicks = newViewModel.item.userData?.playbackPositionTicks ?? 0
 | |
|         lastProgressReportTicks = newViewModel.item.userData?.playbackPositionTicks ?? 0
 | |
| 
 | |
|         let media = VLCMedia(url: newViewModel.streamURL)
 | |
|         media.addOption("--prefetch-buffer-size=1048576")
 | |
|         media.addOption("--network-caching=5000")
 | |
|         
 | |
|         vlcMediaPlayer.media = media
 | |
|         
 | |
|         setupOverlayHostingController(viewModel: newViewModel)
 | |
|         setupViewModelListeners(viewModel: newViewModel)
 | |
|         
 | |
|         newViewModel.getAdjacentEpisodes()
 | |
|         newViewModel.playerOverlayDelegate = self
 | |
|         
 | |
|         let startPercentage = viewModel.item.userData?.playedPercentage ?? 0
 | |
|         
 | |
|         if startPercentage > 0 {
 | |
|             newViewModel.sliderPercentage = startPercentage / 100
 | |
|         }
 | |
|         
 | |
|         viewModel = newViewModel
 | |
|     }
 | |
|     
 | |
|     // MARK: startPlayback
 | |
|     func startPlayback() {
 | |
|         vlcMediaPlayer.play()
 | |
|         
 | |
|         setMediaPlayerTimeAtCurrentSlider()
 | |
|         
 | |
|         viewModel.sendPlayReport()
 | |
|         
 | |
|         restartOverlayDismissTimer()
 | |
|     }
 | |
|     
 | |
|     // MARK: setupViewModelListeners
 | |
|     
 | |
|     private func setupViewModelListeners(viewModel: VideoPlayerViewModel) {
 | |
|         viewModel.$playbackSpeed.sink { newSpeed in
 | |
|             self.vlcMediaPlayer.rate = Float(newSpeed.rawValue)
 | |
|         }.store(in: &viewModelReactCancellables)
 | |
|         
 | |
|         viewModel.$sliderIsScrubbing.sink { sliderIsScrubbing in
 | |
|             if sliderIsScrubbing {
 | |
|                 self.didBeginScrubbing()
 | |
|             } else {
 | |
|                 self.didEndScrubbing()
 | |
|             }
 | |
|         }.store(in: &viewModelReactCancellables)
 | |
|         
 | |
|         viewModel.$selectedAudioStreamIndex.sink { newAudioStreamIndex in
 | |
|             self.didSelectAudioStream(index: newAudioStreamIndex)
 | |
|         }.store(in: &viewModelReactCancellables)
 | |
|         
 | |
|         viewModel.$selectedSubtitleStreamIndex.sink { newSubtitleStreamIndex in
 | |
|             self.didSelectSubtitleStream(index: newSubtitleStreamIndex)
 | |
|         }.store(in: &viewModelReactCancellables)
 | |
|     }
 | |
|     
 | |
|     func setMediaPlayerTimeAtCurrentSlider() {
 | |
|         // Necessary math as VLCMediaPlayer doesn't work well
 | |
|         //     by just setting the position
 | |
|         let videoPosition = Double(vlcMediaPlayer.time.intValue / 1000)
 | |
|         let videoDuration = Double(viewModel.item.runTimeTicks! / 10_000_000)
 | |
|         let secondsScrubbedTo = round(viewModel.sliderPercentage * videoDuration)
 | |
|         let newPositionOffset = secondsScrubbedTo - videoPosition
 | |
|         
 | |
|         if newPositionOffset > 0 {
 | |
|             vlcMediaPlayer.jumpForward(Int32(newPositionOffset))
 | |
|         } else {
 | |
|             vlcMediaPlayer.jumpBackward(Int32(abs(newPositionOffset)))
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| // MARK: Show/Hide Overlay
 | |
| extension VLCPlayerViewController {
 | |
|     
 | |
|     private func showOverlay() {
 | |
|         guard let overlayHostingController = currentOverlayHostingController else { return }
 | |
|         
 | |
|         guard overlayHostingController.view.alpha != 1 else { return }
 | |
|         
 | |
|         UIView.animate(withDuration: 0.2) {
 | |
|             overlayHostingController.view.alpha = 1
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     private func hideOverlay() {
 | |
|         guard let overlayHostingController = currentOverlayHostingController else { return }
 | |
|         
 | |
|         guard overlayHostingController.view.alpha != 0 else { return }
 | |
|         
 | |
|         UIView.animate(withDuration: 0.2) {
 | |
|             overlayHostingController.view.alpha = 0
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     private func toggleOverlay() {
 | |
|         guard let overlayHostingController = currentOverlayHostingController else { return }
 | |
|         
 | |
|         if overlayHostingController.view.alpha < 1 {
 | |
|             showOverlay()
 | |
|         } else {
 | |
|             hideOverlay()
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| // MARK: OverlayTimer
 | |
| extension VLCPlayerViewController {
 | |
|     
 | |
|     private func restartOverlayDismissTimer(interval: Double = 3) {
 | |
|         self.overlayDismissTimer?.invalidate()
 | |
|         self.overlayDismissTimer = Timer.scheduledTimer(timeInterval: interval, target: self, selector: #selector(dismissTimerFired), userInfo: nil, repeats: false)
 | |
|     }
 | |
|     
 | |
|     @objc private func dismissTimerFired() {
 | |
|         self.hideOverlay()
 | |
|     }
 | |
|     
 | |
|     private func stopOverlayDismissTimer() {
 | |
|         self.overlayDismissTimer?.invalidate()
 | |
|     }
 | |
| }
 | |
| 
 | |
| // MARK: VLCMediaPlayerDelegate
 | |
| extension VLCPlayerViewController: VLCMediaPlayerDelegate {
 | |
|     
 | |
|     
 | |
|     // MARK: mediaPlayerStateChanged
 | |
|     func mediaPlayerStateChanged(_ aNotification: Notification!) {
 | |
|         
 | |
|         // Don't show buffering if paused, usually here while scrubbing
 | |
|         if vlcMediaPlayer.state == .buffering && viewModel.playerState == .paused {
 | |
|             return
 | |
|         }
 | |
|         
 | |
|         viewModel.playerState = vlcMediaPlayer.state
 | |
|         
 | |
|         if vlcMediaPlayer.state == VLCMediaPlayerState.ended {
 | |
|             if viewModel.autoPlayNextItem && viewModel.shouldShowAutoPlayNextItem && viewModel.nextItemVideoPlayerViewModel != nil {
 | |
|                 didSelectNextItem()
 | |
|             } else {
 | |
|                 didSelectClose()
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     // MARK: mediaPlayerTimeChanged
 | |
|     func mediaPlayerTimeChanged(_ aNotification: Notification!) {
 | |
|         
 | |
|         guard !viewModel.sliderIsScrubbing else {
 | |
|             lastPlayerTicks = currentPlayerTicks
 | |
|             return
 | |
|         }
 | |
|         
 | |
|         viewModel.sliderPercentage = Double(vlcMediaPlayer.position)
 | |
|         
 | |
|         // Have to manually set playing because VLCMediaPlayer doesn't
 | |
|         // properly set it itself
 | |
|         if abs(currentPlayerTicks - lastPlayerTicks) >= 10_000 {
 | |
|             viewModel.playerState = VLCMediaPlayerState.playing
 | |
|         }
 | |
|         
 | |
|         // If needing to fix subtitle streams during playback
 | |
|         if vlcMediaPlayer.currentVideoSubTitleIndex != viewModel.selectedSubtitleStreamIndex && viewModel.subtitlesEnabled {
 | |
|             didSelectSubtitleStream(index: viewModel.selectedSubtitleStreamIndex)
 | |
|             didSelectAudioStream(index: viewModel.selectedAudioStreamIndex)
 | |
|         }
 | |
|         
 | |
|         lastPlayerTicks = currentPlayerTicks
 | |
|         
 | |
|         // Send progress report every 5 seconds
 | |
|         if abs(lastProgressReportTicks - currentPlayerTicks) >= 500_000_000 {
 | |
|             viewModel.sendProgressReport()
 | |
|             
 | |
|             lastProgressReportTicks = currentPlayerTicks
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| // MARK: PlayerOverlayDelegate
 | |
| extension VLCPlayerViewController: PlayerOverlayDelegate {
 | |
|     
 | |
|     func didSelectAudioStream(index: Int) {
 | |
|         vlcMediaPlayer.currentAudioTrackIndex = Int32(index)
 | |
|         
 | |
|         viewModel.sendProgressReport()
 | |
|         
 | |
|         lastProgressReportTicks = currentPlayerTicks
 | |
|     }
 | |
|     
 | |
|     func didSelectSubtitleStream(index: Int) {
 | |
|         if viewModel.subtitlesEnabled {
 | |
|             vlcMediaPlayer.currentVideoSubTitleIndex = Int32(index)
 | |
|         } else {
 | |
|             vlcMediaPlayer.currentVideoSubTitleIndex = -1
 | |
|         }
 | |
|         
 | |
|         viewModel.sendProgressReport()
 | |
|         
 | |
|         lastProgressReportTicks = currentPlayerTicks
 | |
|     }
 | |
|     
 | |
|     func didSelectClose() {
 | |
|         vlcMediaPlayer.stop()
 | |
|         
 | |
|         viewModel.sendStopReport()
 | |
|         
 | |
|         dismiss(animated: true, completion: nil)
 | |
|     }
 | |
|     
 | |
|     func didSelectGoogleCast() {
 | |
|         print("didSelectCast")
 | |
|     }
 | |
|     
 | |
|     func didSelectAirplay() {
 | |
|         print("didSelectAirplay")
 | |
|     }
 | |
|     
 | |
|     func didSelectCaptions() {
 | |
|         
 | |
|         viewModel.subtitlesEnabled = !viewModel.subtitlesEnabled
 | |
|         
 | |
|         if viewModel.subtitlesEnabled {
 | |
|             vlcMediaPlayer.currentVideoSubTitleIndex = Int32(viewModel.selectedSubtitleStreamIndex)
 | |
|         } else {
 | |
|             vlcMediaPlayer.currentVideoSubTitleIndex = -1
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     // TODO: Implement properly in overlays
 | |
|     func didSelectMenu() {
 | |
|         stopOverlayDismissTimer()
 | |
|     }
 | |
|     
 | |
|     // TODO: Implement properly in overlays
 | |
|     func didDeselectMenu() {
 | |
|         restartOverlayDismissTimer()
 | |
|     }
 | |
|     
 | |
|     func didSelectBackward() {
 | |
|         vlcMediaPlayer.jumpBackward(jumpBackwardLength.rawValue)
 | |
|         
 | |
|         restartOverlayDismissTimer()
 | |
|         
 | |
|         viewModel.sendProgressReport()
 | |
|         
 | |
|         self.lastProgressReportTicks = currentPlayerTicks
 | |
|     }
 | |
|     
 | |
|     func didSelectForward() {
 | |
|         vlcMediaPlayer.jumpForward(jumpForwardLength.rawValue)
 | |
|         
 | |
|         restartOverlayDismissTimer()
 | |
|         
 | |
|         viewModel.sendProgressReport()
 | |
|         
 | |
|         self.lastProgressReportTicks = currentPlayerTicks
 | |
|     }
 | |
|     
 | |
|     func didSelectMain() {
 | |
|         
 | |
|         switch viewModel.playerState {
 | |
|         case .buffering:
 | |
|             vlcMediaPlayer.play()
 | |
|             restartOverlayDismissTimer()
 | |
|         case .playing:
 | |
|             viewModel.sendPauseReport(paused: true)
 | |
|             vlcMediaPlayer.pause()
 | |
|             restartOverlayDismissTimer(interval: 5)
 | |
|         case .paused:
 | |
|             viewModel.sendPauseReport(paused: false)
 | |
|             vlcMediaPlayer.play()
 | |
|             restartOverlayDismissTimer()
 | |
|         default: ()
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     func didGenerallyTap() {
 | |
|         toggleOverlay()
 | |
|         
 | |
|         restartOverlayDismissTimer(interval: 5)
 | |
|     }
 | |
|     
 | |
|     func didBeginScrubbing() {
 | |
|         stopOverlayDismissTimer()
 | |
|     }
 | |
|     
 | |
|     func didEndScrubbing() {
 | |
|         setMediaPlayerTimeAtCurrentSlider()
 | |
|         
 | |
|         restartOverlayDismissTimer()
 | |
|         
 | |
|         viewModel.sendProgressReport()
 | |
|         
 | |
|         self.lastProgressReportTicks = currentPlayerTicks
 | |
|     }
 | |
|     
 | |
|     func didSelectPreviousItem() {
 | |
|         setupMediaPlayer(newViewModel: viewModel.previousItemVideoPlayerViewModel!)
 | |
|         startPlayback()
 | |
|     }
 | |
|     
 | |
|     func didSelectNextItem() {
 | |
|         setupMediaPlayer(newViewModel: viewModel.nextItemVideoPlayerViewModel!)
 | |
|         startPlayback()
 | |
|     }
 | |
| }
 |