我写了这个扩展。它将检查当前片段和目的地,如果两者相同,它只会关闭抽屉。但是关闭抽屉时存在一些动画问题。
fun NavigationView.setupWithUniqueFragment(navController: NavController) {
this.setNavigationItemSelectedListener(object : NavigationView.OnNavigationItemSelectedListener {
override fun onNavigationItemSelected(item: MenuItem): Boolean {
val parent = this@setupWithUniqueFragment.parent
if (item.itemId == navController.currentDestination?.id) {
if (parent is DrawerLayout) {
parent.closeDrawer(this@setupWithUniqueFragment, true)
}
return true
}
val handled = NavigationUI.onNavDestinationSelected(item, navController)
if (handled) {
if (parent is DrawerLayout) {
parent.closeDrawer(this@setupWithUniqueFragment, true)
}
}
return handled
}
})
val weakReference = WeakReference<NavigationView>(this@setupWithUniqueFragment)
navController.addOnDestinationChangedListener(
object : NavController.OnDestinationChangedListener {
override fun onDestinationChanged(
controller: NavController,
destination: NavDestination, arguments: Bundle?
) {
val view = weakReference.get()
if (view == null) {
navController.removeOnDestinationChangedListener(this)
return
}
val menu = view.menu
var h = 0
val size = menu.size()
while (h < size) {
val item = menu.getItem(h)
item.isChecked = matchDestination(destination, item.itemId)
h++
}
}
})
}
internal fun matchDestination(
destination: NavDestination,
@IdRes destId: Int
): Boolean {
var currentDestination: NavDestination? = destination
while (currentDestination!!.id != destId && currentDestination.parent != null) {
currentDestination = currentDestination.parent
}
return currentDestination.id == destId
}