From today, I officially start to analyze the basic usage and implementation principle of Navigation library. I wasn’t really going to learn this library, but recently the group has made it possible to componentize the process of jumping between collating pages. As it happens, the page jump of our business is almost realized in the way of single Activity + Fragment. So I put forward a suggestion, is it possible to study the possibility of Navigation library landing in our business scene? Therefore, I first need to investigate the status quo of this Navigation library, as well as its advantages and disadvantages, and then evaluate whether it can be implemented into business.

Now, I have successfully exported a survey document of Navigation and sent it to the big guys. Of course, I have a basic understanding of its usage and implementation principle. Finally, I thought I had not written a blog for a long time, so I wrote this article for record, and also for your reference.

This article is more content, will be divided into two parts to analyze. The main content of the previous chapter is:

  1. Basic use of Navigation.
  2. Navigation basic structure and basic explanation of core classes.
  3. Graph file analysis. This includes parsing the meaning of each element, the process of the node inflate, and the meaning of the NavDestination

The main content of the second part is:

  1. The realization of page jump logic.
  2. Navigator’s analysis. We’ll focus on FragmentNavigator.
  3. How to customize the Navigator and how to pass parameters to the page.
  4. Some design aesthetics and “flaws” of Navigation. I put the defects in quotation marks to indicate that they are only my own thoughts and do not represent a problem with Navigation design.

As can be seen from the above list, the first part mainly introduces the basic related content, while the second part focuses on analyzing the realization principle of Navigation.

Reference article:

  1. The official documentation for Navigation
  2. [Navigation with Jetpack] Go where you want, compass of Android world
  3. Navigation source code parsing

The source references for this article are from version 2.3.5.

1. An overview of the

According to the official documentation, the main purpose of Navigation library is to support the user to enter and exit different page interaction. In layman’s terms, it supports switching between different fragments within an Activity, and also supports jumping between activities. The Activity’s jump system supports this by default, so this is not the point of Navigation. When choosing and learning, we must have taken its ability to switch fragments into consideration, which is also the focus of this article.

Using Navigation library to switch fragments mainly has the following advantages:

  1. Can process Fragment transactions to ensure the correctness of the Fragment life cycle.
  2. By default, round-trip operations are handled correctly.
  3. Support custom Fragment special animation.
  4. DeepLink jumps are supported.
  5. Parameters can be freely passed between pages, not only Parcelable and Serializable, but also any data type.

For the advantages listed above, I would like to add the first point. When Navigation switches to fragments, it uses the FragmentTransaction replace method. Therefore, if you jump from Fragment A to Fragment B, The Fragment life cycle will go to onDestroyView, and when it returns to Fragment A, it will go back to onCreateView. This should never be acceptable because we typically do a lot of initialization in the onCreateView method. For this, there are generally two solutions on the network:

  1. Overwrite the Fragment’s onCreateView so that it will only initialize the View once, with the relevant judgment.
  2. Rewrite FragmentNavigator to use show and hide schemes to control Fragment switching.

I would like to say that the two schemes mentioned above have their own advantages and disadvantages as follows:

plan advantages disadvantages
Plan a Fragment life cycle is guaranteed

Phase go to onStop, resource can

To release
The state of the page could go wrong, say, a second time

So onCreateView is actually using the View from the last View,

The state of the View could be problematic.
Scheme 2 Does not appear on return retune

Using the onCreateView method

The problem
Since it is hidden, the last Fragment’s life cycle does not have any

Change, so resources are not effectively released.

From the above analysis, both schemes have advantages and disadvantages. So is there a better way to get the life cycle right without introducing other problems? This is possible at the time. In fact, we can modify plan 2 to use setMaxLifecycle to change the Fragment lifecycle to accommodate the advantages and disadvantages of both solutions. The specific implementation and analysis of this scheme will be introduced in the next chapter, here to sell a mystery.

In addition to the functional advantages, I think it has the advantage of a design, it is to jump into a visual process design solutions, the existence of graph file not only make a newcomer to a completely unfamiliar page can quickly understand, at the same time also will jump the realization of the process configuration, make the follow-up maintenance and extension work of low cost.

2. Basic usage

Since Navigation is taught hand in hand, we should first understand how to use Navigation, because only by understanding its characteristics can we better understand and analyze its implementation principle, and further, we can apply what we have learned.

I have always believed that the purpose of source learning is not to install force, but has the following two benefits:

  1. More in-depth understanding of the implementation principle of the library, when there is a problem in the process of using the library, you can quickly identify the cause of the problem from the source point of view, and give an effective solution.
  2. Apply what you have learned to practice. We can apply some of the design ideas and implementation details of the library to real business scenarios, making our business code as elegant as the official library.

