在Android Navigation Architecture Component(Android Jetpack)中,是否可以有条件地设置startDestination?

95

我正在使用NavigationAndroid Jetpack在屏幕之间进行导航。 现在,我想要动态地设置startDestination。

我有一个名为MainActivity的Activity, 以及两个Fragment,FragmentA和FragmentB。

var isAllSetUp : Boolean = // It is dynamic and I’m getting this from Preferences.

    If(isAllSetUp)
    {
     // show FragmentA
    }
    else
    {
     //show FragmentB
    }

我想使用导航架构组件来设置上述流程。目前我已经使用了以下的startDestination,但它不能满足我的要求。
<navigation xmlns:android="http://schemas.android.com/apk/res/android"
   xmlns:app="http://schemas.android.com/apk/res-auto"
   xmlns:tools="http://schemas.android.com/tools"
   android:id="@+id/lrf_navigation"
   app:startDestination="@id/fragmentA">

   <fragment
       android:id="@+id/fragmentA"
       android:name="com.mindinventory.FragmentA"
       android:label="fragment_a"
       tools:layout="@layout/fragment_a" />
</navigation>

使用Android Navigation Architecture组件,是否可以有条件地设置startDestination?


1
#导航架构组件 #导航图 - Akash Patel
请查看以下链接:https://dev59.com/kFUK5IYBdhLWcg3w2zCR - AskNilesh
嗨@NileshRathod, 如果我先打开FragmentA再打开FragmentB,它可以工作。但是我想以编程方式设置StartDestination。你有任何解决方案吗? - Akash Patel
尝试使用 NavigationUI.onNavDestinationSelected(menuItem, navController.getNavController()); - AskNilesh
1
我想在没有菜单或任何其他导航控件的情况下完成上述操作。也许没有办法有条件地设置startDestination。所以现在我按照你的建议使用了[https://dev59.com/kFUK5IYBdhLWcg3w2zCR]。 - Akash Patel
这个答案中添加了一个动态扩展:动态更改目的地 - Morteza Rastgoo
5个回答

125

终于,我找到了解决我的问题的方法...

将下面的代码放在ActivityonCreate()方法中。

Kotlin代码

val navHostFragment = (supportFragmentManager.findFragmentById(R.id.home_nav_fragment) as NavHostFragment)
val inflater = navHostFragment.navController.navInflater
val graph = inflater.inflate(R.navigation.nav_main)
//graph.addArgument("argument", NavArgument)
graph.setStartDestination(R.id.fragment1)
//or
//graph.setStartDestination(R.id.fragment2)

navHostFragment.navController.graph = graph

Java 代码

NavHostFragment navHostFragment = (NavHostFragment) getSupportFragmentManager().findFragmentById(R.id.home_nav_fragment);  // Hostfragment
NavInflater inflater = navHostFragment.getNavController().getNavInflater();
NavGraph graph = inflater.inflate(R.navigation.nav_main);
//graph.addArgument("argument", NavArgument)
graph.setStartDestination(R.id.fragment1);

navHostFragment.getNavController().setGraph(graph);
navHostFragment.getNavController().getGraph().setDefaultArguments(getIntent().getExtras());


NavigationView navigationView = findViewById(R.id.navigationView);
NavigationUI.setupWithNavController(navigationView, navHostFragment.getNavController());

补充信息

根据 @artnest 建议,从布局中删除 app:navGraph 属性。删除后的样子如下:

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <androidx.fragment.app.FragmentContainerView
        android:id="@+id/home_nav_fragment"
        android:name="androidx.navigation.fragment.NavHostFragment"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        app:defaultNavHost="true" />

</FrameLayout>
在使用FragmentContainerView的情况下,如果使用了fragment标记,则上述更改保持不变。

19
首先,您需要从 Activity 布局的 XML 中删除 app:navGraph NavHostFragment 属性,因为 NavHostFragment 会在宿主 Activity 的 onCreate() 方法调用之后实例化并设置导航图。 同时,如果您使用 NavigationUI.setupWithNavController 方法,例如用于 Toolbar 或 BottomNavigationView,则需要将此代码片段放置在 NavigationUI.setupWithNavController 方法调用之前。 - theme_an
2
我还要补充一点,在 Kotlin 代码中保留返回按钮行为,你仍然需要一种方法将 navHostFragment 设置为默认的导航宿主。可以通过以下事务完成:supportFragmentManager.beginTransaction().setPrimaryNavigationFragment(navHostFragment).commit() - Chris
1
方法 graph.setDefaultArguments() 不再存在,您也不需要最后的两行代码(NavigationUI)。请参见我的答案获取更新后的版本。 - Danish Khan
在设置图表时是否可以添加过渡动画? - Maddy
1
深度链接是否有效,因为当您跳转到深度目标时,它会添加startDestination?这个设置会添加什么? - uberchilly
这对我实际上并不起作用。@DanishKhan的答案却可以。 - Ben Butterworth

60

根据Akash的答案,一些API已经更改、不可用或者不再必要。现在变得简单了一些。

MainActivity.java:

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

  NavHostFragment navHost = (NavHostFragment) getSupportFragmentManager().findFragmentById(R.id.fragment_main_nav_host);
  NavController navController = navHost.getNavController();

  NavInflater navInflater = navController.getNavInflater();
  NavGraph graph = navInflater.inflate(R.navigation.navigation_main);

  if (false) {
    graph.setStartDestination(R.id.oneFragment);
  } else {
    graph.setStartDestination(R.id.twoFragment);
  }

  navController.setGraph(graph);
}

