package com.github.libretube.services import android.app.NotificationManager import android.app.Service import android.content.Context import android.content.Intent import android.os.IBinder import android.support.v4.media.session.MediaSessionCompat import com.github.libretube.obj.Streams import com.github.libretube.preferences.PreferenceHelper import com.github.libretube.preferences.PreferenceKeys import com.github.libretube.util.DescriptionAdapter import com.github.libretube.util.RetrofitInstance import com.google.android.exoplayer2.C import com.google.android.exoplayer2.ExoPlayer import com.google.android.exoplayer2.MediaItem import com.google.android.exoplayer2.Player import com.google.android.exoplayer2.audio.AudioAttributes import com.google.android.exoplayer2.ext.mediasession.MediaSessionConnector import com.google.android.exoplayer2.ui.PlayerNotificationManager import kotlinx.coroutines.launch import kotlinx.coroutines.runBlocking import java.lang.Exception /** * Loads the selected videos audio in background mode with a notification area. */ class BackgroundMode : Service() { /** * The response that gets when called the Api. */ private var response: Streams? = null /** * The [ExoPlayer] player. Followed tutorial [here](https://developer.android.com/codelabs/exoplayer-intro) */ private var player: ExoPlayer? = null private var playWhenReadyPlayer = true /** * The [MediaSessionCompat] for the [response]. */ private lateinit var mediaSession: MediaSessionCompat /** * The [MediaSessionConnector] to connect with the [mediaSession] and implement it with the [player]. */ private lateinit var mediaSessionConnector: MediaSessionConnector /** * The [PlayerNotificationManager] to load the [mediaSession] content on it. */ private var playerNotification: PlayerNotificationManager? = null /** * The [AudioAttributes] handle the audio focus of the [player] */ private lateinit var audioAttributes: AudioAttributes /** * Initializes the [player] with the [MediaItem]. */ override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int { try { val videoId = intent?.getStringExtra("videoId")!! val seekToPosition = intent.getLongExtra("seekToPosition", 0L) playOnBackgroundMode(this, videoId, seekToPosition) } catch (e: Exception) { stopService(intent) } return super.onStartCommand(intent, flags, startId) } /** * Gets the video data and prepares the [player]. */ private fun playOnBackgroundMode( c: Context, videoId: String, seekToPosition: Long = 0 ) { runBlocking { val job = launch { response = RetrofitInstance.api.getStreams(videoId) } // Wait until the job is done, to load correctly later in the player job.join() initializePlayer(c) initializePlayerNotification(c) player?.apply { playWhenReady = playWhenReadyPlayer prepare() } if (seekToPosition != 0L) player?.seekTo(seekToPosition) } } /** * create the player */ private fun initializePlayer(c: Context) { audioAttributes = AudioAttributes.Builder() .setUsage(C.USAGE_MEDIA) .setContentType(C.CONTENT_TYPE_MUSIC) .build() if (player == null) { player = ExoPlayer.Builder(c) .setAudioAttributes(audioAttributes, true) .build() } /** * Listens for changed playbackStates (e.g. pause, end) * Plays the next video when the current one ended */ player!!.addListener(object : Player.Listener { override fun onPlaybackStateChanged(@Player.State state: Int) { val autoplay = PreferenceHelper.getBoolean(PreferenceKeys.AUTO_PLAY, false) if (state == Player.STATE_ENDED) { if (autoplay) playNextVideo(c) } } }) setMediaItem(c) } /** * Plays the first related video to the current (used when the playback of the current video ended) */ private fun playNextVideo(c: Context) { if (response!!.relatedStreams!!.isNotEmpty()) { val videoId = response!! .relatedStreams!![0].url!! .replace("/watch?v=", "") // destroy old player and its notification playerNotification = null player = null // kill old notification val notificationManager = c.getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager notificationManager.cancel(1) // play new video on background playOnBackgroundMode(c, videoId) } } /** * Initializes the [playerNotification] attached to the [player] and shows it. */ private fun initializePlayerNotification(c: Context) { playerNotification = PlayerNotificationManager .Builder(c, 1, "background_mode") // set the description of the notification .setMediaDescriptionAdapter( DescriptionAdapter( response?.title!!, response?.uploader!!, response?.thumbnailUrl!!, c ) ) .build() playerNotification?.apply { setPlayer(player) setUseNextAction(false) setUsePreviousAction(false) setUseStopAction(true) setColorized(true) setMediaSessionToken(mediaSession.sessionToken) } } /** * Sets the [MediaItem] with the [response] into the [player]. Also creates a [MediaSessionConnector] * with the [mediaSession] and attach it to the [player]. */ private fun setMediaItem(c: Context) { response?.let { val mediaItem = MediaItem.Builder().setUri(it.hls!!).build() player?.setMediaItem(mediaItem) } mediaSession = MediaSessionCompat(c, this.javaClass.name) mediaSession.isActive = true mediaSessionConnector = MediaSessionConnector(mediaSession) mediaSessionConnector.setPlayer(player) } override fun onBind(p0: Intent?): IBinder? { TODO("Not yet implemented") } }