编辑 2:虽然仍然没有一流的支持(截至撰写本文时),但 Google 现在已经更新了他们的示例,并提供了一个他们认为现在应该如何解决的示例:https ://github.com/googlesamples/android-architecture-组件/树/主/NavigationAdvancedSample
主要原因是您只使用一个NavHostFragment
来保存应用程序的整个后台堆栈。
解决方案是每个选项卡都应该拥有自己的后台堆栈。
- 在您的主布局中,用
FrameLayout
.
- 每个选项卡片段都是一个
NavHostFragment
并包含自己的导航图,以使每个选项卡片段都有自己的后退堆栈。
- 添加一个
BottomNavigationView.OnNavigationItemSelectedListener
toBottomNavigtionView
来处理每个 FrameLayout 的可见性。
这也照顾了您的“...我不喜欢将所有这些片段保存在内存中...”,因为NavHostFragment
默认情况下使用Navigation fragmentTransaction.replace()
,即您将始终只有与NavHostFragment
s 一样多的片段。其余的只是在导航图的后面堆栈中。
编辑:谷歌正在开发本地实现https://issuetracker.google.com/issues/80029773#comment25
更详细
假设您有BottomNavigationView
2 个菜单选项,Dogs
并且Cats
.
<menu xmlns:android="http://schemas.android.com/apk/res/android">
<item android:id="@+id/dogMenu"
.../>
<item android:id="@+id/catMenu"
.../>
</menu>
然后你需要 2 个导航图,比如dog_navigation_graph.xml
和cat_navigation_graph.xml
。
可能dog_navigation_graph
看起来像
<navigation
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@+id/dog_navigation_graph"
app:startDestination="@id/dogMenu">
</navigation>
和对应的 for cat_navigation_graph
。
在您的activity_main.xml
中,添加 2NavHostFragment
秒
<FrameLayout
android:id="@+id/frame_dog"
...>
<fragment
android:id="@+id/dog_navigation_host_fragment"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:name="androidx.navigation.fragment.NavHostFragment"
app:navGraph="@navigation/dog_navigation_graph"
app:defaultNavHost="true"/>
</FrameLayout>
并在下面为您的 cat 添加相应的内容NavHostFragment
。在你的猫框架布局上,设置android:visibility="invisible"
现在,在你MainActivity
的onCreateView
你可以
bottom_navigation_view.setOnNavigationItemSelectedListener { item ->
when (item.itemId) {
R.id.dogMenu -> showHostView(host = 0)
R.id.catMenu -> showHostView(host = 1)
}
return@setOnNavigationItemSelectedListener true
}
所做showHostView()
的只是切换FrameLayout
包装NavHostFragment
s 的 s 的可见性。所以一定要以某种方式保存它们,例如onCreateView
val hostViews = arrayListOf<FrameLayout>() // Member variable of MainActivity
hostViews.apply {
add(findViewById(R.id.frame_dog))
add(findViewById(R.id.frame_cat))
}
现在很容易切换hostViews
应该可见和不可见的。