使用 Fragments / Toolbar 返回导航
Back navigation with Fragments / Toolbar
我现在正在为这个问题挠头....我有一个 ActionBarActivity
加载初始 Fragment
- 原始菜单在 activity 中膨胀。现在,我有一个导航栏,当一个项目被选中时,它会加载一个不同的片段并将其添加到后台堆栈。
执行此操作时,我需要设置几项内容:
- 将主页设置为向上指示器
- 使主菜单中的选项无效 activity
- Set 的 Fragment 选项为 true
- 确保向上指示器正确导航回原始片段
发生了一些相当奇怪的事情 - 向上指示器只出现一次并且不像后退按钮那样工作,虽然我已经使新菜单无效并膨胀,但新菜单得到 附加 到原来的 Activity 菜单。
编辑: 好的,我已经解决了附加问题 - 忘记在 onCreateOptionsMenu
方法中添加 menu.clear()
。
我的导航抽屉布局对所有菜单项都有 onClick
方法,这些方法会触发另一个 Fragment 的加载:
public void navItemClick(View view) {
switch (view.getId()) {
case R.id.ripSMS:
mNavigationDrawer.toggleHome(false);
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
FragmentTransaction mTrans = getSupportFragmentManager().beginTransaction();
mTrans.replace(R.id.voiceover_frame_layout,new MessageFragment(),"main_ui")
.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_FADE).addToBackStack("msg").commit();
break;
case R.id.ripEmail:
break;
case R.id.ripSettings:
break;
}
mNavigationDrawer.closeDrawer();
}
切换首页:
public void toggleHome(boolean show) {
mDrawerToggle.setDrawerIndicatorEnabled(show);
if (!show) {
mDrawerLayout.setDrawerLockMode(DrawerLayout.LOCK_MODE_LOCKED_CLOSED);
} else {
mDrawerLayout.setDrawerLockMode(DrawerLayout.LOCK_MODE_UNLOCKED);
}
}
一旦项目被触发,onCreate
包含无效和 hasOptions 代码:
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
getActivity().invalidateOptionsMenu();
setHasOptionsMenu(true);
}
然后 onCreateOptionsMenu
展开另一个菜单布局(包含一个名为设置的项目)。
如前所述,这只能部分工作一次——我第一次使用该项目加载片段时,我得到了后退图标,但它也不起作用(这是在 onOptionsItemSelected
中设置的,以检查home item press - 它什么都不做)。当我按下后退按钮时,它会将我带回正确的位置。但是,如果我向后导航,即使代码贯穿 onCreate
!
,后退箭头现在也会显示得更长
好的,经过反复试验,我设法解决了这个问题。进行了两项更改:
- 实施
addOnBackStackChangedListener
ActionBarDrawerToggle's
setToolbarNavigationClickListener
需要设置
因为我只有一个 activity(其他都是 Fragment classes)我将后台监听器添加到父 Activity 的 onCreate
方法中:
getSupportFragmentManager().addOnBackStackChangedListener(new FragmentManager.OnBackStackChangedListener() {
@Override
public void onBackStackChanged() {
if (getSupportFragmentManager().getBackStackEntryCount() > 0) {
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
} else {
getSupportActionBar().setDisplayHomeAsUpEnabled(false);
}
}
});
这解决了返回片段时消失的后退箭头。最后将侦听器添加到我的 NavigationDrawer 设置 class:
mDrawerToggle.setToolbarNavigationClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
getActivity().onBackPressed();
}
});
我想我唯一的问题是关于将 onOptionsItemSelected
方法与 android.R.id.home
一起使用的所有问题,但这对我来说从来没有用过。这当然可能是我实现事情的方式,但如果有人看到任何明显的原因请告诉我!
这些步骤可帮助您在加载片段时在工具栏中显示后退按钮。并在点击时弹出。
将 setNavigationOnClickListener
设置为您 activity 中的工具栏。
final DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
toggle = new ActionBarDrawerToggle(
this, drawer, toolbar, R.string.navigation_drawer_open, R.string.navigation_drawer_close);
drawer.setDrawerListener(toggle);
toggle.syncState();
toolbar.setNavigationOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if(getSupportFragmentManager().getBackStackEntryCount() > 0){
getSupportFragmentManager().popBackStack();
}else {
drawer.openDrawer(GravityCompat.START);
}
}
});
在你Activity中实施FragmentManager.OnBackStackChangedListener
。并在 OnCreate()
中用 SupportFragmentManager
注册
getSupportFragmentManager().addOnBackStackChangedListener(this);
OnBackStackChangedListener
实现方法:
@Override
public void onBackStackChanged() {
if(getSupportFragmentManager().getBackStackEntryCount() > 0){
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
}else {
getSupportActionBar().setDisplayHomeAsUpEnabled(false);
toggle.syncState();
}
}
对我来说,上面的答案还不够,但我使用了@Hamz4h_,并在深入了解 ActionBarDrawerToggle class 后添加了更多答案。我只是从 activity 的 onCreate:
调用我的这个方法
private void initNavigationElements() {
final ActionBarDrawerToggle toggle = new ActionBarDrawerToggle(
this, mBinding.drawerLayout, mBinding.appBarMain.toolbar, R.string.navigation_drawer_open, R.string.navigation_drawer_close);
mBinding.drawerLayout.addDrawerListener(toggle);
// Tricking the toggle by giving it its own arrow as a custom indicator.
// It will use it when setDrawerIndicatorEnabled is called with false
toggle.setHomeAsUpIndicator(toggle.getDrawerArrowDrawable());
toggle.syncState();
getSupportFragmentManager().addOnBackStackChangedListener(new FragmentManager.OnBackStackChangedListener() {
@Override
public void onBackStackChanged() {
DrawerArrowDrawable drawerArrowDrawable = toggle.getDrawerArrowDrawable();
if (getSupportFragmentManager().getBackStackEntryCount() > 0) {
// 1 - Display as arrow
drawerArrowDrawable.setProgress(1);
toggle.setDrawerIndicatorEnabled(false);
} else {
// 2 - Display as arrow menu
drawerArrowDrawable.setProgress(0);
toggle.setDrawerIndicatorEnabled(true);
}
}
});
toggle.setToolbarNavigationClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// This is called only when setDrawerIndicatorEnabled is set as false, meaning we are not at the "root" fragment.
getSupportFragmentManager().popBackStackImmediate();
}
});
}
希望这会对某人有所帮助:)
我现在正在为这个问题挠头....我有一个 ActionBarActivity
加载初始 Fragment
- 原始菜单在 activity 中膨胀。现在,我有一个导航栏,当一个项目被选中时,它会加载一个不同的片段并将其添加到后台堆栈。
执行此操作时,我需要设置几项内容:
- 将主页设置为向上指示器
- 使主菜单中的选项无效 activity
- Set 的 Fragment 选项为 true
- 确保向上指示器正确导航回原始片段
发生了一些相当奇怪的事情 - 向上指示器只出现一次并且不像后退按钮那样工作,虽然我已经使新菜单无效并膨胀,但新菜单得到 附加 到原来的 Activity 菜单。
编辑: 好的,我已经解决了附加问题 - 忘记在 onCreateOptionsMenu
方法中添加 menu.clear()
。
我的导航抽屉布局对所有菜单项都有 onClick
方法,这些方法会触发另一个 Fragment 的加载:
public void navItemClick(View view) {
switch (view.getId()) {
case R.id.ripSMS:
mNavigationDrawer.toggleHome(false);
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
FragmentTransaction mTrans = getSupportFragmentManager().beginTransaction();
mTrans.replace(R.id.voiceover_frame_layout,new MessageFragment(),"main_ui")
.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_FADE).addToBackStack("msg").commit();
break;
case R.id.ripEmail:
break;
case R.id.ripSettings:
break;
}
mNavigationDrawer.closeDrawer();
}
切换首页:
public void toggleHome(boolean show) {
mDrawerToggle.setDrawerIndicatorEnabled(show);
if (!show) {
mDrawerLayout.setDrawerLockMode(DrawerLayout.LOCK_MODE_LOCKED_CLOSED);
} else {
mDrawerLayout.setDrawerLockMode(DrawerLayout.LOCK_MODE_UNLOCKED);
}
}
一旦项目被触发,onCreate
包含无效和 hasOptions 代码:
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
getActivity().invalidateOptionsMenu();
setHasOptionsMenu(true);
}
然后 onCreateOptionsMenu
展开另一个菜单布局(包含一个名为设置的项目)。
如前所述,这只能部分工作一次——我第一次使用该项目加载片段时,我得到了后退图标,但它也不起作用(这是在 onOptionsItemSelected
中设置的,以检查home item press - 它什么都不做)。当我按下后退按钮时,它会将我带回正确的位置。但是,如果我向后导航,即使代码贯穿 onCreate
!
好的,经过反复试验,我设法解决了这个问题。进行了两项更改:
- 实施
addOnBackStackChangedListener
ActionBarDrawerToggle's
setToolbarNavigationClickListener
需要设置
因为我只有一个 activity(其他都是 Fragment classes)我将后台监听器添加到父 Activity 的 onCreate
方法中:
getSupportFragmentManager().addOnBackStackChangedListener(new FragmentManager.OnBackStackChangedListener() {
@Override
public void onBackStackChanged() {
if (getSupportFragmentManager().getBackStackEntryCount() > 0) {
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
} else {
getSupportActionBar().setDisplayHomeAsUpEnabled(false);
}
}
});
这解决了返回片段时消失的后退箭头。最后将侦听器添加到我的 NavigationDrawer 设置 class:
mDrawerToggle.setToolbarNavigationClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
getActivity().onBackPressed();
}
});
我想我唯一的问题是关于将 onOptionsItemSelected
方法与 android.R.id.home
一起使用的所有问题,但这对我来说从来没有用过。这当然可能是我实现事情的方式,但如果有人看到任何明显的原因请告诉我!
这些步骤可帮助您在加载片段时在工具栏中显示后退按钮。并在点击时弹出。
将 setNavigationOnClickListener
设置为您 activity 中的工具栏。
final DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
toggle = new ActionBarDrawerToggle(
this, drawer, toolbar, R.string.navigation_drawer_open, R.string.navigation_drawer_close);
drawer.setDrawerListener(toggle);
toggle.syncState();
toolbar.setNavigationOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if(getSupportFragmentManager().getBackStackEntryCount() > 0){
getSupportFragmentManager().popBackStack();
}else {
drawer.openDrawer(GravityCompat.START);
}
}
});
在你Activity中实施FragmentManager.OnBackStackChangedListener
。并在 OnCreate()
SupportFragmentManager
注册
getSupportFragmentManager().addOnBackStackChangedListener(this);
OnBackStackChangedListener
实现方法:
@Override
public void onBackStackChanged() {
if(getSupportFragmentManager().getBackStackEntryCount() > 0){
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
}else {
getSupportActionBar().setDisplayHomeAsUpEnabled(false);
toggle.syncState();
}
}
对我来说,上面的答案还不够,但我使用了@Hamz4h_,并在深入了解 ActionBarDrawerToggle class 后添加了更多答案。我只是从 activity 的 onCreate:
调用我的这个方法private void initNavigationElements() {
final ActionBarDrawerToggle toggle = new ActionBarDrawerToggle(
this, mBinding.drawerLayout, mBinding.appBarMain.toolbar, R.string.navigation_drawer_open, R.string.navigation_drawer_close);
mBinding.drawerLayout.addDrawerListener(toggle);
// Tricking the toggle by giving it its own arrow as a custom indicator.
// It will use it when setDrawerIndicatorEnabled is called with false
toggle.setHomeAsUpIndicator(toggle.getDrawerArrowDrawable());
toggle.syncState();
getSupportFragmentManager().addOnBackStackChangedListener(new FragmentManager.OnBackStackChangedListener() {
@Override
public void onBackStackChanged() {
DrawerArrowDrawable drawerArrowDrawable = toggle.getDrawerArrowDrawable();
if (getSupportFragmentManager().getBackStackEntryCount() > 0) {
// 1 - Display as arrow
drawerArrowDrawable.setProgress(1);
toggle.setDrawerIndicatorEnabled(false);
} else {
// 2 - Display as arrow menu
drawerArrowDrawable.setProgress(0);
toggle.setDrawerIndicatorEnabled(true);
}
}
});
toggle.setToolbarNavigationClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// This is called only when setDrawerIndicatorEnabled is set as false, meaning we are not at the "root" fragment.
getSupportFragmentManager().popBackStackImmediate();
}
});
}
希望这会对某人有所帮助:)