是否可以使用 Android 导航架构组件(Android Jetpack)有条件地设置 startDestination?

Posted

技术标签:

【中文标题】是否可以使用 Android 导航架构组件(Android Jetpack)有条件地设置 startDestination?【英文标题】:Is it possible to set startDestination conditionally using Android Navigation Architecture Component(Android Jetpack)? 【发布时间】:2019-01-26 11:56:27 【问题描述】:

我正在使用android Jetpack 中的Navigation 在屏幕之间导航。 现在我想动态设置 startDestination。

我有一个名为 MainActivity 的活动 还有两个 Fragment,FragmentA 和 FragmentB。

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

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

我想使用导航架构组件设置上述流程。目前我已经使用了 startDestionation 如下,但它没有满足我的要求。

<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 导航架构组件有条件地设置 startDestination?

【问题讨论】:

#navigation-architecture-component #navigation-graph 查看***.com/questions/51173002/… 嗨@NileshRathod,如果我先打开FragmentA 然后打开FragmentB,这是可行的。但我想以编程方式设置 StartDestination。你有同样的解决方案吗? 试试这个NavigationUI.onNavDestinationSelected(menuItem, navController.getNavController()); 我想在没有菜单或任何其他导航控件的情况下执行上述操作。也许没有办法有条件地设置 startDestination。所以现在我按照你的建议使用了 [***.com/questions/51173002/…。 【参考方案1】:

终于,我的查询得到了解决方案……

将下面的代码放入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.startDestination = R.id.fragment1
//or
//graph.startDestination = 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_
    android:layout_>

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

</FrameLayout>

在使用fragment 标记而不是FragmentContainerView 的情况下,上述更改保持不变

【讨论】:

首先,您需要从 Activity 布局 xml 中删除 app:navGraph NavHostFragment 属性,因为 NavHostFragment 在主机 Activity onCreate() 调用后实例化并设置导航图。此外,如果您使用 NavigationUI.setupWithNavController 方法,例如对于 Toolbar 或 BottomNavigationView,您需要将此代码 sn-p 放在 NavigationUI.setupWithNavController 方法调用的上方。 我还要补充一点,以保留 Kotlin 代码中的后退按钮行为,您仍然需要一种方法将 navHostFragment 设置为默认导航主机。这可以通过这个事务来完成:supportFragmentManager.beginTransaction().setPrimaryNavigationFragment(navHostFragment).commit() 方法 graph.setDefaultArguments() 不再存在,您也不需要最后两行 (NavigationUI)。有关更新版本,请参阅 my answer。 设置图表时是否可以添加过渡动画? 当您跳转到深度目的地时,深度链接是否会起作用,因为它会添加 startDestination?这个设置会增加什么?【参考方案2】:

自 Akash's answer 以来,某些 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_
  android:layout_
  tools:context=".MainActivity">

  <!-- Following line omitted inside <fragment> -->
  <!-- app:navGraph="@navigation/navigation_main" -->
  <fragment
    android:id="@+id/fragment_main_nav_host"
    android:layout_
    android:layout_
    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>

【讨论】:

如果这个 startDestination 需要一个捆绑包怎么办? 那么,我将如何从twoFragment 回到oneFragment,从后面的堆栈中删除twoFragment?我的意思是,导航 API 相当于 getSupportFragmentManager().beginTransaction().replace(R.id.nav_host_fragment, new OneFragment()).commit();?另外,你能告诉我把setSupportActionBar()NavigationUI.setupWithNavController()放在哪里吗?谢谢? 如果 startDestination 需要捆绑包怎么办? @DamienLocque @Manikanta 使用 navController.setGraph(graph, bundle) 使用 navController.setGraph(graph, intent.extras)【参考方案3】:

这可以通过导航操作来完成。因为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 启动时,只需使用 action id 进行导航,它将删除堆栈中的 FragmentA,并跳转到 FragmentB。看来,fragmentB 是您的起始目的地。

if(!isAllSetUp)

  // FragmentB
  navController.navigate(R.id.action_a_to_b)

【讨论】:

谢谢,非常干净的解决方案!您也可以在片段的 onCreate 中执行此操作。我在那里做,因为我也希望在我的情况下从抽屉菜单调用片段时也发生这种情况。 虽然 navigate() 不会立即杀死片段(即使它在过程中从后台堆栈中删除)。所以,我的应用程序实际上在 onViewCreated 中崩溃了。我修复了它,但是当它导航离开时,提前退出 onViewCreated 似乎有点丑陋。【参考方案4】:

这不是答案,而只是以更简洁明了的方式复制@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

【讨论】:

如果你使用这个解决方案,如果活动不再存在,你会崩溃java.lang.IllegalStateException: unknown destination during restore: com.x:id/navigation_on_boarding 你试过了吗?我已经在使用它了,它完全没有副作用,请您详细说明您已经尝试过的内容并导致拥有illegalStateException,以便我可以在我的设置中检查它并通过反馈回复您【参考方案5】:

您可以通过编程方式设置起始目的地,但是,大多数情况下,您的起始逻辑会参考某个远程端点。如果您不在屏幕上显示任何内容,您的 UI 会看起来很糟糕。

我所做的总是显示启动画面。它将确定下一个要显示的屏幕。

例如,在上图中,您可以在 Splash Screen State 中询问是否有已保存的 LoginToken。如果它为空,则导航到登录屏幕。

登录屏幕完成后,您分析结果保存令牌并导航到您的下一个片段主屏幕。

在主屏幕中按下后退按钮时,您将向启动屏幕发送一条结果消息,指示它完成应用程序。

要弹出 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)

【讨论】:

如果您从通知中进行深层链接,您的后台也将包含启动画面 起初我的初始屏幕包含 NavController 指令,但是当使用来自反应组件的异步回调时出现问题......有时,它们不会在初始屏幕之前按时到达初始屏幕已经做出了决定。我发现异步方向处理程序最好与主要活动生命周期相关联,因为它可以在启动屏幕(AKA 主页)和它的任何子方向之间的任何变化中存活下来。可以选择在重新连接之前清除发布者的缓存,但这已经太侵入 imo。

以上是关于是否可以使用 Android 导航架构组件(Android Jetpack)有条件地设置 startDestination?的主要内容,如果未能解决你的问题,请参考以下文章

如何使用导航架构组件查找子片段

关于android导航架构组件的问题

导航架构组件 - 对话框片段

导航架构组件 - 使用 Fragment 将参数数据传递给 startDestination

Android 导航架构组件:如何将捆绑数据传递给 startDestination

导航架构组件 - 带有 lambda 表达式的动作导航