/* * 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.flat import android.animation.ObjectAnimator import android.graphics.PorterDuff import android.os.Bundle import android.view.View 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.FragmentFlatPlayerPlaybackControlsBinding import code.name.monkey.retromusic.extensions.* import code.name.monkey.retromusic.fragments.base.AbsPlayerControlsFragment import code.name.monkey.retromusic.fragments.base.goToAlbum import code.name.monkey.retromusic.fragments.base.goToArtist import code.name.monkey.retromusic.helper.MusicPlayerRemote import code.name.monkey.retromusic.helper.MusicProgressViewUpdateHelper import code.name.monkey.retromusic.helper.MusicProgressViewUpdateHelper.Callback 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 class FlatPlaybackControlsFragment : AbsPlayerControlsFragment(R.layout.fragment_flat_player_playback_controls), Callback { private var lastPlaybackControlsColor: Int = 0 private var lastDisabledPlaybackControlsColor: Int = 0 private lateinit var progressViewUpdateHelper: MusicProgressViewUpdateHelper private var _binding: FragmentFlatPlayerPlaybackControlsBinding? = null private val binding get() = _binding!! override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) progressViewUpdateHelper = MusicProgressViewUpdateHelper(this) } override fun onViewCreated(view: View, savedInstanceState: Bundle?) { super.onViewCreated(view, savedInstanceState) _binding = FragmentFlatPlayerPlaybackControlsBinding.bind(view) setUpMusicControllers() binding.title.isSelected = true binding.text.isSelected = true binding.title.setOnClickListener { goToAlbum(requireActivity()) } binding.text.setOnClickListener { goToArtist(requireActivity()) } } override fun onResume() { super.onResume() progressViewUpdateHelper.start() } override fun onPause() { super.onPause() progressViewUpdateHelper.stop() } public override fun show() { binding.playPauseButton.animate() .scaleX(1f) .scaleY(1f) .setInterpolator(DecelerateInterpolator()) .start() } public override fun hide() { binding.playPauseButton.apply { scaleX = 0f scaleY = 0f rotation = 0f } } override fun setColor(color: MediaNotificationProcessor) { if (ATHUtil.isWindowBackgroundDark(requireContext())) { lastPlaybackControlsColor = MaterialValueHelper.getSecondaryTextColor(requireContext(), false) lastDisabledPlaybackControlsColor = MaterialValueHelper.getSecondaryDisabledTextColor(requireContext(), false) } else { lastPlaybackControlsColor = MaterialValueHelper.getPrimaryTextColor(requireContext(), true) lastDisabledPlaybackControlsColor = MaterialValueHelper.getPrimaryDisabledTextColor(requireContext(), true) } val colorFinal = if (PreferenceUtil.isAdaptiveColor) { color.primaryTextColor } else { ThemeStore.accentColor(requireContext()).ripAlpha() } updateTextColors(colorFinal) volumeFragment?.setTintable(colorFinal) binding.progressSlider.applyColor(colorFinal) updateRepeatState() updateShuffleState() } private fun updateTextColors(color: Int) { val isDark = ColorUtil.isColorLight(color) val darkColor = ColorUtil.darkenColor(color) val colorPrimary = MaterialValueHelper.getPrimaryTextColor(context, isDark) val colorSecondary = MaterialValueHelper.getSecondaryTextColor(context, ColorUtil.isColorLight(darkColor)) TintHelper.setTintAuto(binding.playPauseButton, colorPrimary, false) TintHelper.setTintAuto(binding.playPauseButton, color, true) binding.title.setBackgroundColor(color) binding.title.setTextColor(colorPrimary) binding.text.setBackgroundColor(darkColor) binding.text.setTextColor(colorSecondary) binding.songInfo.setBackgroundColor(darkColor) binding.songInfo.setTextColor(colorSecondary) } override fun onServiceConnected() { updatePlayPauseDrawableState() updateRepeatState() updateShuffleState() updateSong() } override fun onPlayingMetaChanged() { super.onPlayingMetaChanged() updateSong() } override fun onPlayStateChanged() { updatePlayPauseDrawableState() } private fun setUpPlayPauseFab() { binding.playPauseButton.setOnClickListener(PlayPauseButtonOnClickHandler()) } private fun updatePlayPauseDrawableState() { if (MusicPlayerRemote.isPlaying) { binding.playPauseButton.setImageResource(R.drawable.ic_pause) } else { binding.playPauseButton.setImageResource(R.drawable.ic_play_arrow_white_32dp) } } private fun setUpMusicControllers() { setUpPlayPauseFab() setUpRepeatButton() setUpShuffleButton() setUpProgressSlider() } private fun updateSong() { val song = MusicPlayerRemote.currentSong binding.title.text = song.title binding.text.text = song.artistName if (PreferenceUtil.isSongInfo) { binding.songInfo.text = getSongInfo(song) binding.songInfo.show() } else { binding.songInfo.hide() } } override fun onRepeatModeChanged() { updateRepeatState() } override fun onShuffleModeChanged() { updateShuffleState() } 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 ) } } } 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 ) } } 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 } }