308 lines
11 KiB
Kotlin
308 lines
11 KiB
Kotlin
/*
|
|
* Copyright (c) 2020 Hemanth Savarla.
|
|
*
|
|
* Licensed under the GNU General Public License v3
|
|
*
|
|
* This is free software: you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or (at your option) any later version.
|
|
*
|
|
* This software is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
|
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
* See the GNU General Public License for more details.
|
|
*
|
|
*/
|
|
package code.name.monkey.retromusic.fragments.player.plain
|
|
|
|
import android.animation.ObjectAnimator
|
|
import android.graphics.PorterDuff
|
|
import android.os.Bundle
|
|
import android.view.View
|
|
import android.view.animation.AccelerateInterpolator
|
|
import android.view.animation.DecelerateInterpolator
|
|
import android.view.animation.LinearInterpolator
|
|
import android.widget.SeekBar
|
|
import code.name.monkey.appthemehelper.ThemeStore
|
|
import code.name.monkey.appthemehelper.util.ATHUtil
|
|
import code.name.monkey.appthemehelper.util.ColorUtil
|
|
import code.name.monkey.appthemehelper.util.MaterialValueHelper
|
|
import code.name.monkey.appthemehelper.util.TintHelper
|
|
import code.name.monkey.retromusic.R
|
|
import code.name.monkey.retromusic.databinding.FragmentPlainControlsFragmentBinding
|
|
import code.name.monkey.retromusic.extensions.applyColor
|
|
import code.name.monkey.retromusic.extensions.getSongInfo
|
|
import code.name.monkey.retromusic.extensions.hide
|
|
import code.name.monkey.retromusic.extensions.show
|
|
import code.name.monkey.retromusic.fragments.base.AbsPlayerControlsFragment
|
|
import code.name.monkey.retromusic.helper.MusicPlayerRemote
|
|
import code.name.monkey.retromusic.helper.MusicProgressViewUpdateHelper
|
|
import code.name.monkey.retromusic.helper.PlayPauseButtonOnClickHandler
|
|
import code.name.monkey.retromusic.misc.SimpleOnSeekbarChangeListener
|
|
import code.name.monkey.retromusic.service.MusicService
|
|
import code.name.monkey.retromusic.util.MusicUtil
|
|
import code.name.monkey.retromusic.util.PreferenceUtil
|
|
import code.name.monkey.retromusic.util.color.MediaNotificationProcessor
|
|
|
|
/**
|
|
* @author Hemanth S (h4h13).
|
|
*/
|
|
|
|
class PlainPlaybackControlsFragment :
|
|
AbsPlayerControlsFragment(R.layout.fragment_plain_controls_fragment) {
|
|
|
|
private var lastPlaybackControlsColor: Int = 0
|
|
private var lastDisabledPlaybackControlsColor: Int = 0
|
|
private lateinit var progressViewUpdateHelper: MusicProgressViewUpdateHelper
|
|
private var _binding: FragmentPlainControlsFragmentBinding? = null
|
|
private val binding get() = _binding!!
|
|
|
|
override fun onPlayStateChanged() {
|
|
updatePlayPauseDrawableState()
|
|
}
|
|
|
|
override fun onRepeatModeChanged() {
|
|
updateRepeatState()
|
|
}
|
|
|
|
override fun onShuffleModeChanged() {
|
|
updateShuffleState()
|
|
}
|
|
|
|
override fun onServiceConnected() {
|
|
updatePlayPauseDrawableState()
|
|
updateRepeatState()
|
|
updateShuffleState()
|
|
updateSong()
|
|
}
|
|
|
|
override fun onPlayingMetaChanged() {
|
|
super.onPlayingMetaChanged()
|
|
updateSong()
|
|
}
|
|
|
|
private fun updateSong() {
|
|
if (PreferenceUtil.isSongInfo) {
|
|
binding.songInfo.text = getSongInfo(MusicPlayerRemote.currentSong)
|
|
binding.songInfo.show()
|
|
} else {
|
|
binding.songInfo.hide()
|
|
}
|
|
}
|
|
|
|
override fun onCreate(savedInstanceState: Bundle?) {
|
|
super.onCreate(savedInstanceState)
|
|
progressViewUpdateHelper = MusicProgressViewUpdateHelper(this)
|
|
}
|
|
|
|
override fun onResume() {
|
|
super.onResume()
|
|
progressViewUpdateHelper.start()
|
|
}
|
|
|
|
override fun onPause() {
|
|
super.onPause()
|
|
progressViewUpdateHelper.stop()
|
|
}
|
|
|
|
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
|
|
super.onViewCreated(view, savedInstanceState)
|
|
_binding = FragmentPlainControlsFragmentBinding.bind(view)
|
|
setUpMusicControllers()
|
|
|
|
binding.playPauseButton.setOnClickListener {
|
|
if (MusicPlayerRemote.isPlaying) {
|
|
MusicPlayerRemote.pauseSong()
|
|
} else {
|
|
MusicPlayerRemote.resumePlaying()
|
|
}
|
|
showBounceAnimation()
|
|
}
|
|
}
|
|
|
|
private fun setUpPlayPauseFab() {
|
|
binding.playPauseButton.setOnClickListener(PlayPauseButtonOnClickHandler())
|
|
}
|
|
|
|
private fun setUpMusicControllers() {
|
|
setUpPlayPauseFab()
|
|
setUpPrevNext()
|
|
setUpRepeatButton()
|
|
setUpShuffleButton()
|
|
setUpProgressSlider()
|
|
}
|
|
|
|
private fun setUpPrevNext() {
|
|
updatePrevNextColor()
|
|
binding.nextButton.setOnClickListener { MusicPlayerRemote.playNextSong() }
|
|
binding.previousButton.setOnClickListener { MusicPlayerRemote.back() }
|
|
}
|
|
|
|
private fun updatePrevNextColor() {
|
|
binding.nextButton.setColorFilter(lastPlaybackControlsColor, PorterDuff.Mode.SRC_IN)
|
|
binding.previousButton.setColorFilter(lastPlaybackControlsColor, PorterDuff.Mode.SRC_IN)
|
|
}
|
|
|
|
override fun setColor(color: MediaNotificationProcessor) {
|
|
val colorBg = ATHUtil.resolveColor(requireContext(), android.R.attr.colorBackground)
|
|
if (ColorUtil.isColorLight(colorBg)) {
|
|
lastPlaybackControlsColor =
|
|
MaterialValueHelper.getSecondaryTextColor(requireContext(), true)
|
|
lastDisabledPlaybackControlsColor =
|
|
MaterialValueHelper.getSecondaryDisabledTextColor(requireContext(), true)
|
|
} else {
|
|
lastPlaybackControlsColor =
|
|
MaterialValueHelper.getPrimaryTextColor(requireContext(), false)
|
|
lastDisabledPlaybackControlsColor =
|
|
MaterialValueHelper.getPrimaryDisabledTextColor(requireContext(), false)
|
|
}
|
|
|
|
val colorFinal = if (PreferenceUtil.isAdaptiveColor) {
|
|
color.primaryTextColor
|
|
} else {
|
|
ThemeStore.accentColor(requireContext())
|
|
}
|
|
volumeFragment?.setTintable(colorFinal)
|
|
binding.progressSlider.applyColor(colorFinal)
|
|
|
|
TintHelper.setTintAuto(
|
|
binding.playPauseButton,
|
|
MaterialValueHelper.getPrimaryTextColor(
|
|
requireContext(),
|
|
ColorUtil.isColorLight(colorFinal)
|
|
),
|
|
false
|
|
)
|
|
TintHelper.setTintAuto(binding.playPauseButton, colorFinal, true)
|
|
|
|
updateRepeatState()
|
|
updateShuffleState()
|
|
updatePrevNextColor()
|
|
}
|
|
|
|
private fun setUpShuffleButton() {
|
|
binding.shuffleButton.setOnClickListener { MusicPlayerRemote.toggleShuffleMode() }
|
|
}
|
|
|
|
override fun updateShuffleState() {
|
|
when (MusicPlayerRemote.shuffleMode) {
|
|
MusicService.SHUFFLE_MODE_SHUFFLE -> binding.shuffleButton.setColorFilter(
|
|
lastPlaybackControlsColor,
|
|
PorterDuff.Mode.SRC_IN
|
|
)
|
|
else -> binding.shuffleButton.setColorFilter(
|
|
lastDisabledPlaybackControlsColor,
|
|
PorterDuff.Mode.SRC_IN
|
|
)
|
|
}
|
|
}
|
|
|
|
private fun setUpRepeatButton() {
|
|
binding.repeatButton.setOnClickListener { MusicPlayerRemote.cycleRepeatMode() }
|
|
}
|
|
|
|
override fun updateRepeatState() {
|
|
when (MusicPlayerRemote.repeatMode) {
|
|
MusicService.REPEAT_MODE_NONE -> {
|
|
binding.repeatButton.setImageResource(R.drawable.ic_repeat)
|
|
binding.repeatButton.setColorFilter(
|
|
lastDisabledPlaybackControlsColor,
|
|
PorterDuff.Mode.SRC_IN
|
|
)
|
|
}
|
|
MusicService.REPEAT_MODE_ALL -> {
|
|
binding.repeatButton.setImageResource(R.drawable.ic_repeat)
|
|
binding.repeatButton.setColorFilter(
|
|
lastPlaybackControlsColor,
|
|
PorterDuff.Mode.SRC_IN
|
|
)
|
|
}
|
|
MusicService.REPEAT_MODE_THIS -> {
|
|
binding.repeatButton.setImageResource(R.drawable.ic_repeat_one)
|
|
binding.repeatButton.setColorFilter(
|
|
lastPlaybackControlsColor,
|
|
PorterDuff.Mode.SRC_IN
|
|
)
|
|
}
|
|
}
|
|
}
|
|
|
|
public override fun show() {
|
|
binding.playPauseButton.animate()
|
|
.scaleX(1f)
|
|
.scaleY(1f)
|
|
.rotation(360f)
|
|
.setInterpolator(DecelerateInterpolator())
|
|
.start()
|
|
}
|
|
|
|
public override fun hide() {
|
|
binding.playPauseButton.apply {
|
|
scaleX = 0f
|
|
scaleY = 0f
|
|
rotation = 0f
|
|
}
|
|
}
|
|
|
|
private fun showBounceAnimation() {
|
|
binding.playPauseButton.apply {
|
|
clearAnimation()
|
|
scaleX = 0.9f
|
|
scaleY = 0.9f
|
|
visibility = View.VISIBLE
|
|
pivotX = (width / 2).toFloat()
|
|
pivotY = (height / 2).toFloat()
|
|
|
|
animate().setDuration(200)
|
|
.setInterpolator(DecelerateInterpolator())
|
|
.scaleX(1.1f)
|
|
.scaleY(1.1f)
|
|
.withEndAction {
|
|
animate().setDuration(200)
|
|
.setInterpolator(AccelerateInterpolator())
|
|
.scaleX(1f)
|
|
.scaleY(1f)
|
|
.alpha(1f).start()
|
|
}.start()
|
|
}
|
|
}
|
|
|
|
private fun updatePlayPauseDrawableState() {
|
|
if (MusicPlayerRemote.isPlaying) {
|
|
binding.playPauseButton.setImageResource(R.drawable.ic_pause)
|
|
} else {
|
|
binding.playPauseButton.setImageResource(R.drawable.ic_play_arrow_white_32dp)
|
|
}
|
|
}
|
|
|
|
override fun setUpProgressSlider() {
|
|
binding.progressSlider.setOnSeekBarChangeListener(object : SimpleOnSeekbarChangeListener() {
|
|
override fun onProgressChanged(seekBar: SeekBar, progress: Int, fromUser: Boolean) {
|
|
if (fromUser) {
|
|
MusicPlayerRemote.seekTo(progress)
|
|
onUpdateProgressViews(
|
|
MusicPlayerRemote.songProgressMillis,
|
|
MusicPlayerRemote.songDurationMillis
|
|
)
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
override fun onUpdateProgressViews(progress: Int, total: Int) {
|
|
binding.progressSlider.max = total
|
|
|
|
val animator = ObjectAnimator.ofInt(binding.progressSlider, "progress", progress)
|
|
animator.duration = SLIDER_ANIMATION_TIME
|
|
animator.interpolator = LinearInterpolator()
|
|
animator.start()
|
|
|
|
binding.songTotalTime.text = MusicUtil.getReadableDurationString(total.toLong())
|
|
binding.songCurrentProgress.text = MusicUtil.getReadableDurationString(progress.toLong())
|
|
}
|
|
|
|
override fun onDestroyView() {
|
|
super.onDestroyView()
|
|
_binding = null
|
|
}
|
|
}
|