Okay, that’s a bit of digression, so let’s get back to business. This section mainly introduces the following contents:

  1. Demo effect display.
  2. Preparation – Introduces and defines the use of Graph, NavHostFragment.
  3. Use of several jump logic –action and deepLink jumps, and activity jumps.
  4. Special attention:popUpToandpopUpToInclusiveThe difference with traditional jump.

(1). The Demo show

Before we introduce the formal usage, let’s use a real Demo to get a feel for the effect. Here’s what it looks like:

Let me show you the structure of this Demo. This Demo has three fragments: NavConatainerFragment, NavChildFragmentA, and NavChildFragmentB. NavConatainerFragment can use Navigation to jump to other two fragments and return to NavConatainerFragment successfully from other two fragments.

The full code for this Demo can be found at NavigationDemo.

(2) Introduction of.graph file

Graph file is very important in Navigation. Because it is the configuration file of all jump processes, the complete jump process between pages can be reflected in this file, including the standardization and legitimacy of the subsequent dynamic jump using codes. So creating and defining graph files is the first step in learning Navigation.

The graph file exists as XML in the project, so it needs to be created in the RES directory. The basic creation process looks like this:

  1. inresCreate a directory namednavigationThe directory.
  2. Then right click on thenavigationDirectory, selectNew->Navigation Resource File, so you can create a graph file.

In the graph file you created for the first time, the basic content is as follows:

The structure in the screenshot is as familiar as it could be. We can get two things from the screenshot:

  1. The root element of the graph file is Navigation.
  2. android:idsaidnavigationUnique identifier of. This unique identifier is very important, in the world of Navigation, not only fragment and activity are destinations, Navigation is also a destination. We’ll talk more about what a NavDestination is.

Beyond that, I’ll focus on the other elements in the Graph file:

The element name role
navigation The following element of graph file must be setidandstartDestination. When NavHostFragment

When the graph file is loaded, it is based onstartDestinationNavigate to the specified page.
action Represents a jump behavior, which can be used as a child of navigation or otherwise

The child element of destination(fragment or activity) must be setidanddestinationBelong to

Sex, where id is provided to other destinations to find specific jump behavior; Destination said

The jump falls to the id of a specific destination. In addition to these properties, there areenterAnimandexitAnimwith

To define page entry and exit animations, as wellpopUpTo andpopUpToInclusive Used to deal with cycle

Loop jump.
deepLink Similar to action, this represents a jump action and can be used as a child of navigation or as a child of navigation

As a child element of another destination(fragment or activity). You can do that by settingurior

actionProperty to indicate which page to jump to.
fragment One of the children of navigation, representing a page

Destination. Among them,idProperty represents the unique identity of the current page to give to the action

The element defines the specific landing page;nameProperty represents the full path for a specific Fragment.
activity One of the children of navigation, representing a page, similar to a fragment.
include One of the children of navigation, used to introduce another graph file. This element benefits graph

File independence, thus facilitating the split and reuse of jump processes.

I’ve listed common elements in the table above, and I’ll add a few things here:

  1. Navigation elementsstartDestinationProperty sets the ID of the fragment or activity element, indicating that when the graph file is first loaded or jumped to, it defaults to the specified page. As mentioned above, navigation itself is onedestinationAre in the same class as fragments and activities. But navigation does not host the Ui itself, so it needs a destination with a Ui.
  2. The action element, when a child of the Fragment element, means that it is just oneLocal action, can be used only on the fragment page corresponding to the fragment element. When the action element is a child of navigation, it is aGlobal actionAll fragments and actions under the navigation element can be used. And itsdestination Property sets the ID of the fragment or activity element.
  3. The deeLink element can be set in two places: as a child of the Fragment and activity elements; As a child of navigation. These two places mean different things. When it is a child of the Fragment and activity elements, it means that other pages can jump to the fragment and Activity page using the corresponding link, which is configured heredeepLink; When used as a child of navigation, indicates that other pages can jump to its graph using the corresponding linkstartDestinationPage. Pay attention to,DeepLink does not support transitions. You need to customize deepLink transitions if necessary.
  4. If we want to jump from a page in one graph file to a page in another graph file, we must use the first graph fileThe include elementBring in another graph file.

In the meantime, I’ve only listed some of the elements here, and some of the less commonly used ones are not included. Such as:

  1. Dialog element: the page itself has a destination, i.e. Navigation allows us to jump from a page to a dialog. However, I personally don’t recommend using this method for Navigation, as I have an attitude towards Navigation that it can’t be used or used entirely.
  2. Argument element: This can be used as a child of the Fragment or activity element to pass specified arguments to the page. As such, I personally do not recommend using this element to pass parameters to a page because it is too limited. There are other ways to pass parameters flexibly, which will be covered in the next article. For more information on this element, you can refer to the official documentation: Passing Data between Destinations.

We’ve already covered the meaning of the Graph file. I’ll use the above Demo as an example to get a feel for the definition of the graph file:

