繁体   English   中英

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

[英]Back navigation with Fragments / Toolbar

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

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

  1. 将主页设置为向上指示符
  2. 使主活动中的选项菜单无效
  3. 对于Fragment,Set的选项为true
  4. 确保向上指示器正确导航回原始片段

发生了一些比较奇怪的事情 - 向上指示器仅出现一次并且不像后退按钮那样,虽然我已经使新菜单无效并且充气,新菜单会附加到原始活动菜单。

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

我的导航抽屉布局对所有菜单项都有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现在更长的后箭头也会显示!

好的,所以我设法在一些试验和错误后解决这个问题。 两个变化:

  1. 实现addOnBackStackChangedListener
  2. 需要设置ActionBarDrawerToggle's setToolbarNavigationClickListener

由于我只有一个活动(其他一切都是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();
            }
        });

我想我唯一的问题是指向使用android.R.id.homeonOptionsItemSelected方法,但这对我来说从来没有用过。 它可能是我当然实现的方式,但如果有人看到任何明显的原因,请让我知道!

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

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

在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类之后添加了更多。 我只是从活动的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();
        }
    });
}

希望这会帮助别人:)

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM