J'utilise Navigation à partir de Android Jetpack pour naviguer entre les écrans. Maintenant, je veux définir startDestination dynamiquement.
J'ai une activité appelée MainActivity et deux fragments, FragmentA et FragmentB.
var isAllSetUp : Boolean = // It is dynamic and I’m getting this from Preferences.
If(isAllSetUp)
{
// show FragmentA
}
else
{
//show FragmentB
}
Je souhaite définir le flux ci-dessus à l'aide du composant d'architecture de navigation. Actuellement, j'ai utilisé startDestionation comme ci-dessous, mais cela ne répond pas à mes besoins.
<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>
Est-il possible de définir startDestination de manière conditionnelle à l'aide de Android composant d'architecture de navigation?)
Enfin, j'ai une solution à ma requête ...
Mettez le code ci-dessous dans la méthode onCreate()
de Activity
.
Code de Kotlin
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
Code 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());
Certaines des API ont changé, ne sont pas disponibles ou ne sont plus nécessaires depuis réponse d'Akash . C'est un peu plus simple maintenant.
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>
Cela peut être fait avec une action de navigation. Parce que fragmentA est votre destination de départ, définissez une action dans fragmentA.
Notez ces deux champs: 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>
Lorsque votre MainActivity a démarré, il vous suffit de naviguer avec l’identificateur d’action, il supprimera fragmentA de la pile et passera à fragmentB. Apparemment, fragmentB est votre destination de départ.
if(!isAllSetUp)
{
// FragmentB
navController.navigate(R.id.action_a_to_b)
}