funkwhale-app-android/app/src/main/java/audio/funkwhale/ffa/fragments/TracksFragment.kt

278 lines
8.7 KiB
Kotlin
Raw Normal View History

package audio.funkwhale.ffa.fragments
2019-08-19 16:50:33 +02:00
import android.graphics.PorterDuff
import android.graphics.PorterDuffColorFilter
2019-08-19 16:50:33 +02:00
import android.os.Bundle
2020-06-13 16:45:58 +02:00
import android.view.Gravity
2021-07-16 10:03:52 +02:00
import android.view.LayoutInflater
2019-08-19 16:50:33 +02:00
import android.view.View
2021-07-16 10:03:52 +02:00
import android.view.ViewGroup
2020-06-13 16:45:58 +02:00
import androidx.appcompat.widget.PopupMenu
import androidx.lifecycle.lifecycleScope
2023-01-10 13:56:20 +01:00
import androidx.navigation.fragment.navArgs
2019-08-19 16:50:33 +02:00
import androidx.recyclerview.widget.RecyclerView
import audio.funkwhale.ffa.R
import audio.funkwhale.ffa.adapters.FavoriteListener
import audio.funkwhale.ffa.adapters.TracksAdapter
2021-07-16 10:03:52 +02:00
import audio.funkwhale.ffa.databinding.FragmentTracksBinding
import audio.funkwhale.ffa.model.Track
import audio.funkwhale.ffa.repositories.FavoritedRepository
import audio.funkwhale.ffa.repositories.FavoritesRepository
import audio.funkwhale.ffa.repositories.TracksRepository
2021-09-09 09:56:15 +02:00
import audio.funkwhale.ffa.utils.Command
import audio.funkwhale.ffa.utils.CommandBus
import audio.funkwhale.ffa.utils.CoverArt
2021-09-09 09:56:15 +02:00
import audio.funkwhale.ffa.utils.Event
import audio.funkwhale.ffa.utils.EventBus
import audio.funkwhale.ffa.utils.Request
import audio.funkwhale.ffa.utils.RequestBus
import audio.funkwhale.ffa.utils.Response
import audio.funkwhale.ffa.utils.getMetadata
import audio.funkwhale.ffa.utils.maybeNormalizeUrl
import audio.funkwhale.ffa.utils.toast
import audio.funkwhale.ffa.utils.wait
import com.google.android.exoplayer2.offline.Download
import com.google.android.exoplayer2.offline.DownloadManager
import com.preference.PowerPreference
2020-06-14 00:42:45 +02:00
import jp.wasabeef.picasso.transformations.RoundedCornersTransformation
import kotlinx.coroutines.Dispatchers.IO
2019-08-19 16:50:33 +02:00
import kotlinx.coroutines.Dispatchers.Main
import kotlinx.coroutines.launch
import kotlinx.coroutines.withContext
import org.koin.java.KoinJavaComponent.inject
2019-08-19 16:50:33 +02:00
2021-07-16 10:03:52 +02:00
class TracksFragment : FFAFragment<Track, TracksAdapter>() {
2023-01-10 13:56:20 +01:00
private val args by navArgs<TracksFragmentArgs>()
private val exoDownloadManager: DownloadManager by inject(DownloadManager::class.java)
2021-07-16 10:03:52 +02:00
override val recycler: RecyclerView get() = binding.tracks
private var _binding: FragmentTracksBinding? = null
private val binding get() = _binding!!
private lateinit var favoritesRepository: FavoritesRepository
private lateinit var favoritedRepository: FavoritedRepository
2019-08-19 16:50:33 +02:00
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
favoritesRepository = FavoritesRepository(context)
favoritedRepository = FavoritedRepository(context)
2023-01-10 13:56:20 +01:00
repository = TracksRepository(context, args.album.id)
adapter = TracksAdapter(layoutInflater, context, FavoriteListener(favoritesRepository))
2019-08-19 16:50:33 +02:00
watchEventBus()
}
override fun onDataFetched(data: List<Track>) {
when {
data.all { it.downloaded } -> {
2021-07-16 10:03:52 +02:00
binding.title.setCompoundDrawablesWithIntrinsicBounds(R.drawable.downloaded, 0, 0, 0)
binding.title.compoundDrawables.forEach {
it?.colorFilter =
PorterDuffColorFilter(
requireContext().getColor(R.color.downloaded),
PorterDuff.Mode.SRC_IN
)
}
}
data.all { it.cached } -> {
2021-07-16 10:03:52 +02:00
binding.title.setCompoundDrawablesWithIntrinsicBounds(R.drawable.downloaded, 0, 0, 0)
binding.title.compoundDrawables.forEach {
it?.colorFilter =
PorterDuffColorFilter(
requireContext().getColor(R.color.cached),
PorterDuff.Mode.SRC_IN
)
}
}
else -> {
2021-07-16 10:03:52 +02:00
binding.title.setCompoundDrawablesWithIntrinsicBounds(0, 0, 0, 0)
}
}
}
2021-07-16 10:03:52 +02:00
override fun onCreateView(
inflater: LayoutInflater,
container: ViewGroup?,
savedInstanceState: Bundle?
): View {
_binding = FragmentTracksBinding.inflate(inflater)
swiper = binding.swiper
when (PowerPreference.getDefaultFile().getString("play_order")) {
"in_order" -> binding.play.text = getString(R.string.playback_play)
else -> binding.play.text = getString(R.string.playback_shuffle)
}
2021-07-16 10:03:52 +02:00
return binding.root
}
override fun onDestroyView() {
super.onDestroyView()
_binding = null
}
2019-08-19 16:50:33 +02:00
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
CoverArt.requestCreator(maybeNormalizeUrl(args.album.cover()))
2019-08-19 16:50:33 +02:00
.noFade()
.fit()
.centerCrop()
2020-06-14 00:42:45 +02:00
.transform(RoundedCornersTransformation(16, 0))
2021-07-16 10:03:52 +02:00
.into(binding.cover)
2019-08-19 16:50:33 +02:00
2023-01-10 13:56:20 +01:00
binding.artist.text = args.album.artist.name
binding.title.text = args.album.title
2019-08-19 16:50:33 +02:00
}
override fun onResume() {
super.onResume()
lifecycleScope.launch(Main) {
2019-08-19 16:50:33 +02:00
RequestBus.send(Request.GetCurrentTrack).wait<Response.CurrentTrack>()?.let { response ->
adapter.currentTrack = response.track
adapter.notifyDataSetChanged()
2019-08-19 16:50:33 +02:00
}
refreshDownloadedTracks()
2019-08-19 16:50:33 +02:00
}
var coverHeight: Float? = null
2021-07-16 10:03:52 +02:00
binding.scroller.setOnScrollChangeListener { _: View?, _: Int, scrollY: Int, _: Int, _: Int ->
if (coverHeight == null) {
2021-07-16 10:03:52 +02:00
coverHeight = binding.cover.measuredHeight.toFloat()
}
2021-07-16 10:03:52 +02:00
binding.cover.translationY = (scrollY / 2).toFloat()
coverHeight?.let { height ->
2021-07-16 10:03:52 +02:00
binding.cover.alpha = (height - scrollY.toFloat()) / height
}
}
when (PowerPreference.getDefaultFile().getString("play_order")) {
2021-07-16 10:03:52 +02:00
"in_order" -> binding.play.text = getString(R.string.playback_play)
else -> binding.play.text = getString(R.string.playback_shuffle)
}
2021-07-16 10:03:52 +02:00
binding.play.setOnClickListener {
when (PowerPreference.getDefaultFile().getString("play_order")) {
"in_order" -> CommandBus.send(Command.ReplaceQueue(adapter.data))
else -> CommandBus.send(Command.ReplaceQueue(adapter.data.shuffled()))
}
2019-08-19 16:50:33 +02:00
context.toast("All tracks were added to your queue")
}
2020-06-13 16:45:58 +02:00
context?.let { context ->
2021-07-16 10:03:52 +02:00
binding.actions.setOnClickListener {
PopupMenu(
context,
binding.actions,
Gravity.START,
R.attr.actionOverflowMenuStyle,
0
).apply {
2020-06-13 16:45:58 +02:00
inflate(R.menu.album)
2019-08-19 16:50:33 +02:00
menu.findItem(R.id.play_secondary)?.let { item ->
when (PowerPreference.getDefaultFile().getString("play_order")) {
"in_order" -> item.title = getString(R.string.playback_shuffle)
else -> item.title = getString(R.string.playback_play)
}
}
2020-06-13 16:45:58 +02:00
setOnMenuItemClickListener {
when (it.itemId) {
2021-09-09 09:56:15 +02:00
R.id.play_secondary -> when (
PowerPreference.getDefaultFile()
.getString("play_order")
) {
"in_order" -> CommandBus.send(Command.ReplaceQueue(adapter.data.shuffled()))
else -> CommandBus.send(Command.ReplaceQueue(adapter.data))
}
2020-06-13 16:45:58 +02:00
R.id.add_to_queue -> {
when (PowerPreference.getDefaultFile().getString("play_order")) {
"in_order" -> CommandBus.send(Command.AddToQueue(adapter.data))
else -> CommandBus.send(Command.AddToQueue(adapter.data.shuffled()))
}
2020-06-13 16:45:58 +02:00
context.toast("All tracks were added to your queue")
}
R.id.download -> CommandBus.send(Command.PinTracks(adapter.data))
}
true
}
show()
}
}
2019-08-19 16:50:33 +02:00
}
}
private fun watchEventBus() {
lifecycleScope.launch(IO) {
2019-10-31 16:17:37 +01:00
EventBus.get().collect { message ->
2022-08-26 14:06:41 +02:00
if (message is Event.DownloadChanged) {
refreshDownloadedTrack(message.download)
}
}
}
lifecycleScope.launch(Main) {
CommandBus.get().collect { command ->
2022-08-26 14:06:41 +02:00
if (command is Command.RefreshTrack) {
refreshCurrentTrack(command.track)
}
}
}
}
private suspend fun refreshDownloadedTracks() {
val downloaded = TracksRepository.getDownloadedIds(exoDownloadManager) ?: listOf()
withContext(Main) {
2022-12-09 09:49:41 +01:00
adapter.setUnfilteredData(
adapter.data.map {
it.downloaded = downloaded.contains(it.id)
it
}.toMutableList()
)
adapter.notifyDataSetChanged()
}
}
private suspend fun refreshDownloadedTrack(download: Download) {
if (download.state == Download.STATE_COMPLETED) {
download.getMetadata()?.let { info ->
adapter.data.withIndex().associate { it.value to it.index }.filter { it.key.id == info.id }
.toList().getOrNull(0)?.let { match ->
withContext(Main) {
adapter.data[match.second].downloaded = true
adapter.notifyItemChanged(match.second)
}
}
2019-08-19 16:50:33 +02:00
}
}
}
private fun refreshCurrentTrack(track: Track?) {
track?.let {
adapter.currentTrack?.current = false
adapter.currentTrack = track.apply {
current = true
}
adapter.notifyDataSetChanged()
}
}
2021-06-26 13:36:32 +02:00
}