我在android中使用新的导航架构组件,我在移动到一个新的片段后被困在清除导航堆栈。

例子: 我在loginFragment中,我想要这个片段从堆栈中清除,当我导航到home片段时,这样用户在按下返回按钮时就不会返回到loginFragment。

我使用一个简单的NavHostFragment.findNavController(Fragment).navigate(R.id.homeFragment)来导航。

当前代码:

mAuth.signInWithCredential(credential)
            .addOnCompleteListener(getActivity(), new OnCompleteListener<AuthResult>() {
                @Override
                public void onComplete(@NonNull Task<AuthResult> task) {
                    if (task.isSuccessful()) {
                        NavHostFragment.findNavController(LoginFragment.this).navigate(R.id.homeFragment);
                    } else {
                        Log.w(TAG, "signInWithCredential:failure", task.getException());
                    }
                }
            });

我尝试使用导航()中的NavOptions,但返回按钮仍然将我送回loginFragment

NavOptions.Builder navBuilder = new NavOptions.Builder();
NavOptions navOptions = navBuilder.setPopUpTo(R.id.homeFragment, false).build();   
             NavHostFragment.findNavController(LoginFragment.this).navigate(R.id.homeFragment, null, navOptions);

当前回答

在我的例子中,我使用了两个不同的活动,它们有各自的导航图。我的第一个活动是“nav_graph”的主机,并具有处理身份验证的片段,第二个活动是“nav_graph_home”的主机。在这里你可以看到我为nav_graph所做的设置。

nav_graph例子

然后回到登录片段的代码中,我写了这样的代码:

findNavController().navigate(R.id.action_logInFragment_to_nav_graph_home)

用户登录并点击后退按钮后,应用程序将关闭。记住设置弹出行为,让它弹出到当前导航图,其中包含登录片段,而不包括它。

编辑: 在此之后,向上按钮仍然出现在顶部栏中。为了避免这种行为,我们需要告诉第一个活动哪些片段是顶级的。要做到这一点,只需在“setupActionBarWithNavController()”方法的参数列表中添加除导航主机片段之外的应用程序栏配置,其中包含一组您的第一个导航图和第二个导航图的主片段。你的代码应该是这样的:

class MainActivity : AppCompatActivity(R.layout.activity_main) {
private lateinit var navController: NavController

override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)

    // Retrieve NavController from the NavHostFragment
    val navHostFragment = supportFragmentManager
        .findFragmentById(R.id.nav_host_fragment) as NavHostFragment
    navController = navHostFragment.navController

    // Set up the action bar for use with the NavController
    setupActionBarWithNavController(navController, AppBarConfiguration(setOf(R.id.logInFragment,R.id.homeFragment)))
}

/**
 * Handle navigation when the user chooses Up from the action bar.
 */
override fun onSupportNavigateUp(): Boolean {
    return navController.navigateUp() || super.onSupportNavigateUp()
}

}

这是我第一次投稿,希望能有所帮助。

其他回答

    NavController navController 
    =Navigation.findNavController(requireActivity(),          
    R.id.nav_host_fragment);// initialize navcontroller

    if (navController.getCurrentDestination().getId() == 
     R.id.my_current_frag) //for avoid crash
  {
    NavDirections action = 
    DailyInfoSelectorFragmentDirections.actionGoToDestionationFragment();

    //for clear current fragment from stack
    NavOptions options = new 
    NavOptions.Builder().setPopUpTo(R.id.my_current_frag, true).build();
    navController.navigate(action, options);
    }

我花了一段时间来防止返回按钮回到我的开始片段,在我的情况下,这是一个只应该出现一次的介绍消息。

简单的解决方案是创建一个全局操作,指向用户应该停留的目的地。你必须正确地设置app:popUpTo="…" -将它设置为你想要弹出的目的地。对我来说,这是我的介绍信息。设置app:popUpToInclusive="true"

以上的解决方案都不适合我。 在花了几个小时之后,以下是我的解决方案:

注意:我有多个nav_graphs和在不同的nav_graphs片段之间切换。

在xml中定义你的动作如下:

<行动 android: id =“@ id / your_action_id” 应用:目的地= " @ id / the_fragment_id_you_want_to_navigate_to” 应用:popUpTo = " @ id / nav_graph_which_contains_destination_fragment” 应用:popUpToInclusive = " true " / >

从Java/Kotlin代码中使用上述操作导航:

findNavController (R.id.your_nav_name) ?。应用{ 导航(R.id.your_action_id) backQueue.clear () }

对于androidx.compose版本1.2.0+

我在使用较低版本时遇到了一些问题,但1.2 +(在撰写本文时是beta版)工作得很好。

Compose中navGraph的更好语法:

navController.navigate(item.name) {

    navController.graph.startDestinationRoute?.let { route ->
        // use saveState = false to NOT save the state for the popped route
        popUpTo(route) { saveState = true }
    }

    launchSingleTop = true

    restoreState = true
}

为Jetpack撰写❤️

navHostController.navigate(Routes.HOME) {
                            this.popUpTo(Routes.ONBOARDING) {
                                this.inclusive = true
                            }
                     }