From 3171c82dbc076cec94d4598b200c148221f20570 Mon Sep 17 00:00:00 2001 From: Isira Seneviratne Date: Sun, 19 Feb 2023 19:52:20 +0530 Subject: [PATCH 1/7] Use lifecycleScope extension. --- .../ui/dialogs/AddToPlaylistDialog.kt | 3 +- .../ui/dialogs/CreatePlaylistDialog.kt | 30 +++++++++---------- .../ui/dialogs/RenamePlaylistDialog.kt | 4 +-- .../ui/fragments/CommentsRepliesFragment.kt | 4 +-- .../libretube/ui/fragments/PlayerFragment.kt | 5 ++-- .../libretube/ui/preferences/MainSettings.kt | 9 ++++-- .../ui/sheets/PlaylistOptionsBottomSheet.kt | 17 +++++------ .../ui/sheets/VideoOptionsBottomSheet.kt | 9 +++--- 8 files changed, 40 insertions(+), 41 deletions(-) diff --git a/app/src/main/java/com/github/libretube/ui/dialogs/AddToPlaylistDialog.kt b/app/src/main/java/com/github/libretube/ui/dialogs/AddToPlaylistDialog.kt index b422068c4..a825720b5 100644 --- a/app/src/main/java/com/github/libretube/ui/dialogs/AddToPlaylistDialog.kt +++ b/app/src/main/java/com/github/libretube/ui/dialogs/AddToPlaylistDialog.kt @@ -18,7 +18,6 @@ import com.github.libretube.extensions.toastFromMainThread import com.github.libretube.ui.models.PlaylistViewModel import com.github.libretube.util.PlayingQueue import com.google.android.material.dialog.MaterialAlertDialogBuilder -import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch @@ -85,7 +84,7 @@ class AddToPlaylistDialog( private fun addToPlaylist(playlistId: String) { val appContext = context?.applicationContext ?: return - CoroutineScope(Dispatchers.IO).launch { + lifecycleScope.launch(Dispatchers.IO) { val streams = when { videoId != null -> listOf( RetrofitInstance.api.getStreams(videoId).toStreamItem(videoId) diff --git a/app/src/main/java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt b/app/src/main/java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt index 0f8000fc1..d051eba3c 100644 --- a/app/src/main/java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt +++ b/app/src/main/java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt @@ -4,12 +4,12 @@ import android.app.Dialog import android.os.Bundle import android.widget.Toast import androidx.fragment.app.DialogFragment +import androidx.lifecycle.lifecycleScope import com.github.libretube.R import com.github.libretube.api.PlaylistsHelper import com.github.libretube.databinding.DialogCreatePlaylistBinding import com.github.libretube.extensions.toastFromMainThread import com.google.android.material.dialog.MaterialAlertDialogBuilder -import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch import kotlinx.coroutines.withContext @@ -28,17 +28,18 @@ class CreatePlaylistDialog( val appContext = context?.applicationContext playlistUrl?.queryParameter("list")?.let { - CoroutineScope(Dispatchers.IO).launch { - val playlistId = PlaylistsHelper.clonePlaylist(requireContext(), it)?.also { - withContext(Dispatchers.Main) { - onSuccess.invoke() - } + lifecycleScope.launch { + val playlistId = withContext(Dispatchers.IO) { + PlaylistsHelper.clonePlaylist(requireContext(), it) + } + if (playlistId != null) { + onSuccess() } appContext?.toastFromMainThread( if (playlistId != null) R.string.playlistCloned else R.string.server_error ) + dismiss() } - dismiss() } ?: run { Toast.makeText(context, R.string.invalid_url, Toast.LENGTH_SHORT).show() } @@ -53,16 +54,15 @@ class CreatePlaylistDialog( binding.createNewPlaylist.setOnClickListener(null) val listName = binding.playlistName.text.toString() if (listName != "") { - CoroutineScope(Dispatchers.IO).launch { - val playlistId = PlaylistsHelper.createPlaylist( - listName, - requireContext().applicationContext - ) - withContext(Dispatchers.Main) { - if (playlistId != null) onSuccess.invoke() + lifecycleScope.launch { + val playlistId = withContext(Dispatchers.IO) { + PlaylistsHelper.createPlaylist(listName, requireContext()) } + if (playlistId != null) { + onSuccess() + } + dismiss() } - dismiss() } else { Toast.makeText(context, R.string.emptyPlaylistName, Toast.LENGTH_LONG).show() } diff --git a/app/src/main/java/com/github/libretube/ui/dialogs/RenamePlaylistDialog.kt b/app/src/main/java/com/github/libretube/ui/dialogs/RenamePlaylistDialog.kt index 15ba841d5..c67e03492 100644 --- a/app/src/main/java/com/github/libretube/ui/dialogs/RenamePlaylistDialog.kt +++ b/app/src/main/java/com/github/libretube/ui/dialogs/RenamePlaylistDialog.kt @@ -6,13 +6,13 @@ import android.text.InputType import android.util.Log import android.widget.Toast import androidx.fragment.app.DialogFragment +import androidx.lifecycle.lifecycleScope import com.github.libretube.R import com.github.libretube.api.PlaylistsHelper import com.github.libretube.databinding.DialogTextPreferenceBinding import com.github.libretube.extensions.TAG import com.github.libretube.extensions.toastFromMainThread import com.google.android.material.dialog.MaterialAlertDialogBuilder -import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch @@ -41,7 +41,7 @@ class RenamePlaylistDialog( } if (input == currentPlaylistName) return@setPositiveButton val appContext = requireContext().applicationContext - CoroutineScope(Dispatchers.IO).launch { + lifecycleScope.launch(Dispatchers.IO) { val success = try { PlaylistsHelper.renamePlaylist(playlistId, binding.input.text.toString()) } catch (e: Exception) { diff --git a/app/src/main/java/com/github/libretube/ui/fragments/CommentsRepliesFragment.kt b/app/src/main/java/com/github/libretube/ui/fragments/CommentsRepliesFragment.kt index 343163f27..328146d1d 100644 --- a/app/src/main/java/com/github/libretube/ui/fragments/CommentsRepliesFragment.kt +++ b/app/src/main/java/com/github/libretube/ui/fragments/CommentsRepliesFragment.kt @@ -8,6 +8,7 @@ import android.view.ViewGroup import androidx.core.view.updatePadding import androidx.fragment.app.Fragment import androidx.fragment.app.activityViewModels +import androidx.lifecycle.lifecycleScope import androidx.recyclerview.widget.LinearLayoutManager import com.github.libretube.api.JsonHelper import com.github.libretube.api.RetrofitInstance @@ -19,7 +20,6 @@ import com.github.libretube.extensions.TAG import com.github.libretube.ui.adapters.CommentsAdapter import com.github.libretube.ui.extensions.filterNonEmptyComments import com.github.libretube.ui.models.CommentsViewModel -import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch import kotlinx.coroutines.withContext @@ -89,7 +89,7 @@ class CommentsRepliesFragment : Fragment() { nextPage: String, onFinished: (CommentsPage) -> Unit ) { - CoroutineScope(Dispatchers.IO).launch { + lifecycleScope.launch(Dispatchers.IO) { if (isLoading) return@launch isLoading = true repliesPage = try { diff --git a/app/src/main/java/com/github/libretube/ui/fragments/PlayerFragment.kt b/app/src/main/java/com/github/libretube/ui/fragments/PlayerFragment.kt index fe3f589b4..b32d5682e 100644 --- a/app/src/main/java/com/github/libretube/ui/fragments/PlayerFragment.kt +++ b/app/src/main/java/com/github/libretube/ui/fragments/PlayerFragment.kt @@ -111,7 +111,6 @@ import java.io.IOException import java.util.* import java.util.concurrent.Executors import kotlin.math.abs -import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch import kotlinx.coroutines.withContext @@ -663,7 +662,7 @@ class PlayerFragment : Fragment(R.layout.fragment_player), OnlinePlayerOptions { } if (PlayingQueue.isEmpty()) { - CoroutineScope(Dispatchers.IO).launch { + lifecycleScope.launch(Dispatchers.IO) { if (playlistId != null) { PlayingQueue.insertPlaylist(playlistId!!, streams.toStreamItem(videoId!!)) } else if (channelId != null) { @@ -728,7 +727,7 @@ class PlayerFragment : Fragment(R.layout.fragment_player), OnlinePlayerOptions { * fetch the segments for SponsorBlock */ private fun fetchSponsorBlockSegments() { - CoroutineScope(Dispatchers.IO).launch { + lifecycleScope.launch(Dispatchers.IO) { runCatching { val categories = PlayerHelper.getSponsorBlockCategories() if (categories.isEmpty()) return@runCatching diff --git a/app/src/main/java/com/github/libretube/ui/preferences/MainSettings.kt b/app/src/main/java/com/github/libretube/ui/preferences/MainSettings.kt index 4641fe33e..46f329822 100644 --- a/app/src/main/java/com/github/libretube/ui/preferences/MainSettings.kt +++ b/app/src/main/java/com/github/libretube/ui/preferences/MainSettings.kt @@ -4,6 +4,7 @@ import android.os.Bundle import androidx.annotation.StringRes import androidx.fragment.app.Fragment import androidx.fragment.app.commitNow +import androidx.lifecycle.lifecycleScope import androidx.preference.Preference import com.github.libretube.BuildConfig import com.github.libretube.R @@ -12,9 +13,9 @@ import com.github.libretube.ui.activities.SettingsActivity import com.github.libretube.ui.base.BasePreferenceFragment import com.github.libretube.ui.dialogs.UpdateAvailableDialog import com.google.android.material.snackbar.Snackbar -import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch +import kotlinx.coroutines.withContext class MainSettings : BasePreferenceFragment() { override val titleResourceId: Int = R.string.settings @@ -84,10 +85,12 @@ class MainSettings : BasePreferenceFragment() { // checking for update: yes -> dialog, no -> snackBar update?.setOnPreferenceClickListener { - CoroutineScope(Dispatchers.IO).launch { + lifecycleScope.launch { // check for update val updateInfo = try { - RetrofitInstance.externalApi.getUpdateInfo() + withContext(Dispatchers.IO) { + RetrofitInstance.externalApi.getUpdateInfo() + } } catch (e: Exception) { showSnackBar(R.string.unknown_error) return@launch diff --git a/app/src/main/java/com/github/libretube/ui/sheets/PlaylistOptionsBottomSheet.kt b/app/src/main/java/com/github/libretube/ui/sheets/PlaylistOptionsBottomSheet.kt index 61c512a6e..4354ce33b 100644 --- a/app/src/main/java/com/github/libretube/ui/sheets/PlaylistOptionsBottomSheet.kt +++ b/app/src/main/java/com/github/libretube/ui/sheets/PlaylistOptionsBottomSheet.kt @@ -1,6 +1,7 @@ package com.github.libretube.ui.sheets import android.os.Bundle +import androidx.lifecycle.lifecycleScope import com.github.libretube.R import com.github.libretube.api.PlaylistsHelper import com.github.libretube.api.RetrofitInstance @@ -8,7 +9,6 @@ import com.github.libretube.db.DatabaseHolder import com.github.libretube.enums.PlaylistType import com.github.libretube.enums.ShareObjectType import com.github.libretube.extensions.awaitQuery -import com.github.libretube.extensions.query import com.github.libretube.extensions.toID import com.github.libretube.extensions.toastFromMainThread import com.github.libretube.helpers.BackgroundHelper @@ -16,10 +16,10 @@ import com.github.libretube.obj.ShareData import com.github.libretube.ui.dialogs.DeletePlaylistDialog import com.github.libretube.ui.dialogs.RenamePlaylistDialog import com.github.libretube.ui.dialogs.ShareDialog -import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch import kotlinx.coroutines.runBlocking +import kotlinx.coroutines.withContext class PlaylistOptionsBottomSheet( private val playlistId: String, @@ -68,7 +68,7 @@ class PlaylistOptionsBottomSheet( // Clone the playlist to the users Piped account getString(R.string.clonePlaylist) -> { val appContext = context?.applicationContext - CoroutineScope(Dispatchers.IO).launch { + lifecycleScope.launch(Dispatchers.IO) { val playlistId = PlaylistsHelper.clonePlaylist( requireContext().applicationContext, playlistId @@ -95,12 +95,9 @@ class PlaylistOptionsBottomSheet( .show(parentFragmentManager, null) } else -> { - CoroutineScope(Dispatchers.IO).launch { + lifecycleScope.launch(Dispatchers.IO) { if (isBookmarked) { - query { - DatabaseHolder.Database.playlistBookmarkDao() - .deleteById(playlistId) - } + DatabaseHolder.Database.playlistBookmarkDao().deleteById(playlistId) } else { val bookmark = try { RetrofitInstance.api.getPlaylist(playlistId) @@ -109,8 +106,10 @@ class PlaylistOptionsBottomSheet( }.toPlaylistBookmark(playlistId) DatabaseHolder.Database.playlistBookmarkDao().insertAll(bookmark) } + withContext(Dispatchers.Main) { + dismiss() + } } - dismiss() } } } diff --git a/app/src/main/java/com/github/libretube/ui/sheets/VideoOptionsBottomSheet.kt b/app/src/main/java/com/github/libretube/ui/sheets/VideoOptionsBottomSheet.kt index b58c81bbe..b9851f3ce 100644 --- a/app/src/main/java/com/github/libretube/ui/sheets/VideoOptionsBottomSheet.kt +++ b/app/src/main/java/com/github/libretube/ui/sheets/VideoOptionsBottomSheet.kt @@ -2,6 +2,7 @@ package com.github.libretube.ui.sheets import android.os.Bundle import android.util.Log +import androidx.lifecycle.lifecycleScope import androidx.navigation.fragment.NavHostFragment import com.github.libretube.R import com.github.libretube.api.RetrofitInstance @@ -20,7 +21,6 @@ import com.github.libretube.ui.dialogs.DownloadDialog import com.github.libretube.ui.dialogs.ShareDialog import com.github.libretube.ui.fragments.SubscriptionsFragment import com.github.libretube.util.PlayingQueue -import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch @@ -77,11 +77,10 @@ class VideoOptionsBottomSheet( shareDialog.show(parentFragmentManager, ShareDialog::class.java.name) } getString(R.string.play_next) -> { - CoroutineScope(Dispatchers.IO).launch { + lifecycleScope.launch(Dispatchers.IO) { try { PlayingQueue.addAsNext( - RetrofitInstance.api.getStreams(videoId) - .toStreamItem(videoId) + RetrofitInstance.api.getStreams(videoId).toStreamItem(videoId) ) } catch (e: Exception) { e.printStackTrace() @@ -89,7 +88,7 @@ class VideoOptionsBottomSheet( } } getString(R.string.add_to_queue) -> { - CoroutineScope(Dispatchers.IO).launch { + lifecycleScope.launch(Dispatchers.IO) { try { PlayingQueue.add( RetrofitInstance.api.getStreams(videoId) From d1fd030ab0cb734acd6f06a67ce84a3050384d68 Mon Sep 17 00:00:00 2001 From: Isira Seneviratne Date: Sun, 19 Feb 2023 19:52:37 +0530 Subject: [PATCH 2/7] Use viewModelScope extension. --- .../com/github/libretube/ui/models/CommentsViewModel.kt | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/app/src/main/java/com/github/libretube/ui/models/CommentsViewModel.kt b/app/src/main/java/com/github/libretube/ui/models/CommentsViewModel.kt index 33d6c2fb1..becb96b71 100644 --- a/app/src/main/java/com/github/libretube/ui/models/CommentsViewModel.kt +++ b/app/src/main/java/com/github/libretube/ui/models/CommentsViewModel.kt @@ -3,16 +3,15 @@ package com.github.libretube.ui.models import android.util.Log import androidx.lifecycle.MutableLiveData import androidx.lifecycle.ViewModel +import androidx.lifecycle.viewModelScope import com.github.libretube.api.RetrofitInstance import com.github.libretube.api.obj.CommentsPage import com.github.libretube.extensions.TAG import com.github.libretube.ui.extensions.filterNonEmptyComments -import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch class CommentsViewModel : ViewModel() { - private val scope = CoroutineScope(Dispatchers.IO) private var isLoading = false val commentsPage = MutableLiveData() @@ -25,7 +24,7 @@ class CommentsViewModel : ViewModel() { fun fetchComments() { videoId ?: return - scope.launch { + viewModelScope.launch(Dispatchers.IO) { isLoading = true val response = try { RetrofitInstance.api.getComments(videoId!!) @@ -42,7 +41,7 @@ class CommentsViewModel : ViewModel() { fun fetchNextComments() { if (isLoading || nextPage == null || videoId == null) return - scope.launch { + viewModelScope.launch(Dispatchers.IO) { isLoading = true val response = try { RetrofitInstance.api.getCommentsNextPage(videoId!!, nextPage!!) From 38b3de6bed1e9252aa0b951790ff9a973c620aed Mon Sep 17 00:00:00 2001 From: Isira Seneviratne Date: Mon, 20 Feb 2023 20:06:15 +0530 Subject: [PATCH 3/7] Move use of lifecycleScope to BaseBottomSheet. --- .../libretube/ui/sheets/BaseBottomSheet.kt | 19 +++++----- .../ui/sheets/ChannelOptionsBottomSheet.kt | 7 ++-- .../ui/sheets/PlaylistOptionsBottomSheet.kt | 38 ++++++++----------- .../ui/sheets/VideoOptionsBottomSheet.kt | 30 +++++++-------- 4 files changed, 41 insertions(+), 53 deletions(-) diff --git a/app/src/main/java/com/github/libretube/ui/sheets/BaseBottomSheet.kt b/app/src/main/java/com/github/libretube/ui/sheets/BaseBottomSheet.kt index 162e80719..2a0020ce6 100644 --- a/app/src/main/java/com/github/libretube/ui/sheets/BaseBottomSheet.kt +++ b/app/src/main/java/com/github/libretube/ui/sheets/BaseBottomSheet.kt @@ -4,10 +4,12 @@ import android.os.Bundle import android.view.LayoutInflater import android.view.View import android.view.ViewGroup +import androidx.lifecycle.lifecycleScope import androidx.recyclerview.widget.LinearLayoutManager import com.github.libretube.databinding.BottomSheetBinding import com.github.libretube.obj.BottomSheetItem import com.github.libretube.ui.adapters.BottomSheetAdapter +import kotlinx.coroutines.launch open class BaseBottomSheet : ExpandedBottomSheet() { private lateinit var items: List @@ -30,19 +32,16 @@ open class BaseBottomSheet : ExpandedBottomSheet() { binding.optionsRecycler.adapter = BottomSheetAdapter(items, listener) } - fun setItems(items: List, listener: ((index: Int) -> Unit)?) = apply { + fun setItems(items: List, listener: (suspend (index: Int) -> Unit)?) = apply { this.items = items this.listener = { index -> - listener?.invoke(index) - dialog?.dismiss() + lifecycleScope.launch { + dialog?.hide() + listener?.invoke(index) + } } } - fun setSimpleItems(titles: List, listener: ((index: Int) -> Unit)?) = apply { - this.items = titles.map { BottomSheetItem(it) } - this.listener = { index -> - listener?.invoke(index) - dialog?.dismiss() - } - } + fun setSimpleItems(titles: List, listener: (suspend (index: Int) -> Unit)?) = + setItems(titles.map { BottomSheetItem(it) }, listener) } diff --git a/app/src/main/java/com/github/libretube/ui/sheets/ChannelOptionsBottomSheet.kt b/app/src/main/java/com/github/libretube/ui/sheets/ChannelOptionsBottomSheet.kt index eaf86e581..073ca23d5 100644 --- a/app/src/main/java/com/github/libretube/ui/sheets/ChannelOptionsBottomSheet.kt +++ b/app/src/main/java/com/github/libretube/ui/sheets/ChannelOptionsBottomSheet.kt @@ -11,7 +11,8 @@ import com.github.libretube.helpers.BackgroundHelper import com.github.libretube.helpers.NavigationHelper import com.github.libretube.obj.ShareData import com.github.libretube.ui.dialogs.ShareDialog -import kotlinx.coroutines.runBlocking +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.withContext /** * Dialog with different options for a selected video. @@ -41,7 +42,7 @@ class ChannelOptionsBottomSheet( } getString(R.string.play_latest_videos) -> { try { - val channel = runBlocking { + val channel = withContext(Dispatchers.IO) { RetrofitInstance.api.getChannel(channelId) } channel.relatedStreams.firstOrNull()?.url?.toID()?.let { @@ -57,7 +58,7 @@ class ChannelOptionsBottomSheet( } getString(R.string.playOnBackground) -> { try { - val channel = runBlocking { + val channel = withContext(Dispatchers.IO) { RetrofitInstance.api.getChannel(channelId) } channel.relatedStreams.firstOrNull()?.url?.toID()?.let { diff --git a/app/src/main/java/com/github/libretube/ui/sheets/PlaylistOptionsBottomSheet.kt b/app/src/main/java/com/github/libretube/ui/sheets/PlaylistOptionsBottomSheet.kt index 4354ce33b..94468f543 100644 --- a/app/src/main/java/com/github/libretube/ui/sheets/PlaylistOptionsBottomSheet.kt +++ b/app/src/main/java/com/github/libretube/ui/sheets/PlaylistOptionsBottomSheet.kt @@ -1,7 +1,6 @@ package com.github.libretube.ui.sheets import android.os.Bundle -import androidx.lifecycle.lifecycleScope import com.github.libretube.R import com.github.libretube.api.PlaylistsHelper import com.github.libretube.api.RetrofitInstance @@ -17,8 +16,6 @@ import com.github.libretube.ui.dialogs.DeletePlaylistDialog import com.github.libretube.ui.dialogs.RenamePlaylistDialog import com.github.libretube.ui.dialogs.ShareDialog import kotlinx.coroutines.Dispatchers -import kotlinx.coroutines.launch -import kotlinx.coroutines.runBlocking import kotlinx.coroutines.withContext class PlaylistOptionsBottomSheet( @@ -55,28 +52,26 @@ class PlaylistOptionsBottomSheet( when (optionsList[which]) { // play the playlist in the background getString(R.string.playOnBackground) -> { - runBlocking { - val playlist = PlaylistsHelper.getPlaylist(playlistId) - if (playlist.relatedStreams.isEmpty()) return@runBlocking + val playlist = withContext(Dispatchers.IO) { + PlaylistsHelper.getPlaylist(playlistId) + } + playlist.relatedStreams.firstOrNull()?.let { BackgroundHelper.playOnBackground( - context = requireContext(), - videoId = playlist.relatedStreams[0].url!!.toID(), + requireContext(), + it.url!!.toID(), playlistId = playlistId ) } } // Clone the playlist to the users Piped account getString(R.string.clonePlaylist) -> { - val appContext = context?.applicationContext - lifecycleScope.launch(Dispatchers.IO) { - val playlistId = PlaylistsHelper.clonePlaylist( - requireContext().applicationContext, - playlistId - ) - appContext?.toastFromMainThread( - if (playlistId != null) R.string.playlistCloned else R.string.server_error - ) + val context = requireContext() + val playlistId = withContext(Dispatchers.IO) { + PlaylistsHelper.clonePlaylist(context, playlistId) } + context.toastFromMainThread( + if (playlistId != null) R.string.playlistCloned else R.string.server_error + ) } // share the playlist getString(R.string.share) -> { @@ -87,7 +82,7 @@ class PlaylistOptionsBottomSheet( getString(R.string.deletePlaylist) -> { DeletePlaylistDialog(playlistId, playlistType) { // try to refresh the playlists in the library on deletion success - onDelete.invoke() + onDelete() }.show(parentFragmentManager, null) } getString(R.string.renamePlaylist) -> { @@ -95,20 +90,17 @@ class PlaylistOptionsBottomSheet( .show(parentFragmentManager, null) } else -> { - lifecycleScope.launch(Dispatchers.IO) { + withContext(Dispatchers.IO) { if (isBookmarked) { DatabaseHolder.Database.playlistBookmarkDao().deleteById(playlistId) } else { val bookmark = try { RetrofitInstance.api.getPlaylist(playlistId) } catch (e: Exception) { - return@launch + return@withContext }.toPlaylistBookmark(playlistId) DatabaseHolder.Database.playlistBookmarkDao().insertAll(bookmark) } - withContext(Dispatchers.Main) { - dismiss() - } } } } diff --git a/app/src/main/java/com/github/libretube/ui/sheets/VideoOptionsBottomSheet.kt b/app/src/main/java/com/github/libretube/ui/sheets/VideoOptionsBottomSheet.kt index b9851f3ce..cf87c75e4 100644 --- a/app/src/main/java/com/github/libretube/ui/sheets/VideoOptionsBottomSheet.kt +++ b/app/src/main/java/com/github/libretube/ui/sheets/VideoOptionsBottomSheet.kt @@ -2,7 +2,6 @@ package com.github.libretube.ui.sheets import android.os.Bundle import android.util.Log -import androidx.lifecycle.lifecycleScope import androidx.navigation.fragment.NavHostFragment import com.github.libretube.R import com.github.libretube.api.RetrofitInstance @@ -22,7 +21,7 @@ import com.github.libretube.ui.dialogs.ShareDialog import com.github.libretube.ui.fragments.SubscriptionsFragment import com.github.libretube.util.PlayingQueue import kotlinx.coroutines.Dispatchers -import kotlinx.coroutines.launch +import kotlinx.coroutines.withContext /** * Dialog with different options for a selected video. @@ -77,26 +76,23 @@ class VideoOptionsBottomSheet( shareDialog.show(parentFragmentManager, ShareDialog::class.java.name) } getString(R.string.play_next) -> { - lifecycleScope.launch(Dispatchers.IO) { - try { - PlayingQueue.addAsNext( - RetrofitInstance.api.getStreams(videoId).toStreamItem(videoId) - ) - } catch (e: Exception) { - e.printStackTrace() + try { + val streamItem = withContext(Dispatchers.IO) { + RetrofitInstance.api.getStreams(videoId).toStreamItem(videoId) } + PlayingQueue.addAsNext(streamItem) + } catch (e: Exception) { + e.printStackTrace() } } getString(R.string.add_to_queue) -> { - lifecycleScope.launch(Dispatchers.IO) { - try { - PlayingQueue.add( - RetrofitInstance.api.getStreams(videoId) - .toStreamItem(videoId) - ) - } catch (e: Exception) { - e.printStackTrace() + try { + val streamItem = withContext(Dispatchers.IO) { + RetrofitInstance.api.getStreams(videoId).toStreamItem(videoId) } + PlayingQueue.add(streamItem) + } catch (e: Exception) { + e.printStackTrace() } } getString(R.string.mark_as_watched) -> { From 9a36a832a659c2412f637d1051f98e1e0d2d3413 Mon Sep 17 00:00:00 2001 From: Isira Seneviratne Date: Mon, 20 Feb 2023 20:20:27 +0530 Subject: [PATCH 4/7] Hide clone dialog before starting the clone. --- .../com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/src/main/java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt b/app/src/main/java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt index d051eba3c..06a9ace5f 100644 --- a/app/src/main/java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt +++ b/app/src/main/java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt @@ -29,6 +29,7 @@ class CreatePlaylistDialog( playlistUrl?.queryParameter("list")?.let { lifecycleScope.launch { + dialog?.hide() val playlistId = withContext(Dispatchers.IO) { PlaylistsHelper.clonePlaylist(requireContext(), it) } @@ -38,7 +39,6 @@ class CreatePlaylistDialog( appContext?.toastFromMainThread( if (playlistId != null) R.string.playlistCloned else R.string.server_error ) - dismiss() } } ?: run { Toast.makeText(context, R.string.invalid_url, Toast.LENGTH_SHORT).show() From 40abc051b2fe51b5f9c036b1c70faa83c290c9c4 Mon Sep 17 00:00:00 2001 From: Isira Seneviratne Date: Mon, 20 Feb 2023 20:40:59 +0530 Subject: [PATCH 5/7] Dismiss dialogs after the long-running operations complete. --- .../java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt | 1 + .../main/java/com/github/libretube/ui/sheets/BaseBottomSheet.kt | 1 + 2 files changed, 2 insertions(+) diff --git a/app/src/main/java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt b/app/src/main/java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt index 06a9ace5f..3f3d50f0b 100644 --- a/app/src/main/java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt +++ b/app/src/main/java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt @@ -39,6 +39,7 @@ class CreatePlaylistDialog( appContext?.toastFromMainThread( if (playlistId != null) R.string.playlistCloned else R.string.server_error ) + dismiss() } } ?: run { Toast.makeText(context, R.string.invalid_url, Toast.LENGTH_SHORT).show() diff --git a/app/src/main/java/com/github/libretube/ui/sheets/BaseBottomSheet.kt b/app/src/main/java/com/github/libretube/ui/sheets/BaseBottomSheet.kt index 2a0020ce6..140adc68d 100644 --- a/app/src/main/java/com/github/libretube/ui/sheets/BaseBottomSheet.kt +++ b/app/src/main/java/com/github/libretube/ui/sheets/BaseBottomSheet.kt @@ -38,6 +38,7 @@ open class BaseBottomSheet : ExpandedBottomSheet() { lifecycleScope.launch { dialog?.hide() listener?.invoke(index) + dismiss() } } } From 10d6ad7633312117e2b906fc48ff64164b1ee286 Mon Sep 17 00:00:00 2001 From: Isira Seneviratne Date: Tue, 21 Feb 2023 18:32:27 +0530 Subject: [PATCH 6/7] Hide clone, create and rename playlist dialogs. --- .../ui/dialogs/CreatePlaylistDialog.kt | 3 +- .../ui/dialogs/RenamePlaylistDialog.kt | 60 +++++++++++-------- 2 files changed, 37 insertions(+), 26 deletions(-) diff --git a/app/src/main/java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt b/app/src/main/java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt index 3f3d50f0b..eb76d7f8a 100644 --- a/app/src/main/java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt +++ b/app/src/main/java/com/github/libretube/ui/dialogs/CreatePlaylistDialog.kt @@ -29,7 +29,7 @@ class CreatePlaylistDialog( playlistUrl?.queryParameter("list")?.let { lifecycleScope.launch { - dialog?.hide() + requireDialog().hide() val playlistId = withContext(Dispatchers.IO) { PlaylistsHelper.clonePlaylist(requireContext(), it) } @@ -56,6 +56,7 @@ class CreatePlaylistDialog( val listName = binding.playlistName.text.toString() if (listName != "") { lifecycleScope.launch { + requireDialog().hide() val playlistId = withContext(Dispatchers.IO) { PlaylistsHelper.createPlaylist(listName, requireContext()) } diff --git a/app/src/main/java/com/github/libretube/ui/dialogs/RenamePlaylistDialog.kt b/app/src/main/java/com/github/libretube/ui/dialogs/RenamePlaylistDialog.kt index c67e03492..e3125104b 100644 --- a/app/src/main/java/com/github/libretube/ui/dialogs/RenamePlaylistDialog.kt +++ b/app/src/main/java/com/github/libretube/ui/dialogs/RenamePlaylistDialog.kt @@ -1,6 +1,7 @@ package com.github.libretube.ui.dialogs import android.app.Dialog +import android.content.DialogInterface import android.os.Bundle import android.text.InputType import android.util.Log @@ -15,6 +16,7 @@ import com.github.libretube.extensions.toastFromMainThread import com.google.android.material.dialog.MaterialAlertDialogBuilder import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch +import kotlinx.coroutines.withContext class RenamePlaylistDialog( private val playlistId: String, @@ -29,34 +31,42 @@ class RenamePlaylistDialog( return MaterialAlertDialogBuilder(requireContext()) .setTitle(R.string.renamePlaylist) .setView(binding.root) - .setPositiveButton(R.string.okay) { _, _ -> - val input = binding.input.text.toString() - if (input == "") { - Toast.makeText( - context, - R.string.emptyPlaylistName, - Toast.LENGTH_SHORT - ).show() - return@setPositiveButton - } - if (input == currentPlaylistName) return@setPositiveButton - val appContext = requireContext().applicationContext - lifecycleScope.launch(Dispatchers.IO) { - val success = try { - PlaylistsHelper.renamePlaylist(playlistId, binding.input.text.toString()) - } catch (e: Exception) { - Log.e(TAG(), e.toString()) - e.localizedMessage?.let { appContext.toastFromMainThread(it) } - return@launch + .setPositiveButton(R.string.okay, null) + .setNegativeButton(R.string.cancel, null) + .show() + .apply { + getButton(DialogInterface.BUTTON_POSITIVE).setOnClickListener { + val input = binding.input.text.toString() + if (input == "") { + Toast.makeText( + context, + R.string.emptyPlaylistName, + Toast.LENGTH_SHORT + ).show() + return@setOnClickListener } - if (success) { - appContext.toastFromMainThread(R.string.success) - } else { - appContext.toastFromMainThread(R.string.server_error) + if (input == currentPlaylistName) return@setOnClickListener + val appContext = requireContext().applicationContext + + lifecycleScope.launch { + requireDialog().hide() + val success = try { + withContext(Dispatchers.IO) { + PlaylistsHelper.renamePlaylist(playlistId, input) + } + } catch (e: Exception) { + Log.e(TAG(), e.toString()) + e.localizedMessage?.let { appContext.toastFromMainThread(it) } + return@launch + } + if (success) { + appContext.toastFromMainThread(R.string.success) + } else { + appContext.toastFromMainThread(R.string.server_error) + } + dismiss() } } } - .setNegativeButton(R.string.cancel, null) - .show() } } From f5f6bb235627091bfb29bcaeaa1bf814f5b76684 Mon Sep 17 00:00:00 2001 From: Bnyro Date: Wed, 22 Feb 2023 10:24:52 +0100 Subject: [PATCH 7/7] Fix adding videos to a playlist --- .../ui/dialogs/AddToPlaylistDialog.kt | 42 +++++++++---------- 1 file changed, 21 insertions(+), 21 deletions(-) diff --git a/app/src/main/java/com/github/libretube/ui/dialogs/AddToPlaylistDialog.kt b/app/src/main/java/com/github/libretube/ui/dialogs/AddToPlaylistDialog.kt index a825720b5..26a954c3b 100644 --- a/app/src/main/java/com/github/libretube/ui/dialogs/AddToPlaylistDialog.kt +++ b/app/src/main/java/com/github/libretube/ui/dialogs/AddToPlaylistDialog.kt @@ -18,7 +18,6 @@ import com.github.libretube.extensions.toastFromMainThread import com.github.libretube.ui.models.PlaylistViewModel import com.github.libretube.util.PlayingQueue import com.google.android.material.dialog.MaterialAlertDialogBuilder -import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch /** @@ -75,34 +74,35 @@ class AddToPlaylistDialog( binding.addToPlaylist.setOnClickListener { val index = binding.playlistsSpinner.selectedItemPosition viewModel.lastSelectedPlaylistId = response[index].id!! - addToPlaylist(response[index].id!!) - dialog?.dismiss() + dialog?.hide() + lifecycleScope.launch { + addToPlaylist(response[index].id!!) + dialog?.dismiss() + } } } } } - private fun addToPlaylist(playlistId: String) { + private suspend fun addToPlaylist(playlistId: String) { val appContext = context?.applicationContext ?: return - lifecycleScope.launch(Dispatchers.IO) { - val streams = when { - videoId != null -> listOf( - RetrofitInstance.api.getStreams(videoId).toStreamItem(videoId) - ) - else -> PlayingQueue.getStreams() - } - - val success = try { - PlaylistsHelper.addToPlaylist(playlistId, *streams.toTypedArray()) - } catch (e: Exception) { - Log.e(TAG(), e.toString()) - appContext.toastFromMainThread(R.string.unknown_error) - return@launch - } - appContext.toastFromMainThread( - if (success) R.string.added_to_playlist else R.string.fail + val streams = when { + videoId != null -> listOf( + RetrofitInstance.api.getStreams(videoId).toStreamItem(videoId) ) + else -> PlayingQueue.getStreams() } + + val success = try { + PlaylistsHelper.addToPlaylist(playlistId, *streams.toTypedArray()) + } catch (e: Exception) { + Log.e(TAG(), e.toString()) + appContext.toastFromMainThread(R.string.unknown_error) + return + } + appContext.toastFromMainThread( + if (success) R.string.added_to_playlist else R.string.fail + ) } private fun Fragment?.runOnUiThread(action: () -> Unit) {