Improve home page loading behavior

This commit is contained in:
Bnyro 2023-02-12 15:48:38 +01:00
parent cd21f64be2
commit 883f14fd56

View File

@ -1,10 +1,10 @@
package com.github.libretube.ui.fragments package com.github.libretube.ui.fragments
import android.os.Bundle import android.os.Bundle
import android.util.Log
import android.view.LayoutInflater import android.view.LayoutInflater
import android.view.View import android.view.View
import android.view.ViewGroup import android.view.ViewGroup
import androidx.fragment.app.activityViewModels
import androidx.lifecycle.lifecycleScope import androidx.lifecycle.lifecycleScope
import androidx.navigation.fragment.findNavController import androidx.navigation.fragment.findNavController
import androidx.recyclerview.widget.GridLayoutManager import androidx.recyclerview.widget.GridLayoutManager
@ -14,21 +14,23 @@ import com.github.libretube.R
import com.github.libretube.api.PlaylistsHelper import com.github.libretube.api.PlaylistsHelper
import com.github.libretube.api.RetrofitInstance import com.github.libretube.api.RetrofitInstance
import com.github.libretube.api.SubscriptionHelper import com.github.libretube.api.SubscriptionHelper
import com.github.libretube.constants.PreferenceKeys
import com.github.libretube.databinding.FragmentHomeBinding import com.github.libretube.databinding.FragmentHomeBinding
import com.github.libretube.db.DatabaseHolder import com.github.libretube.db.DatabaseHolder
import com.github.libretube.extensions.awaitQuery import com.github.libretube.extensions.awaitQuery
import com.github.libretube.extensions.toastFromMainThread
import com.github.libretube.helpers.LocaleHelper import com.github.libretube.helpers.LocaleHelper
import com.github.libretube.helpers.PreferenceHelper
import com.github.libretube.ui.adapters.PlaylistBookmarkAdapter import com.github.libretube.ui.adapters.PlaylistBookmarkAdapter
import com.github.libretube.ui.adapters.PlaylistsAdapter import com.github.libretube.ui.adapters.PlaylistsAdapter
import com.github.libretube.ui.adapters.VideosAdapter import com.github.libretube.ui.adapters.VideosAdapter
import com.github.libretube.ui.base.BaseFragment import com.github.libretube.ui.base.BaseFragment
import kotlinx.coroutines.CancellationException import com.github.libretube.ui.models.SubscriptionsViewModel
import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.launch import kotlinx.coroutines.withContext
class HomeFragment : BaseFragment() { class HomeFragment : BaseFragment() {
private lateinit var binding: FragmentHomeBinding private lateinit var binding: FragmentHomeBinding
private val subscriptionsViewModel: SubscriptionsViewModel by activityViewModels()
override fun onCreateView( override fun onCreateView(
inflater: LayoutInflater, inflater: LayoutInflater,
@ -60,102 +62,106 @@ class HomeFragment : BaseFragment() {
binding.refresh.setOnRefreshListener { binding.refresh.setOnRefreshListener {
binding.refresh.isRefreshing = true binding.refresh.isRefreshing = true
lifecycleScope.launch(Dispatchers.IO) { fetchHomeFeed()
fetchHome()
}
} }
lifecycleScope.launch(Dispatchers.IO) { fetchHomeFeed()
fetchHome() }
private fun fetchHomeFeed() {
lifecycleScope.launchWhenCreated {
withContext(Dispatchers.IO) {
loadTrending()
loadBookmarks()
}
}
lifecycleScope.launchWhenCreated {
withContext(Dispatchers.IO) {
loadFeed()
loadPlaylists()
}
} }
} }
private suspend fun fetchHome() { private suspend fun loadTrending() {
runOrError { val region = LocaleHelper.getTrendingRegion(requireContext())
val feed = SubscriptionHelper.getFeed().take(20) val trending = RetrofitInstance.api.getTrending(region).take(10)
if (feed.isEmpty()) return@runOrError if (trending.isEmpty()) return
runOnUiThread {
makeVisible(binding.featuredRV, binding.featuredTV)
binding.featuredRV.layoutManager = LinearLayoutManager(
context,
LinearLayoutManager.HORIZONTAL,
false
)
binding.featuredRV.adapter = VideosAdapter(
feed.toMutableList(),
forceMode = VideosAdapter.Companion.ForceMode.HOME
)
}
}
runOrError { runOnUiThread {
val region = LocaleHelper.getTrendingRegion(requireContext()) makeVisible(binding.trendingRV, binding.trendingTV)
val trending = RetrofitInstance.api.getTrending(region).take(10) binding.trendingRV.layoutManager = GridLayoutManager(context, 2)
if (trending.isEmpty()) return@runOrError binding.trendingRV.adapter = VideosAdapter(
runOnUiThread { trending.toMutableList(),
makeVisible(binding.trendingRV, binding.trendingTV) forceMode = VideosAdapter.Companion.ForceMode.TRENDING
binding.trendingRV.layoutManager = GridLayoutManager(context, 2) )
binding.trendingRV.adapter = VideosAdapter(
trending.toMutableList(),
forceMode = VideosAdapter.Companion.ForceMode.TRENDING
)
}
} }
}
runOrError { private suspend fun loadFeed() {
val playlists = PlaylistsHelper.getPlaylists().take(20) val feed = if (
if (playlists.isEmpty()) return@runOrError PreferenceHelper.getBoolean(PreferenceKeys.SAVE_FEED, false) &&
runOnUiThread { !subscriptionsViewModel.videoFeed.value.isNullOrEmpty()
makeVisible(binding.playlistsRV, binding.playlistsTV) ) {
binding.playlistsRV.layoutManager = LinearLayoutManager(context) subscriptionsViewModel.videoFeed.value.orEmpty()
binding.playlistsRV.adapter = PlaylistsAdapter( } else {
playlists.toMutableList(), SubscriptionHelper.getFeed()
PlaylistsHelper.getPrivatePlaylistType() }.take(20)
) if (feed.isEmpty()) return
binding.playlistsRV.adapter?.registerAdapterDataObserver(object : runOnUiThread {
RecyclerView.AdapterDataObserver() { makeVisible(binding.featuredRV, binding.featuredTV)
override fun onItemRangeRemoved(positionStart: Int, itemCount: Int) { binding.featuredRV.layoutManager = LinearLayoutManager(
super.onItemRangeRemoved(positionStart, itemCount) context,
if (itemCount == 0) { LinearLayoutManager.HORIZONTAL,
binding.playlistsRV.visibility = View.GONE false
binding.playlistsTV.visibility = View.GONE )
} binding.featuredRV.adapter = VideosAdapter(
feed.toMutableList(),
forceMode = VideosAdapter.Companion.ForceMode.HOME
)
}
}
private fun loadBookmarks() {
val bookmarkedPlaylists = awaitQuery {
DatabaseHolder.Database.playlistBookmarkDao().getAll()
}
if (bookmarkedPlaylists.isEmpty()) return
runOnUiThread {
makeVisible(binding.bookmarksTV, binding.bookmarksRV)
binding.bookmarksRV.layoutManager = LinearLayoutManager(
context,
LinearLayoutManager.HORIZONTAL,
false
)
binding.bookmarksRV.adapter = PlaylistBookmarkAdapter(
bookmarkedPlaylists,
PlaylistBookmarkAdapter.Companion.BookmarkMode.HOME
)
}
}
private suspend fun loadPlaylists() {
val playlists = PlaylistsHelper.getPlaylists().take(20)
if (playlists.isEmpty()) return
runOnUiThread {
makeVisible(binding.playlistsRV, binding.playlistsTV)
binding.playlistsRV.layoutManager = LinearLayoutManager(context)
binding.playlistsRV.adapter = PlaylistsAdapter(
playlists.toMutableList(),
PlaylistsHelper.getPrivatePlaylistType()
)
binding.playlistsRV.adapter?.registerAdapterDataObserver(object :
RecyclerView.AdapterDataObserver() {
override fun onItemRangeRemoved(positionStart: Int, itemCount: Int) {
super.onItemRangeRemoved(positionStart, itemCount)
if (itemCount == 0) {
binding.playlistsRV.visibility = View.GONE
binding.playlistsTV.visibility = View.GONE
} }
}) }
} })
}
runOrError {
val bookmarkedPlaylists = awaitQuery {
DatabaseHolder.Database.playlistBookmarkDao().getAll()
}
if (bookmarkedPlaylists.isEmpty()) return@runOrError
runOnUiThread {
makeVisible(binding.bookmarksTV, binding.bookmarksRV)
binding.bookmarksRV.layoutManager = LinearLayoutManager(
context,
LinearLayoutManager.HORIZONTAL,
false
)
binding.bookmarksRV.adapter = PlaylistBookmarkAdapter(
bookmarkedPlaylists,
PlaylistBookmarkAdapter.Companion.BookmarkMode.HOME
)
}
}
}
private fun runOrError(action: suspend () -> Unit) {
lifecycleScope.launch(Dispatchers.IO) {
try {
action.invoke()
// Can be caused due to activity launch in front view from PiP mode.
} catch (e: CancellationException) {
Log.e("fetching home tab", e.toString())
} catch (e: Exception) {
e.localizedMessage?.let { context?.toastFromMainThread(it) }
Log.e("fetching home tab", e.toString())
}
} }
} }