2022-09-19 23:37:55 +05:30
|
|
|
package com.github.libretube.util
|
|
|
|
|
2022-11-06 16:20:04 +05:30
|
|
|
import android.util.Log
|
2022-11-20 21:39:07 +05:30
|
|
|
import com.github.libretube.api.PlaylistsHelper
|
2022-10-23 18:13:47 +05:30
|
|
|
import com.github.libretube.api.RetrofitInstance
|
2022-10-23 15:03:35 +05:30
|
|
|
import com.github.libretube.api.obj.StreamItem
|
2022-10-23 18:39:58 +05:30
|
|
|
import com.github.libretube.extensions.move
|
2022-10-23 17:09:15 +05:30
|
|
|
import com.github.libretube.extensions.toID
|
2022-10-23 18:13:47 +05:30
|
|
|
import kotlinx.coroutines.CoroutineScope
|
|
|
|
import kotlinx.coroutines.Dispatchers
|
|
|
|
import kotlinx.coroutines.launch
|
2022-10-23 15:03:35 +05:30
|
|
|
|
2022-09-19 23:37:55 +05:30
|
|
|
object PlayingQueue {
|
2022-10-23 17:09:15 +05:30
|
|
|
private val queue = mutableListOf<StreamItem>()
|
|
|
|
private var currentStream: StreamItem? = null
|
2023-01-16 22:55:12 +05:30
|
|
|
private val scope = CoroutineScope(Dispatchers.IO)
|
2023-01-13 22:50:00 +05:30
|
|
|
|
|
|
|
/**
|
|
|
|
* Listener that gets called when the user selects an item from the queue
|
|
|
|
*/
|
2022-11-06 16:20:04 +05:30
|
|
|
private var onQueueTapListener: (StreamItem) -> Unit = {}
|
2023-01-13 22:50:00 +05:30
|
|
|
|
2022-11-17 21:15:07 +05:30
|
|
|
var repeatQueue: Boolean = false
|
2022-10-23 17:09:15 +05:30
|
|
|
|
2023-01-14 21:29:21 +05:30
|
|
|
fun clear() = queue.clear()
|
|
|
|
|
2022-10-23 17:09:15 +05:30
|
|
|
fun add(vararg streamItem: StreamItem) {
|
2023-01-17 00:32:07 +05:30
|
|
|
for (stream in streamItem) {
|
2023-02-27 23:08:36 +05:30
|
|
|
if (currentStream?.url?.toID() == stream.url?.toID() || stream.title.isNullOrBlank()) continue
|
2023-01-17 00:32:07 +05:30
|
|
|
// remove if already present
|
|
|
|
queue.remove(stream)
|
|
|
|
queue.add(stream)
|
2022-10-23 17:09:15 +05:30
|
|
|
}
|
2022-09-19 23:37:55 +05:30
|
|
|
}
|
|
|
|
|
2022-10-23 17:09:15 +05:30
|
|
|
fun addAsNext(streamItem: StreamItem) {
|
|
|
|
if (currentStream == streamItem) return
|
|
|
|
if (queue.contains(streamItem)) queue.remove(streamItem)
|
2022-09-19 23:37:55 +05:30
|
|
|
queue.add(
|
2022-10-23 17:09:15 +05:30
|
|
|
currentIndex() + 1,
|
2023-05-09 22:11:05 +05:30
|
|
|
streamItem,
|
2022-09-19 23:37:55 +05:30
|
|
|
)
|
|
|
|
}
|
2022-09-19 23:43:25 +05:30
|
|
|
|
2023-06-11 18:20:05 +05:30
|
|
|
// return the next item, or if repeating enabled, the first one of the queue
|
2023-03-18 21:58:48 +05:30
|
|
|
fun getNext(): String? = queue.getOrNull(currentIndex() + 1)?.url?.toID()
|
|
|
|
?: queue.firstOrNull()?.url?.toID()?.takeIf { repeatQueue }
|
2022-09-20 01:07:30 +05:30
|
|
|
|
2023-06-11 18:20:05 +05:30
|
|
|
// return the previous item, or if repeating enabled, the last one of the queue
|
2023-03-18 21:58:48 +05:30
|
|
|
fun getPrev(): String? = queue.getOrNull(currentIndex() - 1)?.url?.toID()
|
2023-06-11 18:20:05 +05:30
|
|
|
?: queue.lastOrNull()?.url?.toID()?.takeIf { repeatQueue }
|
2022-09-20 01:07:30 +05:30
|
|
|
|
2023-06-11 18:20:05 +05:30
|
|
|
fun hasPrev() = getPrev() != null
|
2022-11-24 21:19:38 +05:30
|
|
|
|
2023-06-11 18:20:05 +05:30
|
|
|
fun hasNext() = getNext() != null
|
2022-09-20 01:07:30 +05:30
|
|
|
|
2022-10-23 17:09:15 +05:30
|
|
|
fun updateCurrent(streamItem: StreamItem) {
|
|
|
|
currentStream = streamItem
|
2023-01-14 21:29:21 +05:30
|
|
|
if (!contains(streamItem)) queue.add(0, streamItem)
|
2022-09-19 23:43:25 +05:30
|
|
|
}
|
2022-09-20 01:13:13 +05:30
|
|
|
|
|
|
|
fun isNotEmpty() = queue.isNotEmpty()
|
2022-09-23 17:42:41 +05:30
|
|
|
|
2022-10-23 17:09:15 +05:30
|
|
|
fun isEmpty() = queue.isEmpty()
|
|
|
|
|
|
|
|
fun size() = queue.size
|
2022-09-23 17:42:41 +05:30
|
|
|
|
2023-03-18 21:58:48 +05:30
|
|
|
fun currentIndex(): Int = queue.indexOfFirst {
|
|
|
|
it.url?.toID() == currentStream?.url?.toID()
|
|
|
|
}.takeIf { it >= 0 } ?: 0
|
2022-09-23 17:42:41 +05:30
|
|
|
|
2023-01-13 22:50:00 +05:30
|
|
|
fun getCurrent(): StreamItem? = currentStream
|
|
|
|
|
2022-10-23 18:13:47 +05:30
|
|
|
fun contains(streamItem: StreamItem) = queue.any { it.url?.toID() == streamItem.url?.toID() }
|
2022-10-23 17:09:15 +05:30
|
|
|
|
2023-01-04 23:19:06 +05:30
|
|
|
// only returns a copy of the queue, no write access
|
|
|
|
fun getStreams() = queue.toList()
|
2022-10-23 18:13:47 +05:30
|
|
|
|
2022-10-29 02:34:44 +05:30
|
|
|
fun setStreams(streams: List<StreamItem>) {
|
|
|
|
queue.clear()
|
|
|
|
queue.addAll(streams)
|
|
|
|
}
|
|
|
|
|
2022-10-23 18:24:02 +05:30
|
|
|
fun remove(index: Int) = queue.removeAt(index)
|
|
|
|
|
2022-10-23 18:39:58 +05:30
|
|
|
fun move(from: Int, to: Int) = queue.move(from, to)
|
|
|
|
|
2022-10-23 18:13:47 +05:30
|
|
|
private fun fetchMoreFromPlaylist(playlistId: String, nextPage: String?) {
|
|
|
|
var playlistNextPage: String? = nextPage
|
2023-01-16 22:55:12 +05:30
|
|
|
scope.launch {
|
2022-10-23 18:13:47 +05:30
|
|
|
while (playlistNextPage != null) {
|
|
|
|
RetrofitInstance.authApi.getPlaylistNextPage(
|
|
|
|
playlistId,
|
2023-05-09 22:11:05 +05:30
|
|
|
playlistNextPage!!,
|
2022-10-23 18:13:47 +05:30
|
|
|
).apply {
|
|
|
|
add(
|
2023-05-09 22:11:05 +05:30
|
|
|
*this.relatedStreams.toTypedArray(),
|
2022-10-23 18:13:47 +05:30
|
|
|
)
|
|
|
|
playlistNextPage = this.nextpage
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fun insertPlaylist(playlistId: String, newCurrentStream: StreamItem) {
|
2023-01-16 22:55:12 +05:30
|
|
|
scope.launch {
|
2022-10-23 18:13:47 +05:30
|
|
|
try {
|
2022-12-01 19:07:44 +05:30
|
|
|
val playlist = PlaylistsHelper.getPlaylist(playlistId)
|
2023-03-18 21:58:48 +05:30
|
|
|
add(*playlist.relatedStreams.toTypedArray())
|
2022-10-23 18:13:47 +05:30
|
|
|
updateCurrent(newCurrentStream)
|
2022-11-20 21:39:07 +05:30
|
|
|
if (playlist.nextpage == null) return@launch
|
|
|
|
fetchMoreFromPlaylist(playlistId, playlist.nextpage)
|
2022-10-23 18:13:47 +05:30
|
|
|
} catch (e: Exception) {
|
|
|
|
e.printStackTrace()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-11-06 16:20:04 +05:30
|
|
|
|
2022-12-02 19:15:52 +05:30
|
|
|
private fun fetchMoreFromChannel(channelId: String, nextPage: String?) {
|
|
|
|
var channelNextPage: String? = nextPage
|
2023-01-16 22:55:12 +05:30
|
|
|
scope.launch {
|
2022-12-02 19:15:52 +05:30
|
|
|
while (channelNextPage != null) {
|
|
|
|
RetrofitInstance.api.getChannelNextPage(channelId, nextPage!!).apply {
|
2023-01-19 05:31:01 +05:30
|
|
|
add(*relatedStreams.toTypedArray())
|
2022-12-02 19:15:52 +05:30
|
|
|
channelNextPage = this.nextpage
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fun insertChannel(channelId: String, newCurrentStream: StreamItem) {
|
2023-01-16 22:55:12 +05:30
|
|
|
scope.launch {
|
|
|
|
runCatching {
|
2022-12-02 19:15:52 +05:30
|
|
|
val channel = RetrofitInstance.api.getChannel(channelId)
|
2023-01-19 05:31:01 +05:30
|
|
|
add(*channel.relatedStreams.toTypedArray())
|
2022-12-02 19:15:52 +05:30
|
|
|
updateCurrent(newCurrentStream)
|
|
|
|
if (channel.nextpage == null) return@launch
|
|
|
|
fetchMoreFromChannel(channelId, channel.nextpage)
|
2023-01-16 22:55:12 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fun insertByVideoId(videoId: String) {
|
|
|
|
scope.launch {
|
|
|
|
runCatching {
|
|
|
|
val streams = RetrofitInstance.api.getStreams(videoId.toID())
|
|
|
|
add(streams.toStreamItem(videoId))
|
2022-12-02 19:15:52 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-06 16:20:04 +05:30
|
|
|
fun onQueueItemSelected(index: Int) {
|
|
|
|
try {
|
|
|
|
val streamItem = queue[index]
|
|
|
|
updateCurrent(streamItem)
|
|
|
|
onQueueTapListener.invoke(streamItem)
|
|
|
|
} catch (e: Exception) {
|
|
|
|
Log.e("Queue on tap", "lifecycle already ended")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fun setOnQueueTapListener(listener: (StreamItem) -> Unit) {
|
|
|
|
onQueueTapListener = listener
|
|
|
|
}
|
|
|
|
|
2022-11-17 21:15:07 +05:30
|
|
|
fun resetToDefaults() {
|
|
|
|
repeatQueue = false
|
2022-11-06 16:20:04 +05:30
|
|
|
onQueueTapListener = {}
|
|
|
|
}
|
2022-09-19 23:37:55 +05:30
|
|
|
}
|