我在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);

当前回答

对于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
}

其他回答

对于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
}

首先,向动作标签添加属性app:popUpTo='your_nav_graph_id'和app:popUpToInclusive="true"。

<fragment
    android:id="@+id/signInFragment"
    android:name="com.glee.incog2.android.fragment.SignInFragment"
    android:label="fragment_sign_in"
    tools:layout="@layout/fragment_sign_in" >
    <action
        android:id="@+id/action_signInFragment_to_usersFragment"
        app:destination="@id/usersFragment"
        app:launchSingleTop="true"
        app:popUpTo="@+id/main_nav_graph"
        app:popUpToInclusive="true" />
</fragment>

其次,使用上述操作作为参数导航到目的地。

findNavController(fragment).navigate(
     SignInFragmentDirections.actionSignInFragmentToUserNameFragment())

更多信息请参见文档。

注意:如果你使用方法导航(@IdRes int resId),你不会得到想要的结果。因此,我使用方法导航(@NonNull NavDirections方向)。

在我的例子中,我使用了两个不同的活动,它们有各自的导航图。我的第一个活动是“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()
}

}

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

你可以像这样重写base activity的back pressed:

override fun onBackPressed() {

  val navigationController = nav_host_fragment.findNavController()
  if (navigationController.currentDestination?.id == R.id.firstFragment) {
    finish()
  } else if (navigationController.currentDestination?.id == R.id.secondFragment) {
    // do nothing
  } else {
    super.onBackPressed()
  }

}

我想你的问题是关于如何使用Pop Behavior / Pop to / app:popUpTo (in xml)

在文档, 在导航前弹出一个给定的目的地。这将从后堆栈中弹出所有不匹配的目的地,直到找到该目的地。

示例(简单的求职应用程序) 我的start_screen_nav图是这样的:

startScreenFragment (start) -> loginFragment -> EmployerMainFragment

                            -> loginFragment -> JobSeekerMainFragment

如果我想导航到EmployerMainFragment并弹出所有包括startScreenFragment,那么代码将是:

        <action
            android:id="@+id/action_loginFragment_to_employerMainFragment"
            app:destination="@id/employerMainFragment"

            app:popUpTo="@+id/startScreenFragment"
            app:popUpToInclusive="true" />

如果我想导航到EmployerMainFragment并弹出所有排除startScreenFragment的代码,那么代码将是:

        <action
            android:id="@+id/action_loginFragment_to_employerMainFragment"
            app:destination="@id/employerMainFragment"

            app:popUpTo="@+id/startScreenFragment"/>

如果我想导航到EmployerMainFragment和弹出loginFragment而不是startScreenFragment,那么代码将是:

        <action
            android:id="@+id/action_loginFragment_to_employerMainFragment"
            app:destination="@id/employerMainFragment"

            app:popUpTo="@+id/loginFragment"
            app:popUpToInclusive="true"/>

OR

        <action
            android:id="@+id/action_loginFragment_to_employerMainFragment"
            app:destination="@id/employerMainFragment"

            app:popUpTo="@+id/startScreenFragment"/>