LibreTube/app/src/main/java/xyz/btcland/libretube/PlayerFragment.kt

398 lines
18 KiB
Kotlin
Raw Normal View History

package xyz.btcland.libretube
2021-12-14 02:58:17 +05:30
import android.R.attr
2021-12-14 21:45:53 +05:30
import android.R.attr.*
import android.os.Bundle
import androidx.fragment.app.Fragment
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import androidx.constraintlayout.motion.widget.MotionLayout
import androidx.constraintlayout.widget.ConstraintSet
import androidx.recyclerview.widget.RecyclerView
import com.google.android.exoplayer2.ExoPlayer
import com.google.android.exoplayer2.MediaItem
2021-12-14 02:58:17 +05:30
import com.google.android.exoplayer2.MediaItem.SubtitleConfiguration
import com.google.android.exoplayer2.source.MediaSource
import com.google.android.exoplayer2.ui.PlayerView
2021-12-14 02:58:17 +05:30
import com.google.android.exoplayer2.ui.StyledPlayerControlView
import com.google.android.exoplayer2.ui.StyledPlayerView
2021-12-18 16:34:14 +05:30
import okhttp3.*
import java.io.IOException
import kotlin.math.abs
2021-12-14 02:58:17 +05:30
import com.google.android.exoplayer2.util.MimeTypes
import com.google.common.collect.ImmutableList
2021-12-14 21:45:53 +05:30
import android.app.ActionBar
2021-12-14 02:58:17 +05:30
import android.content.DialogInterface
2021-12-14 21:45:53 +05:30
import android.content.Intent
import android.content.pm.ActivityInfo
2021-12-14 02:58:17 +05:30
import android.graphics.Color
import android.widget.*
import androidx.core.net.toUri
import com.google.android.exoplayer2.C.SELECTION_FLAG_DEFAULT
import com.google.android.exoplayer2.MediaItem.fromUri
import com.google.android.exoplayer2.Player.REPEAT_MODE_ONE
import com.google.android.exoplayer2.source.MergingMediaSource
import com.google.android.exoplayer2.trackselection.DefaultTrackSelector
import com.google.android.exoplayer2.source.ProgressiveMediaSource
import com.google.android.exoplayer2.upstream.DataSource
import com.google.android.exoplayer2.upstream.DefaultHttpDataSource
import android.widget.PopupWindow
import android.widget.TextView
import android.graphics.drawable.Drawable
import com.google.android.exoplayer2.util.Util
import android.graphics.drawable.ColorDrawable
2021-12-18 16:34:14 +05:30
import android.util.Log
2021-12-14 02:58:17 +05:30
import androidx.appcompat.app.AlertDialog
2021-12-14 21:45:53 +05:30
import androidx.constraintlayout.widget.ConstraintLayout
import androidx.core.text.PrecomputedTextCompat
2021-12-18 16:34:14 +05:30
import androidx.lifecycle.lifecycleScope
2021-12-14 02:58:17 +05:30
import androidx.recyclerview.widget.LinearLayoutManager
2021-12-17 17:52:55 +05:30
import com.google.android.exoplayer2.Player
import com.google.android.exoplayer2.Player.STATE_IDLE
2021-12-14 21:45:53 +05:30
import com.google.android.exoplayer2.source.DefaultMediaSourceFactory
2021-12-14 02:58:17 +05:30
import com.google.android.exoplayer2.util.RepeatModeUtil
import com.google.android.exoplayer2.ui.TimeBar
import com.google.android.exoplayer2.ui.TimeBar.OnScrubListener
2021-12-18 16:34:14 +05:30
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.launch
import retrofit2.HttpException
import retrofit2.Retrofit
import retrofit2.converter.jackson.JacksonConverterFactory
import retrofit2.http.GET
import retrofit2.http.Path
import xyz.btcland.libretube.obj.PipedStream
2021-12-14 02:58:17 +05:30
// TODO: Rename parameter arguments, choose names that match
// the fragment initialization parameters, e.g. ARG_ITEM_NUMBER
private const val ARG_PARAM1 = "param1"
private const val ARG_PARAM2 = "param2"
/**
* A simple [Fragment] subclass.
* Use the [PlayerFragment.newInstance] factory method to
* create an instance of this fragment.
*/
class PlayerFragment : Fragment() {
// TODO: Rename and change types of parameters
private var videoId: String? = null
private var param2: String? = null
private var lastProgress: Float = 0.toFloat()
private var sId: Int=0
private var eId: Int=0
2021-12-14 02:58:17 +05:30
private var paused =false
2021-12-14 21:45:53 +05:30
private var isFullScreen = false
private var whichQuality = 0
2021-12-14 02:58:17 +05:30
private lateinit var exoPlayerView: StyledPlayerView
2021-12-17 17:52:55 +05:30
private lateinit var motionLayout: MotionLayout
2021-12-14 21:45:53 +05:30
private lateinit var exoPlayer: ExoPlayer
private lateinit var mediaSource: MediaSource
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
arguments?.let {
videoId = it.getString("videoId")
}
}
override fun onCreateView(
inflater: LayoutInflater, container: ViewGroup?,
savedInstanceState: Bundle?
): View? {
// Inflate the layout for this fragment
return inflater.inflate(R.layout.fragment_player, container, false)
}
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
val mainActivity = activity as MainActivity
mainActivity.findViewById<FrameLayout>(R.id.container).visibility=View.VISIBLE
2021-12-17 17:52:55 +05:30
val playerMotionLayout = view.findViewById<MotionLayout>(R.id.playerMotionLayout)
motionLayout = playerMotionLayout
exoPlayerView = view.findViewById(R.id.player)
view.findViewById<TextView>(R.id.textTest).text = videoId
playerMotionLayout.addTransitionListener(object: MotionLayout.TransitionListener {
override fun onTransitionStarted(
motionLayout: MotionLayout?,
startId: Int,
endId: Int
) {
}
override fun onTransitionChange(motionLayout: MotionLayout?, startId: Int, endId: Int, progress: Float) {
val mainActivity = activity as MainActivity
val mainMotionLayout = mainActivity.findViewById<MotionLayout>(R.id.mainMotionLayout)
mainMotionLayout.progress = abs(progress)
eId=endId
sId=startId
}
override fun onTransitionCompleted(motionLayout: MotionLayout?, currentId: Int) {
println(currentId)
val mainActivity = activity as MainActivity
val mainMotionLayout = mainActivity.findViewById<MotionLayout>(R.id.mainMotionLayout)
if (currentId==eId) {
2021-12-14 02:58:17 +05:30
view.findViewById<ImageButton>(R.id.quality_select).visibility =View.GONE
view.findViewById<ImageButton>(R.id.close_imageButton).visibility =View.GONE
view.findViewById<TextView>(R.id.quality_text).visibility =View.GONE
mainMotionLayout.progress = 1.toFloat()
2021-12-16 03:54:40 +05:30
mainActivity.supportActionBar?.show()
}else if(currentId==sId){
2021-12-14 02:58:17 +05:30
view.findViewById<ImageButton>(R.id.quality_select).visibility =View.VISIBLE
view.findViewById<ImageButton>(R.id.close_imageButton).visibility =View.VISIBLE
view.findViewById<TextView>(R.id.quality_text).visibility =View.VISIBLE
mainMotionLayout.progress = 0.toFloat()
2021-12-16 03:54:40 +05:30
mainActivity.supportActionBar?.hide()
}
}
override fun onTransitionTrigger(
motionLayout: MotionLayout?,
triggerId: Int,
positive: Boolean,
progress: Float
) {
}
})
playerMotionLayout.progress=1.toFloat()
playerMotionLayout.transitionToStart()
fetchJson(view)
view.findViewById<ImageView>(R.id.close_imageView).setOnClickListener{
val mainActivity = activity as MainActivity
mainActivity.supportFragmentManager.beginTransaction()
.remove(this)
.commit()
}
2021-12-14 02:58:17 +05:30
view.findViewById<ImageButton>(R.id.close_imageButton).setOnClickListener{
val mainActivity = activity as MainActivity
mainActivity.supportFragmentManager.beginTransaction()
.remove(this)
.commit()
}
val playImageView = view.findViewById<ImageView>(R.id.play_imageView)
playImageView.setOnClickListener{
paused = if(paused){
2021-12-15 15:54:12 +05:30
playImageView.setImageResource(R.drawable.ic_pause)
2021-12-14 02:58:17 +05:30
exoPlayer.play()
false
}else {
2021-12-15 15:54:12 +05:30
playImageView.setImageResource(R.drawable.ic_play)
2021-12-14 02:58:17 +05:30
exoPlayer.pause()
true
}
}
2021-12-18 16:34:14 +05:30
//FullScreen button trigger
2021-12-14 21:45:53 +05:30
view.findViewById<ImageButton>(R.id.fullscreen).setOnClickListener{
2021-12-16 03:54:40 +05:30
//remember to hide everything when new shit added
2021-12-15 15:54:12 +05:30
if (!isFullScreen){
2021-12-18 16:34:14 +05:30
/*view.findViewById<ScrollView>(R.id.scrollView2).visibility = View.GONE
2021-12-15 15:54:12 +05:30
view.findViewById<LinearLayout>(R.id.linLayout).visibility = View.GONE
2021-12-18 16:34:14 +05:30
view.findViewById<TextView>(R.id.textTest).visibility = View.GONE*/
2021-12-16 03:54:40 +05:30
//view.findViewById<ConstraintLayout>(R.id.main_container).visibility = View.GONE
2021-12-15 15:54:12 +05:30
with(motionLayout) {
getConstraintSet(R.id.start).constrainHeight(R.id.player, -1)
2021-12-16 03:54:40 +05:30
//getTransition(R.id.yt_transition).isEnabled = false
2021-12-15 15:54:12 +05:30
}
isFullScreen=true
}else{
2021-12-18 16:34:14 +05:30
/*view.findViewById<ScrollView>(R.id.scrollView2).visibility = View.VISIBLE
2021-12-15 15:54:12 +05:30
view.findViewById<LinearLayout>(R.id.linLayout).visibility = View.VISIBLE
2021-12-18 16:34:14 +05:30
view.findViewById<TextView>(R.id.textTest).visibility = View.VISIBLE*/
2021-12-16 03:54:40 +05:30
//view.findViewById<ConstraintLayout>(R.id.main_container).visibility = View.VISIBLE
2021-12-15 15:54:12 +05:30
with(motionLayout) {
getConstraintSet(R.id.start).constrainHeight(R.id.player, 0)
2021-12-16 03:54:40 +05:30
//getTransition(R.id.yt_transition).isEnabled = true
2021-12-15 15:54:12 +05:30
}
isFullScreen=false
}
2021-12-14 21:45:53 +05:30
}
2021-12-14 02:58:17 +05:30
}
2021-12-14 21:45:53 +05:30
override fun onStop() {
super.onStop()
2021-12-14 02:58:17 +05:30
try {
2021-12-28 01:37:07 +05:30
(activity as MainActivity).supportActionBar?.show()
2021-12-14 02:58:17 +05:30
exoPlayer.stop()
}catch (e: Exception){}
}
companion object {
/**
* Use this factory method to create a new instance of
* this fragment using the provided parameters.
*
* @param param1 Parameter 1.
* @param param2 Parameter 2.
* @return A new instance of fragment PlayerFragment.
*/
// TODO: Rename and change types and number of parameters
@JvmStatic
fun newInstance(param1: String, param2: String) =
PlayerFragment().apply {
arguments = Bundle().apply {
putString(ARG_PARAM1, param1)
putString(ARG_PARAM2, param2)
}
}
}
private fun fetchJson(view: View) {
fun run() {
2021-12-18 16:34:14 +05:30
lifecycleScope.launchWhenCreated {
val response = try {
RetrofitInstance.api.getStreams(videoId!!)
}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")
return@launchWhenCreated
}
var videosNameArray: Array<CharSequence> = arrayOf()
videosNameArray += "HLS"
for (vid in response.videoStreams!!){
val name = vid.quality +" "+ vid.format
videosNameArray += name
}
runOnUiThread {
var subtitle = mutableListOf<SubtitleConfiguration>()
if(response.subtitles!!.isNotEmpty()){
subtitle?.add(SubtitleConfiguration.Builder(response.subtitles!![0].url!!.toUri())
.setMimeType(response.subtitles!![0].mimeType!!) // The correct MIME type (required).
.setLanguage(response.subtitles!![0].code) // The subtitle language (optional).
.build())}
val mediaItem: MediaItem = MediaItem.Builder()
.setUri(response.hls)
.setSubtitleConfigurations(subtitle)
.build()
exoPlayer = ExoPlayer.Builder(view.context)
.build()
exoPlayerView.setShowSubtitleButton(true)
exoPlayerView.setShowNextButton(false)
exoPlayerView.setShowPreviousButton(false)
//exoPlayerView.controllerShowTimeoutMs = 1500
exoPlayerView.controllerHideOnTouch = true
exoPlayerView.player = exoPlayer
exoPlayer.setMediaItem(mediaItem)
///exoPlayer.getMediaItemAt(5)
exoPlayer.prepare()
exoPlayer.play()
view.findViewById<TextView>(R.id.title_textView).text = response.title
view.findViewById<ImageButton>(R.id.quality_select).setOnClickListener{
//Dialog for quality selection
val builder: AlertDialog.Builder? = activity?.let {
AlertDialog.Builder(it)
}
builder!!.setTitle(R.string.choose_quality_dialog)
.setItems(videosNameArray,
DialogInterface.OnClickListener { _, which ->
whichQuality = which
if(response.subtitles!!.isNotEmpty()) {
var subtitle =
mutableListOf<SubtitleConfiguration>()
subtitle?.add(
SubtitleConfiguration.Builder(response.subtitles!![0].url!!.toUri())
.setMimeType(response.subtitles!![0].mimeType!!) // The correct MIME type (required).
.setLanguage(response.subtitles!![0].code) // The subtitle language (optional).
.build()
)
}
if(which==0){
val mediaItem: MediaItem = MediaItem.Builder()
.setUri(response.hls)
.setSubtitleConfigurations(subtitle)
.build()
exoPlayer.setMediaItem(mediaItem)
}else{
val dataSourceFactory: DataSource.Factory =
DefaultHttpDataSource.Factory()
val videoItem: MediaItem = MediaItem.Builder()
.setUri(response.videoStreams[which-1].url)
.setSubtitleConfigurations(subtitle)
.build()
val videoSource: MediaSource = DefaultMediaSourceFactory(dataSourceFactory)
.createMediaSource(videoItem)
var audioSource: MediaSource = DefaultMediaSourceFactory(dataSourceFactory)
.createMediaSource(fromUri(response.audioStreams!![0].url!!))
if (response.videoStreams[which-1].quality=="720p" || response.videoStreams[which-1].quality=="1080p" || response.videoStreams[which-1].quality=="480p" ){
audioSource = ProgressiveMediaSource.Factory(dataSourceFactory)
.createMediaSource(fromUri(response.audioStreams!![getMostBitRate(response.audioStreams)].url!!))
}
val mergeSource: MediaSource = MergingMediaSource(videoSource,audioSource)
exoPlayer.setMediaSource(mergeSource)
}
view.findViewById<TextView>(R.id.quality_text).text=videosNameArray[which]
})
val dialog: AlertDialog? = builder?.create()
dialog?.show()
}
//Listener for play and pause icon change
exoPlayer!!.addListener(object : com.google.android.exoplayer2.Player.Listener {
override fun onPlayerStateChanged(playWhenReady: Boolean,playbackState: Int) {
if (playWhenReady && playbackState == Player.STATE_READY) {
// media actually playing
view.findViewById<ImageView>(R.id.play_imageView).setImageResource(R.drawable.ic_pause)
} else if (playWhenReady) {
// might be idle (plays after prepare()),
// buffering (plays when data available)
// or ended (plays when seek away from end)
view.findViewById<ImageView>(R.id.play_imageView).setImageResource(R.drawable.ic_play)
} else {
// player paused in any state
view.findViewById<ImageView>(R.id.play_imageView).setImageResource(R.drawable.ic_play)
}
}
})
}
}
}
run()
}
2021-12-18 16:34:14 +05:30
private fun Fragment?.runOnUiThread(action: () -> Unit) {
this ?: return
if (!isAdded) return // Fragment not attached to an Activity
activity?.runOnUiThread(action)
}
2021-12-14 02:58:17 +05:30
2021-12-18 16:34:14 +05:30
fun getMostBitRate(audios: List<PipedStream>):Int{
2021-12-14 02:58:17 +05:30
var bitrate =0
var index = 0
for ((i, audio) in audios.withIndex()){
2021-12-18 16:34:14 +05:30
val q = audio.quality!!.replace(" kbps","").toInt()
2021-12-14 02:58:17 +05:30
if (q>bitrate){
bitrate=q
index = i
}
}
return index
}
2021-12-14 21:45:53 +05:30
override fun onResume() {
super.onResume()
}
}