使用片段/工具栏返回导航

时间:2015-03-08 18:53:07

标签: android fragment android-toolbar android-actionbaractivity fragment-backstack

我现在正用这个抓挠我的头......我有ActionBarActivity加载一个初始Fragment - 原始菜单在活动中膨胀。现在,我有一个导航栏,当选择一个项目时,它会加载一个不同的片段并将其添加到backstack。

当我这样做时,我想设置几件事情:

  1. 将主页设置为向上指示符
  2. 使主要活动
  3. 中的选项菜单无效
  4. 将片段
  5. 的选项设置为true
  6. 确保向上指示符正确导航回原始片段
  7. 出现了一些比较奇怪的事情 - 向上指示器仅出现一次并且不像后退按钮那样,虽然我已经使新菜单无效并且充气,新菜单会追加到原来的活动菜单。

    编辑:确定我已解决了附加问题 - 忘记在menu.clear()方法中添加onCreateOptionsMenu

    我的导航抽屉布局对所有菜单项都有onClick个方法,这些方法会触发另一个片段的加载:

    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();
        }
    

    toggleHome:

    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包含invalidate和hasOptions代码:

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        getActivity().invalidateOptionsMenu();
        setHasOptionsMenu(true);
    }
    

    onCreateOptionsMenu然后膨胀另一个菜单布局(包含一个名为settings的项目)。

    如前所述,这只是部分工作一次 - 我第一次使用该项加载片段时,我得到了后面的图标,但它也没有工作(这是在onOptionsItemSelected内设置来检查对于家庭项目按 - 它什么都不做)。当我按下后退按钮时,它会将我带回正确的位置。但是,如果我向后导航,即使代码通过onCreate

    ,现在更长的后箭头也会显示

3 个答案:

答案 0 :(得分:12)

好的,所以我设法在经过一些试验和错误后解决了这个问题。做了两处改动:

  1. 实施addOnBackStackChangedListener
  2. ActionBarDrawerToggle's setToolbarNavigationClickListener需要设置
  3. 由于我只有一个活动(其他一切都是Fragment类),我将backstack监听器添加到Parent Activity的onCreate方法中:

    getSupportFragmentManager().addOnBackStackChangedListener(new FragmentManager.OnBackStackChangedListener() {
            @Override
            public void onBackStackChanged() {
                if (getSupportFragmentManager().getBackStackEntryCount() > 0) {
                    getSupportActionBar().setDisplayHomeAsUpEnabled(true);
                } else {
                    getSupportActionBar().setDisplayHomeAsUpEnabled(false);
                }
            }
        });
    

    当回到片段时,这解决了消失的后退箭头。最后将监听器添加到NavigationDrawer的安装类:

    mDrawerToggle.setToolbarNavigationClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    getActivity().onBackPressed();
                }
            });
    

    我想我唯一的问题就是使用onOptionsItemSelected方法与android.R.id.home一起使用的所有内容,但这对我来说无效。这可能是我当然实施的方式,但如果有人看到任何明显的原因,请让我知道!

答案 1 :(得分:5)

这些步骤可帮助您在加载片段时显示工具栏中的后退按钮。并在点击时弹出。

setNavigationOnClickListener设置为您活动中的工具栏。

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);
          }
        }
    });

在您的活动中实施FragmentManager.OnBackStackChangedListener。并在SupportFragmentManager

中使用OnCreate()进行注册
 getSupportFragmentManager().addOnBackStackChangedListener(this);

OnBackStackChangedListener实施方法:

@Override
public void onBackStackChanged() {
    if(getSupportFragmentManager().getBackStackEntryCount() > 0){
        getSupportActionBar().setDisplayHomeAsUpEnabled(true);
    }else {
        getSupportActionBar().setDisplayHomeAsUpEnabled(false);
        toggle.syncState();
    }
}

答案 2 :(得分:1)

对我来说,上面的答案还不够,但我已经使用了@ Hamz4h_并在深入了解ActionBarDrawerToggle类后添加了更多。我只是从活动的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();
        }
    });
}

希望这会有所帮助:)