onNavigationDrawerItemSelected

onNavigationDrawerItemSelected

我已经使用Navigation Drawer Fragment和MainActivity的模板实现设置了一个新项目。

它为我提供了以下相关方法:

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    Intent intent = getIntent();
    token = intent.getStringExtra(EXTRA_TOKEN);

    mNavigationDrawerFragment = (NavigationDrawerFragment)
            getSupportFragmentManager().findFragmentById(R.id.navigation_drawer);
    mNavigationDrawerFragment.activityMain = this;

    mTitle = getTitle();

    // Set up the drawer.
    mNavigationDrawerFragment.setUp(
            R.id.navigation_drawer,
            (DrawerLayout) findViewById(R.id.drawer_layout));
}

我的MainActivity由启动活动启动,该启动活动通过EXTRA_TOKEN获取保存的访问令牌。

这是MainAcitivity中Navigation Drawer项目选择侦听器的替代:
@Override
public void onNavigationDrawerItemSelected(int position) {
    // update the main content by replacing fragments
    FragmentManager fragmentManager = getSupportFragmentManager();
    onSectionAttached(position + 1);

    switch(position) {
        case 0:
            fragmentManager.beginTransaction()
                    .replace(R.id.container, FeedFragment.newInstance(token, ""))
                    .commit();
            break;

        case 1:
            fragmentManager.beginTransaction()
                    .replace(R.id.container, PeopleFragment.newInstance("", ""))
                    .commit();
            break;

        case 2:
            if(qbloggedin) {
                fragmentManager.beginTransaction()
                        .replace(R.id.container, MessagesFragment.newInstance(token, ""))
                        .commit();
            }
            break;

        default:
            break;
    }
}

根据NavDrawer中选择的项目,它会启动三个不同的片段。在实例化新片段时,将token字符串传递到其构造函数中,该构造函数将保存在片段的类中以供进一步使用。

但是,在应用程序的首次启动时,似乎onNavigationDrawerItemSelectedonCreate之前被调用了!这导致我将零值令牌传递给片段,导致片段全部混乱。

这怎么可能?据我了解,应该尚未设置NavigationDrawerFragment!

我在onCreateonNavigationDrawerItemSelected switch position = 0上都设置了断点。 onNavigationDrawerItemSelected实际上在onCreate之前被命中。

如何确保在尝试处理onNavigationDrawerItemSelected之前先获取令牌?

任何帮助,将不胜感激。

最佳答案

我相信我已经发现了这个问题,因为对任何搜索此内容却找不到答案的人来说,这都是发生在我身上的。

如果您使用Android Studio DrawerActivity,那么将为他们创建样板代码。在activity_main.xml或您的DrawerActivity设置为其内容视图的任何XML中的此代码中,都有一个标记。

在onCreate()中调用setContentView()时,将自动创建此片段,因此从技术上讲,仍将首先调用onCreate(),但随后在创建其他任何对象之前,先调用onNavigationDrawerItemSelected()方法。由于setContentView通常保持在最前面,因此在尝试将片段的状态存储在抽屉中时会引起问题。

只需将检查checkedInstanceBundle的所有代码移到setContentView()上方,即可解决该问题。

带有注释的示例:

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    // THIS IS WHERE YOU CHECK FOR SAVED INSTANCE
    // Check for frag
    if (savedInstanceState != null) {
        Log.i(TAG, "Get QuestionDayFragment");
        mQuestionDaysFragment = (QuestionDaysFragment) getSupportFragmentManager().getFragment(savedInstanceState, QUESTION_DAY_FRAGMENT);
    }

    // View injection
    setContentView(R.layout.activity_main);
    ButterKnife.inject(this);

    // THIS IS WHERE THE CODE WAS BEFORE
    // THIS WOULD BE CALLED AFTER onNavigationDrawerItemSelected()

    // Singleton injection
    LifeboxApplication.graph().inject(this);

    // Toolbar
    setSupportActionBar(mToolbar);

    // FB
    uiHelper = new UiLifecycleHelper(this, callback);
    uiHelper.onCreate(savedInstanceState);

    // Drawer
    mNavigationDrawerFragment = (NavigationDrawerFragment) getSupportFragmentManager().findFragmentById(R.id.navigation_drawer);
    mTitle = getTitle();
    mNavigationDrawerFragment.setUp(R.id.navigation_drawer, (DrawerLayout) findViewById(R.id.drawer_layout));

}

07-24 15:19