
Use viewLifecycleOwner for all fragments that rely on databinding. This should solve some possible memory leaks, as viewLifecycleOwner is tied to the lifecycle of the fragment views instead of the fragments them selves.
157 lines
5.6 KiB
Kotlin
157 lines
5.6 KiB
Kotlin
package org.oxycblt.auxio
|
|
|
|
import android.content.res.ColorStateList
|
|
import android.os.Bundle
|
|
import android.view.LayoutInflater
|
|
import android.view.MenuItem
|
|
import android.view.View
|
|
import android.view.ViewGroup
|
|
import androidx.core.graphics.ColorUtils
|
|
import androidx.fragment.app.Fragment
|
|
import androidx.fragment.app.activityViewModels
|
|
import androidx.navigation.NavController
|
|
import androidx.navigation.NavOptions
|
|
import androidx.navigation.fragment.NavHostFragment
|
|
import androidx.navigation.fragment.findNavController
|
|
import com.google.android.material.bottomnavigation.BottomNavigationView
|
|
import org.oxycblt.auxio.databinding.FragmentMainBinding
|
|
import org.oxycblt.auxio.detail.DetailViewModel
|
|
import org.oxycblt.auxio.music.Song
|
|
import org.oxycblt.auxio.playback.PlaybackViewModel
|
|
import org.oxycblt.auxio.ui.Accent
|
|
import org.oxycblt.auxio.ui.isLandscape
|
|
import org.oxycblt.auxio.ui.isTablet
|
|
import org.oxycblt.auxio.ui.resolveAttr
|
|
import org.oxycblt.auxio.ui.toColor
|
|
|
|
/**
|
|
* The primary "Home" [Fragment] for Auxio.
|
|
*/
|
|
class MainFragment : Fragment() {
|
|
private val playbackModel: PlaybackViewModel by activityViewModels()
|
|
private val detailModel: DetailViewModel by activityViewModels()
|
|
|
|
override fun onCreateView(
|
|
inflater: LayoutInflater,
|
|
container: ViewGroup?,
|
|
savedInstanceState: Bundle?
|
|
): View {
|
|
val binding = FragmentMainBinding.inflate(inflater)
|
|
|
|
val colorActive = Accent.get().color.toColor(requireContext())
|
|
val colorInactive = ColorUtils.setAlphaComponent(colorActive, 150)
|
|
|
|
// Set up the tints for the navigation icons + text
|
|
val navTints = ColorStateList(
|
|
arrayOf(
|
|
intArrayOf(-android.R.attr.state_checked),
|
|
intArrayOf(android.R.attr.state_checked)
|
|
),
|
|
intArrayOf(colorInactive, colorActive)
|
|
)
|
|
|
|
val navController = (
|
|
childFragmentManager.findFragmentById(R.id.explore_nav_host)
|
|
as NavHostFragment?
|
|
)?.findNavController()
|
|
|
|
// --- UI SETUP ---
|
|
|
|
binding.lifecycleOwner = viewLifecycleOwner
|
|
|
|
// Speed up the slide-in effect on the controls view, solely to improve the UX
|
|
// and maybe hide the problem where the main view will snap-shrink before the compact
|
|
// view can slide.
|
|
(binding.controlsContainer as ViewGroup).layoutTransition.setDuration(150)
|
|
|
|
binding.navBar.apply {
|
|
itemIconTintList = navTints
|
|
itemTextColor = navTints
|
|
|
|
if (isTablet(resources) && !isLandscape(resources)) {
|
|
labelVisibilityMode = BottomNavigationView.LABEL_VISIBILITY_LABELED
|
|
}
|
|
|
|
navController?.let { controller ->
|
|
binding.navBar.setOnItemSelectedListener { item ->
|
|
navigateWithItem(controller, item)
|
|
}
|
|
}
|
|
|
|
// BottomNavigationView is a special little snowflake and doesn't like it when
|
|
// we set the background in XML
|
|
setBackgroundColor(R.attr.colorSurface.resolveAttr(requireContext()))
|
|
}
|
|
|
|
// --- VIEWMODEL SETUP ---
|
|
|
|
// Change CompactPlaybackFragment's visibility here so that an animation occurs.
|
|
handleCompactPlaybackVisibility(binding, playbackModel.song.value)
|
|
|
|
playbackModel.song.observe(viewLifecycleOwner) { song ->
|
|
handleCompactPlaybackVisibility(binding, song)
|
|
}
|
|
|
|
detailModel.navToItem.observe(viewLifecycleOwner) { item ->
|
|
if (item != null && navController != null) {
|
|
val curDest = navController.currentDestination?.id
|
|
|
|
// SongsFragment and SettingsFragment have no navigation pathways, so correct
|
|
// them to the library tab instead.
|
|
if (curDest == R.id.songs_fragment || curDest == R.id.settings_fragment) {
|
|
binding.navBar.selectedItemId = R.id.library_fragment
|
|
}
|
|
}
|
|
}
|
|
|
|
playbackModel.setupPlayback(requireContext())
|
|
|
|
logD("Fragment Created.")
|
|
|
|
return binding.root
|
|
}
|
|
|
|
/**
|
|
* Custom navigator code that has proper animations, unlike BottomNavigationView.setupWithNavController().
|
|
*/
|
|
private fun navigateWithItem(navController: NavController, item: MenuItem): Boolean {
|
|
if (navController.currentDestination!!.id != item.itemId) {
|
|
// Create custom NavOptions myself so that animations work
|
|
val options = NavOptions.Builder()
|
|
.setLaunchSingleTop(true)
|
|
.setEnterAnim(R.animator.nav_default_enter_anim)
|
|
.setExitAnim(R.animator.nav_default_exit_anim)
|
|
.setPopEnterAnim(R.animator.nav_default_pop_enter_anim)
|
|
.setPopExitAnim(R.animator.nav_default_pop_exit_anim)
|
|
.build()
|
|
|
|
return try {
|
|
navController.navigate(item.itemId, null, options)
|
|
true
|
|
} catch (e: IllegalArgumentException) {
|
|
false
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
/**
|
|
* Handle the visibility of CompactPlaybackFragment. Done here so that there's a nice animation.
|
|
*/
|
|
private fun handleCompactPlaybackVisibility(binding: FragmentMainBinding, song: Song?) {
|
|
if (song == null) {
|
|
logD("Hiding CompactPlaybackFragment since no song is being played.")
|
|
|
|
binding.compactPlayback.visibility = if (isLandscape(resources)) {
|
|
View.INVISIBLE
|
|
} else {
|
|
View.GONE
|
|
}
|
|
|
|
playbackModel.disableAnimation()
|
|
} else {
|
|
binding.compactPlayback.visibility = View.VISIBLE
|
|
}
|
|
}
|
|
}
|