2022-06-03 01:33:36 +05:30
|
|
|
package com.github.libretube.fragments
|
2021-12-12 17:38:23 +05:30
|
|
|
|
2022-02-13 22:43:26 +05:30
|
|
|
import android.annotation.SuppressLint
|
2022-06-01 12:20:02 +05:30
|
|
|
import android.app.NotificationManager
|
2022-06-26 21:19:42 +05:30
|
|
|
import android.app.PictureInPictureParams
|
2022-02-13 22:43:26 +05:30
|
|
|
import android.content.Context
|
2022-03-05 11:56:54 +05:30
|
|
|
import android.content.Intent
|
2021-12-14 21:45:53 +05:30
|
|
|
import android.content.pm.ActivityInfo
|
2022-05-14 17:01:33 +05:30
|
|
|
import android.graphics.Rect
|
2022-03-15 21:36:42 +05:30
|
|
|
import android.net.Uri
|
2022-01-28 18:31:41 +05:30
|
|
|
import android.os.Build
|
2022-02-27 00:27:05 +05:30
|
|
|
import android.os.Build.VERSION.SDK_INT
|
2022-02-26 22:49:42 +05:30
|
|
|
import android.os.Bundle
|
2022-07-08 21:19:32 +05:30
|
|
|
import android.os.Handler
|
|
|
|
import android.os.Looper
|
2022-06-20 17:52:28 +05:30
|
|
|
import android.os.PowerManager
|
2022-06-01 11:32:16 +05:30
|
|
|
import android.support.v4.media.session.MediaSessionCompat
|
2022-01-28 18:31:41 +05:30
|
|
|
import android.text.Html
|
2022-05-16 15:41:22 +05:30
|
|
|
import android.text.TextUtils
|
2021-12-18 16:34:14 +05:30
|
|
|
import android.util.Log
|
2022-02-26 22:49:42 +05:30
|
|
|
import android.view.LayoutInflater
|
|
|
|
import android.view.View
|
|
|
|
import android.view.ViewGroup
|
2022-05-21 13:32:04 +05:30
|
|
|
import android.widget.Toast
|
2022-02-26 22:49:42 +05:30
|
|
|
import androidx.constraintlayout.motion.widget.MotionLayout
|
|
|
|
import androidx.core.net.toUri
|
2022-02-05 00:25:05 +05:30
|
|
|
import androidx.core.os.bundleOf
|
2022-03-28 04:08:00 +05:30
|
|
|
import androidx.core.view.isVisible
|
2022-02-26 22:49:42 +05:30
|
|
|
import androidx.fragment.app.Fragment
|
2021-12-18 16:34:14 +05:30
|
|
|
import androidx.lifecycle.lifecycleScope
|
2022-01-28 18:31:41 +05:30
|
|
|
import androidx.recyclerview.widget.GridLayoutManager
|
2022-05-08 16:03:01 +05:30
|
|
|
import androidx.recyclerview.widget.LinearLayoutManager
|
2022-06-03 01:33:36 +05:30
|
|
|
import com.github.libretube.R
|
2022-07-01 20:25:21 +05:30
|
|
|
import com.github.libretube.activities.MainActivity
|
|
|
|
import com.github.libretube.activities.hideKeyboard
|
2022-06-13 19:45:02 +05:30
|
|
|
import com.github.libretube.adapters.ChaptersAdapter
|
2022-05-08 16:03:01 +05:30
|
|
|
import com.github.libretube.adapters.CommentsAdapter
|
2022-02-01 21:22:06 +05:30
|
|
|
import com.github.libretube.adapters.TrendingAdapter
|
2022-07-02 22:01:56 +05:30
|
|
|
import com.github.libretube.databinding.ExoStyledPlayerControlViewBinding
|
2022-07-01 00:35:31 +05:30
|
|
|
import com.github.libretube.databinding.FragmentPlayerBinding
|
2022-06-03 00:40:16 +05:30
|
|
|
import com.github.libretube.dialogs.AddtoPlaylistDialog
|
|
|
|
import com.github.libretube.dialogs.DownloadDialog
|
2022-06-05 14:22:35 +05:30
|
|
|
import com.github.libretube.dialogs.ShareDialog
|
2022-06-12 22:56:38 +05:30
|
|
|
import com.github.libretube.obj.ChapterSegment
|
2022-02-01 21:22:06 +05:30
|
|
|
import com.github.libretube.obj.PipedStream
|
2022-06-29 12:34:11 +05:30
|
|
|
import com.github.libretube.obj.Playlist
|
2022-05-16 15:41:22 +05:30
|
|
|
import com.github.libretube.obj.Segment
|
|
|
|
import com.github.libretube.obj.Segments
|
2022-06-26 14:06:34 +05:30
|
|
|
import com.github.libretube.obj.SponsorBlockPrefs
|
2022-06-18 14:02:05 +05:30
|
|
|
import com.github.libretube.obj.StreamItem
|
2022-06-01 11:55:12 +05:30
|
|
|
import com.github.libretube.obj.Streams
|
2022-02-13 22:43:26 +05:30
|
|
|
import com.github.libretube.obj.Subscribe
|
2022-07-02 21:53:24 +05:30
|
|
|
import com.github.libretube.preferences.PreferenceHelper
|
2022-06-28 20:26:37 +05:30
|
|
|
import com.github.libretube.services.IS_DOWNLOAD_RUNNING
|
2022-05-27 07:04:44 +05:30
|
|
|
import com.github.libretube.util.CronetHelper
|
2022-06-14 15:30:58 +05:30
|
|
|
import com.github.libretube.util.DescriptionAdapter
|
2022-06-03 00:40:16 +05:30
|
|
|
import com.github.libretube.util.RetrofitInstance
|
2022-06-10 18:33:48 +05:30
|
|
|
import com.github.libretube.util.formatShort
|
2022-07-08 20:56:45 +05:30
|
|
|
import com.github.libretube.views.DoubleClickListener
|
2022-05-13 19:25:31 +05:30
|
|
|
import com.google.android.exoplayer2.C
|
2022-06-03 22:13:15 +05:30
|
|
|
import com.google.android.exoplayer2.DefaultLoadControl
|
2022-02-26 22:49:42 +05:30
|
|
|
import com.google.android.exoplayer2.ExoPlayer
|
|
|
|
import com.google.android.exoplayer2.MediaItem
|
|
|
|
import com.google.android.exoplayer2.MediaItem.SubtitleConfiguration
|
|
|
|
import com.google.android.exoplayer2.MediaItem.fromUri
|
|
|
|
import com.google.android.exoplayer2.Player
|
2022-05-13 19:25:31 +05:30
|
|
|
import com.google.android.exoplayer2.audio.AudioAttributes
|
2022-04-15 12:08:53 +05:30
|
|
|
import com.google.android.exoplayer2.ext.cronet.CronetDataSource
|
2022-06-01 11:32:16 +05:30
|
|
|
import com.google.android.exoplayer2.ext.mediasession.MediaSessionConnector
|
2022-02-26 22:49:42 +05:30
|
|
|
import com.google.android.exoplayer2.source.DefaultMediaSourceFactory
|
|
|
|
import com.google.android.exoplayer2.source.MediaSource
|
|
|
|
import com.google.android.exoplayer2.source.MergingMediaSource
|
|
|
|
import com.google.android.exoplayer2.source.ProgressiveMediaSource
|
2022-06-03 00:55:33 +05:30
|
|
|
import com.google.android.exoplayer2.ui.AspectRatioFrameLayout
|
2022-06-01 11:32:16 +05:30
|
|
|
import com.google.android.exoplayer2.ui.PlayerNotificationManager
|
2022-02-26 22:49:42 +05:30
|
|
|
import com.google.android.exoplayer2.ui.StyledPlayerView
|
|
|
|
import com.google.android.exoplayer2.upstream.DataSource
|
2022-04-15 12:08:53 +05:30
|
|
|
import com.google.android.exoplayer2.upstream.DefaultDataSource
|
2022-02-26 22:49:42 +05:30
|
|
|
import com.google.android.exoplayer2.upstream.DefaultHttpDataSource
|
2022-05-13 10:02:16 +05:30
|
|
|
import com.google.android.exoplayer2.util.RepeatModeUtil
|
2022-07-08 21:36:48 +05:30
|
|
|
import com.google.android.exoplayer2.video.VideoSize
|
2022-02-13 22:43:26 +05:30
|
|
|
import com.google.android.material.button.MaterialButton
|
2022-05-20 20:01:14 +05:30
|
|
|
import com.google.android.material.dialog.MaterialAlertDialogBuilder
|
2022-02-26 22:49:42 +05:30
|
|
|
import com.squareup.picasso.Picasso
|
2022-06-29 12:34:11 +05:30
|
|
|
import kotlinx.coroutines.CoroutineScope
|
|
|
|
import kotlinx.coroutines.Dispatchers
|
|
|
|
import kotlinx.coroutines.launch
|
2022-06-24 20:56:36 +05:30
|
|
|
import org.chromium.net.CronetEngine
|
|
|
|
import retrofit2.HttpException
|
2022-02-26 22:49:42 +05:30
|
|
|
import java.io.IOException
|
2022-04-15 12:08:53 +05:30
|
|
|
import java.util.concurrent.Executors
|
2022-02-26 22:49:42 +05:30
|
|
|
import kotlin.math.abs
|
2022-04-15 12:08:53 +05:30
|
|
|
|
2022-02-10 16:39:34 +05:30
|
|
|
var isFullScreen = false
|
2022-06-26 20:50:53 +05:30
|
|
|
var isMiniPlayerVisible = false
|
2022-04-15 12:08:53 +05:30
|
|
|
|
2021-12-12 17:38:23 +05:30
|
|
|
class PlayerFragment : Fragment() {
|
2022-02-05 21:20:16 +05:30
|
|
|
|
2022-02-05 00:25:05 +05:30
|
|
|
private val TAG = "PlayerFragment"
|
2022-07-01 00:35:31 +05:30
|
|
|
private lateinit var binding: FragmentPlayerBinding
|
2022-07-02 22:01:56 +05:30
|
|
|
private lateinit var playerBinding: ExoStyledPlayerControlViewBinding
|
2022-07-01 00:35:31 +05:30
|
|
|
|
2021-12-12 17:38:23 +05:30
|
|
|
private var videoId: String? = null
|
2022-06-29 12:34:11 +05:30
|
|
|
private var playlistId: String? = null
|
2022-03-27 11:49:17 +05:30
|
|
|
private var sId: Int = 0
|
|
|
|
private var eId: Int = 0
|
|
|
|
private var paused = false
|
2021-12-14 21:45:53 +05:30
|
|
|
private var whichQuality = 0
|
2022-06-17 16:51:55 +05:30
|
|
|
private var transitioning = false
|
2022-06-17 16:55:17 +05:30
|
|
|
private var autoplay = false
|
2022-06-03 00:55:33 +05:30
|
|
|
private var isZoomed: Boolean = false
|
2021-12-12 20:31:44 +05:30
|
|
|
|
2022-06-14 18:39:47 +05:30
|
|
|
private var isSubscribed: Boolean = false
|
2022-02-13 22:43:26 +05:30
|
|
|
|
2022-05-09 00:48:04 +05:30
|
|
|
private var commentsAdapter: CommentsAdapter? = null
|
2022-05-20 21:15:16 +05:30
|
|
|
private var commentsLoaded: Boolean? = false
|
2022-05-08 23:02:13 +05:30
|
|
|
private var nextPage: String? = null
|
2022-05-09 00:48:04 +05:30
|
|
|
private var isLoading = true
|
2021-12-14 02:58:17 +05:30
|
|
|
private lateinit var exoPlayerView: StyledPlayerView
|
2021-12-14 21:45:53 +05:30
|
|
|
private lateinit var exoPlayer: ExoPlayer
|
2022-05-16 15:41:22 +05:30
|
|
|
private lateinit var segmentData: Segments
|
2022-06-12 22:56:38 +05:30
|
|
|
private var relatedStreamsEnabled = true
|
2022-06-29 12:52:22 +05:30
|
|
|
|
2022-06-29 12:34:11 +05:30
|
|
|
private var relatedStreams: List<StreamItem>? = arrayListOf()
|
|
|
|
private var nextStreamId: String? = null
|
|
|
|
private var playlistStreamIds: MutableList<String> = arrayListOf()
|
2022-06-29 12:52:22 +05:30
|
|
|
private var playlistNextPage: String? = null
|
|
|
|
|
2022-06-24 20:56:36 +05:30
|
|
|
private var isPlayerLocked: Boolean = false
|
2021-12-12 20:31:44 +05:30
|
|
|
|
2022-06-01 11:32:16 +05:30
|
|
|
private lateinit var mediaSession: MediaSessionCompat
|
|
|
|
private lateinit var mediaSessionConnector: MediaSessionConnector
|
|
|
|
private lateinit var playerNotification: PlayerNotificationManager
|
|
|
|
|
2022-06-14 15:30:58 +05:30
|
|
|
private lateinit var title: String
|
|
|
|
private lateinit var uploader: String
|
|
|
|
private lateinit var thumbnailUrl: String
|
2022-06-26 14:06:34 +05:30
|
|
|
private val sponsorBlockPrefs = SponsorBlockPrefs()
|
2022-06-14 15:30:58 +05:30
|
|
|
|
2021-12-12 17:38:23 +05:30
|
|
|
override fun onCreate(savedInstanceState: Bundle?) {
|
|
|
|
super.onCreate(savedInstanceState)
|
|
|
|
arguments?.let {
|
|
|
|
videoId = it.getString("videoId")
|
2022-06-29 12:34:11 +05:30
|
|
|
playlistId = it.getString("playlistId")
|
2021-12-12 17:38:23 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
override fun onCreateView(
|
2022-03-27 11:49:17 +05:30
|
|
|
inflater: LayoutInflater,
|
|
|
|
container: ViewGroup?,
|
2021-12-12 17:38:23 +05:30
|
|
|
savedInstanceState: Bundle?
|
2022-07-01 00:55:40 +05:30
|
|
|
): View {
|
2022-07-01 00:35:31 +05:30
|
|
|
binding = FragmentPlayerBinding.inflate(layoutInflater, container, false)
|
2022-07-02 22:01:56 +05:30
|
|
|
playerBinding = binding.player.binding
|
2021-12-12 17:38:23 +05:30
|
|
|
// Inflate the layout for this fragment
|
2022-07-01 00:35:31 +05:30
|
|
|
return binding.root
|
2021-12-12 17:38:23 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
|
|
|
|
super.onViewCreated(view, savedInstanceState)
|
2022-03-15 14:21:31 +05:30
|
|
|
hideKeyboard()
|
2022-03-28 04:08:00 +05:30
|
|
|
|
2022-06-26 14:06:34 +05:30
|
|
|
setSponsorBlockPrefs()
|
2022-06-29 19:37:58 +05:30
|
|
|
createExoPlayer(view)
|
2022-06-14 15:51:18 +05:30
|
|
|
initializeTransitionLayout(view)
|
2022-06-29 19:37:58 +05:30
|
|
|
playVideo(view)
|
2022-06-14 15:51:18 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
private fun initializeTransitionLayout(view: View) {
|
2022-03-27 11:49:17 +05:30
|
|
|
videoId = videoId!!.replace("/watch?v=", "")
|
2022-07-01 00:35:31 +05:30
|
|
|
|
2021-12-13 01:47:55 +05:30
|
|
|
val mainActivity = activity as MainActivity
|
2022-07-01 00:35:31 +05:30
|
|
|
mainActivity.binding.container.visibility = View.VISIBLE
|
|
|
|
|
|
|
|
exoPlayerView = binding.player
|
2022-07-01 00:56:16 +05:30
|
|
|
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.playerMotionLayout.addTransitionListener(object : MotionLayout.TransitionListener {
|
2021-12-12 17:38:23 +05:30
|
|
|
override fun onTransitionStarted(
|
|
|
|
motionLayout: MotionLayout?,
|
|
|
|
startId: Int,
|
|
|
|
endId: Int
|
|
|
|
) {
|
|
|
|
}
|
|
|
|
|
2022-04-15 12:08:53 +05:30
|
|
|
override fun onTransitionChange(
|
|
|
|
motionLayout: MotionLayout?,
|
|
|
|
startId: Int,
|
|
|
|
endId: Int,
|
|
|
|
progress: Float
|
|
|
|
) {
|
2021-12-12 17:38:23 +05:30
|
|
|
val mainActivity = activity as MainActivity
|
2022-04-15 12:08:53 +05:30
|
|
|
val mainMotionLayout =
|
2022-07-01 00:35:31 +05:30
|
|
|
mainActivity.binding.mainMotionLayout
|
2021-12-12 17:38:23 +05:30
|
|
|
mainMotionLayout.progress = abs(progress)
|
2022-06-24 18:46:14 +05:30
|
|
|
exoPlayerView.hideController()
|
2022-03-27 11:49:17 +05:30
|
|
|
eId = endId
|
|
|
|
sId = startId
|
2021-12-12 17:38:23 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
override fun onTransitionCompleted(motionLayout: MotionLayout?, currentId: Int) {
|
|
|
|
println(currentId)
|
|
|
|
val mainActivity = activity as MainActivity
|
2022-04-15 12:08:53 +05:30
|
|
|
val mainMotionLayout =
|
2022-07-01 00:35:31 +05:30
|
|
|
mainActivity.binding.mainMotionLayout
|
2022-03-27 11:49:17 +05:30
|
|
|
if (currentId == eId) {
|
2022-06-26 20:50:53 +05:30
|
|
|
isMiniPlayerVisible = true
|
2022-06-19 17:09:41 +05:30
|
|
|
exoPlayerView.useController = false
|
2022-06-24 18:46:14 +05:30
|
|
|
mainMotionLayout.progress = 1F
|
2022-03-27 11:49:17 +05:30
|
|
|
} else if (currentId == sId) {
|
2022-06-26 20:50:53 +05:30
|
|
|
isMiniPlayerVisible = false
|
2022-06-19 17:09:41 +05:30
|
|
|
exoPlayerView.useController = true
|
2022-06-24 18:46:14 +05:30
|
|
|
mainMotionLayout.progress = 0F
|
2021-12-12 17:38:23 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
override fun onTransitionTrigger(
|
2022-07-01 00:55:40 +05:30
|
|
|
MotionLayout: MotionLayout?,
|
2021-12-12 17:38:23 +05:30
|
|
|
triggerId: Int,
|
|
|
|
positive: Boolean,
|
|
|
|
progress: Float
|
|
|
|
) {
|
|
|
|
}
|
|
|
|
})
|
2022-06-03 22:13:15 +05:30
|
|
|
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.playerMotionLayout.progress = 1.toFloat()
|
|
|
|
binding.playerMotionLayout.transitionToStart()
|
2022-06-14 15:51:18 +05:30
|
|
|
|
2022-07-01 00:35:31 +05:30
|
|
|
binding.closeImageView.setOnClickListener {
|
2022-06-26 20:50:53 +05:30
|
|
|
isMiniPlayerVisible = false
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.playerMotionLayout.transitionToEnd()
|
2021-12-13 01:47:55 +05:30
|
|
|
val mainActivity = activity as MainActivity
|
2022-06-06 21:45:54 +05:30
|
|
|
mainActivity.requestedOrientation = ActivityInfo.SCREEN_ORIENTATION_USER_PORTRAIT
|
2021-12-13 01:47:55 +05:30
|
|
|
mainActivity.supportFragmentManager.beginTransaction()
|
|
|
|
.remove(this)
|
|
|
|
.commit()
|
|
|
|
}
|
2022-07-02 22:01:56 +05:30
|
|
|
playerBinding.closeImageButton.setOnClickListener {
|
2022-06-26 20:52:06 +05:30
|
|
|
isMiniPlayerVisible = false
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.playerMotionLayout.transitionToEnd()
|
2021-12-14 02:58:17 +05:30
|
|
|
val mainActivity = activity as MainActivity
|
2022-06-06 21:45:54 +05:30
|
|
|
mainActivity.requestedOrientation = ActivityInfo.SCREEN_ORIENTATION_USER_PORTRAIT
|
2021-12-14 02:58:17 +05:30
|
|
|
mainActivity.supportFragmentManager.beginTransaction()
|
|
|
|
.remove(this)
|
|
|
|
.commit()
|
|
|
|
}
|
2022-07-01 00:35:31 +05:30
|
|
|
binding.playImageView.setOnClickListener {
|
2022-03-27 11:49:17 +05:30
|
|
|
paused = if (paused) {
|
2022-07-01 00:35:31 +05:30
|
|
|
binding.playImageView.setImageResource(R.drawable.ic_pause)
|
2021-12-14 02:58:17 +05:30
|
|
|
exoPlayer.play()
|
|
|
|
false
|
2022-03-27 11:49:17 +05:30
|
|
|
} else {
|
2022-07-01 00:35:31 +05:30
|
|
|
binding.playImageView.setImageResource(R.drawable.ic_play)
|
2021-12-14 02:58:17 +05:30
|
|
|
exoPlayer.pause()
|
|
|
|
true
|
|
|
|
}
|
|
|
|
}
|
2022-03-28 04:08:00 +05:30
|
|
|
|
2022-06-28 20:23:18 +05:30
|
|
|
// video description and chapters toggle
|
2022-07-01 00:35:31 +05:30
|
|
|
binding.playerTitleLayout.setOnClickListener {
|
|
|
|
binding.playerDescriptionArrow.animate().rotationBy(180F).setDuration(250).start()
|
|
|
|
binding.descLinLayout.visibility =
|
|
|
|
if (binding.descLinLayout.isVisible) View.GONE else View.VISIBLE
|
2022-03-28 04:08:00 +05:30
|
|
|
}
|
2022-05-08 16:29:44 +05:30
|
|
|
|
2022-07-01 18:42:00 +05:30
|
|
|
binding.commentsToggle.setOnClickListener {
|
2022-07-01 20:25:21 +05:30
|
|
|
toggleComments()
|
|
|
|
}
|
2022-05-08 16:29:44 +05:30
|
|
|
|
2022-06-27 22:32:10 +05:30
|
|
|
// FullScreen button trigger
|
2022-07-02 22:01:56 +05:30
|
|
|
playerBinding.fullscreen.setOnClickListener {
|
2022-06-24 18:33:29 +05:30
|
|
|
exoPlayerView.hideController()
|
2022-07-03 21:52:04 +05:30
|
|
|
var scaleFactor: Float? = null
|
2022-03-27 11:49:17 +05:30
|
|
|
if (!isFullScreen) {
|
2022-07-03 21:43:52 +05:30
|
|
|
// go to fullscreen mode
|
2022-07-01 00:55:40 +05:30
|
|
|
with(binding.playerMotionLayout) {
|
2021-12-15 15:54:12 +05:30
|
|
|
getConstraintSet(R.id.start).constrainHeight(R.id.player, -1)
|
2022-03-27 11:49:17 +05:30
|
|
|
enableTransition(R.id.yt_transition, false)
|
2021-12-15 15:54:12 +05:30
|
|
|
}
|
2022-06-27 22:32:10 +05:30
|
|
|
|
2022-07-01 00:35:31 +05:30
|
|
|
binding.mainContainer.isClickable = true
|
|
|
|
binding.linLayout.visibility = View.GONE
|
2022-07-02 22:01:56 +05:30
|
|
|
playerBinding.fullscreen.setImageResource(R.drawable.ic_fullscreen_exit)
|
|
|
|
playerBinding.exoTitle.visibility = View.VISIBLE
|
2022-06-27 22:32:10 +05:30
|
|
|
|
2022-07-03 21:52:04 +05:30
|
|
|
scaleFactor = 1.3F
|
|
|
|
|
2022-02-10 13:52:05 +05:30
|
|
|
val mainActivity = activity as MainActivity
|
2022-07-05 00:49:16 +05:30
|
|
|
val fullscreenOrientationPref = PreferenceHelper
|
2022-07-05 01:00:13 +05:30
|
|
|
.getString(requireContext(), "fullscreen_orientation", "ratio")
|
|
|
|
Log.e(TAG, fullscreenOrientationPref.toString())
|
|
|
|
val orientation = when (fullscreenOrientationPref) {
|
2022-07-05 00:49:16 +05:30
|
|
|
"ratio" -> {
|
|
|
|
val videoSize = exoPlayer.videoSize
|
|
|
|
// probably a youtube shorts video
|
2022-07-05 01:00:13 +05:30
|
|
|
Log.e(TAG, videoSize.height.toString() + " " + videoSize.width.toString())
|
2022-07-05 00:49:16 +05:30
|
|
|
if (videoSize.height > videoSize.width) ActivityInfo.SCREEN_ORIENTATION_USER_PORTRAIT
|
|
|
|
// a video with normal aspect ratio
|
|
|
|
else ActivityInfo.SCREEN_ORIENTATION_USER_LANDSCAPE
|
|
|
|
}
|
|
|
|
"auto" -> ActivityInfo.SCREEN_ORIENTATION_USER
|
|
|
|
"landscape" -> ActivityInfo.SCREEN_ORIENTATION_USER_LANDSCAPE
|
|
|
|
"portrait" -> ActivityInfo.SCREEN_ORIENTATION_USER_PORTRAIT
|
|
|
|
else -> ActivityInfo.SCREEN_ORIENTATION_USER_LANDSCAPE
|
|
|
|
}
|
2022-07-05 01:00:13 +05:30
|
|
|
mainActivity.requestedOrientation = orientation
|
2022-04-15 12:08:53 +05:30
|
|
|
} else {
|
2022-07-03 21:43:52 +05:30
|
|
|
// leave fullscreen mode
|
2022-07-01 00:55:40 +05:30
|
|
|
with(binding.playerMotionLayout) {
|
2021-12-15 15:54:12 +05:30
|
|
|
getConstraintSet(R.id.start).constrainHeight(R.id.player, 0)
|
2022-03-27 11:49:17 +05:30
|
|
|
enableTransition(R.id.yt_transition, true)
|
2021-12-15 15:54:12 +05:30
|
|
|
}
|
2022-06-27 22:32:10 +05:30
|
|
|
|
2022-07-01 00:35:31 +05:30
|
|
|
binding.mainContainer.isClickable = false
|
|
|
|
binding.linLayout.visibility = View.VISIBLE
|
2022-07-02 22:01:56 +05:30
|
|
|
playerBinding.fullscreen.setImageResource(R.drawable.ic_fullscreen)
|
|
|
|
playerBinding.exoTitle.visibility = View.INVISIBLE
|
2022-06-27 22:32:10 +05:30
|
|
|
|
2022-07-03 21:52:04 +05:30
|
|
|
scaleFactor = 1F
|
|
|
|
|
2022-02-10 13:52:05 +05:30
|
|
|
val mainActivity = activity as MainActivity
|
2022-06-06 21:45:54 +05:30
|
|
|
mainActivity.requestedOrientation = ActivityInfo.SCREEN_ORIENTATION_USER_PORTRAIT
|
2021-12-15 15:54:12 +05:30
|
|
|
}
|
2022-06-27 22:32:10 +05:30
|
|
|
isFullScreen = !isFullScreen
|
2022-07-03 21:52:04 +05:30
|
|
|
|
|
|
|
// scale the exo player center controls
|
|
|
|
playerBinding.exoPlayPause.scaleX = scaleFactor
|
|
|
|
playerBinding.exoPlayPause.scaleY = scaleFactor
|
2021-12-14 21:45:53 +05:30
|
|
|
}
|
2022-05-08 23:02:13 +05:30
|
|
|
|
2022-06-03 00:55:33 +05:30
|
|
|
// switching between original aspect ratio (black bars) and zoomed to fill device screen
|
2022-07-02 22:01:56 +05:30
|
|
|
playerBinding.aspectRatioButton.setOnClickListener {
|
2022-06-03 01:06:58 +05:30
|
|
|
if (isZoomed) {
|
2022-06-03 00:55:33 +05:30
|
|
|
exoPlayerView.resizeMode = AspectRatioFrameLayout.RESIZE_MODE_FIT
|
|
|
|
isZoomed = false
|
|
|
|
} else {
|
|
|
|
exoPlayerView.resizeMode = AspectRatioFrameLayout.RESIZE_MODE_ZOOM
|
|
|
|
isZoomed = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-24 20:56:36 +05:30
|
|
|
// lock and unlock the player
|
2022-07-02 22:01:56 +05:30
|
|
|
playerBinding.lockPlayer.setOnClickListener {
|
2022-06-24 20:56:36 +05:30
|
|
|
// change the locked/unlocked icon
|
|
|
|
if (!isPlayerLocked) {
|
2022-07-02 22:01:56 +05:30
|
|
|
playerBinding.lockPlayer.setImageResource(R.drawable.ic_locked)
|
2022-06-24 20:56:36 +05:30
|
|
|
} else {
|
2022-07-02 22:01:56 +05:30
|
|
|
playerBinding.lockPlayer.setImageResource(R.drawable.ic_unlocked)
|
2022-06-24 20:56:36 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
// show/hide all the controls
|
|
|
|
lockPlayer(isPlayerLocked)
|
|
|
|
|
|
|
|
// change locked status
|
|
|
|
isPlayerLocked = !isPlayerLocked
|
|
|
|
}
|
|
|
|
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.playerScrollView.viewTreeObserver
|
2022-05-08 23:02:13 +05:30
|
|
|
.addOnScrollChangedListener {
|
2022-07-01 00:55:40 +05:30
|
|
|
if (binding.playerScrollView.getChildAt(0).bottom
|
|
|
|
== (binding.playerScrollView.height + binding.playerScrollView.scrollY) &&
|
2022-05-21 13:32:04 +05:30
|
|
|
nextPage != null
|
|
|
|
) {
|
2022-05-08 23:02:13 +05:30
|
|
|
fetchNextComments()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.commentsRecView.layoutManager = LinearLayoutManager(view.context)
|
|
|
|
binding.commentsRecView.setItemViewCacheSize(20)
|
2022-05-08 16:03:01 +05:30
|
|
|
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.relatedRecView.layoutManager =
|
2022-04-15 12:08:53 +05:30
|
|
|
GridLayoutManager(view.context, resources.getInteger(R.integer.grid_items))
|
2021-12-13 01:47:55 +05:30
|
|
|
}
|
2021-12-12 20:31:44 +05:30
|
|
|
|
2022-06-12 22:56:38 +05:30
|
|
|
private fun toggleComments() {
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.commentsRecView.visibility =
|
|
|
|
if (binding.commentsRecView.isVisible) View.GONE else View.VISIBLE
|
|
|
|
binding.relatedRecView.visibility =
|
|
|
|
if (binding.relatedRecView.isVisible) View.GONE else View.VISIBLE
|
2022-06-12 22:56:38 +05:30
|
|
|
if (!commentsLoaded!!) fetchComments()
|
|
|
|
}
|
|
|
|
|
2022-06-20 17:52:28 +05:30
|
|
|
override fun onPause() {
|
|
|
|
// pause the player if the screen is turned off
|
2022-06-26 01:51:22 +05:30
|
|
|
val pausePlayerOnScreenOffEnabled = PreferenceHelper.getBoolean(
|
|
|
|
requireContext(),
|
|
|
|
"pause_screen_off",
|
|
|
|
false
|
|
|
|
)
|
2022-06-20 17:52:28 +05:30
|
|
|
|
|
|
|
// check whether the screen is on
|
|
|
|
val pm = context?.getSystemService(Context.POWER_SERVICE) as PowerManager
|
|
|
|
val isScreenOn = pm.isInteractive
|
|
|
|
|
|
|
|
// pause player if screen off and setting enabled
|
2022-06-29 19:37:58 +05:30
|
|
|
if (
|
2022-07-01 14:41:24 +05:30
|
|
|
this::exoPlayer.isInitialized && !isScreenOn && pausePlayerOnScreenOffEnabled
|
2022-06-29 19:37:58 +05:30
|
|
|
) {
|
2022-06-20 17:52:28 +05:30
|
|
|
exoPlayer.pause()
|
|
|
|
}
|
|
|
|
super.onPause()
|
|
|
|
}
|
|
|
|
|
2022-02-06 21:51:37 +05:30
|
|
|
override fun onDestroy() {
|
|
|
|
super.onDestroy()
|
2021-12-14 02:58:17 +05:30
|
|
|
try {
|
2022-07-02 22:34:19 +05:30
|
|
|
saveWatchPosition()
|
2022-06-05 15:12:33 +05:30
|
|
|
mediaSession.isActive = false
|
|
|
|
mediaSession.release()
|
|
|
|
mediaSessionConnector.setPlayer(null)
|
|
|
|
playerNotification.setPlayer(null)
|
|
|
|
val notificationManager = context?.getSystemService(
|
2022-06-01 12:22:38 +05:30
|
|
|
Context.NOTIFICATION_SERVICE
|
|
|
|
) as NotificationManager
|
2022-06-06 20:12:46 +05:30
|
|
|
notificationManager.cancel(1)
|
2022-06-05 15:12:33 +05:30
|
|
|
exoPlayer.release()
|
2022-04-15 12:08:53 +05:30
|
|
|
} catch (e: Exception) {
|
|
|
|
}
|
2021-12-13 01:47:55 +05:30
|
|
|
}
|
2021-12-12 20:31:44 +05:30
|
|
|
|
2022-07-02 22:34:19 +05:30
|
|
|
// save the watch position if video isn't finished and option enabled
|
|
|
|
private fun saveWatchPosition() {
|
|
|
|
val watchPositionsEnabled = PreferenceHelper.getBoolean(
|
|
|
|
requireContext(),
|
|
|
|
"watch_positions_toggle",
|
|
|
|
true
|
|
|
|
)
|
|
|
|
if (watchPositionsEnabled && exoPlayer.currentPosition != exoPlayer.duration) {
|
|
|
|
PreferenceHelper.saveWatchPosition(
|
|
|
|
requireContext(),
|
|
|
|
videoId!!,
|
|
|
|
exoPlayer.currentPosition
|
|
|
|
)
|
|
|
|
} else if (watchPositionsEnabled) {
|
|
|
|
// delete watch position if video has ended
|
|
|
|
PreferenceHelper.removeWatchPosition(requireContext(), videoId!!)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-20 03:52:10 +05:30
|
|
|
private fun checkForSegments() {
|
2022-06-26 14:06:34 +05:30
|
|
|
if (!exoPlayer.isPlaying || !sponsorBlockPrefs.sponsorBlockEnabled) return
|
2022-05-16 15:41:22 +05:30
|
|
|
|
|
|
|
exoPlayerView.postDelayed(this::checkForSegments, 100)
|
|
|
|
|
2022-06-24 20:56:36 +05:30
|
|
|
if (!::segmentData.isInitialized || segmentData.segments.isEmpty()) {
|
2022-05-16 15:41:22 +05:30
|
|
|
return
|
2022-06-24 20:56:36 +05:30
|
|
|
}
|
2022-05-16 15:41:22 +05:30
|
|
|
|
|
|
|
segmentData.segments.forEach { segment: Segment ->
|
|
|
|
val segmentStart = (segment.segment!![0] * 1000.0f).toLong()
|
2022-05-21 13:32:04 +05:30
|
|
|
val segmentEnd = (segment.segment[1] * 1000.0f).toLong()
|
2022-05-16 15:41:22 +05:30
|
|
|
val currentPosition = exoPlayer.currentPosition
|
2022-05-20 03:52:10 +05:30
|
|
|
if (currentPosition in segmentStart until segmentEnd) {
|
2022-06-26 14:06:34 +05:30
|
|
|
if (sponsorBlockPrefs.sponsorNotificationsEnabled) {
|
2022-05-23 00:07:09 +05:30
|
|
|
Toast.makeText(context, R.string.segment_skipped, Toast.LENGTH_SHORT).show()
|
|
|
|
}
|
2022-05-20 03:52:10 +05:30
|
|
|
exoPlayer.seekTo(segmentEnd)
|
2022-05-16 15:41:22 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-29 19:37:58 +05:30
|
|
|
private fun playVideo(view: View) {
|
2021-12-12 20:31:44 +05:30
|
|
|
fun run() {
|
2021-12-18 16:34:14 +05:30
|
|
|
lifecycleScope.launchWhenCreated {
|
|
|
|
val response = try {
|
|
|
|
RetrofitInstance.api.getStreams(videoId!!)
|
2022-03-27 11:49:17 +05:30
|
|
|
} catch (e: IOException) {
|
2021-12-18 16:34:14 +05:30
|
|
|
println(e)
|
|
|
|
Log.e(TAG, "IOException, you might not have internet connection")
|
2022-04-15 12:08:53 +05:30
|
|
|
Toast.makeText(context, R.string.unknown_error, Toast.LENGTH_SHORT).show()
|
2021-12-18 16:34:14 +05:30
|
|
|
return@launchWhenCreated
|
|
|
|
} catch (e: HttpException) {
|
|
|
|
Log.e(TAG, "HttpException, unexpected response")
|
2022-04-15 12:08:53 +05:30
|
|
|
Toast.makeText(context, R.string.server_error, Toast.LENGTH_SHORT).show()
|
2021-12-18 16:34:14 +05:30
|
|
|
return@launchWhenCreated
|
|
|
|
}
|
2022-06-14 15:30:58 +05:30
|
|
|
// for the notification description adapter
|
|
|
|
title = response.title!!
|
|
|
|
uploader = response.uploader!!
|
|
|
|
thumbnailUrl = response.thumbnailUrl!!
|
|
|
|
|
2022-06-29 12:34:11 +05:30
|
|
|
// save whether related streams and autoplay are enabled
|
2022-06-26 14:38:10 +05:30
|
|
|
autoplay = PreferenceHelper.getBoolean(requireContext(), "autoplay", false)
|
2022-06-26 15:50:56 +05:30
|
|
|
relatedStreamsEnabled =
|
|
|
|
PreferenceHelper.getBoolean(requireContext(), "related_streams_toggle", true)
|
2022-06-17 16:51:55 +05:30
|
|
|
// save related streams for autoplay
|
|
|
|
relatedStreams = response.relatedStreams
|
2022-06-29 12:34:11 +05:30
|
|
|
|
2022-06-14 15:30:58 +05:30
|
|
|
runOnUiThread {
|
2022-06-26 21:19:42 +05:30
|
|
|
// set media sources for the player
|
2022-06-14 15:30:58 +05:30
|
|
|
setResolutionAndSubtitles(view, response)
|
2022-06-28 20:02:26 +05:30
|
|
|
prepareExoPlayerView()
|
2022-06-26 21:19:42 +05:30
|
|
|
initializePlayerView(view, response)
|
2022-07-02 22:34:19 +05:30
|
|
|
seekToWatchPosition()
|
2022-07-03 15:43:38 +05:30
|
|
|
exoPlayer.prepare()
|
2022-06-14 15:30:58 +05:30
|
|
|
exoPlayer.play()
|
2022-06-26 21:37:51 +05:30
|
|
|
exoPlayerView.useController = true
|
2022-06-14 15:30:58 +05:30
|
|
|
initializePlayerNotification(requireContext())
|
|
|
|
fetchSponsorBlockSegments()
|
2022-06-26 21:19:42 +05:30
|
|
|
// show comments if related streams disabled
|
2022-06-14 15:30:58 +05:30
|
|
|
if (!relatedStreamsEnabled) toggleComments()
|
2022-06-29 12:34:11 +05:30
|
|
|
// prepare for autoplay
|
|
|
|
initAutoPlay()
|
2022-07-01 21:41:12 +05:30
|
|
|
val watchHistoryEnabled =
|
|
|
|
PreferenceHelper.getBoolean(requireContext(), "Watch_history_toggle", true)
|
2022-07-01 23:18:20 +05:30
|
|
|
if (watchHistoryEnabled) {
|
2022-07-01 21:41:12 +05:30
|
|
|
PreferenceHelper.addToWatchHistory(requireContext(), videoId!!, response)
|
2022-07-01 23:18:20 +05:30
|
|
|
}
|
2022-06-29 12:34:11 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
run()
|
|
|
|
}
|
|
|
|
|
2022-07-02 22:34:19 +05:30
|
|
|
private fun seekToWatchPosition() {
|
2022-07-03 15:43:38 +05:30
|
|
|
// seek to saved watch position if available
|
2022-07-02 22:34:19 +05:30
|
|
|
val watchPositions = PreferenceHelper.getWatchPositions(requireContext())
|
2022-07-07 19:28:51 +05:30
|
|
|
var position: Long? = null
|
2022-07-02 22:34:19 +05:30
|
|
|
watchPositions.forEach {
|
2022-07-07 19:28:51 +05:30
|
|
|
if (it.videoId == videoId) position = it.position
|
2022-07-02 22:34:19 +05:30
|
|
|
}
|
2022-07-03 15:43:38 +05:30
|
|
|
// support for time stamped links
|
2022-07-07 19:28:51 +05:30
|
|
|
val timeStamp: Long? = arguments?.getLong("timeStamp")
|
|
|
|
if (timeStamp != null && timeStamp != 0L) {
|
|
|
|
position = timeStamp * 1000
|
2022-07-03 15:43:38 +05:30
|
|
|
}
|
2022-07-07 19:28:51 +05:30
|
|
|
if (position != null) exoPlayer.seekTo(position!!)
|
2022-07-02 22:34:19 +05:30
|
|
|
}
|
|
|
|
|
2022-06-29 12:59:23 +05:30
|
|
|
// the function is working recursively
|
2022-06-29 12:34:11 +05:30
|
|
|
private fun initAutoPlay() {
|
|
|
|
// save related streams for autoplay
|
|
|
|
if (autoplay) {
|
2022-06-29 12:52:22 +05:30
|
|
|
// if it's a playlist use the next video
|
2022-06-29 12:34:11 +05:30
|
|
|
if (playlistId != null) {
|
2022-06-29 12:52:22 +05:30
|
|
|
lateinit var playlist: Playlist // var for saving the list in
|
2022-06-29 12:59:23 +05:30
|
|
|
// runs only the first time when starting a video from a playlist
|
2022-06-29 12:34:11 +05:30
|
|
|
if (playlistStreamIds.isEmpty()) {
|
|
|
|
CoroutineScope(Dispatchers.IO).launch {
|
|
|
|
// fetch the playlists videos
|
|
|
|
playlist = RetrofitInstance.api.getPlaylist(playlistId!!)
|
2022-06-29 12:59:23 +05:30
|
|
|
// save the playlist urls in the array
|
2022-06-29 12:34:11 +05:30
|
|
|
playlist.relatedStreams?.forEach { video ->
|
|
|
|
playlistStreamIds += video.url?.replace("/watch?v=", "")!!
|
|
|
|
}
|
2022-06-29 12:59:23 +05:30
|
|
|
// save playlistNextPage for usage if video is not contained
|
2022-06-29 12:52:22 +05:30
|
|
|
playlistNextPage = playlist.nextpage
|
2022-06-29 12:59:23 +05:30
|
|
|
// restart the function after videos are loaded
|
2022-06-29 12:52:22 +05:30
|
|
|
initAutoPlay()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// if the playlists contain the video, then save the next video as next stream
|
|
|
|
else if (playlistStreamIds.contains(videoId)) {
|
|
|
|
val index = playlistStreamIds.indexOf(videoId)
|
|
|
|
// check whether there's a next video
|
|
|
|
if (index + 1 <= playlistStreamIds.size) {
|
|
|
|
nextStreamId = playlistStreamIds[index + 1]
|
|
|
|
}
|
|
|
|
// fetch the next page of the playlist if the video isn't contained
|
|
|
|
} else if (playlistNextPage != null) {
|
|
|
|
CoroutineScope(Dispatchers.IO).launch {
|
|
|
|
RetrofitInstance.api.getPlaylistNextPage(playlistId!!, playlistNextPage!!)
|
2022-06-29 12:59:23 +05:30
|
|
|
// append all the playlist item urls to the array
|
2022-06-29 12:52:22 +05:30
|
|
|
playlist.relatedStreams?.forEach { video ->
|
|
|
|
playlistStreamIds += video.url?.replace("/watch?v=", "")!!
|
2022-06-29 12:34:11 +05:30
|
|
|
}
|
2022-06-29 12:59:23 +05:30
|
|
|
// save playlistNextPage for usage if video is not contained
|
2022-06-29 12:52:22 +05:30
|
|
|
playlistNextPage = playlist.nextpage
|
2022-06-29 12:59:23 +05:30
|
|
|
// restart the function after videos are loaded
|
2022-06-29 12:52:22 +05:30
|
|
|
initAutoPlay()
|
2022-06-29 12:34:11 +05:30
|
|
|
}
|
|
|
|
}
|
2022-06-29 12:59:23 +05:30
|
|
|
// else: the video must be the last video of the playlist so nothing happens
|
|
|
|
|
2022-06-29 12:52:22 +05:30
|
|
|
// if it's not a playlist then use the next related video
|
2022-06-29 12:34:11 +05:30
|
|
|
} else if (relatedStreams != null && relatedStreams!!.isNotEmpty()) {
|
|
|
|
// save next video from related streams for autoplay
|
2022-07-01 14:41:24 +05:30
|
|
|
nextStreamId = relatedStreams!![0].url!!.replace("/watch?v=", "")
|
2022-06-29 12:34:11 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-29 12:59:23 +05:30
|
|
|
// used for autoplay and skipping to next video
|
2022-06-29 12:34:11 +05:30
|
|
|
private fun playNextVideo() {
|
2022-06-29 12:59:23 +05:30
|
|
|
// check whether there is a new video in the queue
|
|
|
|
// by making sure that the next and the current video aren't the same
|
|
|
|
if (videoId != nextStreamId) {
|
|
|
|
// save the id of the next stream as videoId and load the next video
|
|
|
|
videoId = nextStreamId
|
2022-06-29 19:37:58 +05:30
|
|
|
playVideo(view!!)
|
2022-06-29 12:59:23 +05:30
|
|
|
}
|
2022-06-29 12:34:11 +05:30
|
|
|
}
|
|
|
|
|
2022-06-26 14:06:34 +05:30
|
|
|
private fun setSponsorBlockPrefs() {
|
|
|
|
sponsorBlockPrefs.sponsorBlockEnabled =
|
|
|
|
PreferenceHelper.getBoolean(requireContext(), "sb_enabled_key", true)
|
|
|
|
sponsorBlockPrefs.sponsorNotificationsEnabled =
|
|
|
|
PreferenceHelper.getBoolean(requireContext(), "sb_notifications_key", true)
|
|
|
|
sponsorBlockPrefs.introEnabled =
|
|
|
|
PreferenceHelper.getBoolean(requireContext(), "intro_category_key", false)
|
|
|
|
sponsorBlockPrefs.selfPromoEnabled =
|
|
|
|
PreferenceHelper.getBoolean(requireContext(), "selfpromo_category_key", false)
|
|
|
|
sponsorBlockPrefs.interactionEnabled =
|
|
|
|
PreferenceHelper.getBoolean(requireContext(), "interaction_category_key", false)
|
|
|
|
sponsorBlockPrefs.sponsorsEnabled =
|
|
|
|
PreferenceHelper.getBoolean(requireContext(), "sponsors_category_key", true)
|
|
|
|
sponsorBlockPrefs.outroEnabled =
|
|
|
|
PreferenceHelper.getBoolean(requireContext(), "outro_category_key", false)
|
|
|
|
sponsorBlockPrefs.fillerEnabled =
|
|
|
|
PreferenceHelper.getBoolean(requireContext(), "filler_category_key", false)
|
|
|
|
sponsorBlockPrefs.musicOffTopicEnabled =
|
|
|
|
PreferenceHelper.getBoolean(requireContext(), "music_offtopic_category_key", false)
|
|
|
|
sponsorBlockPrefs.previewEnabled =
|
|
|
|
PreferenceHelper.getBoolean(requireContext(), "preview_category_key", false)
|
|
|
|
}
|
|
|
|
|
2022-06-14 15:30:58 +05:30
|
|
|
private fun fetchSponsorBlockSegments() {
|
|
|
|
fun run() {
|
2022-06-29 19:37:58 +05:30
|
|
|
lifecycleScope.launch(Dispatchers.IO) {
|
2022-06-26 14:06:34 +05:30
|
|
|
if (sponsorBlockPrefs.sponsorBlockEnabled) {
|
2022-05-16 16:04:13 +05:30
|
|
|
val categories: ArrayList<String> = arrayListOf()
|
2022-06-26 14:06:34 +05:30
|
|
|
if (sponsorBlockPrefs.introEnabled) {
|
2022-05-16 16:04:13 +05:30
|
|
|
categories.add("intro")
|
|
|
|
}
|
2022-06-26 14:06:34 +05:30
|
|
|
if (sponsorBlockPrefs.selfPromoEnabled) {
|
2022-05-16 16:04:13 +05:30
|
|
|
categories.add("selfpromo")
|
|
|
|
}
|
2022-06-26 14:06:34 +05:30
|
|
|
if (sponsorBlockPrefs.interactionEnabled) {
|
2022-05-16 16:04:13 +05:30
|
|
|
categories.add("interaction")
|
|
|
|
}
|
2022-06-26 14:06:34 +05:30
|
|
|
if (sponsorBlockPrefs.sponsorsEnabled) {
|
2022-05-16 16:04:13 +05:30
|
|
|
categories.add("sponsor")
|
|
|
|
}
|
2022-06-26 14:06:34 +05:30
|
|
|
if (sponsorBlockPrefs.outroEnabled) {
|
2022-05-16 16:04:13 +05:30
|
|
|
categories.add("outro")
|
|
|
|
}
|
2022-06-26 14:06:34 +05:30
|
|
|
if (sponsorBlockPrefs.fillerEnabled) {
|
2022-06-14 17:20:02 +05:30
|
|
|
categories.add("filler")
|
|
|
|
}
|
2022-06-26 14:06:34 +05:30
|
|
|
if (sponsorBlockPrefs.musicOffTopicEnabled) {
|
2022-06-14 17:20:02 +05:30
|
|
|
categories.add("music_offtopic")
|
|
|
|
}
|
2022-06-26 14:06:34 +05:30
|
|
|
if (sponsorBlockPrefs.previewEnabled) {
|
2022-06-14 17:20:02 +05:30
|
|
|
categories.add("preview")
|
|
|
|
}
|
2022-05-20 03:52:10 +05:30
|
|
|
if (categories.size > 0) {
|
2022-05-16 16:04:13 +05:30
|
|
|
segmentData = try {
|
|
|
|
RetrofitInstance.api.getSegments(
|
|
|
|
videoId!!,
|
|
|
|
"[\"" + TextUtils.join("\",\"", categories) + "\"]"
|
|
|
|
)
|
|
|
|
} catch (e: IOException) {
|
|
|
|
println(e)
|
|
|
|
Log.e(TAG, "IOException, you might not have internet connection")
|
2022-06-29 19:37:58 +05:30
|
|
|
return@launch
|
2022-05-16 16:04:13 +05:30
|
|
|
} catch (e: HttpException) {
|
|
|
|
Log.e(TAG, "HttpException, unexpected response")
|
2022-06-29 19:37:58 +05:30
|
|
|
return@launch
|
2022-05-16 15:41:22 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-06-01 11:55:12 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
run()
|
|
|
|
}
|
2022-05-16 15:41:22 +05:30
|
|
|
|
2022-06-03 22:13:15 +05:30
|
|
|
private fun prepareExoPlayerView() {
|
2022-06-05 15:12:33 +05:30
|
|
|
exoPlayerView.apply {
|
|
|
|
setShowSubtitleButton(true)
|
|
|
|
setShowNextButton(false)
|
|
|
|
setShowPreviousButton(false)
|
|
|
|
setRepeatToggleModes(RepeatModeUtil.REPEAT_TOGGLE_MODE_ALL)
|
|
|
|
// controllerShowTimeoutMs = 1500
|
|
|
|
controllerHideOnTouch = true
|
2022-06-26 21:37:51 +05:30
|
|
|
useController = false
|
2022-06-05 15:12:33 +05:30
|
|
|
player = exoPlayer
|
|
|
|
}
|
2022-06-03 22:13:15 +05:30
|
|
|
}
|
|
|
|
|
2022-06-01 11:55:12 +05:30
|
|
|
private fun initializePlayerView(view: View, response: Streams) {
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.playerViewsInfo.text =
|
2022-06-10 18:33:48 +05:30
|
|
|
context?.getString(R.string.views, response.views.formatShort()) +
|
|
|
|
" • " + response.uploadDate
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.textLike.text = response.likes.formatShort()
|
|
|
|
binding.textDislike.text = response.dislikes.formatShort()
|
|
|
|
Picasso.get().load(response.uploaderAvatar).into(binding.playerChannelImage)
|
|
|
|
binding.playerChannelName.text = response.uploader
|
2022-06-03 22:13:15 +05:30
|
|
|
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.titleTextView.text = response.title
|
|
|
|
binding.playerTitle.text = response.title
|
|
|
|
binding.playerDescription.text = response.description
|
2022-06-03 22:13:15 +05:30
|
|
|
|
2022-07-02 22:01:56 +05:30
|
|
|
playerBinding.exoTitle.text = response.title
|
2022-06-27 22:32:10 +05:30
|
|
|
|
2022-07-08 20:56:45 +05:30
|
|
|
enableDoubleTapToSeek()
|
|
|
|
|
2022-07-03 15:43:38 +05:30
|
|
|
// init the chapters recyclerview
|
|
|
|
if (response.chapters != null) initializeChapters(response.chapters)
|
|
|
|
|
2022-06-03 22:13:15 +05:30
|
|
|
// Listener for play and pause icon change
|
2022-06-14 18:39:47 +05:30
|
|
|
exoPlayer.addListener(object : Player.Listener {
|
2022-06-03 22:13:15 +05:30
|
|
|
override fun onIsPlayingChanged(isPlaying: Boolean) {
|
2022-06-26 14:06:34 +05:30
|
|
|
if (isPlaying && sponsorBlockPrefs.sponsorBlockEnabled) {
|
2022-06-03 22:13:15 +05:30
|
|
|
exoPlayerView.postDelayed(
|
|
|
|
this@PlayerFragment::checkForSegments,
|
|
|
|
100
|
|
|
|
)
|
2022-06-03 02:24:18 +05:30
|
|
|
}
|
2022-06-03 22:13:15 +05:30
|
|
|
}
|
2022-06-16 01:33:32 +05:30
|
|
|
|
2022-07-03 16:58:24 +05:30
|
|
|
override fun onVideoSizeChanged(
|
|
|
|
videoSize: VideoSize
|
|
|
|
) {
|
|
|
|
// Set new width/height of view
|
|
|
|
// height or width must be cast to float as int/int will give 0
|
|
|
|
|
2022-07-03 21:57:15 +05:30
|
|
|
// Redraw the player container with the new layout height
|
2022-07-08 21:36:48 +05:30
|
|
|
val params = binding.player.layoutParams
|
|
|
|
params.height = videoSize.height / videoSize.width * params.width
|
|
|
|
binding.player.layoutParams = params
|
|
|
|
binding.player.requestLayout()
|
2022-07-03 16:58:24 +05:30
|
|
|
}
|
|
|
|
|
2022-06-14 18:39:47 +05:30
|
|
|
@Deprecated(message = "Deprecated", level = DeprecationLevel.HIDDEN)
|
2022-06-03 22:13:15 +05:30
|
|
|
override fun onPlayerStateChanged(
|
|
|
|
playWhenReady: Boolean,
|
|
|
|
playbackState: Int
|
|
|
|
) {
|
|
|
|
exoPlayerView.keepScreenOn = !(
|
|
|
|
playbackState == Player.STATE_IDLE ||
|
|
|
|
playbackState == Player.STATE_ENDED ||
|
|
|
|
!playWhenReady
|
|
|
|
)
|
2022-06-03 02:24:18 +05:30
|
|
|
|
2022-06-17 16:51:55 +05:30
|
|
|
// check if video has ended, next video is available and autoplay is enabled.
|
2022-06-17 17:03:10 +05:30
|
|
|
if (
|
2022-06-17 17:02:00 +05:30
|
|
|
playbackState == Player.STATE_ENDED &&
|
2022-06-29 12:34:11 +05:30
|
|
|
nextStreamId != null &&
|
2022-06-17 17:02:00 +05:30
|
|
|
!transitioning &&
|
|
|
|
autoplay
|
|
|
|
) {
|
2022-06-17 16:51:55 +05:30
|
|
|
transitioning = true
|
2022-06-29 12:34:11 +05:30
|
|
|
// check whether autoplay is enabled
|
|
|
|
if (autoplay) playNextVideo()
|
2022-06-17 16:51:55 +05:30
|
|
|
}
|
|
|
|
|
2022-06-03 22:13:15 +05:30
|
|
|
if (playWhenReady && playbackState == Player.STATE_READY) {
|
|
|
|
// media actually playing
|
2022-06-17 16:51:55 +05:30
|
|
|
transitioning = false
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.playImageView.setImageResource(R.drawable.ic_pause)
|
2022-06-03 22:13:15 +05:30
|
|
|
} else if (playWhenReady) {
|
|
|
|
// might be idle (plays after prepare()),
|
|
|
|
// buffering (plays when data available)
|
|
|
|
// or ended (plays when seek away from end)
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.playImageView.setImageResource(R.drawable.ic_play)
|
2022-06-03 22:13:15 +05:30
|
|
|
} else {
|
|
|
|
// player paused in any state
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.playImageView.setImageResource(R.drawable.ic_play)
|
2022-06-03 02:24:18 +05:30
|
|
|
}
|
2022-06-03 02:15:45 +05:30
|
|
|
}
|
2022-06-03 22:13:15 +05:30
|
|
|
})
|
|
|
|
|
|
|
|
// share button
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.relPlayerShare.setOnClickListener {
|
2022-06-21 01:17:35 +05:30
|
|
|
val shareDialog = ShareDialog(videoId!!, false)
|
2022-06-05 14:22:35 +05:30
|
|
|
shareDialog.show(childFragmentManager, "ShareDialog")
|
2022-06-03 22:13:15 +05:30
|
|
|
}
|
|
|
|
// check if livestream
|
|
|
|
if (response.duration!! > 0) {
|
|
|
|
// download clicked
|
2022-07-01 00:35:31 +05:30
|
|
|
binding.relPlayerDownload.setOnClickListener {
|
2022-06-03 22:13:15 +05:30
|
|
|
if (!IS_DOWNLOAD_RUNNING) {
|
|
|
|
val newFragment = DownloadDialog()
|
2022-06-14 18:39:47 +05:30
|
|
|
val bundle = Bundle()
|
2022-06-03 22:13:15 +05:30
|
|
|
bundle.putString("video_id", videoId)
|
|
|
|
newFragment.arguments = bundle
|
2022-07-08 01:50:24 +05:30
|
|
|
newFragment.show(childFragmentManager, "DownloadDialog")
|
2022-06-03 22:13:15 +05:30
|
|
|
} else {
|
|
|
|
Toast.makeText(context, R.string.dlisinprogress, Toast.LENGTH_SHORT)
|
|
|
|
.show()
|
2022-06-01 11:55:12 +05:30
|
|
|
}
|
|
|
|
}
|
2022-06-03 22:13:15 +05:30
|
|
|
} else {
|
|
|
|
Toast.makeText(context, R.string.cannotDownload, Toast.LENGTH_SHORT).show()
|
|
|
|
}
|
2022-06-03 02:15:45 +05:30
|
|
|
|
2022-06-03 22:13:15 +05:30
|
|
|
if (response.hls != null) {
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.relPlayerVlc.setOnClickListener {
|
2022-06-28 20:47:25 +05:30
|
|
|
// start an intent with video as mimetype using the hls stream
|
|
|
|
val uri: Uri = Uri.parse(response.hls)
|
|
|
|
val intent = Intent()
|
|
|
|
|
|
|
|
intent.action = Intent.ACTION_VIEW
|
|
|
|
intent.setDataAndType(uri, "video/*")
|
|
|
|
intent.putExtra(Intent.EXTRA_TITLE, title)
|
|
|
|
intent.putExtra("title", title)
|
|
|
|
intent.putExtra("artist", uploader)
|
|
|
|
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
|
|
|
|
|
|
|
|
startActivity(intent)
|
2022-06-03 02:15:45 +05:30
|
|
|
}
|
2022-06-03 22:13:15 +05:30
|
|
|
}
|
2022-06-12 22:56:38 +05:30
|
|
|
if (relatedStreamsEnabled) {
|
2022-06-28 20:47:25 +05:30
|
|
|
// only show related streams if enabled
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.relatedRecView.adapter = TrendingAdapter(
|
2022-06-12 22:56:38 +05:30
|
|
|
response.relatedStreams!!,
|
|
|
|
childFragmentManager
|
|
|
|
)
|
|
|
|
}
|
2022-06-28 20:47:25 +05:30
|
|
|
// set video description
|
2022-06-03 22:13:15 +05:30
|
|
|
val description = response.description!!
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.playerDescription.text =
|
2022-06-03 22:13:15 +05:30
|
|
|
// detect whether the description is html formatted
|
|
|
|
if (description.contains("<") && description.contains(">")) {
|
|
|
|
if (SDK_INT >= Build.VERSION_CODES.N) {
|
|
|
|
Html.fromHtml(description, Html.FROM_HTML_MODE_COMPACT)
|
|
|
|
.trim()
|
2022-06-03 02:15:45 +05:30
|
|
|
} else {
|
2022-06-03 22:13:15 +05:30
|
|
|
Html.fromHtml(description).trim()
|
2022-06-03 02:15:45 +05:30
|
|
|
}
|
2022-06-03 22:13:15 +05:30
|
|
|
} else {
|
|
|
|
description
|
|
|
|
}
|
2022-06-03 02:15:45 +05:30
|
|
|
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.playerChannel.setOnClickListener {
|
2022-06-03 22:13:15 +05:30
|
|
|
val activity = view.context as MainActivity
|
|
|
|
val bundle = bundleOf("channel_id" to response.uploaderUrl)
|
2022-07-01 21:59:47 +05:30
|
|
|
activity.navController.navigate(R.id.channelFragment, bundle)
|
2022-07-01 00:55:40 +05:30
|
|
|
activity.binding.mainMotionLayout.transitionToEnd()
|
|
|
|
binding.playerMotionLayout.transitionToEnd()
|
2022-06-03 22:13:15 +05:30
|
|
|
}
|
2022-06-26 14:38:10 +05:30
|
|
|
val token = PreferenceHelper.getToken(requireContext())
|
|
|
|
if (token != "") {
|
2022-06-03 22:13:15 +05:30
|
|
|
val channelId = response.uploaderUrl?.replace("/channel/", "")
|
2022-07-01 00:55:40 +05:30
|
|
|
isSubscribed(binding.playerSubscribe, channelId!!)
|
|
|
|
binding.save.setOnClickListener {
|
2022-06-03 22:13:15 +05:30
|
|
|
val newFragment = AddtoPlaylistDialog()
|
2022-06-14 18:39:47 +05:30
|
|
|
val bundle = Bundle()
|
2022-06-03 22:13:15 +05:30
|
|
|
bundle.putString("videoId", videoId)
|
|
|
|
newFragment.arguments = bundle
|
|
|
|
newFragment.show(childFragmentManager, "AddToPlaylist")
|
2022-06-01 11:55:12 +05:30
|
|
|
}
|
2022-06-03 02:15:45 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-08 20:56:45 +05:30
|
|
|
private fun enableDoubleTapToSeek() {
|
|
|
|
val seekIncrement =
|
|
|
|
PreferenceHelper.getString(requireContext(), "seek_increment", "5")?.toLong()!! * 1000
|
|
|
|
|
2022-07-08 21:36:48 +05:30
|
|
|
// enable rewind button
|
2022-07-08 21:19:32 +05:30
|
|
|
binding.rewindFL.setOnClickListener(
|
2022-07-08 20:56:45 +05:30
|
|
|
DoubleClickListener(
|
|
|
|
callback = object : DoubleClickListener.Callback {
|
|
|
|
override fun doubleClicked() {
|
2022-07-08 21:19:32 +05:30
|
|
|
binding.rewindBTN.visibility = View.VISIBLE
|
2022-07-08 20:56:45 +05:30
|
|
|
exoPlayer.seekTo(exoPlayer.currentPosition - seekIncrement)
|
2022-07-08 21:19:32 +05:30
|
|
|
Handler(Looper.getMainLooper()).postDelayed({
|
2022-07-08 21:36:48 +05:30
|
|
|
binding.rewindBTN.visibility = View.INVISIBLE
|
|
|
|
}, 700)
|
|
|
|
}
|
|
|
|
|
|
|
|
override fun singleClicked() {
|
|
|
|
if (exoPlayerView.isControllerFullyVisible) exoPlayerView.hideController()
|
|
|
|
else exoPlayerView.showController()
|
2022-07-08 20:56:45 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
2022-07-08 21:36:48 +05:30
|
|
|
// enable fast forward button
|
2022-07-08 21:19:32 +05:30
|
|
|
binding.forwardFL.setOnClickListener(
|
2022-07-08 20:56:45 +05:30
|
|
|
DoubleClickListener(
|
|
|
|
callback = object : DoubleClickListener.Callback {
|
|
|
|
override fun doubleClicked() {
|
2022-07-08 21:19:32 +05:30
|
|
|
binding.forwardBTN.visibility = View.VISIBLE
|
2022-07-08 20:56:45 +05:30
|
|
|
exoPlayer.seekTo(exoPlayer.currentPosition + seekIncrement)
|
2022-07-08 21:19:32 +05:30
|
|
|
Handler(Looper.getMainLooper()).postDelayed({
|
|
|
|
binding.forwardBTN.visibility = View.INVISIBLE
|
2022-07-08 21:36:48 +05:30
|
|
|
}, 700)
|
|
|
|
}
|
|
|
|
|
|
|
|
override fun singleClicked() {
|
|
|
|
if (exoPlayerView.isControllerFullyVisible) exoPlayerView.hideController()
|
|
|
|
else exoPlayerView.showController()
|
2022-07-08 20:56:45 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-06-12 22:56:38 +05:30
|
|
|
private fun initializeChapters(chapters: List<ChapterSegment>) {
|
2022-06-13 19:45:02 +05:30
|
|
|
if (chapters.isNotEmpty()) {
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.chaptersRecView.layoutManager =
|
2022-06-13 19:45:02 +05:30
|
|
|
LinearLayoutManager(this.context, LinearLayoutManager.HORIZONTAL, false)
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.chaptersRecView.adapter = ChaptersAdapter(chapters, exoPlayer)
|
|
|
|
binding.chaptersRecView.visibility = View.VISIBLE
|
2022-06-12 22:56:38 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-16 01:30:36 +05:30
|
|
|
private fun setMediaSource(
|
|
|
|
subtitle: MutableList<SubtitleConfiguration>,
|
2022-06-16 14:08:36 +05:30
|
|
|
videoUri: Uri,
|
|
|
|
audioUrl: String
|
2022-06-16 01:30:36 +05:30
|
|
|
) {
|
|
|
|
val dataSourceFactory: DataSource.Factory =
|
|
|
|
DefaultHttpDataSource.Factory()
|
|
|
|
val videoItem: MediaItem = MediaItem.Builder()
|
|
|
|
.setUri(videoUri)
|
|
|
|
.setSubtitleConfigurations(subtitle)
|
|
|
|
.build()
|
|
|
|
val videoSource: MediaSource =
|
|
|
|
DefaultMediaSourceFactory(dataSourceFactory)
|
|
|
|
.createMediaSource(videoItem)
|
2022-07-01 14:41:24 +05:30
|
|
|
val audioSource: MediaSource =
|
2022-06-16 01:30:36 +05:30
|
|
|
ProgressiveMediaSource.Factory(dataSourceFactory)
|
2022-06-16 14:08:36 +05:30
|
|
|
.createMediaSource(fromUri(audioUrl))
|
2022-06-16 01:30:36 +05:30
|
|
|
val mergeSource: MediaSource =
|
|
|
|
MergingMediaSource(videoSource, audioSource)
|
|
|
|
exoPlayer.setMediaSource(mergeSource)
|
|
|
|
}
|
|
|
|
|
2022-06-03 02:15:45 +05:30
|
|
|
private fun setResolutionAndSubtitles(view: View, response: Streams) {
|
2022-06-26 15:50:56 +05:30
|
|
|
val videoFormatPreference =
|
|
|
|
PreferenceHelper.getString(requireContext(), "player_video_format", "WEBM")
|
2022-06-26 01:51:22 +05:30
|
|
|
val defres = PreferenceHelper.getString(requireContext(), "default_res", "")!!
|
2022-06-16 14:08:36 +05:30
|
|
|
|
2022-06-03 02:15:45 +05:30
|
|
|
var videosNameArray: Array<CharSequence> = arrayOf()
|
2022-06-16 01:30:36 +05:30
|
|
|
var videosUrlArray: Array<Uri> = arrayOf()
|
|
|
|
|
2022-06-16 14:08:36 +05:30
|
|
|
// append hls to list if available
|
|
|
|
if (response.hls != null) {
|
2022-07-03 16:58:24 +05:30
|
|
|
videosNameArray += getString(R.string.hls)
|
2022-06-16 14:08:36 +05:30
|
|
|
videosUrlArray += response.hls.toUri()
|
|
|
|
}
|
|
|
|
|
2022-06-03 02:15:45 +05:30
|
|
|
for (vid in response.videoStreams!!) {
|
2022-06-16 14:08:36 +05:30
|
|
|
// append quality to list if it has the preferred format (e.g. MPEG)
|
|
|
|
if (vid.format.equals(videoFormatPreference)) { // preferred format
|
2022-06-16 01:30:36 +05:30
|
|
|
videosNameArray += vid.quality!!
|
|
|
|
videosUrlArray += vid.url!!.toUri()
|
2022-06-16 14:08:36 +05:30
|
|
|
} else if (vid.quality.equals("LBRY") && vid.format.equals("MP4")) { // LBRY MP4 format)
|
|
|
|
videosNameArray += "LBRY MP4"
|
|
|
|
videosUrlArray += vid.url!!.toUri()
|
2022-06-16 01:30:36 +05:30
|
|
|
}
|
2022-06-03 02:15:45 +05:30
|
|
|
}
|
2022-06-16 01:30:36 +05:30
|
|
|
// create a list of subtitles
|
2022-06-14 18:39:47 +05:30
|
|
|
val subtitle = mutableListOf<SubtitleConfiguration>()
|
2022-06-19 01:14:22 +05:30
|
|
|
response.subtitles!!.forEach {
|
2022-06-03 02:15:45 +05:30
|
|
|
subtitle.add(
|
2022-06-19 01:14:22 +05:30
|
|
|
SubtitleConfiguration.Builder(it.url!!.toUri())
|
|
|
|
.setMimeType(it.mimeType!!) // The correct MIME type (required).
|
|
|
|
.setLanguage(it.code) // The subtitle language (optional).
|
2022-06-03 02:15:45 +05:30
|
|
|
.build()
|
|
|
|
)
|
|
|
|
}
|
2022-06-16 01:30:36 +05:30
|
|
|
// set resolution in the beginning
|
2022-06-03 02:15:45 +05:30
|
|
|
when {
|
2022-06-16 01:30:36 +05:30
|
|
|
// search for the default resolution in the videoNamesArray, select quality if found
|
2022-06-03 02:15:45 +05:30
|
|
|
defres != "" -> {
|
|
|
|
run lit@{
|
2022-06-16 01:30:36 +05:30
|
|
|
videosNameArray.forEachIndexed { index, pipedStream ->
|
|
|
|
if (pipedStream.contains(defres)) {
|
2022-06-16 14:08:36 +05:30
|
|
|
val videoUri = videosUrlArray[index]
|
|
|
|
val audioUrl = getMostBitRate(response.audioStreams!!)
|
|
|
|
setMediaSource(subtitle, videoUri, audioUrl)
|
2022-07-02 22:01:56 +05:30
|
|
|
playerBinding.qualityText.text = videosNameArray[index]
|
2022-06-03 02:15:45 +05:30
|
|
|
return@lit
|
2022-06-16 20:16:06 +05:30
|
|
|
} else if (response.hls != null) {
|
2022-06-03 02:15:45 +05:30
|
|
|
val mediaItem: MediaItem = MediaItem.Builder()
|
|
|
|
.setUri(response.hls)
|
|
|
|
.setSubtitleConfigurations(subtitle)
|
|
|
|
.build()
|
|
|
|
exoPlayer.setMediaItem(mediaItem)
|
2022-06-16 20:16:06 +05:30
|
|
|
} else {
|
|
|
|
Toast.makeText(
|
|
|
|
context,
|
|
|
|
getString(R.string.unknown_error),
|
|
|
|
Toast.LENGTH_LONG
|
|
|
|
).show()
|
2022-06-03 02:15:45 +05:30
|
|
|
}
|
2022-03-27 11:49:17 +05:30
|
|
|
}
|
2022-06-01 11:55:12 +05:30
|
|
|
}
|
|
|
|
}
|
2022-06-16 14:08:36 +05:30
|
|
|
// if defres doesn't match use hls if available
|
2022-06-03 02:15:45 +05:30
|
|
|
response.hls != null -> {
|
|
|
|
val mediaItem: MediaItem = MediaItem.Builder()
|
|
|
|
.setUri(response.hls)
|
|
|
|
.setSubtitleConfigurations(subtitle)
|
|
|
|
.build()
|
|
|
|
exoPlayer.setMediaItem(mediaItem)
|
|
|
|
}
|
2022-06-16 14:08:36 +05:30
|
|
|
// otherwise use the first list entry
|
2022-06-03 02:15:45 +05:30
|
|
|
else -> {
|
2022-06-16 14:08:36 +05:30
|
|
|
val videoUri = videosUrlArray[0]
|
|
|
|
val audioUrl = getMostBitRate(response.audioStreams!!)
|
|
|
|
setMediaSource(subtitle, videoUri, audioUrl)
|
2022-07-02 22:01:56 +05:30
|
|
|
playerBinding.qualityText.text = videosNameArray[0]
|
2022-06-03 02:15:45 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-07 19:34:32 +05:30
|
|
|
playerBinding.qualityLinLayout.setOnClickListener {
|
2022-06-03 02:15:45 +05:30
|
|
|
// Dialog for quality selection
|
|
|
|
val builder: MaterialAlertDialogBuilder? = activity?.let {
|
|
|
|
MaterialAlertDialogBuilder(it)
|
2021-12-18 16:34:14 +05:30
|
|
|
}
|
2022-06-14 18:39:47 +05:30
|
|
|
val lastPosition = exoPlayer.currentPosition
|
2022-06-03 02:15:45 +05:30
|
|
|
builder!!.setTitle(R.string.choose_quality_dialog)
|
|
|
|
.setItems(
|
2022-06-14 18:39:47 +05:30
|
|
|
videosNameArray
|
|
|
|
) { _, which ->
|
|
|
|
whichQuality = which
|
2022-06-16 14:08:36 +05:30
|
|
|
if (
|
2022-07-03 16:58:24 +05:30
|
|
|
videosNameArray[which] == getString(R.string.hls) ||
|
2022-06-16 14:08:36 +05:30
|
|
|
videosNameArray[which] == "LBRY HLS"
|
|
|
|
) {
|
|
|
|
// no need to merge sources if using hls
|
2022-06-14 18:39:47 +05:30
|
|
|
val mediaItem: MediaItem = MediaItem.Builder()
|
2022-06-16 14:08:36 +05:30
|
|
|
.setUri(videosUrlArray[which])
|
2022-06-14 18:39:47 +05:30
|
|
|
.setSubtitleConfigurations(subtitle)
|
|
|
|
.build()
|
|
|
|
exoPlayer.setMediaItem(mediaItem)
|
|
|
|
} else {
|
2022-06-16 14:08:36 +05:30
|
|
|
val videoUri = videosUrlArray[which]
|
|
|
|
val audioUrl = getMostBitRate(response.audioStreams!!)
|
|
|
|
setMediaSource(subtitle, videoUri, audioUrl)
|
2022-06-03 02:15:45 +05:30
|
|
|
}
|
2022-06-14 18:39:47 +05:30
|
|
|
exoPlayer.seekTo(lastPosition)
|
2022-07-02 22:01:56 +05:30
|
|
|
playerBinding.qualityText.text = videosNameArray[which]
|
2022-06-14 18:39:47 +05:30
|
|
|
}
|
2022-06-03 02:15:45 +05:30
|
|
|
val dialog = builder.create()
|
|
|
|
dialog.show()
|
2021-12-12 20:31:44 +05:30
|
|
|
}
|
|
|
|
}
|
2022-02-13 22:43:26 +05:30
|
|
|
|
2022-06-01 11:32:16 +05:30
|
|
|
private fun createExoPlayer(view: View) {
|
2022-06-26 15:50:56 +05:30
|
|
|
val playbackSpeed =
|
|
|
|
PreferenceHelper.getString(requireContext(), "playback_speed", "1F")?.toFloat()
|
2022-06-18 21:51:30 +05:30
|
|
|
// multiply by thousand: s -> ms
|
2022-06-26 15:50:56 +05:30
|
|
|
val bufferingGoal =
|
|
|
|
PreferenceHelper.getString(requireContext(), "buffering_goal", "50")?.toInt()!! * 1000
|
|
|
|
val seekIncrement =
|
|
|
|
PreferenceHelper.getString(requireContext(), "seek_increment", "5")?.toLong()!! * 1000
|
2022-06-14 23:31:27 +05:30
|
|
|
|
2022-06-01 11:32:16 +05:30
|
|
|
val cronetEngine: CronetEngine = CronetHelper.getCronetEngine()
|
|
|
|
val cronetDataSourceFactory: CronetDataSource.Factory =
|
|
|
|
CronetDataSource.Factory(cronetEngine, Executors.newCachedThreadPool())
|
|
|
|
|
|
|
|
val dataSourceFactory = DefaultDataSource.Factory(
|
|
|
|
requireContext(),
|
|
|
|
cronetDataSourceFactory
|
|
|
|
)
|
|
|
|
|
2022-06-03 22:13:15 +05:30
|
|
|
// handles the audio focus
|
2022-06-01 11:32:16 +05:30
|
|
|
val audioAttributes = AudioAttributes.Builder()
|
|
|
|
.setUsage(C.USAGE_MEDIA)
|
|
|
|
.setContentType(C.CONTENT_TYPE_MOVIE)
|
|
|
|
.build()
|
|
|
|
|
2022-06-03 22:13:15 +05:30
|
|
|
// handles the duration of media to retain in the buffer prior to the current playback position (for fast backward seeking)
|
|
|
|
val loadControl = DefaultLoadControl.Builder()
|
|
|
|
// cache the last three minutes
|
|
|
|
.setBackBuffer(1000 * 60 * 3, true)
|
2022-06-14 23:31:27 +05:30
|
|
|
.setBufferDurationsMs(
|
2022-06-30 17:08:21 +05:30
|
|
|
1000 * 10, // exo default is 50s
|
2022-06-18 21:51:30 +05:30
|
|
|
bufferingGoal,
|
2022-06-14 23:31:27 +05:30
|
|
|
DefaultLoadControl.DEFAULT_BUFFER_FOR_PLAYBACK_MS,
|
|
|
|
DefaultLoadControl.DEFAULT_BUFFER_FOR_PLAYBACK_AFTER_REBUFFER_MS
|
|
|
|
)
|
2022-06-03 22:13:15 +05:30
|
|
|
.build()
|
|
|
|
|
2022-06-01 11:32:16 +05:30
|
|
|
exoPlayer = ExoPlayer.Builder(view.context)
|
|
|
|
.setMediaSourceFactory(DefaultMediaSourceFactory(dataSourceFactory))
|
2022-06-03 22:13:15 +05:30
|
|
|
.setLoadControl(loadControl)
|
2022-06-18 21:51:30 +05:30
|
|
|
.setSeekBackIncrementMs(seekIncrement)
|
|
|
|
.setSeekForwardIncrementMs(seekIncrement)
|
2022-06-01 11:32:16 +05:30
|
|
|
.build()
|
|
|
|
|
|
|
|
exoPlayer.setAudioAttributes(audioAttributes, true)
|
2022-06-07 20:22:06 +05:30
|
|
|
|
|
|
|
exoPlayer.setPlaybackSpeed(playbackSpeed!!)
|
2022-06-01 11:32:16 +05:30
|
|
|
}
|
|
|
|
|
2022-06-05 15:12:33 +05:30
|
|
|
private fun initializePlayerNotification(c: Context) {
|
2022-06-01 11:32:16 +05:30
|
|
|
mediaSession = MediaSessionCompat(c, this.javaClass.name)
|
2022-06-05 15:12:33 +05:30
|
|
|
mediaSession.apply {
|
|
|
|
isActive = true
|
|
|
|
}
|
|
|
|
|
2022-06-01 11:32:16 +05:30
|
|
|
mediaSessionConnector = MediaSessionConnector(mediaSession)
|
|
|
|
mediaSessionConnector.setPlayer(exoPlayer)
|
|
|
|
|
2022-06-01 15:33:00 +05:30
|
|
|
playerNotification = PlayerNotificationManager
|
2022-06-06 20:12:46 +05:30
|
|
|
.Builder(c, 1, "background_mode")
|
2022-06-14 15:30:58 +05:30
|
|
|
.setMediaDescriptionAdapter(
|
2022-06-19 01:14:22 +05:30
|
|
|
DescriptionAdapter(title, uploader, thumbnailUrl, requireContext())
|
2022-06-14 15:30:58 +05:30
|
|
|
)
|
2022-06-01 12:20:02 +05:30
|
|
|
.build()
|
2022-06-05 15:12:33 +05:30
|
|
|
|
|
|
|
playerNotification.apply {
|
|
|
|
setPlayer(exoPlayer)
|
|
|
|
setUsePreviousAction(false)
|
2022-06-30 19:32:55 +05:30
|
|
|
setUseStopAction(true)
|
2022-06-05 15:12:33 +05:30
|
|
|
setMediaSessionToken(mediaSession.sessionToken)
|
|
|
|
}
|
2022-06-01 11:32:16 +05:30
|
|
|
}
|
|
|
|
|
2022-06-24 20:56:36 +05:30
|
|
|
private fun lockPlayer(isLocked: Boolean) {
|
2022-07-03 21:43:52 +05:30
|
|
|
val visibility = if (isLocked) View.VISIBLE else View.GONE
|
2022-07-02 22:01:56 +05:30
|
|
|
playerBinding.exoTopBarRight.visibility = visibility
|
|
|
|
playerBinding.exoPlayPause.visibility = visibility
|
|
|
|
playerBinding.exoBottomBar.visibility = visibility
|
2022-07-03 21:43:52 +05:30
|
|
|
playerBinding.closeImageButton.visibility = visibility
|
|
|
|
playerBinding.exoTitle.visibility = visibility
|
2022-06-24 20:56:36 +05:30
|
|
|
}
|
|
|
|
|
2022-03-27 11:49:17 +05:30
|
|
|
private fun isSubscribed(button: MaterialButton, channel_id: String) {
|
2022-02-13 22:43:26 +05:30
|
|
|
@SuppressLint("ResourceAsColor")
|
|
|
|
fun run() {
|
|
|
|
lifecycleScope.launchWhenCreated {
|
|
|
|
val response = try {
|
2022-06-26 14:38:10 +05:30
|
|
|
val token = PreferenceHelper.getToken(requireContext())
|
2022-07-03 00:25:22 +05:30
|
|
|
RetrofitInstance.authApi.isSubscribed(
|
2022-04-15 12:08:53 +05:30
|
|
|
channel_id,
|
2022-06-26 14:38:10 +05:30
|
|
|
token
|
2022-04-15 12:08:53 +05:30
|
|
|
)
|
2022-03-27 11:49:17 +05:30
|
|
|
} catch (e: IOException) {
|
2022-02-13 22:43:26 +05:30
|
|
|
println(e)
|
|
|
|
Log.e(TAG, "IOException, you might not have internet connection")
|
|
|
|
return@launchWhenCreated
|
|
|
|
} catch (e: HttpException) {
|
|
|
|
Log.e(TAG, "HttpException, unexpected response")
|
|
|
|
return@launchWhenCreated
|
|
|
|
}
|
2022-02-15 00:23:01 +05:30
|
|
|
|
2022-02-13 22:43:26 +05:30
|
|
|
runOnUiThread {
|
2022-03-27 11:49:17 +05:30
|
|
|
if (response.subscribed == true) {
|
|
|
|
isSubscribed = true
|
|
|
|
button.text = getString(R.string.unsubscribe)
|
2022-02-13 22:43:26 +05:30
|
|
|
}
|
2022-03-27 11:49:17 +05:30
|
|
|
if (response.subscribed != null) {
|
|
|
|
button.setOnClickListener {
|
|
|
|
if (isSubscribed) {
|
|
|
|
unsubscribe(channel_id)
|
|
|
|
button.text = getString(R.string.subscribe)
|
|
|
|
} else {
|
|
|
|
subscribe(channel_id)
|
|
|
|
button.text = getString(R.string.unsubscribe)
|
|
|
|
}
|
2022-04-15 12:08:53 +05:30
|
|
|
}
|
2022-03-27 11:49:17 +05:30
|
|
|
}
|
2022-02-13 22:43:26 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
run()
|
|
|
|
}
|
|
|
|
|
2022-03-27 11:49:17 +05:30
|
|
|
private fun subscribe(channel_id: String) {
|
2022-02-13 22:43:26 +05:30
|
|
|
fun run() {
|
|
|
|
lifecycleScope.launchWhenCreated {
|
|
|
|
val response = try {
|
2022-06-26 14:38:10 +05:30
|
|
|
val token = PreferenceHelper.getToken(requireContext())
|
2022-07-03 00:25:22 +05:30
|
|
|
RetrofitInstance.authApi.subscribe(
|
2022-06-26 14:38:10 +05:30
|
|
|
token,
|
2022-04-15 12:08:53 +05:30
|
|
|
Subscribe(channel_id)
|
|
|
|
)
|
2022-03-27 11:49:17 +05:30
|
|
|
} catch (e: IOException) {
|
2022-02-13 22:43:26 +05:30
|
|
|
println(e)
|
|
|
|
Log.e(TAG, "IOException, you might not have internet connection")
|
|
|
|
return@launchWhenCreated
|
|
|
|
} catch (e: HttpException) {
|
|
|
|
Log.e(TAG, "HttpException, unexpected response$e")
|
|
|
|
return@launchWhenCreated
|
|
|
|
}
|
2022-03-27 11:49:17 +05:30
|
|
|
isSubscribed = true
|
2022-02-13 22:43:26 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
run()
|
|
|
|
}
|
2022-04-15 12:08:53 +05:30
|
|
|
|
2022-03-27 11:49:17 +05:30
|
|
|
private fun unsubscribe(channel_id: String) {
|
2022-02-13 22:43:26 +05:30
|
|
|
fun run() {
|
|
|
|
lifecycleScope.launchWhenCreated {
|
|
|
|
val response = try {
|
2022-06-26 14:38:10 +05:30
|
|
|
val token = PreferenceHelper.getToken(requireContext())
|
2022-07-03 00:25:22 +05:30
|
|
|
RetrofitInstance.authApi.unsubscribe(
|
2022-06-26 14:38:10 +05:30
|
|
|
token,
|
2022-04-15 12:08:53 +05:30
|
|
|
Subscribe(channel_id)
|
|
|
|
)
|
2022-03-27 11:49:17 +05:30
|
|
|
} catch (e: IOException) {
|
2022-02-13 22:43:26 +05:30
|
|
|
println(e)
|
|
|
|
Log.e(TAG, "IOException, you might not have internet connection")
|
|
|
|
return@launchWhenCreated
|
|
|
|
} catch (e: HttpException) {
|
|
|
|
Log.e(TAG, "HttpException, unexpected response")
|
|
|
|
return@launchWhenCreated
|
|
|
|
}
|
2022-03-27 11:49:17 +05:30
|
|
|
isSubscribed = false
|
2022-02-13 22:43:26 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
run()
|
|
|
|
}
|
|
|
|
|
2021-12-18 16:34:14 +05:30
|
|
|
private fun Fragment?.runOnUiThread(action: () -> Unit) {
|
2021-12-12 20:31:44 +05:30
|
|
|
this ?: return
|
|
|
|
if (!isAdded) return // Fragment not attached to an Activity
|
|
|
|
activity?.runOnUiThread(action)
|
|
|
|
}
|
2021-12-14 02:58:17 +05:30
|
|
|
|
2022-06-16 14:08:36 +05:30
|
|
|
private fun getMostBitRate(audios: List<PipedStream>): String {
|
2022-03-27 11:49:17 +05:30
|
|
|
var bitrate = 0
|
2021-12-14 02:58:17 +05:30
|
|
|
var index = 0
|
2022-03-27 11:49:17 +05:30
|
|
|
for ((i, audio) in audios.withIndex()) {
|
|
|
|
val q = audio.quality!!.replace(" kbps", "").toInt()
|
2022-04-15 12:08:53 +05:30
|
|
|
if (q > bitrate) {
|
2022-03-27 11:49:17 +05:30
|
|
|
bitrate = q
|
2021-12-14 02:58:17 +05:30
|
|
|
index = i
|
|
|
|
}
|
|
|
|
}
|
2022-06-16 14:08:36 +05:30
|
|
|
return audios[index].url!!
|
2021-12-14 02:58:17 +05:30
|
|
|
}
|
2021-12-14 21:45:53 +05:30
|
|
|
|
2022-05-20 21:15:16 +05:30
|
|
|
private fun fetchComments() {
|
|
|
|
lifecycleScope.launchWhenCreated {
|
|
|
|
val commentsResponse = try {
|
|
|
|
RetrofitInstance.api.getComments(videoId!!)
|
|
|
|
} catch (e: IOException) {
|
|
|
|
println(e)
|
|
|
|
Log.e(TAG, "IOException, you might not have internet connection")
|
|
|
|
Toast.makeText(context, R.string.unknown_error, Toast.LENGTH_SHORT).show()
|
|
|
|
return@launchWhenCreated
|
|
|
|
} catch (e: HttpException) {
|
|
|
|
Log.e(TAG, "HttpException, unexpected response")
|
|
|
|
return@launchWhenCreated
|
|
|
|
}
|
2022-06-05 19:12:51 +05:30
|
|
|
commentsAdapter = CommentsAdapter(videoId!!, commentsResponse.comments)
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.commentsRecView.adapter = commentsAdapter
|
2022-05-20 21:15:16 +05:30
|
|
|
nextPage = commentsResponse.nextpage
|
|
|
|
commentsLoaded = true
|
2022-06-03 22:13:15 +05:30
|
|
|
isLoading = false
|
2022-05-20 21:15:16 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-20 03:52:10 +05:30
|
|
|
private fun fetchNextComments() {
|
|
|
|
lifecycleScope.launchWhenCreated {
|
|
|
|
if (!isLoading) {
|
|
|
|
isLoading = true
|
|
|
|
val response = try {
|
|
|
|
RetrofitInstance.api.getCommentsNextPage(videoId!!, nextPage!!)
|
|
|
|
} catch (e: IOException) {
|
|
|
|
println(e)
|
|
|
|
Log.e(TAG, "IOException, you might not have internet connection")
|
|
|
|
return@launchWhenCreated
|
|
|
|
} catch (e: HttpException) {
|
|
|
|
Log.e(TAG, "HttpException, unexpected response," + e.response())
|
|
|
|
return@launchWhenCreated
|
2022-05-08 23:02:13 +05:30
|
|
|
}
|
2022-05-20 03:52:10 +05:30
|
|
|
nextPage = response.nextpage
|
2022-05-21 13:32:04 +05:30
|
|
|
commentsAdapter?.updateItems(response.comments)
|
2022-05-20 03:52:10 +05:30
|
|
|
isLoading = false
|
2022-05-08 23:02:13 +05:30
|
|
|
}
|
2022-05-20 03:52:10 +05:30
|
|
|
}
|
2022-05-08 23:02:13 +05:30
|
|
|
}
|
2022-05-14 16:08:28 +05:30
|
|
|
|
|
|
|
override fun onPictureInPictureModeChanged(isInPictureInPictureMode: Boolean) {
|
|
|
|
super.onPictureInPictureModeChanged(isInPictureInPictureMode)
|
|
|
|
if (isInPictureInPictureMode) {
|
|
|
|
exoPlayerView.hideController()
|
2022-06-19 17:09:41 +05:30
|
|
|
exoPlayerView.useController = false
|
2022-07-04 22:06:47 +05:30
|
|
|
binding.linLayout.visibility = View.GONE
|
|
|
|
|
2022-07-01 00:55:40 +05:30
|
|
|
with(binding.playerMotionLayout) {
|
2022-05-14 16:08:28 +05:30
|
|
|
getConstraintSet(R.id.start).constrainHeight(R.id.player, -1)
|
|
|
|
enableTransition(R.id.yt_transition, false)
|
|
|
|
}
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.mainContainer.isClickable = true
|
2022-07-05 00:49:16 +05:30
|
|
|
|
2022-05-14 17:01:33 +05:30
|
|
|
val mainActivity = activity as MainActivity
|
2022-06-06 21:45:54 +05:30
|
|
|
mainActivity.requestedOrientation = ActivityInfo.SCREEN_ORIENTATION_USER_PORTRAIT
|
2022-05-20 03:52:10 +05:30
|
|
|
isFullScreen = false
|
2022-05-14 16:08:28 +05:30
|
|
|
} else {
|
2022-07-01 00:55:40 +05:30
|
|
|
with(binding.playerMotionLayout) {
|
2022-05-14 16:08:28 +05:30
|
|
|
getConstraintSet(R.id.start).constrainHeight(R.id.player, 0)
|
|
|
|
enableTransition(R.id.yt_transition, true)
|
|
|
|
}
|
2022-07-04 22:06:47 +05:30
|
|
|
|
2022-06-19 17:09:41 +05:30
|
|
|
exoPlayerView.useController = true
|
2022-07-04 22:06:47 +05:30
|
|
|
binding.linLayout.visibility = View.VISIBLE
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.mainContainer.isClickable = false
|
2022-05-14 16:08:28 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fun onUserLeaveHint() {
|
2022-05-14 17:01:33 +05:30
|
|
|
val bounds = Rect()
|
2022-07-01 00:55:40 +05:30
|
|
|
binding.playerScrollView.getHitRect(bounds)
|
2022-05-14 17:01:33 +05:30
|
|
|
|
2022-06-04 18:05:01 +05:30
|
|
|
if (SDK_INT >= Build.VERSION_CODES.O &&
|
2022-07-01 14:41:24 +05:30
|
|
|
exoPlayer.isPlaying && (binding.playerScrollView.getLocalVisibleRect(bounds) || isFullScreen)
|
2022-05-21 13:32:04 +05:30
|
|
|
) {
|
2022-06-26 21:19:42 +05:30
|
|
|
activity?.enterPictureInPictureMode(updatePipParams())
|
2022-05-20 03:52:10 +05:30
|
|
|
}
|
2021-12-14 21:45:53 +05:30
|
|
|
}
|
2022-06-26 21:19:42 +05:30
|
|
|
|
|
|
|
private fun updatePipParams() = PictureInPictureParams.Builder()
|
|
|
|
.setActions(emptyList())
|
|
|
|
.build()
|
2022-03-27 11:49:17 +05:30
|
|
|
}
|