我想在我的应用导航逻辑中实现,就像在Youtube应用中一样。 (BottomNavigationView +片段管理)。我想要这个,因为这些片段很重,所以我希望它们被懒惰初始化然后存储在backstack中,我觉得YouTube就是这样做的。我已经实现了BottomNagivationView,但我在片段管理方面遇到了问题。
我的代码:
bottomNavigationView.setOnTabSelectedListener { position, _ ->
setFragment(OnlinePageFragment.Page.values()[position])
}
其中Pages是枚举
enum class Page(index: Int, val klass: Class<*>) {
ONE(0, OnePageFragment::class.java),
TWO(1, TwoPageFragment::class.java),
THREE(2, ThreePageFragment::class.java)
}
这是我的setFragment函数
fun setFragment(page: OnlinePageFragment.Page) {
var fragment: Fragment? = supportFragmentManager.findFragmentByTag(page.klass.name)
val tag = page.klass.name
if (fragment == null)
fragment = OnlinePageFragment.newInstance(page, null)
val ft = supportFragmentManager.beginTransaction()
with(ft) {
replace(R.id.fragmentContainer, fragment, tag)
addToBackStack(tag)
commit()
}
}
override fun onBackPressed() {
if (supportFragmentManager.backStackEntryCount == 1) finish()
else super.onBackPressed()
}
它有效,但不如YouTube应用程序好。 YouTube应用程序有一些神奇的行为,即每个片段只保留一个事务,而我的应用程序允许创建“无限”的事务堆栈。您对YouTube应用中的工作原理有什么想法吗?
答案 0 :(得分:1)
无需使用视图寻呼机即可进行管理。 我已经实施了,请检查一下。 https://github.com/sandeshsk/BackStackFragmentRedirectsToHome
如果有任何问题,请更新。
这是分配片段的方法
public void addFragment(FragmentManager fragmentManager,
Fragment fragment,
int containerId,boolean isFromHome){
fragmentManager.popBackStack(null,FragmentManager.POP_BACK_STACK_INCLUSIVE);
FragmentTransaction fragmentTransaction=fragmentManager.beginTransaction();
if(isFromHome){
fragmentTransaction.replace(containerId,fragment);
}else{
fragmentTransaction.add(new HomeFragment(),"Home");
fragmentTransaction.addToBackStack("Home");
}
fragmentTransaction.replace(containerId,fragment).commit();
}
这是您的导航项目侦听器
private BottomNavigationView.OnNavigationItemSelectedListener mOnNavigationItemSelectedListener
= new BottomNavigationView.OnNavigationItemSelectedListener() {
@Override
public boolean onNavigationItemSelected(@NonNull MenuItem item) {
switch (item.getItemId()) {
case R.id.navigation_home:
if (getSupportFragmentManager().getBackStackEntryCount() == 0) {
addFragment(getSupportFragmentManager(), new HomeFragment(), R.id.frame, true);
}else{
getSupportFragmentManager().popBackStack();
}
return true;
case R.id.navigation_dashboard:
addFragment(getSupportFragmentManager(),new DashboardFragment(),R.id.frame,false);
return true;
case R.id.navigation_notifications:
addFragment(getSupportFragmentManager(),new NotificationFragment(),R.id.frame,false);
return true;
case R.id.navigation_setting:
addFragment(getSupportFragmentManager(),new SettingFragment(),R.id.frame,false);
return true;
}
return false;
}
};
onBackPressed方法
@Override
public void onBackPressed() {
if(getSupportFragmentManager().getBackStackEntryCount()>0){
navigation.setSelectedItemId(R.id.navigation_home);
}else {
super.onBackPressed();
}
}
答案 1 :(得分:0)
我想这里最好的方法是使用ViewPager
存储根片段。导航栏将更改ViewPager
的页面。
根片段将内容片段存储在其ChildFragmentManagers
中,并实现内容片段之间的导航逻辑。
此外,您还需要实现从活动到根片段的委托背压事件的逻辑,以在其后台堆栈中执行导航。
UPD#1
确保ViewPager中的根片段未被销毁,请使用setOffscreenPageLimit()
UPD#2 如果您不想使用ViewPager,则可以使用此answer中的代码来管理根片段。
答案 2 :(得分:0)
由于san's answer按预期显示了片段,但是每次更改片段时,都有一些bug两次在HomeFragment上调用onCreateView。另外,每次更改页面时,更改页面都会实例化新片段。
因此,我拿起code并修改了setPage
,它的工作效果很好。
方法如下:
private void setPage(Page page) {
FragmentManager fragmentManager = getSupportFragmentManager();
String tag = page.name();
FragmentTransaction transaction = fragmentManager.beginTransaction();
// detach everything
for (Fragment fragment : fragmentManager.getFragments()) {
transaction.detach(fragment);
}
// Retrieve fragment instance, if it was already created
Fragment fragment = fragmentManager.findFragmentByTag(tag);
if (fragment == null) { // If not, crate new instance and add it
try {
fragment = (Fragment) page.clazz.newInstance();
transaction.add(R.id.frame, fragment, tag);
} catch (InstantiationException | IllegalAccessException e) {
e.printStackTrace();
}
} else { // otherwise just attach it
transaction.attach(fragment);
}
transaction.commit();
}
onBackPressed实现
@Override
public void onBackPressed() {
if (!getSupportFragmentManager().findFragmentByTag(Page.HOME.name()).isDetached()) {
super.onBackPressed();
} else {
navigation.setSelectedItemId(R.id.navigation_home);
}
}