Jetpack学习(一) Navigation

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
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,907评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,987评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,298评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,586评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,633评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,488评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,275评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,176评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,619评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,819评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,932评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,655评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,265评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,871评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,994评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,095评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,884评论 2 354

推荐阅读更多精彩内容