<? The XML version = "1.0" encoding = "utf-8"? > <navigation xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" android:id="@+id/default_graph" app:startDestination="@id/fragment_container"> <fragment android:id="@+id/fragment_container" android:name="com.example.navigationdemo.NavContainerFragment"> <action android:id="@+id/action_to_child_a" app:destination="@id/fragment_nav_child_a" /> <action android:id="@+id/action_to_child_b" app:destination="@id/fragment_nav_child_b" /> </fragment> <fragment android:id="@+id/fragment_nav_child_a" android:name="com.example.navigationdemo.NavChildFragmentA" /> <fragment android:id="@+id/fragment_nav_child_b" android:name="com.example.navigationdemo.NavChildFragmentB" /> </navigation>Copy the code

In default_graph, there is only one sub-element, fragment, below the navigation element. Here, I would like to add a few points:

  1. If you want a Fragment or Activity to be redirected to another page, you must declare it in the graph file. For example, hereNavChildFragmentAandNavChildFragmentBAlthough they do not jump to other pages, they will be the landing page of other businesses, so they have to be declared in the file.
  2. There are two actions for NavContainerFragment: jump toNavChildFragmentAandNavChildFragmentBThese two pages. And the definition of theta is thetaLocal action.
  3. In the visualization, a destination represents a node and a local action represents a line. When the number of nodes and the number of actions reaches a certain point, a graph is constructed. This is why a jump flow configuration file is also called a Graph file. You can get it from here. For example, here’s what our Demo looks like:

(3). Introduction of NavHostFragment

Once we have defined the Graph file, which means we have constructed the complete jump flow, who handles and implements the jump flow? That is NavHostFragment, the main character of this section.

NavHostFragment, as an implementation class of Fragment, inherits the characteristics of Fragment naturally. So, to actually use NavHostFragment, you must load it into your Activity. The loading of NavHostFragment can be divided into two types: dynamic loading and static loading. Sounds like a normal Fragment loading? Let’s look at the code implementation. Let’s look at static loading first:

<? The 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"> <androidx.fragment.app.FragmentContainerView android:id="@+id/nav_host_fragment_container" android:name="androidx.navigation.fragment.NavHostFragment" android:layout_width="0dp" android:layout_height="0dp" app:defaultNavHost="true" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintLeft_toLeftOf="parent" app:layout_constraintRight_toRightOf="parent" app:layout_constraintTop_toTopOf="parent" app:navGraph="@navigation/default_graph" /> </androidx.constraintlayout.widget.ConstraintLayout>Copy the code

Here are a few things to note:

  1. Using theFragmentContainerViewTo load the Fragment. One might ask, are traditional fragments also ok? Of course it is possible, but it will lose a lot of features, such as the Fragment transition animation can be broken.
  2. app:defaultNavHost If this parameter is set to true, the back event of the current system is preferentially processed by NavHostFragment.
  3. app:navGraphIndicates the configuration file to be loaded. After setting this property, all the information we configured in the graph file took effect, and we happily jumped to the corresponding Fragment using the corresponding action.

The implementation principle of APP :defaultNavHost and APP :navGraph will be analyzed in the future.

Then, let’s look at the implementation of dynamic loading. Remove the FragmentContainerView attributes android: Name, app:defaultNavHost, and app:navGraph.

<? The 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"> <androidx.fragment.app.FragmentContainerView android:id="@+id/nav_host_fragment_container" android:layout_width="0dp" android:layout_height="0dp" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintLeft_toLeftOf="parent" app:layout_constraintRight_toRightOf="parent" app:layout_constraintTop_toTopOf="parent" /> </androidx.constraintlayout.widget.ConstraintLayout>Copy the code

Then load the Fragment dynamically using code:

class MainActivity : AppCompatActivity() { override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main) val hostFragment = NavHostFragment.create(R.navigation.default_graph) supportFragmentManager.beginTransaction() Add (R.i.D.nav_host_fragment_container, hostFragment) // Only after this attribute is set, the NavHostFragment can successfully intercept the system back event. .setPrimaryNavigationFragment(hostFragment) .commitAllowingStateLoss() } }Copy the code

To implement dynamically loading fragments, we need to pay attention to the following points:

  1. When creating an object using the Create method of NavHostFragment, you need to pass a graph file ID to indicate that the graph file needs to be applied to the Fragment. The create method is implemented by adding a graph file ID parameter to the Fragment argument so that the Fragment can parse and use its configuration when appropriate.
  2. You need to callsetPrimaryNavigationFragmentMethod, and pass NavHostFragment to the system, indicating that the current system’s back events are processed by the Fragment. It is important to note that round trips cannot be handled properly inside NavHostFragment without calling this method.

(4). Page hopping

