美文网首页
Jetpack学习(一) Navigation

Jetpack学习(一) Navigation

作者: 飞哥278999401 | 来源:发表于2021-03-26 17:57 被阅读0次

    Navigation 组件旨在用于具有一个主 Activity 和多个 Fragment 目的地的应用。

    添加Navigation的依赖(在app的build.gradle)

    def nav_version = "2.3.3"
    implementation "androidx.navigation:navigation-fragment-ktx:$nav_version"
    implementation "androidx.navigation:navigation-ui-ktx:$nav_version"
    

    Navigation 组件具有一个名为 Safe Args 的 Gradle 插件,该插件可以生成简单的 object 和 builder 类,以便以类型安全的方式浏览和访问任何关联的参数。我们强烈建议您将 Safe Args 用于导航和数据传递,因为它可以确保类型安全。

    1、建立导航配置

    a、在“Project”窗口中,右键点击 res 目录,然后依次选择 New > Android Resource File。此时系统会显示 New Resource File 对话框。

    b、在 File name 字段中输入名称,例如“nav_graph”。

    c、从 Resource type 下拉列表中选择 Navigation,然后点击 OK。

    2、向Activity添加NavHost

    主 Activity 与导航图相关联,且包含一个负责根据需要交换目的地的 NavHostFragment。在具有多个 Activity 目的地的应用中,每个 Activity 均拥有其自己的导航图。

    使用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">
    
        <androidx.fragment.app.FragmentContainerView
            android:id="@+id/nav_host_fragment"
            android:name="androidx.navigation.fragment.NavHostFragment"
            android:layout_width="0dp"
            android:layout_height="0dp"
            app:layout_constraintLeft_toLeftOf="parent"
            app:layout_constraintRight_toRightOf="parent"
            app:layout_constraintTop_toTopOf="parent"
            app:layout_constraintBottom_toBottomOf="parent"
    
            app:defaultNavHost="true"
            app:navGraph="@navigation/nav_graph" />
    
    
    </androidx.constraintlayout.widget.ConstraintLayout>
    

    1、android:name 属性包含 NavHost 实现的类名称。

    2、app:navGraph 属性将 NavHostFragment 与导航图相关联。导航图会在此 NavHostFragment 中指定用户可以导航到的所有目的地。

    3、app:defaultNavHost="true" 属性确保您的 NavHostFragment 会拦截系统返回按钮。请注意,只能有一个默认 NavHost。如果同一布局(例如,双窗格布局)中有多个宿主,请务必仅指定一个默认 NavHost。

    3、添加页面(fragment)和页面跳转

    在nav_graph配置

    
    <?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/nav_graph"
        app:startDestination="@id/onefragment">
    
    
        <fragment
            android:id="@+id/onefragment"
            android:name="com.example.myapplication.fragment.OneFragment"
          >
    
    
            <action
                android:id="@+id/action_onefragment_to_twofragment"
                app:destination="@id/twofragment" />
        </fragment>
    
        <fragment
            android:id="@+id/twofragment"
            android:name="com.example.myapplication.fragment.TwoFragment"
             />
    
    
    </navigation>
    
    

    <fragment>标签里面id是当前fragment的标识,name是该标签对应的实际的fragment的类。

    <fragment>标签里面的<action>标签,定义导航栏跳转的动作,id代表该动作的命名,用于在代码中引用该动作,destination代表跳转的目的地,对应跳转到的<fragment>里面的id。

    跳转也很简单,获取到当前的NavController,直接调用navigate,填写<action>标签的id即可,如下:

          val navController = findNavController()
          navController.navigate(R.id.action_onefragment_to_twofragment)
    

    4、使用safeArgs传参

    在项目的build.gradle添加

        def nav_version = "2.3.4"
        classpath "androidx.navigation:navigation-safe-args-gradle-plugin:$nav_version"
    

    在app的build.gradle里面添加

    apply plugin: "androidx.navigation.safeargs.kotlin"
    

    使用该插件,比如我当前页面为OneFragment,跳转到TwoFragment
    配置好xml后,点击Build->Make Project会自动生成当前页面的Directions类,我当前页面是OneFragment,则生成OneFragmentDirections。
    会在OneFragmentDirections类,并且生成一个方法,方法名字对应OneFragment的action标签里面的id,比如我的action的id叫action_onefragment_to_twofragment,会生成一个方法actionOnefragmentToTwofragment

    然后在需要跳转到目的地TwoFragment的标签里面添加<argument>标签,用于标注需要从OneFragment传递的参数,<argument>的name,类似bundle传参里面的key, argType指定参数类型,支持的类型跟bundle一样,如下图。

    TwoFragment的<argument>标签配置,会在OneFragmentDirections的actionOnefragmentToTwofragment方法生成待传入参数的方法,有几个<argument>标签,该方法就会变成有几个参数的方法。内部实际上还是bundle传递。在生成一个根据该Fragment名字后加Args的类,比如跳转目的地TwoFragment会生成TwoFragmentArgs,这个类和<argument>标签是对应的。用于目的地TwoFragment取参数。

    
      <fragment
            android:id="@+id/onefragment"
            android:name="com.example.myapplication.fragment.OneFragment"
            >
            <action
                android:id="@+id/action_onefragment_to_twofragment"
                app:destination="@id/twofragment" />
        </fragment>
    
        <fragment
            android:id="@+id/twofragment"
            android:name="com.example.myapplication.fragment.TwoFragment"
            >
    
            <argument
                android:name="amount"
                app:argType="integer"
                android:defaultValue="0"
                />
    
        </fragment>
    

    跳转时

       val navController = findNavController()
       val amount = 123;
    
      val action = OneFragmentDirections.actionOnefragmentToTwofragment(amount);
                navController.navigate(action)
    
    

    actionOnefragmentToTwofragment这个方法在自动生成的代码里实际上是:

    public class OneFragmentDirections private constructor() {
      private data class ActionOnefragmentToTwofragment(
        public val amount: Int = 0
      ) : NavDirections {
        public override fun getActionId(): Int = R.id.action_onefragment_to_twofragment
    
        public override fun getArguments(): Bundle {
          val result = Bundle()
          result.putInt("amount", this.amount)
          return result
        }
      }
    
      public companion object {
        public fun actionOnefragmentToTwofragment(amount: Int = 0): NavDirections =
            ActionOnefragmentToTwofragment(amount)
      }
    }
    
    
    image.png

    5、在目的地TwoFragment取参

    在跳转后的TwoFragment中,直接使用by navArgs()即可取到参数,在<fragment>标签中配置的 <argument>参数

    class TwoFragment : Fragment() {
    
        val args: TwoFragmentArgs by navArgs()
    
        override fun onCreateView(
            inflater: LayoutInflater,
            container: ViewGroup?,
            savedInstanceState: Bundle?
        ): View? {
    
            val view = inflater.inflate(R.layout.fragment_two, container, false)
            return view;
        }
    
        override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
            super.onViewCreated(view, savedInstanceState)
    
            textview.text = args.amount.toString() ;
    
        }
    }
    
    

    6、ProGuard 注意事项

    -keepnames class com.path.to.your.ParcelableArg
    -keepnames class com.path.to.your.SerializableArg
    -keepnames class com.path.to.your.EnumArg
    

    相关文章

      网友评论

          本文标题:Jetpack学习(一) Navigation

          本文链接:https://www.haomeiwen.com/subject/vrpjhltx.html