activity_main.xml:

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout 
  xmlns:android="http://schemas.android.com/apk/res/android"
  xmlns:app="http://schemas.android.com/apk/res-auto"
  xmlns:tools="http://schemas.android.com/tools"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  tools:context=".MainActivity">

  <!-- Following line omitted inside <fragment> -->
  <!-- app:navGraph="@navigation/navigation_main" -->
  <fragment
    android:id="@+id/fragment_main_nav_host"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:name="androidx.navigation.fragment.NavHostFragment"
  />

</androidx.constraintlayout.widget.ConstraintLayout>

navigation_main.xml:

<?xml version="1.0" encoding="utf-8"?>
<!-- Following line omitted inside <navigation>-->
<!-- app:startDestination="@id/oneFragment" -->
<navigation xmlns:android="http://schemas.android.com/apk/res/android"
  xmlns:app="http://schemas.android.com/apk/res-auto"
  xmlns:tools="http://schemas.android.com/tools"
  android:id="@+id/navigation_main"
  >

  <fragment
    android:id="@+id/oneFragment"
    android:name="com.apponymous.apponymous.OneFragment"
    android:label="fragment_one"
    tools:layout="@layout/fragment_one"/>
  <fragment
    android:id="@+id/twoFragment"
    android:name="com.apponymous.apponymous.TwoFragment"
    android:label="fragment_two"
    tools:layout="@layout/fragment_two"/>
</navigation>

1
如果这个 startDestination 需要一个 bundle 呢? - Damien Locque
那么,我该如何从twoFragment返回到oneFragment,并将twoFragment从后退栈中移除呢?我的意思是,有没有Navigation API的等效方法来实现getSupportFragmentManager().beginTransaction().replace(R.id.nav_host_fragment, new OneFragment()).commit();的功能呢? 此外,您能否展示一下在哪里放置setSupportActionBar()NavigationUI.setupWithNavController()?谢谢。 - Oliver
1
如果startDestination需要一个bundle怎么办? - Manikanta
3
使用 navController.setGraph(graph, bundle) - amitavk
使用 navController.setGraph(graph, intent.extras) - Dino Sunny

13

这可以通过导航操作完成。因为fragmentA是起始目的地,所以在fragmentA中定义一个操作。

请注意这两个字段: app:popUpToInclusive="true" app:popUpTo="@id/fragmentA"