Once we have the Graph file and NavHostFragment ready, we can jump to Fragment and Activity. The main content of this section is as follows:

  1. Action Indicates the redirect and precautions.
  2. DeepLink jump and precautions.
  3. Use Safe Args to jump.
  4. How to perform standardized parameter transmission.

(A). The action to jump

When we introduced graph files earlier, we learned that we can set a number of actions for each Fragment, such as the following code:

    <fragment
        android:id="@+id/fragment_container"
        android:name="com.example.navigationdemo.NavContainerFragment">
        <action
            android:id="@+id/action_to_child_a"
            app:destination="@id/fragment_nav_child_a" />
        <action
            android:id="@+id/action_to_child_b"
            app:destination="@id/fragment_nav_child_b" />
    </fragment>
Copy the code

Here we have two actions for NavContainerFragment: Jump to NavChileFragmentA and NavChildFragmentB.

So how do we jump through action? That’s relying on NavController. In Navigation, it is very simple to obtain the NavController object. Java and Kotlin have different methods, as follows:

Kotlin: * fragments. FindNavController () * the findNavController () * Activity in findNavController (viewId: Int) Java:  * NavHostFragment.findNavController(Fragment) * Navigation.findNavController(Activity, @IdRes int viewId) * Navigation.findNavController(View)Copy the code

It looks like the NavController is associated with a View. The NavController is stored as a tag in the root View of the NavHostFragment. The id of the tag is nav_controller_view_tag.

