Я использую навигацию из Android Jetpack для навигации между экранами. Теперь я хочу установить startDestination динамически.

У меня есть активность с именем MainActivity и два фрагмента, 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>

Можно ли установить startDestination условно с помощью компонента Android Navigation Architecture?

45
Akash Patel 20 Авг 2018 в 13:59

5 ответов

Лучший ответ

Наконец, я получил решение для моего запроса ...

Поместите код ниже в onCreate() метод Activity.

Код Котлина

val navHostFragment = home_nav_fragment as NavHostFragment
val inflater = navHostFragment.navController.navInflater
val graph = inflater.inflate(R.navigation.nav_main)
graph.setDefaultArguments(intent.extras)
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.setDefaultArguments(getIntent().getExtras());
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());
56
Somesh Kumar 20 Янв 2020 в 08:19

Это можно сделать с помощью действия навигации. Поскольку фрагментА является вашим начальным пунктом назначения, определите действие в фрагменте А.

Обратите внимание на эти два поля: 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 запущена, просто выполните навигацию с идентификатором действия, она удалит фрагментA из стека и перейдет к фрагменту B. Кажется, фрагмент B - это ваш начальный пункт назначения.

if(!isAllSetUp)
{
  // FragmentB
  navController.navigate(R.id.action_a_to_b)
}
9
LeeR 19 Фев 2019 в 07:16

Вы можете установить начальный пункт назначения программно, однако, в большинстве случаев ваша стартовая логика будет обращаться к некоторой удаленной конечной точке. Если вы ничего не показываете на экране, ваш интерфейс будет выглядеть плохо.

Я всегда показываю заставку. Он определит, какой будет следующий экран для отображения.

enter image description here

Например, на картинке выше вы можете спросить в состоянии экрана-заставки, есть ли сохраненный 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)
1
Pablo Valdes 7 Фев 2020 в 17:06

Это не ответ, а просто репликация ответа @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
0
Bahaa KA 8 Ноя 2019 в 16:57

Некоторые из 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>
27
Danish Khan 1 Янв 2019 в 02:46
51929290