<navigation xmlns:android="http://schemas.android.com/apk/res/android"
   xmlns:app="http://schemas.android.com/apk/res-auto"
   xmlns:tools="http://schemas.android.com/tools"
   android:id="@+id/lrf_navigation"
   app:startDestination="@id/fragmentA">

   <fragment
       android:id="@+id/fragmentA"
       android:name="com.mindinventory.FragmentA"
       android:label="fragment_a"
       tools:layout="@layout/fragment_a">

    <action android:id="@+id/action_a_to_b"
            app:destination="@id/fragmentB"
            app:popUpToInclusive="true"
            app:popUpTo="@id/fragmentA"/>
   <fragment>

   <fragment
       android:id="@+id/fragmentB"
       android:name="com.mindinventory.FragmentB"
       android:label="fragment_b"
       tools:layout="@layout/fragment_b"/>
</navigation>

当MainActivity启动时,只需使用操作id进行导航,它将从堆栈中删除fragmentA,并跳转到fragmentB。看起来,fragmentB是您的起始目标。

if(!isAllSetUp)
{
  // FragmentB
  navController.navigate(R.id.action_a_to_b)
}

谢谢,非常干净的解决方案!显然你也可以在片段的onCreate方法中实现这个功能。我选择在那里实现它,因为我希望在我的情况下,当从抽屉菜单调用该片段时,也能发生这种情况。 - Max
尽管navigate()不会立即杀死片段(即使在此过程中从后堆栈中删除它),但我的应用程序实际上在onViewCreated中崩溃了。我已经修复了它,但当它导航离开时,从onViewCreated中提前返回似乎有点丑陋的hack。 - Max

3

这不是答案,只是对@Akash Patel的答案进行更加清晰明了的复述。

// in your MainActivity
navController = findNavController(R.id.nav_host_fragment)
val graph = navController.navInflater.inflate(R.navigation.nav_graph)
if (Authentication.checkUserLoggedIn()) {
      graph.startDestination = R.id.homeFragment
} else {
      graph.startDestination = R.id.loginFragment
}
navController.graph = graph

1
如果使用此解决方案,如果活动不再存在,则会崩溃。java.lang.IllegalStateException: unknown destination during restore: com.x:id/navigation_on_boarding - rafaelasguerra
你试过了吗?我已经使用它了,完全没有副作用。能否请您详细解释一下您之前尝试的内容,以及导致出现“illegalStateException”的原因,这样我就可以在我的设置中检查并给您反馈。 - Bahaa KA

3
你可以通过编程的方式设置起始目的地,然而,大多数情况下,起始逻辑将会查询一些远程端点。如果不在屏幕上显示任何内容,你的用户界面将会显得很糟糕。
我的做法是始终显示一个启动屏幕。它将确定下一个要显示的屏幕。

enter image description here

例如,在上面的图片中,您可以在启动屏幕状态中询问是否保存了登录令牌。如果为空,则导航到登录屏幕。
一旦完成登录屏幕,然后您分析结果保存令牌并导航到下一个片段主屏幕。
当在主屏幕中按下返回按钮时,您将向启动屏幕发送一个结果消息,指示它完成应用程序。
要弹出1个片段并导航到另一个片段,您可以使用以下代码:
val nextDestination = if (loginSuccess) {
        R.id.action_Dashboard
    } else {
        R.id.action_NotAuthorized
    }

val options = NavOptions.Builder()
        .setPopUpTo(R.id.loginParentFragment, true)
        .build()

findNavController().navigate(nextDestination, null, options)

如果你从通知中进行深度链接,你的返回栈也将包含启动屏幕。 - uberchilly
起初我让闪屏包含NavController的指示,但问题在于当从响应式组件使用异步回调时......有时它们没有及时到达闪屏,导致闪屏已经做出了决定。我发现将异步方向处理程序与主活动生命周期绑定是最好的选择,因为它将在闪屏(也称为主页)和其任何子方向之间的任何更改中保持不变。当然,还有一种选择是在重新连接之前清除发布者的缓存,但我认为这已经太过侵入性了。 - Delark

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接