public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) { // ...... Navigation.setViewNavController(view, mNavController); // When added programmatically, we need to set the NavController on the parent - i.e., // the View that has the ID matching this NavHostFragment. if (view.getParent() ! = null) { mViewParent = (View) view.getParent(); if (mViewParent.getId() == getId()) { Navigation.setViewNavController(mViewParent, mNavController); }} / /... }Copy the code

When you get the NavController object, you can jump directly. The code is as follows:

override fun onViewCreated(view: View, savedInstanceState: Bundle?) { super.onViewCreated(view, SavedInstanceState) mViewGroup = View.findViewById (R.i.ViewGroup) addViewWithClickListener(" Jump to NavChildFragmentA") { FindNavController ().navigate(r.i.action_to_child_A)} addViewWithClickListener(" to NavChildFragmentB") {findNavController().navigate(r.i.action_to_child_A)} addViewWithClickListener(" to NavChildFragmentB") { findNavController().navigate(R.id.action_to_child_b) } }Copy the code

The code is very simple: just call the Navigate method of NavController and pass the ID of the action. This allows the page to jump, but there are a few things to note:

  1. This action can only use the local action of its own Fragment, or the global action in the graph that contains the current Fragment. Using actions elsewhere crashes.
  2. Actions are written statically in the Graph file, so what if we have dynamic requirements? There are three methods: the first method is to write all the actions of the current fragment in the configuration file. The second way is to jump through deepLink, which we’ll talk about in a minute; Generally, the previous two methods can cover all parts of the scene, but some strange logic is not excluded. You need to jump to a special page according to the dynamically delivered data. At this time, you can dynamically add an action to the current Fragment and then jump to it. The following code:
val newId = View.generateViewId() findNavController().currentDestination? .putAction(newId, R.i.d.fragment_nav_child_b) addViewWithClickListener(" Jump to NavChildFragmentB") {findNavController().navigate(newId)}Copy the code

(B). DeepLink jump

To use deepLink to jump to a specific Fragment, perform two steps as follows:

  1. Creates a deepLink for the specified Fragment. Indicates that an external can use this deepLink to jump to itself.
  2. An external call to the Navigate method of NavController passes the specified deepLink.

Suppose we create a deepLink for NavChildFragmentB:

<fragment android:id="@+id/fragment_nav_child_b" android:name="com.example.navigationdemo.NavChildFragmentB"> <! <deepLink app:uri="http://www.jade.com" /> </fragment>Copy the code

We can then jump through the following code:

AddViewWithClickListener (" Use deepLink to jump to NavChildFragmentB") {findNavController().navigate("http://www.jade.com".touri ()) }Copy the code

When we click on the View, it jumps to NavChildFragmentB via deepLink. Is it very simple? However, here are a few things to note:

  1. When creating deepLink for a page, never leave out host, as shown abovehttp://. Because if we don’t add host, then when we match, we just accept host ashttps://andhttp://. For example, up here we tookNavChildFragmentBDeepLink is configured aswww.jade.comExternal can only be usedhttps://www.jade.comandhttp://www.jade.comJump instead of directly usingwww.jade.comThere will be a crash. This is a hidden logic that needs special attention.
  2. DeepLink’s matching rules follow regular expressions. For more details, see the official documentation: Creating Deep Links for Destinations.
  3. If you want to use deepLink to send parameters, you can use the next FragmentargumentsGet a key insideandroid-support-nav:controller:deepLinkIntentIntent, and then retrieve the Uri from it to retrieve the parameter. The code to add deepLink to Arguments is as follows:
public void navigate(@NonNull NavDeepLinkRequest request, @Nullable NavOptions navOptions, @Nullable Navigator.Extras navigatorExtras) { NavDestination.DeepLinkMatch deepLinkMatch = mGraph.matchDeepLink(request); if (deepLinkMatch ! = null) { NavDestination destination = deepLinkMatch.getDestination(); Bundle args = destination.addInDefaultArgs(deepLinkMatch.getMatchingArgs()); if (args == null) { args = new Bundle(); } NavDestination node = deepLinkMatch.getDestination(); Intent intent = new Intent(); intent.setDataAndType(request.getUri(), request.getMimeType()); intent.setAction(request.getAction()); // Add deepLink to arguments. args.putParcelable(KEY_DEEP_LINK_INTENT, intent); navigate(node, args, navOptions, navigatorExtras); } else { throw new IllegalArgumentException("Navigation destination that matches request " + request + " cannot be found  in the navigation graph " + mGraph); }}Copy the code

We’ve just covered the URI attribute, but deepLink has three other attributes, which are as follows:

  1. app:action: is a component of deepLink and is a string. If the value is not empty, the action must be the same to match.
  2. app:mimeType: is a component of deepLink, a media data type that requires type correlation for a successful match. For example, “image/ JPG “matches “image/*”.
  3. android:autoVerify: Asks Google to verify that you are the owner of the corresponding URI. For details, please refer toVerify Android app links. This is not used in Fragment jumping, so it can be ignored.

(C). Safe Args jumps

Safe Args is a Gradle plugin, so introduce it separately. The configuration is as follows: first, introduce the corresponding plugin in the build.gradle file of the project:

dependencies { // ...... / / introducing the sage of the args plugin classpath "androidx. Navigation: navigation - safe - the args - gradle - plugin: 2.3.5"}Copy the code

Gradle = build.gradle = build.gradle = build.gradle

plugins {
    id "androidx.navigation.safeargs"
}
Copy the code

After introducing the Plugin for Safe Args, we edit the graph file with the Fragment dimension, generating relevant classes for us to call. Such as:

    <fragment
        android:id="@+id/fragment_container"
        android:name="com.example.navigationdemo.NavContainerFragment">
        <action
            android:id="@+id/action_to_child_a"
            app:destination="@id/fragment_nav_child_a" />
    </fragment>
Copy the code

We give NavContainerFragment added a jump to NacChildFragmentA action, safe args plugin corresponding to generate a NavContainerFragmentDirections class, This has a method called actionToChildA that we call the navigate method to jump to:

AddViewWithClickListener (" Jump to NavChildFragmentB with SafeArgs ") { findNavController().navigate(NavContainerFragmentDirections.actionToChildA()) }Copy the code

As for how it works, it’s pretty simple. When we add actions to a Fragment in the graph file, the plugin automatically generates a class named Fragment name +Directions for each Fragment. This class contains a number of static methods. The method name is related to the action ID. The method return type is NavDirections. The function of this class is very simple, similar to a Wrapper class, which wraps the action ID and argument.

So, the implementation of Safe Args is pretty simple. The plugin scans the graph file and generates wrapper classes for each Fragment to help jump. One advantage of this is that you won’t crash your App using the wrong action, but I personally don’t recommend this method for jumping:

  1. Gradle plugins themselves depend on gradle versions, and gradle may not be fully compatible with the Safe Args plugin in a production environment, resulting in compilation failures, or the resulting helper classes not being as expected.
  2. Package size. Safe Args generates a lot of classes during compilation, and we essentially just need an action ID to jump to the page. Safe Args generates those classes unnecessarily.
  3. The implementation logic is opaque. Safe Args generates classes that are not found in the project, making it hard to see the implementation logic directly and difficult to troubleshoot if something goes wrong. PS: If we want the corresponding generated class, the only way I can find at present is to decompile Apk, which undoubtedly increases the difficulty of troubleshooting problems.

As for the benefits of Safe Args, I don’t think it’s very important. Because if you use the wrong action, crashes will usually occur during development, so they won’t be brought online. As for other parameter passing issues, the Safe args itself cannot be completely avoided, as it can only detect parameters that can be placed in the Bundle, and nothing else. Therefore, I think the current Safe ARgs is relatively weak, and simple projects can be tried.

(D). How to carry out standardized parameter transmission?

The parameters to be passed are generally divided into two types:

  1. Parameters that can be serialized, such as raw data types, as well as Parcelable and Serializable types.
  2. Non-serializable parameters, such as those that do not implement the Parcelable and Serializable interfaces.

Let’s talk about it case by case. The first parameters that can be serialized are very easy to pass becauseNavControllerthenavigateThe method itself has a Bunble argument that can be used to pass arguments:

The Bundle is put into the Fragment’s argument, so look at the FragmentNavigator’s navigate method in advance. And we’ll focus on that as well.

Non-serializable parameters can be passed through another parameter of the Navigate method, called navigator.extras. FragmentNavigator.Extras is an interface that we can implement ourselves and pass our own parameters to, and then pass the parameters to the Fragment in FragmentNavigator as follows:

@Override public NavDestination navigate(@NonNull Destination destination, @Nullable Bundle args, @Nullable NavOptions navOptions, NavigatorExtras) {// ······ if (navigatorExtras instanceof Extras) {navigatorExtras instanceof Extras = (Extras) navigatorExtras; for (Map.Entry<View, String> sharedElement : extras.getSharedElements().entrySet()) { ft.addSharedElement(sharedElement.getKey(), sharedElement.getValue()); }} //...Copy the code

This is an implementation of FragmentNavigator, and we need to note the following:

  1. FragmentNavigatorThe inside of theNavigator.ExtrasImplementation class nameFragmentNavigator.ExtrasThe implementation class only supports passing key-value pairs of view-Strings. The main purpose is to handle cases of shared elements.
  2. We can follow suit. ReferenceFragmentNavigator.ExtrasTo implement our own Extras class, but this depends on customizationNavigator. The customNavigatorIt will be introduced in the next part.

(5). PopUpTo and popUpToInclusive

Before we introduce popupTo, let’s take a look at how action jumps now:

  1. Enter & Exit: The traditional way to enter a page and exit a page. For example, the actions we configured above are accessed this way. The most important feature of this approach is that when entering a page, regardless of whether the page has instances, a new object will be created and put into the return stack.
  2. PopEnter & popExit: When entering a page, determine whether the current return stack has an instance of the page, and if so, clear all pages above the current instance. ifpopUpToInclusiveIf set to true, the instance itself will also be cleared.

That explains a lot, but how to use it? PopUpTo is a property of the action element.

<? The XML version = "1.0" encoding = "utf-8"? > <navigation xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" android:id="@+id/default_graph" app:startDestination="@id/fragment_container"> <! <fragment Android :id="@+id/ fragment_nav_child_A" android:name="com.example.navigationdemo.NavChildFragmentA"> <action android:id="@+id/action_child_a_to_b_by_popUp" app:destination="@id/fragment_nav_child_b" app:popUpTo="@id/fragment_nav_child_b" app:popUpToInclusive="true" /> </fragment> <fragment android:id="@+id/fragment_nav_child_b" android:name="com.example.navigationdemo.NavChildFragmentB"> <! -- Create deepLink, <deepLink app:uri="www.jade.com" /> <action Android :id="@+id/action_child_b_to_a_by_popUp" app:destination="@id/fragment_nav_child_a" app:popUpTo="@id/fragment_nav_child_a" app:popUpToInclusive="true" /> </fragment> </navigation>Copy the code

App :popUpTo and App :popUpToInclusive = app:popUpToInclusive = App :popUpToInclusive = App :popUpToInclusive = App :popUpToInclusive These two properties have been explained briefly above, but I will explain them here:

  1. App :popUpTo: When jumping to the current page, it will first look for the instance of the page in the back stack. If so, it needs to clear all the pages on the page.
  2. app:popUpToInclusiveIf only app:popUpTo is configured, it will not clear the instance of the page itself, then there are two instances of the page in the return stack, which is not expected. So I’m going to have to putapp:popUpToInclusiveIf set to true, the instance of the page itself can be cleared.

The above description may seem abstract, but let me illustrate it with an example. Suppose I jump from page A to page B, and then from page B to page A. If the popupTo property is not used, the return stack instance is: A’ B A’ “; If we set popupTo, the return stack instance is: A “A”, because when we jump from B to A, we will clear all instances above A, and that instance of B will be automatically cleared; If we have both popupTo and popUpToInclusive configured, then the instance in the return stack is: A “, which is the newly created instance.

From the above description, we can see that the two properties match with the Activity’s singleTask startup mode, but not exactly the same. This is where I have a problem with it. For now, it exists for a good purpose, but it’s like a half-finished product. For example:

  1. In the ABA case, why is the second A page A A new instance?
  2. Why must popUpToInclusive be configured to ensure that there is only one INSTANCE of A?

In addition, action has another property: launchSingleTop. This works much like the Action singleTop. The basic explanation is as follows:

If the current top of the stack element is the target page, no new instance is created; If the current top stack element is not the target page, a new instance is created. Note that this property does not guarantee that there is only one instance in the return stack.

3. Basic structure

The usage of Navigation is covered, and here we are about to start formal source code analysis. But before analyzing, we need to have an overall outline of the execution flow of Navigation, which will be very helpful for us to understand the specific meaning of the source code.

This section mainly consists of the following two parts:

  1. Navigation execution flow.
  2. Core class interpretation in the process.

(1). Execution process

Let’s start by looking at the content of the execution process. I divided the execution process into the following steps:

  1. NavHostFragment loading process.
  2. Page redirection process.
  3. Page return flow.

(A). NavHostFragment loading

The explanation of NavhostFragments has already been explained briefly. But here’s a detailed analysis of what it does when it loads.

The flowchart above explains what is done during the NavHostFragment loading process, which is divided into three main stages. Here, I’d like to add a few things:

  1. The onCreate method calls the onCreateNavController method, which basically adds a bunch of navigators to the NavController. NavController is added by default when it is createdNavGraphNavigatorandActivityNavigator; And in the onCreateNavController method, addDialogFragmentNavigatorandFragmentNavigator. Among themFragmentNavigatorIs to handle jumps between fragments, which we will focus on later.
  2. The onCreate method gives it the graph ID by calling the NavController setGraph method. This process triggers graph file parsing, which will be the starting point for our graph file parsing.
  3. SetGraph not only triggers graph parsing, it also defaults to the page of the graph file’s startDestination tag.

(B). Page hopping process

Previously, we knew that page jumps were mainly implemented with the Navigate method of the NavController. However, before we only stay in the external call level, about the internal call process, and did not have a clear understanding.

In the page hopping process, a Navigator is added. This class deals with the specific logic of page jumps and returns, and we’ll cover it more formally later.

(C). Page return process

Page returns involve the Activity back event. NavHostFragment intercepts the back event and processes it according to its own return stack. The execution process is as follows:

Here we see the Navigator’s popBackStack method, which corresponds to the navigate method, which handles the logic for returning events. We’ll focus on that later.

(2). Core class meaning

In the Navigation framework, there are many core classes to assist the implementation of various functions. Here, we explain the classes involved in the Navigation framework in a unified way for everyone to understand.

(A). NavHostFragment

This Fragment is the container in Navigation. Theoretically, any Fragment that needs to be switched by Navigation must be the child of that Fragment. In addition, the class also maintains some core procedures for Navigation, such as:

  1. Parse the Graph file.
  2. Create and initializeNavController“, which sets the stage for subsequent page jumps.

(B). NavController

This class can be thought of as a controller for page navigation. When we jump, we just take the object of the class and pass the corresponding argument. In general, there are two ways to do jumps.

  1. The action to jump. After the NavController gets the action ID we passed, it will look for the corresponding page in the graph. If it finds the corresponding page, it can jump successfully. If they don’t, they crash. In general, actions that cannot find a page are either invalid or illegal.
  2. DeepLink jump. The NavController will use the information we pass to match the corresponding page. If multiple pages are matched, it will select the one with the highest matching degree. If there is no match, it will crash.

(C). NavDestination

In Navigation, different types of pages are abstracted as NavDestination, which abstracts the following pages:

page NavDestination implementation class Element corresponding to graph file
Activity ActivityNavigator$Destination The activity elements
Fragment FragmentNavigator$Destination Fragments elements
Diglog DiglogNavigator$Destination Dialog elements
There is no specific page NavGraph Navigation elements
In the table above, special attention should be paid toNavGraph.

NavGraph does not represent a specific page; in the graph file, it corresponds to the navigation element. This is usually used in nesting attempts, when we introduce a graph file in a graph file and need to jump from one page of the graph file to another page of the graph file. However, this is all transparent to our external use and does not require perception.

Since the NavDestination represents a page, every attribute we define for the page in the Graph file will have the corresponding field in the NavDestination. For example, there is an array in NavDestination that stores the actions that the page can jump to.

At the same time, NavDestinations and their subclasses generally do not exist on their own, but need to be paired with the Navigator we are going to talk about.

(D). Navigator

A Navigator. As the name implies, the real logic for page switching is maintained in this class. NavController = NavController = NavController = NavController = NavController = NavController = NavController = NavController = NavController = NavController The last is through the Navigator to achieve the page switch.

Let’s look again at what the different subclasses of Navigator mean.

The name of the The corresponding element name
ActivityNavigator activity
DialogFragmentNavigator dialog
FragmentNavigator fragment
NavGraphNavigator navigation

As we can see from the table above, eachNavigatorAll correspond to a graph element. So, if we need to define Navigator, we need to label the corresponding element name, how do we do that? Direct use ofNavigator.NameNotes:

And then in the graph file you can click on the name of the element.

(E). Other classes

In addition to that, there are other classes, and I’ll just briefly introduce them.

The name of the role
NavAction Wrap the action in the Graph file.
NavDeepLink DeepLink package.
NavOptions Encapsulate action properties such as enterAnim and exitAnim.
NavInflater Parse the elements in the Graph file.

One of these classes that we need to pay special attention to is NavInflater, which is one of the more important classes that we’re going to look at in a second. NavInflater helps navigate how Navigation converts elements in the Graph file to the various entity classes in the corresponding code.

4. NavInflater parsing process

As mentioned earlier, the Graph file is a configuration file for a jump flow that defines the jump relationships between each page. So how does this jump relationship work? When we use NavController to jump, how does the configuration file restrict the jump to be expected? It all starts with NavInflater. Let’s look at NavInflater parsing first. First, parsing starts in the onCreate method of NavHostFragment:

Public void onCreate(@nullable Bundle savedInstanceState) {/ ······ · if (mGraphId! = 0) { // Set from onInflate() mNavController.setGraph(mGraphId); } else { // See if it was set by NavHostFragment.create() final Bundle args = getArguments(); final int graphId = args ! = null ? args.getInt(KEY_GRAPH_ID) : 0; final Bundle startDestinationArgs = args ! = null ? args.getBundle(KEY_START_DESTINATION_ARGS) : null; if (graphId ! = 0) { mNavController.setGraph(graphId, startDestinationArgs); }} //...Copy the code

The setGraph method actually does two things:

  1. Create NavInflater objects and parse graph files.
  2. By default, it navigates to the graph filestartDestinationProperty tag page.

For the second thing, let’s skip the analysis here and look at NavInflater parsing. Look directly at the inflate method:

private NavDestination inflate(@NonNull Resources res, @NonNull XmlResourceParser parser, @NonNull AttributeSet attrs, int graphResId) throws XmlPullParserException, IOException { // 1. First, have the corresponding Navigator create the NavDestination based on the node name. Navigator<? > navigator = mNavigatorProvider.getNavigator(parser.getName()); final NavDestination dest = navigator.createDestination(); dest.onInflate(mContext, attrs); final int innerDepth = parser.getDepth() + 1; int type; int depth; While ((type = parser.next())! = XmlPullParser.END_DOCUMENT && ((depth = parser.getDepth()) >= innerDepth || type ! = XmlPullParser.END_TAG)) { if (type ! = XmlPullParser.START_TAG) { continue; } if (depth > innerDepth) { continue; } final String name = parser.getName(); if (TAG_ARGUMENT.equals(name)) { inflateArgumentForDestination(res, dest, attrs, graphResId); } else if (TAG_DEEP_LINK.equals(name)) { inflateDeepLink(res, dest, attrs); } else if (TAG_ACTION.equals(name)) { inflateAction(res, dest, attrs, parser, graphResId); } else if (tag_include.equals (name) &&dest instanceof NavGraph) {// If the current node is an include, then parse the new graph file recursively. And // include the NavGraph as a child node of this NavGrap. final TypedArray a = res.obtainAttributes( attrs, androidx.navigation.R.styleable.NavInclude); final int id = a.getResourceId( androidx.navigation.R.styleable.NavInclude_graph, 0); ((NavGraph) dest).addDestination(inflate(id)); a.recycle(); } else if (dest instanceof NavGraph) {// If the current node is NavGraph, then continue parsing its child elements, The parsed node is the child of the NavGraph ((NavGraph) dest).addDestination(inflate(res, Parser, attrs, graphResId)); } } return dest; }Copy the code

The inflate method has two main steps:

  1. The first is to get the current node name, and then passNavigator Create the corresponding NavDestination. Note in particular that the name of the current node will only beNavDestinationThe element name of the subclass, nothing else.
  2. The next step is to parse all the children of the node. This step requires two special points: first, if the current node isincludeIs another graph file that starts parsing again, invoking the inflate method with only one parameter; If the current node isnavigationAnd that translates toNavGraphYou need to continue recursively parsing its children, which have other navdestinations such as fragments, activities, etc. This calls the inflate method with four parameters.

Inflate method about the content of the basic is like this, in fact, there is also the methods, such as inflateArgumentForDestination, inflateDeepLink etc, these are for NavDestination analytical related attributes and behaviors, If you’re interested, you can go into it, but I won’t expand it here.

We are basically familiar with the parsing process, so what data structure is the node stored after parsing? Here’s a picture to show it:

After the storage is completed, when each page (node) needs to jump, it can use its own action to find the target page from the diagram, so as to realize the page jump. It is important to note that the data structure stored here is actually a tree, and this should be distinguished from the graph visualized in the graph file.

5. To summarize

At this point, Navigation concludes the previous part of the article and I will make a little summary.

  1. Graph file exists as a configuration file for Navigation. The content is composed of nodes (destination) and paths (Action, deepLink) to form a graph.
  2. There are two modes of page hopping: Action and deepLink. It is important to note how these two methods are defined and used.
  3. Every page is based onNavDestination A graph file is resolved to be a graphNavDestinationIs the tree of nodes.

The content of this article is relatively simple, the next part will focus on Navigation and other implementation principles, stay tuned.