diff --git a/app/build.gradle.kts b/app/build.gradle.kts index d696d8b056..f2809da72a 100644 --- a/app/build.gradle.kts +++ b/app/build.gradle.kts @@ -83,6 +83,13 @@ android { } } + /*splits { + abi { + isEnable = true + isUniversalApk = true + } + }*/ + @Suppress("UnstableApiUsage") buildFeatures { viewBinding = true @@ -94,6 +101,9 @@ android { compileOptions { isCoreLibraryDesugaringEnabled = true } + packagingOptions { + doNotStrip("**/*.so") + } lintOptions { isAbortOnError = false sarifReport = true @@ -179,6 +189,11 @@ dependencies { // Desugaring coreLibraryDesugaring(Dependencies.Health.androidDesugarLibs) + + // TODO: Decide how to build / publish this.. + implementation(files("libmpv\\app-release.aar")) + //implementation(files("libmpv\\app-sources.jar")) + //implementation(files("libmpv\\app-javadoc.jar")) } tasks { diff --git a/app/libmpv/app-javadoc.jar b/app/libmpv/app-javadoc.jar new file mode 100644 index 0000000000..95460e2396 Binary files /dev/null and b/app/libmpv/app-javadoc.jar differ diff --git a/app/libmpv/app-release.aar b/app/libmpv/app-release.aar new file mode 100644 index 0000000000..5da3042367 Binary files /dev/null and b/app/libmpv/app-release.aar differ diff --git a/app/libmpv/app-sources.jar b/app/libmpv/app-sources.jar new file mode 100644 index 0000000000..80e8d0e404 Binary files /dev/null and b/app/libmpv/app-sources.jar differ diff --git a/app/src/main/assets/mpv.conf b/app/src/main/assets/mpv.conf new file mode 100644 index 0000000000..70a08601a5 --- /dev/null +++ b/app/src/main/assets/mpv.conf @@ -0,0 +1,13 @@ +# hwdec: try to use hardware decoding +hwdec=mediacodec-copy +hwdec-codecs="h264,hevc,mpeg4,mpeg2video,vp8,vp9" +gpu-dumb-mode=auto +# tls: allow self signed certificate +tls-verify=no +tls-ca-file="" +# demuxer: limit cache to 32 MiB, the default is too high for mobile devices +demuxer-max-bytes=32MiB +demuxer-max-back-bytes=32MiB +# sub: scale subtitles with video +sub-scale-with-window=no +sub-use-margins=no diff --git a/app/src/main/assets/native/ExoPlayerPlugin.js b/app/src/main/assets/native/NativePlayerPlugin.js similarity index 95% rename from app/src/main/assets/native/ExoPlayerPlugin.js rename to app/src/main/assets/native/NativePlayerPlugin.js index 8bfa209d67..306faefd87 100644 --- a/app/src/main/assets/native/ExoPlayerPlugin.js +++ b/app/src/main/assets/native/NativePlayerPlugin.js @@ -1,4 +1,4 @@ -export class ExoPlayerPlugin { +export class NativePlayerPlugin { constructor({ events, playbackManager, loading }) { window['ExoPlayer'] = this; @@ -6,9 +6,9 @@ export class ExoPlayerPlugin { this.playbackManager = playbackManager; this.loading = loading; - this.name = 'ExoPlayer'; + this.name = 'NativePlayer'; this.type = 'mediaplayer'; - this.id = 'exoplayer'; + this.id = 'nativeplayer'; // Prioritize first this.priority = -1; @@ -164,7 +164,7 @@ export class ExoPlayerPlugin { async getDeviceProfile() { var profile = {}; - profile.Name = 'ExoPlayer Stub'; + profile.Name = 'NativePlayer Stub'; profile.MaxStreamingBitrate = 100000000; profile.MaxStaticBitrate = 100000000; profile.MusicStreamingTranscodingBitrate = 320000; diff --git a/app/src/main/assets/native/nativeshell.js b/app/src/main/assets/native/nativeshell.js index ea2b0ce909..58c6dde869 100644 --- a/app/src/main/assets/native/nativeshell.js +++ b/app/src/main/assets/native/nativeshell.js @@ -17,7 +17,7 @@ const features = [ const plugins = [ 'NavigationPlugin', - 'ExoPlayerPlugin', + 'NativePlayerPlugin', 'ExternalPlayerPlugin' ]; diff --git a/app/src/main/java/org/jellyfin/mobile/ApplicationModule.kt b/app/src/main/java/org/jellyfin/mobile/ApplicationModule.kt index 89ef66031d..0e8952d519 100644 --- a/app/src/main/java/org/jellyfin/mobile/ApplicationModule.kt +++ b/app/src/main/java/org/jellyfin/mobile/ApplicationModule.kt @@ -11,12 +11,14 @@ import kotlinx.coroutines.channels.Channel import okhttp3.OkHttpClient import org.jellyfin.mobile.api.DeviceProfileBuilder import org.jellyfin.mobile.bridge.ExternalPlayer +import org.jellyfin.mobile.bridge.NativePlayer import org.jellyfin.mobile.controller.ApiController import org.jellyfin.mobile.fragment.ConnectFragment import org.jellyfin.mobile.fragment.WebViewFragment import org.jellyfin.mobile.media.car.LibraryBrowser import org.jellyfin.mobile.player.PlayerEvent import org.jellyfin.mobile.player.PlayerFragment +import org.jellyfin.mobile.player.mpv.MPVPlayer import org.jellyfin.mobile.player.source.MediaSourceResolver import org.jellyfin.mobile.utils.Constants import org.jellyfin.mobile.utils.PermissionRequestHelper @@ -54,8 +56,9 @@ val applicationModule = module { // Media player helpers single { MediaSourceResolver(get(), get(), get()) } single { DeviceProfileBuilder() } - single { get().getDeviceProfile() } - single(named(ExternalPlayer.DEVICE_PROFILE_NAME)) { get().getExternalPlayerProfile() } + single(named(ExternalPlayer.PLAYER_NAME)) { get().getExternalPlayerProfile() } + single(named(MPVPlayer.PLAYER_NAME)) { get().getMPVPlayerProfile() } + single(named(NativePlayer.PLAYER_NAME)) { get().getExoPlayerProfile() } // ExoPlayer data sources single { DefaultDataSourceFactory(androidApplication(), Util.getUserAgent(androidApplication(), Constants.APP_INFO_NAME)) } diff --git a/app/src/main/java/org/jellyfin/mobile/api/DeviceProfileBuilder.kt b/app/src/main/java/org/jellyfin/mobile/api/DeviceProfileBuilder.kt index b8e173ecd1..92e6242a37 100644 --- a/app/src/main/java/org/jellyfin/mobile/api/DeviceProfileBuilder.kt +++ b/app/src/main/java/org/jellyfin/mobile/api/DeviceProfileBuilder.kt @@ -2,8 +2,9 @@ package org.jellyfin.mobile.api import android.media.MediaCodecList import org.jellyfin.mobile.bridge.ExternalPlayer +import org.jellyfin.mobile.bridge.NativePlayer import org.jellyfin.mobile.player.DeviceCodec -import org.jellyfin.mobile.utils.Constants +import org.jellyfin.mobile.player.mpv.MPVPlayer import org.jellyfin.sdk.model.api.CodecProfile import org.jellyfin.sdk.model.api.ContainerProfile import org.jellyfin.sdk.model.api.DeviceProfile @@ -21,7 +22,7 @@ class DeviceProfileBuilder { require(SUPPORTED_CONTAINER_FORMATS.size == AVAILABLE_VIDEO_CODECS.size && SUPPORTED_CONTAINER_FORMATS.size == AVAILABLE_AUDIO_CODECS.size) } - fun getDeviceProfile(): DeviceProfile { + fun getExoPlayerProfile(): DeviceProfile { val containerProfiles = ArrayList() val directPlayProfiles = ArrayList() val codecProfiles = ArrayList() @@ -62,7 +63,7 @@ class DeviceProfileBuilder { } return DeviceProfile( - name = Constants.APP_INFO_NAME, + name = NativePlayer.PLAYER_NAME, directPlayProfiles = directPlayProfiles, transcodingProfiles = getTranscodingProfiles(), containerProfiles = containerProfiles, @@ -83,8 +84,32 @@ class DeviceProfileBuilder { ) } + fun getMPVPlayerProfile(): DeviceProfile = DeviceProfile( + name = MPVPlayer.PLAYER_NAME, + directPlayProfiles = listOf( + DirectPlayProfile(type = DlnaProfileType.VIDEO), + DirectPlayProfile(type = DlnaProfileType.AUDIO), + ), + transcodingProfiles = getTranscodingProfiles(), + containerProfiles = emptyList(), + codecProfiles = emptyList(), + subtitleProfiles = getSubtitleProfiles(MPV_PLAYER_SUBTITLES.plus("vtt"), MPV_PLAYER_SUBTITLES), + + // TODO: remove redundant defaults after API/SDK is fixed + maxAlbumArtWidth = Int.MAX_VALUE, + maxAlbumArtHeight = Int.MAX_VALUE, + timelineOffsetSeconds = 0, + enableAlbumArtInDidl = false, + enableSingleAlbumArtLimit = false, + enableSingleSubtitleLimit = false, + requiresPlainFolders = false, + requiresPlainVideoItems = false, + enableMsMediaReceiverRegistrar = false, + ignoreTranscodeByteRangeRequests = false, + ) + fun getExternalPlayerProfile(): DeviceProfile = DeviceProfile( - name = ExternalPlayer.DEVICE_PROFILE_NAME, + name = ExternalPlayer.PLAYER_NAME, directPlayProfiles = listOf( DirectPlayProfile(type = DlnaProfileType.VIDEO), DirectPlayProfile(type = DlnaProfileType.AUDIO), @@ -294,5 +319,11 @@ class DeviceProfileBuilder { private val EXTERNAL_PLAYER_SUBTITLES = arrayOf( "ssa", "ass", "srt", "subrip", "idx", "sub", "vtt", "webvtt", "ttml", "pgs", "pgssub", "smi", "smil" ) + // https://github.com/mpv-player/mpv/blob/6857600c47f069aeb68232a745bc8f81d45c9967/player/external_files.c#L35 + private val MPV_PLAYER_SUBTITLES = arrayOf( + "idx", "sub", "srt", "rt", "ssa", "ass", "mks",/* "vtt", */"sup", "scc", "smi", "lrc", "pgs", + // https://ffmpeg.org/general.html#Subtitle-Formats + "aqt", "jss", "txt", "mpsub", "pjs", "sami", "stl" + ) } } diff --git a/app/src/main/java/org/jellyfin/mobile/bridge/ExternalPlayer.kt b/app/src/main/java/org/jellyfin/mobile/bridge/ExternalPlayer.kt index 500d508029..a211e676a4 100644 --- a/app/src/main/java/org/jellyfin/mobile/bridge/ExternalPlayer.kt +++ b/app/src/main/java/org/jellyfin/mobile/bridge/ExternalPlayer.kt @@ -45,7 +45,7 @@ class ExternalPlayer( private val appPreferences: AppPreferences by inject() private val webappFunctionChannel: WebappFunctionChannel by inject() private val mediaSourceResolver: MediaSourceResolver by inject() - private val externalPlayerProfile: DeviceProfile by inject(named(DEVICE_PROFILE_NAME)) + private val externalPlayerProfile: DeviceProfile by inject(named(PLAYER_NAME)) private val videosApi: VideosApi by inject() private val apiClient: ApiClient by inject() @@ -293,6 +293,6 @@ class ExternalPlayer( } companion object { - const val DEVICE_PROFILE_NAME = "Android External Player" + const val PLAYER_NAME = "Android External Player" } } diff --git a/app/src/main/java/org/jellyfin/mobile/bridge/NativePlayer.kt b/app/src/main/java/org/jellyfin/mobile/bridge/NativePlayer.kt index 05206a3897..2f0a7a0860 100644 --- a/app/src/main/java/org/jellyfin/mobile/bridge/NativePlayer.kt +++ b/app/src/main/java/org/jellyfin/mobile/bridge/NativePlayer.kt @@ -18,7 +18,7 @@ class NativePlayer(private val host: NativePlayerHost) : KoinComponent { private val playerEventChannel: Channel by inject(named(PLAYER_EVENT_CHANNEL)) @JavascriptInterface - fun isEnabled() = appPreferences.videoPlayerType == VideoPlayerType.EXO_PLAYER + fun isEnabled() = appPreferences.videoPlayerType in arrayOf(VideoPlayerType.EXO_PLAYER, VideoPlayerType.MPV_PLAYER) @JavascriptInterface fun loadPlayer(args: String) { @@ -61,4 +61,8 @@ class NativePlayer(private val host: NativePlayerHost) : KoinComponent { fun setVolume(volume: Int) { playerEventChannel.trySend(PlayerEvent.SetVolume(volume)) } + + companion object { + const val PLAYER_NAME = Constants.APP_INFO_NAME + } } diff --git a/app/src/main/java/org/jellyfin/mobile/player/PlayerViewModel.kt b/app/src/main/java/org/jellyfin/mobile/player/PlayerViewModel.kt index 9ed337562f..45d1a95c77 100644 --- a/app/src/main/java/org/jellyfin/mobile/player/PlayerViewModel.kt +++ b/app/src/main/java/org/jellyfin/mobile/player/PlayerViewModel.kt @@ -24,15 +24,19 @@ import kotlinx.coroutines.Job import kotlinx.coroutines.channels.Channel import kotlinx.coroutines.delay import kotlinx.coroutines.launch +import org.jellyfin.mobile.AppPreferences import org.jellyfin.mobile.BuildConfig import org.jellyfin.mobile.PLAYER_EVENT_CHANNEL +import org.jellyfin.mobile.bridge.ExternalPlayer +import org.jellyfin.mobile.bridge.NativePlayer +import org.jellyfin.mobile.player.mpv.MPVPlayer import org.jellyfin.mobile.player.source.JellyfinMediaSource import org.jellyfin.mobile.player.source.MediaQueueManager +import org.jellyfin.mobile.settings.VideoPlayerType import org.jellyfin.mobile.utils.Constants import org.jellyfin.mobile.utils.Constants.SUPPORTED_VIDEO_PLAYER_PLAYBACK_ACTIONS import org.jellyfin.mobile.utils.applyDefaultAudioAttributes import org.jellyfin.mobile.utils.applyDefaultLocalAudioAttributes -import org.jellyfin.mobile.utils.getRendererIndexByType import org.jellyfin.mobile.utils.getVolumeLevelPercent import org.jellyfin.mobile.utils.getVolumeRange import org.jellyfin.mobile.utils.scaleInRange @@ -52,6 +56,7 @@ import org.koin.core.qualifier.named import timber.log.Timber class PlayerViewModel(application: Application) : AndroidViewModel(application), KoinComponent, Player.Listener { + private val appPreferences by inject() private val playStateApi by inject() private val lifecycleObserver = PlayerLifecycleObserver(this) @@ -59,22 +64,28 @@ class PlayerViewModel(application: Application) : AndroidViewModel(application), val notificationHelper: PlayerNotificationHelper by lazy { PlayerNotificationHelper(this) } // Media source handling - val mediaQueueManager = MediaQueueManager(this) + val mediaQueueManager = MediaQueueManager( + viewModel = this, playerName = when (appPreferences.videoPlayerType) { + VideoPlayerType.EXO_PLAYER -> NativePlayer.PLAYER_NAME + VideoPlayerType.MPV_PLAYER -> MPVPlayer.PLAYER_NAME + else -> ExternalPlayer.PLAYER_NAME + } + ) val mediaSourceOrNull: JellyfinMediaSource? get() = mediaQueueManager.mediaQueue.value?.jellyfinMediaSource - // ExoPlayer - private val _player = MutableLiveData() + // Player + private val _player = MutableLiveData() private val _playerState = MutableLiveData() - val player: LiveData get() = _player + val player: LiveData get() = _player val playerState: LiveData get() = _playerState private var progressUpdateJob: Job? = null /** - * Returns the current ExoPlayer instance or null + * Returns the current Player instance or null */ - val playerOrNull: ExoPlayer? get() = _player.value + val playerOrNull: Player? get() = _player.value private val playerEventChannel: Channel by inject(named(PLAYER_EVENT_CHANNEL)) @@ -109,24 +120,37 @@ class PlayerViewModel(application: Application) : AndroidViewModel(application), } /** - * Setup a new [SimpleExoPlayer] for video playback, register callbacks and set attributes + * Setup a new [Player] for video playback, register callbacks and set attributes */ fun setupPlayer() { - _player.value = SimpleExoPlayer.Builder(getApplication()).apply { - setTrackSelector(mediaQueueManager.trackSelector) - if (BuildConfig.DEBUG) { - setAnalyticsCollector(AnalyticsCollector(Clock.DEFAULT).apply { - addListener(mediaQueueManager.eventLogger) - }) + _player.value = when (appPreferences.videoPlayerType) { + VideoPlayerType.EXO_PLAYER -> { + SimpleExoPlayer.Builder(getApplication()).apply { + setTrackSelector(mediaQueueManager.trackSelector) + if (BuildConfig.DEBUG) { + setAnalyticsCollector(AnalyticsCollector(Clock.DEFAULT).apply { + addListener(mediaQueueManager.eventLogger) + }) + } + }.build().apply { + addListener(this@PlayerViewModel) + applyDefaultAudioAttributes(C.CONTENT_TYPE_MOVIE) + } + } + VideoPlayerType.MPV_PLAYER -> { + MPVPlayer( + context = getApplication(), + requestAudioFocus = true + ).apply { + addListener(this@PlayerViewModel) + } } - }.build().apply { - addListener(this@PlayerViewModel) - applyDefaultAudioAttributes(C.CONTENT_TYPE_MOVIE) + else -> null } } /** - * Release the current ExoPlayer and stop/release the current MediaSession + * Release the current [Player] and stop/release the current [MediaSession] */ private fun releasePlayer() { notificationHelper.dismissNotification() @@ -141,7 +165,12 @@ class PlayerViewModel(application: Application) : AndroidViewModel(application), fun play(queueItem: MediaQueueManager.QueueItem.Loaded) { val player = playerOrNull ?: return - player.setMediaSource(queueItem.exoMediaSource) + if (player is ExoPlayer) { + player.setMediaSource(queueItem.exoMediaSource) + } + if (player is MPVPlayer) { + player.setMediaItem(queueItem.exoMediaSource.mediaItem) + } player.prepare() val startTime = queueItem.jellyfinMediaSource.startTimeMs if (startTime > 0) { @@ -325,10 +354,6 @@ class PlayerViewModel(application: Application) : AndroidViewModel(application), audioManager.setStreamVolume(stream, scaled, 0) } - fun getPlayerRendererIndex(type: Int): Int { - return playerOrNull?.getRendererIndexByType(type) ?: -1 - } - @SuppressLint("SwitchIntDef") override fun onPlayerStateChanged(playWhenReady: Boolean, playbackState: Int) { val player = playerOrNull ?: return diff --git a/app/src/main/java/org/jellyfin/mobile/player/mpv/MPVPlayer.kt b/app/src/main/java/org/jellyfin/mobile/player/mpv/MPVPlayer.kt new file mode 100644 index 0000000000..c13c6db00d --- /dev/null +++ b/app/src/main/java/org/jellyfin/mobile/player/mpv/MPVPlayer.kt @@ -0,0 +1,1470 @@ +package org.jellyfin.mobile.player.mpv + +import `is`.xyz.libmpv.MPVLib +import android.annotation.SuppressLint +import android.app.Application +import android.content.Context +import android.content.res.AssetManager +import android.media.AudioManager +import android.os.Build +import android.os.Handler +import android.os.Looper +import android.os.Parcelable +import android.view.Surface +import android.view.SurfaceHolder +import android.view.SurfaceView +import android.view.TextureView +import android.view.WindowManager +import androidx.core.content.getSystemService +import com.google.android.exoplayer2.BasePlayer +import com.google.android.exoplayer2.C +import com.google.android.exoplayer2.ExoPlaybackException +import com.google.android.exoplayer2.Format +import com.google.android.exoplayer2.MediaItem +import com.google.android.exoplayer2.MediaMetadata +import com.google.android.exoplayer2.PlaybackParameters +import com.google.android.exoplayer2.Player +import com.google.android.exoplayer2.Timeline +import com.google.android.exoplayer2.audio.AudioAttributes +import com.google.android.exoplayer2.device.DeviceInfo +import com.google.android.exoplayer2.metadata.Metadata +import com.google.android.exoplayer2.source.MediaSource +import com.google.android.exoplayer2.source.ProgressiveMediaSource +import com.google.android.exoplayer2.source.TrackGroup +import com.google.android.exoplayer2.source.TrackGroupArray +import com.google.android.exoplayer2.text.Cue +import com.google.android.exoplayer2.trackselection.TrackSelection +import com.google.android.exoplayer2.trackselection.TrackSelectionArray +import com.google.android.exoplayer2.upstream.DataSource +import com.google.android.exoplayer2.util.Clock +import com.google.android.exoplayer2.util.ExoFlags +import com.google.android.exoplayer2.util.ListenerSet +import com.google.android.exoplayer2.util.MimeTypes +import com.google.android.exoplayer2.util.Util +import com.google.android.exoplayer2.video.VideoSize +import kotlinx.parcelize.Parcelize +import org.json.JSONArray +import org.json.JSONException +import org.json.JSONObject +import java.io.File +import java.io.FileOutputStream +import java.util.concurrent.CopyOnWriteArraySet + +@Suppress("SpellCheckingInspection") +class MPVPlayer( + context: Context, + requestAudioFocus: Boolean +) : BasePlayer(), MPVLib.EventObserver, AudioManager.OnAudioFocusChangeListener { + + private val audioManager: AudioManager by lazy { context.getSystemService()!! } + private var audioFocusCallback: () -> Unit = {} + private var audioFocusRequest = AudioManager.AUDIOFOCUS_REQUEST_FAILED + private val handler = Handler(context.mainLooper) + + init { + require(context is Application) + @Suppress("DEPRECATION") + val mpvDir = File(context.getExternalFilesDir(null) ?: context.filesDir, "mpv") + if (!mpvDir.exists()) { + mpvDir.mkdirs() + } + // https://github.com/mpv-android/mpv-android/commit/12d4d78 + arrayOf("mpv.conf", "subfont.ttf"/*, "cacert.pem"*/).forEach { fileName -> + val file = File(mpvDir, fileName) + if (!file.exists()) { + context.assets.open(fileName, AssetManager.ACCESS_STREAMING).copyTo(FileOutputStream(file)) + } + } + MPVLib.create(context) + MPVLib.setOptionString("config", "yes") + MPVLib.setOptionString("config-dir", mpvDir.path) + // vo: set display fps as reported by android + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { + val wm = context.getSystemService(Context.WINDOW_SERVICE) as WindowManager + + @Suppress("DEPRECATION") + val display = wm.defaultDisplay + val refreshRate = display.mode.refreshRate + MPVLib.setOptionString("override-display-fps", "$refreshRate") + MPVLib.setOptionString("vo-null-fps", "$refreshRate") + } + MPVLib.setOptionString("vo", "gpu") + MPVLib.setOptionString("gpu-context", "android") + MPVLib.setOptionString("ao", "audiotrack,opensles") + + MPVLib.init() + + // hardcoded options + MPVLib.setOptionString("cache", "yes") + MPVLib.setOptionString("cache-pause-initial", "yes") + MPVLib.setOptionString("force-window", "no") + MPVLib.setOptionString("keep-open", "always") + MPVLib.setOptionString("save-position-on-quit", "no") + MPVLib.setOptionString("sub-font-provider", "none") + MPVLib.setOptionString("ytdl", "no") + + MPVLib.addObserver(this) + + // Observe properties + data class Property(val name: String, @MPVLib.Format val format: Int) + arrayOf( + Property("track-list", MPVLib.MPV_FORMAT_STRING), + Property("paused-for-cache", MPVLib.MPV_FORMAT_FLAG), + Property("eof-reached", MPVLib.MPV_FORMAT_FLAG), + Property("seekable", MPVLib.MPV_FORMAT_FLAG), + Property("time-pos", MPVLib.MPV_FORMAT_INT64), + Property("duration", MPVLib.MPV_FORMAT_INT64), + Property("demuxer-cache-time", MPVLib.MPV_FORMAT_INT64), + Property("speed", MPVLib.MPV_FORMAT_DOUBLE) + ).forEach { (name, format) -> + MPVLib.observeProperty(name, format) + } + + if (requestAudioFocus) { + @Suppress("DEPRECATION") + audioFocusRequest = audioManager.requestAudioFocus( + /* listener= */ this, + /* streamType= */ AudioManager.STREAM_MUSIC, + /* durationHint= */ AudioManager.AUDIOFOCUS_GAIN + ) + if (audioFocusRequest != AudioManager.AUDIOFOCUS_REQUEST_GRANTED) { + MPVLib.setPropertyBoolean("pause", true) + } + } + } + + // Listeners and notification. + @Suppress("DEPRECATION") + private val listeners: ListenerSet = ListenerSet( + context.mainLooper, + Clock.DEFAULT + ) { listener: Player.EventListener, flags: ExoFlags -> + listener.onEvents( /* player= */this, Player.Events(flags)) + } + @Suppress("DEPRECATION") + private val videoListeners = CopyOnWriteArraySet() + + // Internal state. + private var internalMediaItems: List? = null + private var internalMediaItem: MediaItem? = null + @Player.State + private var playbackState: Int = Player.STATE_IDLE + private var currentPlayWhenReady: Boolean = false + @Player.RepeatMode + private val repeatMode: Int = REPEAT_MODE_OFF + private var trackGroupArray: TrackGroupArray = TrackGroupArray.EMPTY + private var trackSelectionArray: TrackSelectionArray = TrackSelectionArray() + private var playbackParameters: PlaybackParameters = PlaybackParameters.DEFAULT + + // MPV Custom + private var isPlayerReady: Boolean = false + private var isSeekable: Boolean = false + private var currentPositionMs: Long? = null + private var currentDurationMs: Long? = null + private var currentCacheDurationMs: Long? = null + private var currentTracks: List = emptyList() + private var initialCommands = mutableListOf>() + private var initialSeekTo: Long = 0L + + // mpv events + override fun eventProperty(property: String) { + // Nothing to do... + } + + override fun eventProperty(property: String, value: String) { + handler.post { + when (property) { + "track-list" -> { + val (tracks, newTrackGroupArray, newTrackSelectionArray) = getMPVTracks(value) + currentTracks = tracks + if (isPlayerReady) { + if (newTrackGroupArray != trackGroupArray || newTrackSelectionArray != trackSelectionArray) { + trackGroupArray = newTrackGroupArray + trackSelectionArray = newTrackSelectionArray + listeners.sendEvent(Player.EVENT_TRACKS_CHANGED) { listener -> + listener.onTracksChanged(currentTrackGroups, currentTrackSelections) + } + } + } else { + trackGroupArray = newTrackGroupArray + trackSelectionArray = newTrackSelectionArray + } + } + } + } + } + + override fun eventProperty(property: String, value: Boolean) { + handler.post { + when (property) { + "eof-reached" -> { + if (value && isPlayerReady) { + setPlayerStateAndNotifyIfChanged( + playWhenReady = false, + playWhenReadyChangeReason = Player.PLAY_WHEN_READY_CHANGE_REASON_END_OF_MEDIA_ITEM, + playbackState = Player.STATE_ENDED + ) + resetInternalState() + } + } + "paused-for-cache" -> { + if (isPlayerReady) { + if (value) { + setPlayerStateAndNotifyIfChanged(playbackState = Player.STATE_BUFFERING) + } else { + setPlayerStateAndNotifyIfChanged(playbackState = Player.STATE_READY) + } + } + } + "seekable" -> { + if (isSeekable != value) { + isSeekable = value + listeners.sendEvent(Player.EVENT_TIMELINE_CHANGED) { listener -> + listener.onTimelineChanged(timeline, Player.TIMELINE_CHANGE_REASON_SOURCE_UPDATE) + } + } + } + } + } + } + + override fun eventProperty(property: String, value: Long) { + handler.post { + when (property) { + "time-pos" -> currentPositionMs = value * C.MILLIS_PER_SECOND + "duration" -> { + if (currentDurationMs != value * C.MILLIS_PER_SECOND) { + currentDurationMs = value * C.MILLIS_PER_SECOND + listeners.sendEvent(Player.EVENT_TIMELINE_CHANGED) { listener -> + listener.onTimelineChanged(timeline, Player.TIMELINE_CHANGE_REASON_SOURCE_UPDATE) + } + } + } + "demuxer-cache-time" -> currentCacheDurationMs = value * C.MILLIS_PER_SECOND + } + } + } + + override fun eventProperty(property: String, value: Double) { + handler.post { + when (property) { + "speed" -> { + playbackParameters = getPlaybackParameters().withSpeed(value.toFloat()) + listeners.sendEvent(Player.EVENT_PLAYBACK_PARAMETERS_CHANGED) {listener -> + listener.onPlaybackParametersChanged(getPlaybackParameters()) + } + } + } + } + } + + @SuppressLint("SwitchIntDef") + override fun event(@MPVLib.Event eventId: Int) { + handler.post { + when (eventId) { + MPVLib.MPV_EVENT_START_FILE -> { + if (!isPlayerReady) { + for (command in initialCommands) { + MPVLib.command(command) + } + } + } + MPVLib.MPV_EVENT_SEEK -> { + setPlayerStateAndNotifyIfChanged(playbackState = Player.STATE_BUFFERING) + listeners.sendEvent(Player.EVENT_POSITION_DISCONTINUITY) { listener -> + @Suppress("DEPRECATION") + listener.onPositionDiscontinuity(Player.DISCONTINUITY_REASON_SEEK) + } + } + MPVLib.MPV_EVENT_PLAYBACK_RESTART -> { + if (!isPlayerReady) { + isPlayerReady = true + listeners.sendEvent(Player.EVENT_TRACKS_CHANGED) { listener -> + listener.onTracksChanged(currentTrackGroups, currentTrackSelections) + } + seekTo(C.TIME_UNSET) + if (playWhenReady) { + MPVLib.setPropertyBoolean("pause", false) + } + for (videoListener in videoListeners) { + videoListener.onRenderedFirstFrame() + } + } else { + if (playbackState == Player.STATE_BUFFERING && bufferedPosition > currentPosition) { + setPlayerStateAndNotifyIfChanged(playbackState = Player.STATE_READY) + } + } + } + } + } + } + + override fun eventEndFile(@MPVLib.Reason reason: Int, @MPVLib.Error error: Int) { + // Nothing to do... + } + + private fun setPlayerStateAndNotifyIfChanged( + playWhenReady: Boolean = getPlayWhenReady(), + @Player.PlayWhenReadyChangeReason playWhenReadyChangeReason: Int = Player.PLAY_WHEN_READY_CHANGE_REASON_USER_REQUEST, + @Player.State playbackState: Int = getPlaybackState() + ) { + var playerStateChanged = false + val wasPlaying = isPlaying + if (playbackState != getPlaybackState()) { + this.playbackState = playbackState + listeners.queueEvent(Player.EVENT_PLAYBACK_STATE_CHANGED) { listener -> + listener.onPlaybackStateChanged(playbackState) + } + playerStateChanged = true + } + if (playWhenReady != getPlayWhenReady()) { + this.currentPlayWhenReady = playWhenReady + listeners.queueEvent(Player.EVENT_PLAY_WHEN_READY_CHANGED) { listener -> + listener.onPlayWhenReadyChanged(playWhenReady, playWhenReadyChangeReason) + } + playerStateChanged = true + } + if (playerStateChanged) { + listeners.queueEvent( /* eventFlag= */ C.INDEX_UNSET) { listener -> + @Suppress("DEPRECATION") + listener.onPlayerStateChanged(playWhenReady, playbackState) + } + } + if (wasPlaying != isPlaying) { + listeners.queueEvent(Player.EVENT_IS_PLAYING_CHANGED) { listener -> + listener.onIsPlayingChanged(isPlaying) + } + } + listeners.flushEvents() + } + + /** + * Select a [Track] or disable a [TrackType] in the current player. + * + * @param trackType The [TrackType] + * @param isExternal If track is external or embed in media + * @param index Index to select or [C.INDEX_UNSET] to disable [TrackType] + * @return true if the track is or was already selected + */ + fun selectTrack(@TrackType trackType: String, isExternal: Boolean = false, index: Int): Boolean { + if (index != C.INDEX_UNSET) { + currentTracks.firstOrNull { + it.type == trackType && (if (isExternal) it.title else "${it.ffIndex}") == "$index" + }.let { track -> + if (track != null) { + if (!track.selected) { + MPVLib.setPropertyInt(trackType, track.id) + } + } else { + return false + } + } + } else { + if (currentTracks.indexOfFirst { it.type == trackType && it.selected } != C.INDEX_UNSET) { + MPVLib.setPropertyString(trackType, "no") + } + } + return true + } + + // Timeline wrapper + private val timeline: Timeline = object : Timeline() { + /** + * Returns the number of windows in the timeline. + */ + override fun getWindowCount(): Int { + return 1 + } + + /** + * Populates a [com.google.android.exoplayer2.Timeline.Window] with data for the window at the specified index. + * + * @param windowIndex The index of the window. + * @param window The [com.google.android.exoplayer2.Timeline.Window] to populate. Must not be null. + * @param defaultPositionProjectionUs A duration into the future that the populated window's + * default start position should be projected. + * @return The populated [com.google.android.exoplayer2.Timeline.Window], for convenience. + */ + override fun getWindow(windowIndex: Int, window: Window, defaultPositionProjectionUs: Long): Window { + val currentMediaItem = internalMediaItem ?: MediaItem.Builder().build() + return if (windowIndex == 0) window.set( + /* uid= */ 0, + /* mediaItem= */ currentMediaItem, + /* manifest= */ null, + /* presentationStartTimeMs= */ C.TIME_UNSET, + /* windowStartTimeMs= */ C.TIME_UNSET, + /* elapsedRealtimeEpochOffsetMs= */ C.TIME_UNSET, + /* isSeekable= */ isSeekable, + /* isDynamic= */ !isSeekable, + /* liveConfiguration= */ currentMediaItem.liveConfiguration, + /* defaultPositionUs= */ C.TIME_UNSET, + /* durationUs= */ C.msToUs(currentDurationMs ?: C.TIME_UNSET), + /* firstPeriodIndex= */ windowIndex, + /* lastPeriodIndex= */ windowIndex, + /* positionInFirstPeriodUs= */ C.TIME_UNSET + ) else window + } + + /** + * Returns the number of periods in the timeline. + */ + override fun getPeriodCount(): Int { + return 1 + } + + /** + * Populates a [com.google.android.exoplayer2.Timeline.Period] with data for the period at the specified index. + * + * @param periodIndex The index of the period. + * @param period The [com.google.android.exoplayer2.Timeline.Period] to populate. Must not be null. + * @param setIds Whether [com.google.android.exoplayer2.Timeline.Period.id] and [com.google.android.exoplayer2.Timeline.Period.uid] should be populated. If false, + * the fields will be set to null. The caller should pass false for efficiency reasons unless + * the fields are required. + * @return The populated [com.google.android.exoplayer2.Timeline.Period], for convenience. + */ + override fun getPeriod(periodIndex: Int, period: Period, setIds: Boolean): Period { + return if (periodIndex == 0) period.set( + /* id= */ 0, + /* uid= */ 0, + /* windowIndex= */ periodIndex, + /* durationUs= */ C.msToUs(currentDurationMs ?: C.TIME_UNSET), + /* positionInWindowUs= */ 0 + ) else period + } + + /** + * Returns the index of the period identified by its unique [com.google.android.exoplayer2.Timeline.Period.uid], or [ ][C.INDEX_UNSET] if the period is not in the timeline. + * + * @param uid A unique identifier for a period. + * @return The index of the period, or [C.INDEX_UNSET] if the period was not found. + */ + override fun getIndexOfPeriod(uid: Any): Int { + return if (uid == 0) 0 else C.INDEX_UNSET + } + + /** + * Returns the unique id of the period identified by its index in the timeline. + * + * @param periodIndex The index of the period. + * @return The unique id of the period. + */ + override fun getUidOfPeriod(periodIndex: Int): Any { + return if (periodIndex == 0) 0 else C.INDEX_UNSET + } + } + + // OnAudioFocusChangeListener implementation. + + /** + * Called on the listener to notify it the audio focus for this listener has been changed. + * The focusChange value indicates whether the focus was gained, + * whether the focus was lost, and whether that loss is transient, or whether the new focus + * holder will hold it for an unknown amount of time. + * When losing focus, listeners can use the focus change information to decide what + * behavior to adopt when losing focus. A music player could for instance elect to lower + * the volume of its music stream (duck) for transient focus losses, and pause otherwise. + * @param focusChange the type of focus change, one of [AudioManager.AUDIOFOCUS_GAIN], + * [AudioManager.AUDIOFOCUS_LOSS], [AudioManager.AUDIOFOCUS_LOSS_TRANSIENT] + * and [AudioManager.AUDIOFOCUS_LOSS_TRANSIENT_CAN_DUCK]. + */ + override fun onAudioFocusChange(focusChange: Int) { + when (focusChange) { + AudioManager.AUDIOFOCUS_LOSS, + AudioManager.AUDIOFOCUS_LOSS_TRANSIENT -> { + val oldAudioFocusCallback = audioFocusCallback + val wasPlaying = isPlaying + MPVLib.setPropertyBoolean("pause", true) + setPlayerStateAndNotifyIfChanged( + playWhenReady = false, + playWhenReadyChangeReason = Player.PLAY_WHEN_READY_CHANGE_REASON_AUDIO_FOCUS_LOSS + ) + audioFocusCallback = { + oldAudioFocusCallback() + if (wasPlaying) MPVLib.setPropertyBoolean("pause", false) + } + } + AudioManager.AUDIOFOCUS_LOSS_TRANSIENT_CAN_DUCK -> { + MPVLib.command(arrayOf("multiply", "volume", "$AUDIO_FOCUS_DUCKING")) + audioFocusCallback = { + MPVLib.command(arrayOf("multiply", "volume", "${1f/AUDIO_FOCUS_DUCKING}")) + } + } + AudioManager.AUDIOFOCUS_GAIN -> { + audioFocusCallback() + audioFocusCallback = {} + } + } + } + + // Player implementation. + + /** + * Returns the [Looper] associated with the application thread that's used to access the + * player and on which player events are received. + */ + override fun getApplicationLooper(): Looper { + return handler.looper + } + + /** + * Registers a listener to receive events from the player. The listener's methods will be called + * on the thread that was used to construct the player. However, if the thread used to construct + * the player does not have a [Looper], then the listener will be called on the main thread. + * + * @param listener The listener to register. + */ + @Suppress("DEPRECATION") + override fun addListener(listener: Player.EventListener) { + listeners.add(listener) + } + + /** + * Registers a listener to receive all events from the player. + * + * @param listener The listener to register. + */ + override fun addListener(listener: Player.Listener) { + listeners.add(listener) + videoListeners.add(listener) + } + + /** + * Unregister a listener registered through [.addListener]. The listener will + * no longer receive events from the player. + * + * @param listener The listener to unregister. + */ + @Suppress("DEPRECATION") + override fun removeListener(listener: Player.EventListener) { + listeners.remove(listener) + } + + /** + * Unregister a listener registered through [.addListener]. The listener will no + * longer receive events. + * + * @param listener The listener to unregister. + */ + override fun removeListener(listener: Player.Listener) { + listeners.remove(listener) + videoListeners.remove(listener) + } + + /** + * Clears the playlist and adds the specified [MediaItems][MediaItem]. + * + * @param mediaItems The new [MediaItems][MediaItem]. + * @param resetPosition Whether the playback position should be reset to the default position in + * the first [Timeline.Window]. If false, playback will start from the position defined + * by [.getCurrentWindowIndex] and [.getCurrentPosition]. + */ + override fun setMediaItems(mediaItems: MutableList, resetPosition: Boolean) { + internalMediaItems = mediaItems + } + + /** + * Clears the playlist and adds the specified [MediaItems][MediaItem]. + * + * @param mediaItems The new [MediaItems][MediaItem]. + * @param startWindowIndex The window index to start playback from. If [com.google.android.exoplayer2.C.INDEX_UNSET] is + * passed, the current position is not reset. + * @param startPositionMs The position in milliseconds to start playback from. If [ ][com.google.android.exoplayer2.C.TIME_UNSET] is passed, the default position of the given window is used. In any case, if + * `startWindowIndex` is set to [com.google.android.exoplayer2.C.INDEX_UNSET], this parameter is ignored and the + * position is not reset at all. + * @throws com.google.android.exoplayer2.IllegalSeekPositionException If the provided `startWindowIndex` is not within the + * bounds of the list of media items. + */ + override fun setMediaItems(mediaItems: MutableList, startWindowIndex: Int, startPositionMs: Long) { + TODO("Not yet implemented") + } + + /** + * Adds a list of media items at the given index of the playlist. + * + * @param index The index at which to add the media items. If the index is larger than the size of + * the playlist, the media items are added to the end of the playlist. + * @param mediaItems The [MediaItems][MediaItem] to add. + */ + override fun addMediaItems(index: Int, mediaItems: MutableList) { + TODO("Not yet implemented") + } + + /** + * Moves the media item range to the new index. + * + * @param fromIndex The start of the range to move. + * @param toIndex The first item not to be included in the range (exclusive). + * @param newIndex The new index of the first media item of the range. If the new index is larger + * than the size of the remaining playlist after removing the range, the range is moved to the + * end of the playlist. + */ + override fun moveMediaItems(fromIndex: Int, toIndex: Int, newIndex: Int) { + TODO("Not yet implemented") + } + + /** + * Removes a range of media items from the playlist. + * + * @param fromIndex The index at which to start removing media items. + * @param toIndex The index of the first item to be kept (exclusive). If the index is larger than + * the size of the playlist, media items to the end of the playlist are removed. + */ + override fun removeMediaItems(fromIndex: Int, toIndex: Int) { + TODO("Not yet implemented") + } + + /** + * Returns the player's currently available [com.google.android.exoplayer2.Player.Commands]. + * + * + * The returned [com.google.android.exoplayer2.Player.Commands] are not updated when available commands change. Use [ ][com.google.android.exoplayer2.Player.Listener.onAvailableCommandsChanged] to get an update when the available commands + * change. + * + * + * Executing a command that is not available (for example, calling [.next] if [ ][.COMMAND_SEEK_TO_NEXT_MEDIA_ITEM] is unavailable) will neither throw an exception nor generate + * a [.getPlayerError] player error}. + * + * + * [.COMMAND_SEEK_TO_NEXT_MEDIA_ITEM] and [.COMMAND_SEEK_TO_PREVIOUS_MEDIA_ITEM] + * are unavailable if there is no such [MediaItem]. + * + * @return The currently available [com.google.android.exoplayer2.Player.Commands]. + * @see com.google.android.exoplayer2.Player.Listener.onAvailableCommandsChanged + */ + override fun getAvailableCommands(): Player.Commands { + return permanentAvailableCommands + } + + private fun resetInternalState() { + isPlayerReady = false + isSeekable = false + playbackState = Player.STATE_IDLE + currentPlayWhenReady = false + currentPositionMs = null + currentDurationMs = null + currentCacheDurationMs = null + trackGroupArray = TrackGroupArray.EMPTY + trackSelectionArray = TrackSelectionArray() + playbackParameters = PlaybackParameters.DEFAULT + initialCommands.clear() + initialSeekTo = 0L + } + + /** Prepares the player. */ + override fun prepare() { + internalMediaItems?.firstOrNull { it.playbackProperties?.uri != null }?.let { mediaItem -> + internalMediaItem = mediaItem + resetInternalState() + mediaItem.playbackProperties?.subtitles?.forEach { subtitle -> + initialCommands.add(arrayOf( + /* command= */ "sub-add", + /* url= */ "${subtitle.uri}", + /* flags= */ "auto", + /* title= */ "${subtitle.label}", + /* lang= */ "${subtitle.language}" + )) + } + MPVLib.command(arrayOf("loadfile", "${mediaItem.playbackProperties?.uri}")) + MPVLib.setPropertyBoolean("pause", true) + listeners.sendEvent(Player.EVENT_TIMELINE_CHANGED) { listener -> + listener.onTimelineChanged(timeline, Player.TIMELINE_CHANGE_REASON_PLAYLIST_CHANGED) + } + listeners.sendEvent(Player.EVENT_MEDIA_ITEM_TRANSITION) { listener -> + listener.onMediaItemTransition(mediaItem, Player.MEDIA_ITEM_TRANSITION_REASON_PLAYLIST_CHANGED) + } + setPlayerStateAndNotifyIfChanged(playbackState = Player.STATE_BUFFERING) + } + } + + /** + * Returns the current [playback state][com.google.android.exoplayer2.Player.State] of the player. + * + * @return The current [playback state][com.google.android.exoplayer2.Player.State]. + * @see com.google.android.exoplayer2.Player.Listener.onPlaybackStateChanged + */ + override fun getPlaybackState(): Int { + return playbackState + } + + /** + * Returns the reason why playback is suppressed even though [.getPlayWhenReady] is `true`, or [.PLAYBACK_SUPPRESSION_REASON_NONE] if playback is not suppressed. + * + * @return The current [playback suppression reason][com.google.android.exoplayer2.Player.PlaybackSuppressionReason]. + * @see com.google.android.exoplayer2.Player.Listener.onPlaybackSuppressionReasonChanged + */ + override fun getPlaybackSuppressionReason(): Int { + return PLAYBACK_SUPPRESSION_REASON_NONE + } + + /** + * Returns the error that caused playback to fail. This is the same error that will have been + * reported via [com.google.android.exoplayer2.Player.Listener.onPlayerError] at the time of failure. It + * can be queried using this method until the player is re-prepared. + * + * + * Note that this method will always return `null` if [.getPlaybackState] is not + * [.STATE_IDLE]. + * + * @return The error, or `null`. + * @see com.google.android.exoplayer2.Player.Listener.onPlayerError + */ + override fun getPlayerError(): ExoPlaybackException? { + return null + } + + /** + * Sets whether playback should proceed when [.getPlaybackState] == [.STATE_READY]. + * + * + * If the player is already in the ready state then this method pauses and resumes playback. + * + * @param playWhenReady Whether playback should proceed when ready. + */ + override fun setPlayWhenReady(playWhenReady: Boolean) { + if (currentPlayWhenReady != playWhenReady) { + setPlayerStateAndNotifyIfChanged( + playWhenReady = playWhenReady, + playWhenReadyChangeReason = Player.PLAY_WHEN_READY_CHANGE_REASON_USER_REQUEST + ) + if (isPlayerReady) { + MPVLib.setPropertyBoolean("pause", !playWhenReady) + } + } + } + + /** + * Whether playback will proceed when [.getPlaybackState] == [.STATE_READY]. + * + * @return Whether playback will proceed when ready. + * @see com.google.android.exoplayer2.Player.Listener.onPlayWhenReadyChanged + */ + override fun getPlayWhenReady(): Boolean { + return currentPlayWhenReady + } + + /** + * Sets the [com.google.android.exoplayer2.Player.RepeatMode] to be used for playback. + * + * @param repeatMode The repeat mode. + */ + override fun setRepeatMode(repeatMode: Int) { + TODO("Not yet implemented") + } + + /** + * Returns the current [com.google.android.exoplayer2.Player.RepeatMode] used for playback. + * + * @return The current repeat mode. + * @see com.google.android.exoplayer2.Player.Listener.onRepeatModeChanged + */ + override fun getRepeatMode(): Int { + return repeatMode + } + + /** + * Sets whether shuffling of windows is enabled. + * + * @param shuffleModeEnabled Whether shuffling is enabled. + */ + override fun setShuffleModeEnabled(shuffleModeEnabled: Boolean) { + TODO("Not yet implemented") + } + + /** + * Returns whether shuffling of windows is enabled. + * + * @see com.google.android.exoplayer2.Player.Listener.onShuffleModeEnabledChanged + */ + override fun getShuffleModeEnabled(): Boolean { + return false + } + + /** + * Whether the player is currently loading the source. + * + * @return Whether the player is currently loading the source. + * @see com.google.android.exoplayer2.Player.Listener.onIsLoadingChanged + */ + override fun isLoading(): Boolean { + return false + } + + /** + * Seeks to a position specified in milliseconds in the specified window. + * + * @param windowIndex The index of the window. + * @param positionMs The seek position in the specified window, or [com.google.android.exoplayer2.C.TIME_UNSET] to seek to + * the window's default position. + * @throws com.google.android.exoplayer2.IllegalSeekPositionException If the player has a non-empty timeline and the provided + * `windowIndex` is not within the bounds of the current timeline. + */ + override fun seekTo(windowIndex: Int, positionMs: Long) { + if (windowIndex == 0) { + val seekTo = if (positionMs != C.TIME_UNSET) positionMs / C.MILLIS_PER_SECOND else initialSeekTo + if (isPlayerReady) { + MPVLib.command(arrayOf("seek", "$seekTo", "absolute")) + } else { + initialSeekTo = seekTo + } + } + } + + /** + * Attempts to set the playback parameters. Passing [PlaybackParameters.DEFAULT] resets the + * player to the default, which means there is no speed or pitch adjustment. + * + * + * Playback parameters changes may cause the player to buffer. [ ][com.google.android.exoplayer2.Player.Listener.onPlaybackParametersChanged] will be called whenever the currently + * active playback parameters change. + * + * @param playbackParameters The playback parameters. + */ + override fun setPlaybackParameters(playbackParameters: PlaybackParameters) { + if (getPlaybackParameters().speed != playbackParameters.speed) { + MPVLib.setPropertyDouble("speed", playbackParameters.speed.toDouble()) + } + } + + /** + * Returns the currently active playback parameters. + * + * @see com.google.android.exoplayer2.Player.Listener.onPlaybackParametersChanged + */ + override fun getPlaybackParameters(): PlaybackParameters { + return playbackParameters + } + + override fun stop(reset: Boolean) { + MPVLib.command(arrayOf("stop", "keep-playlist")) + } + + /** + * Releases the player. This method must be called when the player is no longer required. The + * player must not be used after calling this method. + */ + override fun release() { + if (audioFocusRequest == AudioManager.AUDIOFOCUS_REQUEST_GRANTED) { + @Suppress("DEPRECATION") + audioManager.abandonAudioFocus(this) + } + resetInternalState() + MPVLib.destroy() + } + + /** + * Returns the available track groups. + * + * @see com.google.android.exoplayer2.Player.Listener.onTracksChanged + */ + override fun getCurrentTrackGroups(): TrackGroupArray { + return trackGroupArray + } + + /** + * Returns the current track selections. + * + * + * A concrete implementation may include null elements if it has a fixed number of renderer + * components, wishes to report a TrackSelection for each of them, and has one or more renderer + * components that is not assigned any selected tracks. + * + * @see com.google.android.exoplayer2.Player.Listener.onTracksChanged + */ + override fun getCurrentTrackSelections(): TrackSelectionArray { + return trackSelectionArray + } + + /** + * Returns the current static metadata for the track selections. + * + * + * The returned `metadataList` is an immutable list of [Metadata] instances, where + * the elements correspond to the [current track selections][.getCurrentTrackSelections], + * or an empty list if there are no track selections or the selected tracks contain no static + * metadata. + * + * + * This metadata is considered static in that it comes from the tracks' declared Formats, + * rather than being timed (or dynamic) metadata, which is represented within a metadata track. + * + * @see com.google.android.exoplayer2.Player.Listener.onStaticMetadataChanged + */ + override fun getCurrentStaticMetadata(): List { + return emptyList() + } + + /** + * Returns the current combined [MediaMetadata], or [MediaMetadata.EMPTY] if not + * supported. + * + * + * This [MediaMetadata] is a combination of the [MediaItem.mediaMetadata] and the + * static and dynamic metadata sourced from [com.google.android.exoplayer2.Player.Listener.onStaticMetadataChanged] and + * [com.google.android.exoplayer2.metadata.MetadataOutput.onMetadata]. + */ + override fun getMediaMetadata(): MediaMetadata { + return MediaMetadata.EMPTY + } + + /** + * Returns the current [Timeline]. Never null, but may be empty. + * + * @see com.google.android.exoplayer2.Player.Listener.onTimelineChanged + */ + override fun getCurrentTimeline(): Timeline { + return timeline + } + + /** Returns the index of the period currently being played. */ + override fun getCurrentPeriodIndex(): Int { + return currentWindowIndex + } + + /** + * Returns the index of the current [window][Timeline.Window] in the [ ][.getCurrentTimeline], or the prospective window index if the [ ][.getCurrentTimeline] is empty. + */ + override fun getCurrentWindowIndex(): Int { + return timeline.getFirstWindowIndex(shuffleModeEnabled) + } + + /** + * Returns the duration of the current content window or ad in milliseconds, or [ ][com.google.android.exoplayer2.C.TIME_UNSET] if the duration is not known. + */ + override fun getDuration(): Long { + return timeline.getWindow(currentWindowIndex, window).durationMs + } + + /** + * Returns the playback position in the current content window or ad, in milliseconds, or the + * prospective position in milliseconds if the [current timeline][.getCurrentTimeline] is + * empty. + */ + override fun getCurrentPosition(): Long { + return currentPositionMs ?: C.TIME_UNSET + } + + /** + * Returns an estimate of the position in the current content window or ad up to which data is + * buffered, in milliseconds. + */ + override fun getBufferedPosition(): Long { + return currentCacheDurationMs ?: contentPosition + } + + /** + * Returns an estimate of the total buffered duration from the current position, in milliseconds. + * This includes pre-buffered data for subsequent ads and windows. + */ + override fun getTotalBufferedDuration(): Long { + return bufferedPosition + } + + /** Returns whether the player is currently playing an ad. */ + override fun isPlayingAd(): Boolean { + return false + } + + /** + * If [.isPlayingAd] returns true, returns the index of the ad group in the period + * currently being played. Returns [com.google.android.exoplayer2.C.INDEX_UNSET] otherwise. + */ + override fun getCurrentAdGroupIndex(): Int { + return C.INDEX_UNSET + } + + /** + * If [.isPlayingAd] returns true, returns the index of the ad in its ad group. Returns + * [com.google.android.exoplayer2.C.INDEX_UNSET] otherwise. + */ + override fun getCurrentAdIndexInAdGroup(): Int { + return C.INDEX_UNSET + } + + /** + * If [.isPlayingAd] returns `true`, returns the content position that will be + * played once all ads in the ad group have finished playing, in milliseconds. If there is no ad + * playing, the returned position is the same as that returned by [.getCurrentPosition]. + */ + override fun getContentPosition(): Long { + return currentPosition + } + + /** + * If [.isPlayingAd] returns `true`, returns an estimate of the content position in + * the current content window up to which data is buffered, in milliseconds. If there is no ad + * playing, the returned position is the same as that returned by [.getBufferedPosition]. + */ + override fun getContentBufferedPosition(): Long { + return bufferedPosition + } + + /** Returns the attributes for audio playback. */ + override fun getAudioAttributes(): AudioAttributes { + return AudioAttributes.DEFAULT + } + + /** + * Sets the audio volume, with 0 being silence and 1 being unity gain (signal unchanged). + * + * @param audioVolume Linear output gain to apply to all audio channels. + */ + override fun setVolume(audioVolume: Float) { + TODO("Not yet implemented") + } + + /** + * Returns the audio volume, with 0 being silence and 1 being unity gain (signal unchanged). + * + * @return The linear gain applied to all audio channels. + */ + override fun getVolume(): Float { + TODO("Not yet implemented") + } + + /** + * Clears any [Surface], [SurfaceHolder], [SurfaceView] or [TextureView] + * currently set on the player. + */ + override fun clearVideoSurface() { + TODO("Not yet implemented") + } + + /** + * Clears the [Surface] onto which video is being rendered if it matches the one passed. + * Else does nothing. + * + * @param surface The surface to clear. + */ + override fun clearVideoSurface(surface: Surface?) { + TODO("Not yet implemented") + } + + /** + * Sets the [Surface] onto which video will be rendered. The caller is responsible for + * tracking the lifecycle of the surface, and must clear the surface by calling `setVideoSurface(null)` if the surface is destroyed. + * + * + * If the surface is held by a [SurfaceView], [TextureView] or [ ] then it's recommended to use [.setVideoSurfaceView], [ ][.setVideoTextureView] or [.setVideoSurfaceHolder] rather than + * this method, since passing the holder allows the player to track the lifecycle of the surface + * automatically. + * + * @param surface The [Surface]. + */ + override fun setVideoSurface(surface: Surface?) { + TODO("Not yet implemented") + } + + /** + * Sets the [SurfaceHolder] that holds the [Surface] onto which video will be + * rendered. The player will track the lifecycle of the surface automatically. + * + * @param surfaceHolder The surface holder. + */ + override fun setVideoSurfaceHolder(surfaceHolder: SurfaceHolder?) { + TODO("Not yet implemented") + } + + /** + * Clears the [SurfaceHolder] that holds the [Surface] onto which video is being + * rendered if it matches the one passed. Else does nothing. + * + * @param surfaceHolder The surface holder to clear. + */ + override fun clearVideoSurfaceHolder(surfaceHolder: SurfaceHolder?) { + TODO("Not yet implemented") + } + + /** + * Sets the [SurfaceView] onto which video will be rendered. The player will track the + * lifecycle of the surface automatically. + * + * @param surfaceView The surface view. + */ + override fun setVideoSurfaceView(surfaceView: SurfaceView?) { + surfaceView?.holder?.addCallback(surfaceHolder) + } + + /** + * Clears the [SurfaceView] onto which video is being rendered if it matches the one passed. + * Else does nothing. + * + * @param surfaceView The texture view to clear. + */ + override fun clearVideoSurfaceView(surfaceView: SurfaceView?) { + surfaceView?.holder?.removeCallback(surfaceHolder) + } + + /** + * Sets the [TextureView] onto which video will be rendered. The player will track the + * lifecycle of the surface automatically. + * + * @param textureView The texture view. + */ + override fun setVideoTextureView(textureView: TextureView?) { + TODO("Not yet implemented") + } + + /** + * Clears the [TextureView] onto which video is being rendered if it matches the one passed. + * Else does nothing. + * + * @param textureView The texture view to clear. + */ + override fun clearVideoTextureView(textureView: TextureView?) { + TODO("Not yet implemented") + } + + /** + * Gets the size of the video. + * + * + * The video's width and height are `0` if there is no video or its size has not been + * determined yet. + * + * @see com.google.android.exoplayer2.Player.Listener.onVideoSizeChanged + */ + override fun getVideoSize(): VideoSize { + return VideoSize.UNKNOWN + } + + /** Returns the current [Cues][Cue]. This list may be empty. */ + override fun getCurrentCues(): MutableList { + TODO("Not yet implemented") + } + + /** Gets the device information. */ + override fun getDeviceInfo(): DeviceInfo { + TODO("Not yet implemented") + } + + /** + * Gets the current volume of the device. + * + * + * For devices with [local playback][DeviceInfo.PLAYBACK_TYPE_LOCAL], the volume returned + * by this method varies according to the current [stream type][com.google.android.exoplayer2.C.StreamType]. The stream + * type is determined by [AudioAttributes.usage] which can be converted to stream type with + * [Util.getStreamTypeForAudioUsage]. + * + * + * For devices with [remote playback][DeviceInfo.PLAYBACK_TYPE_REMOTE], the volume of the + * remote device is returned. + */ + override fun getDeviceVolume(): Int { + TODO("Not yet implemented") + } + + /** Gets whether the device is muted or not. */ + override fun isDeviceMuted(): Boolean { + TODO("Not yet implemented") + } + + /** + * Sets the volume of the device. + * + * @param volume The volume to set. + */ + override fun setDeviceVolume(volume: Int) { + TODO("Not yet implemented") + } + + /** Increases the volume of the device. */ + override fun increaseDeviceVolume() { + TODO("Not yet implemented") + } + + /** Decreases the volume of the device. */ + override fun decreaseDeviceVolume() { + TODO("Not yet implemented") + } + + /** Sets the mute state of the device. */ + override fun setDeviceMuted(muted: Boolean) { + TODO("Not yet implemented") + } + + private class CurrentTrackSelection( + private val currentTrackGroup: TrackGroup, + private val index: Int + ) : TrackSelection { + /** + * Returns an integer specifying the type of the selection, or [.TYPE_UNSET] if not + * specified. + * + * + * Track selection types are specific to individual applications, but should be defined + * starting from [.TYPE_CUSTOM_BASE] to ensure they don't conflict with any types that may + * be added to the library in the future. + */ + override fun getType(): Int { + return TrackSelection.TYPE_UNSET + } + + /** Returns the [TrackGroup] to which the selected tracks belong. */ + override fun getTrackGroup(): TrackGroup { + return currentTrackGroup + } + + /** Returns the number of tracks in the selection. */ + override fun length(): Int { + return if (index != C.INDEX_UNSET) 1 else 0 + } + + /** + * Returns the format of the track at a given index in the selection. + * + * @param index The index in the selection. + * @return The format of the selected track. + */ + override fun getFormat(index: Int): Format { + return currentTrackGroup.getFormat(index) + } + + /** + * Returns the index in the track group of the track at a given index in the selection. + * + * @param index The index in the selection. + * @return The index of the selected track. + */ + override fun getIndexInTrackGroup(index: Int): Int { + return index + } + + /** + * Returns the index in the selection of the track with the specified format. The format is + * located by identity so, for example, `selection.indexOf(selection.getFormat(index)) == + * index` even if multiple selected tracks have formats that contain the same values. + * + * @param format The format. + * @return The index in the selection, or [C.INDEX_UNSET] if the track with the specified + * format is not part of the selection. + */ + override fun indexOf(format: Format): Int { + return currentTrackGroup.indexOf(format) + } + + /** + * Returns the index in the selection of the track with the specified index in the track group. + * + * @param indexInTrackGroup The index in the track group. + * @return The index in the selection, or [C.INDEX_UNSET] if the track with the specified + * index is not part of the selection. + */ + override fun indexOf(indexInTrackGroup: Int): Int { + return indexInTrackGroup + } + } + + companion object { + /** + * Fraction to which audio volume is ducked on loss of audio focus + */ + private const val AUDIO_FOCUS_DUCKING = 0.5f + + private val permanentAvailableCommands: Player.Commands = Player.Commands.Builder() + .addAll( + COMMAND_PLAY_PAUSE, + COMMAND_SEEK_IN_CURRENT_MEDIA_ITEM, + COMMAND_PREPARE_STOP, + COMMAND_SET_SPEED_AND_PITCH, + COMMAND_GET_CURRENT_MEDIA_ITEM, + COMMAND_GET_MEDIA_ITEMS, + COMMAND_CHANGE_MEDIA_ITEMS, + COMMAND_SET_VIDEO_SURFACE + ) + .build() + + private val surfaceHolder: SurfaceHolder.Callback = object : SurfaceHolder.Callback { + /** + * This is called immediately after the surface is first created. + * Implementations of this should start up whatever rendering code + * they desire. Note that only one thread can ever draw into + * a [Surface], so you should not draw into the Surface here + * if your normal rendering will be in another thread. + * + * @param holder The SurfaceHolder whose surface is being created. + */ + override fun surfaceCreated(holder: SurfaceHolder) { + MPVLib.attachSurface(holder.surface) + MPVLib.setOptionString("force-window", "yes") + MPVLib.setOptionString("vo", "gpu") + } + + /** + * This is called immediately after any structural changes (format or + * size) have been made to the surface. You should at this point update + * the imagery in the surface. This method is always called at least + * once, after [.surfaceCreated]. + * + * @param holder The SurfaceHolder whose surface has changed. + * @param format The new [android.graphics.PixelFormat] of the surface. + * @param width The new width of the surface. + * @param height The new height of the surface. + */ + override fun surfaceChanged(holder: SurfaceHolder, format: Int, width: Int, height: Int) { + MPVLib.setPropertyString("android-surface-size", "${width}x$height") + } + + /** + * This is called immediately before a surface is being destroyed. After + * returning from this call, you should no longer try to access this + * surface. If you have a rendering thread that directly accesses + * the surface, you must ensure that thread is no longer touching the + * Surface before returning from this function. + * + * @param holder The SurfaceHolder whose surface is being destroyed. + */ + override fun surfaceDestroyed(holder: SurfaceHolder) { + MPVLib.setOptionString("vo", "null") + MPVLib.setOptionString("force-window", "no") + MPVLib.detachSurface() + } + } + + @Parcelize + data class Track( + val id: Int, + @TrackType val type: String, + val mimeType: String = when (type) { + TrackType.VIDEO -> MimeTypes.BASE_TYPE_VIDEO + TrackType.AUDIO -> MimeTypes.BASE_TYPE_AUDIO + TrackType.SUBTITLE -> MimeTypes.BASE_TYPE_TEXT + else -> "" + }, + val title: String, + val lang: String, + val external: Boolean, + val selected: Boolean, + val externalFilename: String?, + val ffIndex: Int, + val codec: String, + val width: Int?, + val height: Int? + ) : Parcelable { + fun toFormat() : Format { + return Format.Builder() + .setId(id) + .setContainerMimeType("$mimeType/$codec") + .setSampleMimeType("$mimeType/$codec") + .setCodecs(codec) + .setWidth(width ?: Format.NO_VALUE) + .setHeight(height ?: Format.NO_VALUE) + .build() + } + companion object { + fun fromJSON(json: JSONObject): Track { + return Track( + id = json.optInt("id"), + type = json.optString("type"), + title = json.optString("title"), + lang = json.optString("lang"), + external = json.getBoolean("external"), + selected = json.getBoolean("selected"), + externalFilename = json.optString("external-filename"), + ffIndex = json.optInt("ff-index"), + codec = json.optString("codec"), + width = json.optInt("demux-w").takeIf { it > 0 }, + height = json.optInt("demux-h").takeIf { it > 0 } + ) + } + } + } + + private fun getMPVTracks(trackList: String) : Triple ,TrackGroupArray, TrackSelectionArray> { + val tracks = mutableListOf() + var trackGroupArray = TrackGroupArray.EMPTY + var trackSelectionArray = TrackSelectionArray() + + val trackListVideo = mutableListOf() + val trackListAudio = mutableListOf() + val trackListText = mutableListOf() + var indexCurrentVideo: Int = C.INDEX_UNSET + var indexCurrentAudio: Int = C.INDEX_UNSET + var indexCurrentText: Int = C.INDEX_UNSET + try { + val currentTrackList = JSONArray(trackList) + for (index in 0 until currentTrackList.length()) { + val currentTrack = Track.fromJSON(currentTrackList.getJSONObject(index)) + val currentFormat = currentTrack.toFormat() + when (currentTrack.type) { + TrackType.VIDEO -> { + tracks.add(currentTrack) + trackListVideo.add(currentFormat) + if (currentTrack.selected) { + indexCurrentVideo = trackListVideo.indexOf(currentFormat) + } + } + TrackType.AUDIO -> { + tracks.add(currentTrack) + trackListAudio.add(currentFormat) + if (currentTrack.selected) { + indexCurrentAudio = trackListAudio.indexOf(currentFormat) + } + } + TrackType.SUBTITLE -> { + tracks.add(currentTrack) + trackListText.add(currentFormat) + if (currentTrack.selected) { + indexCurrentText = trackListText.indexOf(currentFormat) + } + } + else -> continue + } + } + val trackGroups = mutableListOf() + val trackSelections = mutableListOf() + if (trackListVideo.isNotEmpty()) { + with(TrackGroup(*trackListVideo.toTypedArray())) { + trackGroups.add(this) + trackSelections.add(CurrentTrackSelection(this, indexCurrentVideo)) + } + } + if (trackListAudio.isNotEmpty()) { + with(TrackGroup(*trackListAudio.toTypedArray())) { + trackGroups.add(this) + trackSelections.add(CurrentTrackSelection(this, indexCurrentAudio)) + } + } + if (trackListText.isNotEmpty()) { + with(TrackGroup(*trackListText.toTypedArray())) { + trackGroups.add(this) + trackSelections.add(CurrentTrackSelection(this, indexCurrentText)) + } + } + if (trackGroups.isNotEmpty()) { + trackGroupArray = TrackGroupArray(*trackGroups.toTypedArray()) + trackSelectionArray = TrackSelectionArray(*trackSelections.toTypedArray()) + } + } catch (e: JSONException) {} + return Triple(tracks, trackGroupArray, trackSelectionArray) + } + + const val PLAYER_NAME = "MPV Player" + + /** + * Merges multiple [subtitleSources] into a single [videoSource] + */ + fun mergeMediaSources( + videoSource: MediaSource, + subtitleSources: Array, + dataSource: DataSource.Factory + ): MediaSource { + return when { + subtitleSources.isEmpty() -> videoSource + else -> { + val subtitles = mutableListOf() + subtitleSources.forEach { subtitleSource -> + subtitleSource.mediaItem.playbackProperties?.subtitles?.forEach { subtitle -> + subtitles.add(subtitle) + } + } + ProgressiveMediaSource.Factory(dataSource) + .createMediaSource( + videoSource.mediaItem.buildUpon() + .setSubtitles(subtitles).build() + ) + } + } + } + } +} diff --git a/app/src/main/java/org/jellyfin/mobile/player/mpv/TrackType.java b/app/src/main/java/org/jellyfin/mobile/player/mpv/TrackType.java new file mode 100644 index 0000000000..abc7e47c05 --- /dev/null +++ b/app/src/main/java/org/jellyfin/mobile/player/mpv/TrackType.java @@ -0,0 +1,14 @@ +package org.jellyfin.mobile.player.mpv; + +import androidx.annotation.StringDef; + +@StringDef({ + TrackType.VIDEO, + TrackType.AUDIO, + TrackType.SUBTITLE, +}) +public @interface TrackType { + String VIDEO = "video"; + String AUDIO = "audio"; + String SUBTITLE = "sub"; +} diff --git a/app/src/main/java/org/jellyfin/mobile/player/source/MediaQueueManager.kt b/app/src/main/java/org/jellyfin/mobile/player/source/MediaQueueManager.kt index 48701461f4..20a72eb2fc 100644 --- a/app/src/main/java/org/jellyfin/mobile/player/source/MediaQueueManager.kt +++ b/app/src/main/java/org/jellyfin/mobile/player/source/MediaQueueManager.kt @@ -6,6 +6,7 @@ import androidx.annotation.CheckResult import androidx.lifecycle.LiveData import androidx.lifecycle.MutableLiveData import com.google.android.exoplayer2.C +import com.google.android.exoplayer2.ExoPlayer import com.google.android.exoplayer2.MediaItem import com.google.android.exoplayer2.source.MediaSource import com.google.android.exoplayer2.source.MergingMediaSource @@ -15,8 +16,11 @@ import com.google.android.exoplayer2.source.hls.HlsMediaSource import com.google.android.exoplayer2.trackselection.DefaultTrackSelector import com.google.android.exoplayer2.util.EventLogger import org.jellyfin.mobile.bridge.PlayOptions +import org.jellyfin.mobile.player.mpv.MPVPlayer import org.jellyfin.mobile.player.PlayerException import org.jellyfin.mobile.player.PlayerViewModel +import org.jellyfin.mobile.player.mpv.TrackType +import org.jellyfin.mobile.utils.getRendererIndexByType import org.jellyfin.sdk.api.client.ApiClient import org.jellyfin.sdk.api.operations.VideosApi import org.jellyfin.sdk.model.api.DeviceProfile @@ -25,14 +29,16 @@ import org.jellyfin.sdk.model.api.PlayMethod import org.koin.core.component.KoinComponent import org.koin.core.component.get import org.koin.core.component.inject +import org.koin.core.qualifier.named import java.util.UUID class MediaQueueManager( private val viewModel: PlayerViewModel, + playerName: String ) : KoinComponent { private val apiClient: ApiClient by inject() private val mediaSourceResolver: MediaSourceResolver by inject() - private val deviceProfile: DeviceProfile by inject() + private val deviceProfile: DeviceProfile by inject(named(playerName)) private val videosApi: VideosApi by inject() private val _mediaQueue: MutableLiveData = MutableLiveData() val mediaQueue: LiveData get() = _mediaQueue @@ -110,7 +116,7 @@ class MediaQueueManager( } /** - * Builds the [MediaSource] to be played by ExoPlayer. + * Builds the [MediaSource] to be played by Player. * * @param source The [JellyfinMediaSource] object containing all necessary info about the item to be played. * @return A [MediaSource]. This can be the media stream of the correct type for the playback method or @@ -121,6 +127,7 @@ class MediaQueueManager( val videoSource = createVideoMediaSource(source) val subtitleSources = createSubtitleMediaSources(source) return when { + viewModel.playerOrNull is MPVPlayer -> MPVPlayer.mergeMediaSources(videoSource, subtitleSources, get()) subtitleSources.isNotEmpty() -> MergingMediaSource(videoSource, *subtitleSources) else -> videoSource } @@ -184,7 +191,14 @@ class MediaQueueManager( val factory = get() return source.getExternalSubtitleStreams().map { stream -> val uri = Uri.parse(apiClient.createUrl(stream.deliveryUrl)) - val mediaItem = MediaItem.Subtitle(uri, stream.mimeType, stream.language, C.SELECTION_FLAG_AUTOSELECT) + val mediaItem = MediaItem.Subtitle( + /* uri= */ uri, + /* mimeType= */ stream.mimeType, + /* language= */ stream.language, + /* selectionFlags= */ C.SELECTION_FLAG_AUTOSELECT, + /* roleFlags= */ 0, + /* label= */ stream.index.toString() + ) factory.setTrackId(stream.index.toString()).createMediaSource(mediaItem, source.runTimeMs) }.toTypedArray() } @@ -206,6 +220,7 @@ class MediaQueueManager( @Suppress("ReturnCount") fun selectAudioTrack(streamIndex: Int, initial: Boolean = false): Boolean { val mediaSource = _mediaQueue.value?.jellyfinMediaSource ?: return false + val player = viewModel.playerOrNull val sourceIndex = mediaSource.audioStreams.binarySearchBy(streamIndex, selector = MediaStream::index) when { @@ -220,22 +235,33 @@ class MediaQueueManager( !mediaSource.selectAudioStream(sourceIndex) -> return false } - // Handle selection in player - val parameters = trackSelector.buildUponParameters() - val rendererIndex = viewModel.getPlayerRendererIndex(C.TRACK_TYPE_AUDIO) - val trackInfo = trackSelector.currentMappedTrackInfo - return if (rendererIndex >= 0 && trackInfo != null) { - val trackGroups = trackInfo.getTrackGroups(rendererIndex) - if (sourceIndex in 0 until trackGroups.length) { - val selection = DefaultTrackSelector.SelectionOverride(sourceIndex, 0) - parameters.setSelectionOverride(rendererIndex, trackGroups, selection) - } else { - parameters.clearSelectionOverride(rendererIndex, trackGroups) - } - parameters.setRendererDisabled(rendererIndex, false) - trackSelector.setParameters(parameters) - true - } else false + // Handle selection in MPVPlayer + if (player is MPVPlayer) { + return player.selectTrack( + trackType = TrackType.AUDIO, + index = mediaSource.selectedSubtitleStream?.index ?: C.INDEX_UNSET + ) + } + + // Handle selection in ExoPlayer + if (player is ExoPlayer) { + val parameters = trackSelector.buildUponParameters() + val rendererIndex = player.getRendererIndexByType(C.TRACK_TYPE_AUDIO) + val trackInfo = trackSelector.currentMappedTrackInfo + return if (rendererIndex >= 0 && trackInfo != null) { + val trackGroups = trackInfo.getTrackGroups(rendererIndex) + if (sourceIndex in 0 until trackGroups.length) { + val selection = DefaultTrackSelector.SelectionOverride(sourceIndex, 0) + parameters.setSelectionOverride(rendererIndex, trackGroups, selection) + } else { + parameters.clearSelectionOverride(rendererIndex, trackGroups) + } + parameters.setRendererDisabled(rendererIndex, false) + trackSelector.setParameters(parameters) + true + } else false + } + return false } /** @@ -247,6 +273,7 @@ class MediaQueueManager( */ fun selectSubtitle(streamIndex: Int, initial: Boolean = false): Boolean { val mediaSource = _mediaQueue.value?.jellyfinMediaSource ?: return false + val player = viewModel.playerOrNull val sourceIndex = mediaSource.subtitleStreams.binarySearchBy(streamIndex, selector = MediaStream::index) when { @@ -256,24 +283,36 @@ class MediaQueueManager( !mediaSource.selectSubtitleStream(sourceIndex) -> return false } - // Handle selection in player - val rendererIndex = viewModel.getPlayerRendererIndex(C.TRACK_TYPE_TEXT) - val trackInfo = trackSelector.currentMappedTrackInfo - return if (rendererIndex >= 0 && trackInfo != null) { - val parameters = trackSelector.buildUponParameters() - val trackGroups = trackInfo.getTrackGroups(rendererIndex) - if (sourceIndex in 0 until trackGroups.length) { - val selection = DefaultTrackSelector.SelectionOverride(sourceIndex, 0) - parameters.setSelectionOverride(rendererIndex, trackGroups, selection) - parameters.setRendererDisabled(rendererIndex, false) - } else { - // No subtitle selected, clear selection overrides and disable renderer - parameters.clearSelectionOverride(rendererIndex, trackGroups) - parameters.setRendererDisabled(rendererIndex, true) - } - trackSelector.setParameters(parameters) - true - } else false + // Handle selection in MPVPlayer + if (player is MPVPlayer) { + return player.selectTrack( + trackType = TrackType.SUBTITLE, + isExternal = mediaSource.selectedSubtitleStream?.isExternal ?: false, + index = mediaSource.selectedSubtitleStream?.index ?: C.INDEX_UNSET + ) + } + + // Handle selection in ExoPlayer + if (player is ExoPlayer) { + val rendererIndex = player.getRendererIndexByType(C.TRACK_TYPE_TEXT) + val trackInfo = trackSelector.currentMappedTrackInfo + return if (rendererIndex >= 0 && trackInfo != null) { + val parameters = trackSelector.buildUponParameters() + val trackGroups = trackInfo.getTrackGroups(rendererIndex) + if (sourceIndex in 0 until trackGroups.length) { + val selection = DefaultTrackSelector.SelectionOverride(sourceIndex, 0) + parameters.setSelectionOverride(rendererIndex, trackGroups, selection) + parameters.setRendererDisabled(rendererIndex, false) + } else { + // No subtitle selected, clear selection overrides and disable renderer + parameters.clearSelectionOverride(rendererIndex, trackGroups) + parameters.setRendererDisabled(rendererIndex, true) + } + trackSelector.setParameters(parameters) + true + } else false + } + return false } /** diff --git a/app/src/main/java/org/jellyfin/mobile/settings/SettingsFragment.kt b/app/src/main/java/org/jellyfin/mobile/settings/SettingsFragment.kt index 541701e548..c1e7523ee3 100644 --- a/app/src/main/java/org/jellyfin/mobile/settings/SettingsFragment.kt +++ b/app/src/main/java/org/jellyfin/mobile/settings/SettingsFragment.kt @@ -75,21 +75,22 @@ class SettingsFragment : Fragment() { val videoPlayerOptions = listOf( SelectionItem(VideoPlayerType.WEB_PLAYER, R.string.video_player_web, R.string.video_player_web_description), SelectionItem(VideoPlayerType.EXO_PLAYER, R.string.video_player_native, R.string.video_player_native_description), + SelectionItem(VideoPlayerType.MPV_PLAYER, R.string.video_player_mpv, R.string.video_player_mpv_description), SelectionItem(VideoPlayerType.EXTERNAL_PLAYER, R.string.video_player_external, R.string.video_player_external_description), ) singleChoice(Constants.PREF_VIDEO_PLAYER_TYPE, videoPlayerOptions) { titleRes = R.string.pref_video_player_type_title initialSelection = VideoPlayerType.WEB_PLAYER defaultOnSelectionChange { selection -> - swipeGesturesPreference.enabled = selection == VideoPlayerType.EXO_PLAYER - rememberBrightnessPreference.enabled = selection == VideoPlayerType.EXO_PLAYER && swipeGesturesPreference.checked - backgroundAudioPreference.enabled = selection == VideoPlayerType.EXO_PLAYER + swipeGesturesPreference.enabled = selection in arrayOf(VideoPlayerType.EXO_PLAYER, VideoPlayerType.MPV_PLAYER) + rememberBrightnessPreference.enabled = selection in arrayOf(VideoPlayerType.EXO_PLAYER, VideoPlayerType.MPV_PLAYER) && swipeGesturesPreference.checked + backgroundAudioPreference.enabled = selection in arrayOf(VideoPlayerType.EXO_PLAYER, VideoPlayerType.MPV_PLAYER) externalPlayerChoicePreference.enabled = selection == VideoPlayerType.EXTERNAL_PLAYER } } swipeGesturesPreference = checkBox(Constants.PREF_EXOPLAYER_ALLOW_SWIPE_GESTURES) { titleRes = R.string.pref_exoplayer_allow_brightness_volume_gesture - enabled = appPreferences.videoPlayerType == VideoPlayerType.EXO_PLAYER + enabled = appPreferences.videoPlayerType in arrayOf(VideoPlayerType.EXO_PLAYER, VideoPlayerType.MPV_PLAYER) defaultValue = true defaultOnCheckedChange { checked -> rememberBrightnessPreference.enabled = checked @@ -97,7 +98,7 @@ class SettingsFragment : Fragment() { } rememberBrightnessPreference = checkBox(Constants.PREF_EXOPLAYER_REMEMBER_BRIGHTNESS) { titleRes = R.string.pref_exoplayer_remember_brightness - enabled = appPreferences.videoPlayerType == VideoPlayerType.EXO_PLAYER && appPreferences.exoPlayerAllowSwipeGestures + enabled = appPreferences.videoPlayerType in arrayOf(VideoPlayerType.EXO_PLAYER, VideoPlayerType.MPV_PLAYER) && appPreferences.exoPlayerAllowSwipeGestures defaultOnCheckedChange { checked -> if (!checked) appPreferences.exoPlayerBrightness = BRIGHTNESS_OVERRIDE_NONE } @@ -105,7 +106,7 @@ class SettingsFragment : Fragment() { backgroundAudioPreference = checkBox(Constants.PREF_EXOPLAYER_ALLOW_BACKGROUND_AUDIO) { titleRes = R.string.pref_exoplayer_allow_background_audio summaryRes = R.string.pref_exoplayer_allow_background_audio_summary - enabled = appPreferences.videoPlayerType == VideoPlayerType.EXO_PLAYER + enabled = appPreferences.videoPlayerType in arrayOf(VideoPlayerType.EXO_PLAYER, VideoPlayerType.MPV_PLAYER) } // Generate available external player options diff --git a/app/src/main/java/org/jellyfin/mobile/settings/VideoPlayerType.java b/app/src/main/java/org/jellyfin/mobile/settings/VideoPlayerType.java index f006ad49e4..c6c1311056 100644 --- a/app/src/main/java/org/jellyfin/mobile/settings/VideoPlayerType.java +++ b/app/src/main/java/org/jellyfin/mobile/settings/VideoPlayerType.java @@ -6,10 +6,12 @@ @StringDef({ VideoPlayerType.WEB_PLAYER, VideoPlayerType.EXO_PLAYER, + VideoPlayerType.MPV_PLAYER, VideoPlayerType.EXTERNAL_PLAYER }) public @interface VideoPlayerType { String WEB_PLAYER = "webui"; String EXO_PLAYER = "exoplayer"; + String MPV_PLAYER = "mpvplayer"; String EXTERNAL_PLAYER = "external"; } diff --git a/app/src/main/java/org/jellyfin/mobile/utils/MediaExtensions.kt b/app/src/main/java/org/jellyfin/mobile/utils/MediaExtensions.kt index 30d31dbd91..143769b2d0 100644 --- a/app/src/main/java/org/jellyfin/mobile/utils/MediaExtensions.kt +++ b/app/src/main/java/org/jellyfin/mobile/utils/MediaExtensions.kt @@ -85,10 +85,10 @@ fun ExoPlayer.getRendererIndexByType(type: Int): Int { for (i in 0 until rendererCount) { if (getRendererType(i) == type) return i } - return -1 + return C.INDEX_UNSET } -fun ExoPlayer.seekToOffset(offsetMs: Long) { +fun Player.seekToOffset(offsetMs: Long) { var positionMs = currentPosition + offsetMs val durationMs = duration if (durationMs != C.TIME_UNSET) { diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index fff38bc8d0..f4b7030a28 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -78,9 +78,11 @@ Video player type Web player Native player + MPV player External player The default HTML video player from the Web UI Based on ExoPlayer, supports more video formats and codecs, and is more integrated into the OS + Based on libmpv, supports more video, audio and subtitle formats, and is more customizable External video playback apps like MX Player and VLC Brightness and volume gestures Remember display brightness