导航库 Navigation 小结

在 Google I/O 2018 上新出现了一个导航组件(Navigation Architecture Component),导航组件类似iOS开发里的StoryBoard,可以可视化的编辑App页面的导航关系。在经过两天的学习后,将心得总结在这里。

还没有看过官方资料的童鞋看这里:
官方文档:The Navigation Architecture Component
官方教程:Navigation Codelab

导航(Navigation)规则

  1. App需要有确定的起始点
  2. 使用一个栈来代表App的导航状态
  3. 向上按钮从不会退出你的App
  4. 在App任务中向上和返回按钮是等价的
  5. 深度链接到目标或导航到相同的目标应产生相同的堆栈

使用

  • 导航是为单Activity多Fragment的应用设计的
  • 导航似乎不能解决数据返回的问题

安装Android Studio最新的预览版 3.2 canary 14

app build.gradle中添加依赖:

    implementation "android.arch.navigation:navigation-fragment-ktx:$nav_version"
    implementation "android.arch.navigation:navigation-fragment:$nav_version" // use -ktx for Kotlin
    implementation "android.arch.navigation:navigation-ui:$nav_version" 
    implementation "android.arch.navigation:navigation-ui-ktx:$nav_version" // use -ktx for Kotlin

添加导航图(类似iOS开发中的StoryBoard):

  • 右击res目录,选择New > Android resource file
  • New Resource对话中输入文件名nav_graph,选择Resource typeNavigation
    点击OK后IDE会在navigation目录下生成nav_graph.xml文件:
<?xml version="1.0" encoding="utf-8"?>
<navigation xmlns:android="http://schemas.android.com/apk/res/android">
</navigation>
导航图编辑器

在Activity布局中指定Navigation的宿主(Host):

?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.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">

    <fragment
        android:id="@+id/my_nav_host_fragment"
        android:name="androidx.navigation.fragment.NavHostFragment"
        app:navGraph="@navigation/nav_graph"
        app:defaultNavHost="true"
        />

</android.support.constraint.ConstraintLayout>

其中,fragment的name一定要是androidx.navigation.fragment.NavHostFragment,app:navGraph输入刚刚生成的导航图位置

覆写onSupportNavigateUp()方法:

@Override
public boolean onSupportNavigateUp() {
    return Navigation.findNavController(this, R.id.nav_host_fragment).navigateUp();
}

添加导航的起始位置和目的位置:

打开导航图编辑器,点击新增按钮,可以新建一个空白目的地或者选择已有的Fragment或者Activity,完成后页面的预览图就会显示在编辑器里,同时IDE会给它指定默认的id等属性。点击Fragment右边的手柄不放开,将它拖动到另一个页面,一个操作(Action)就创建好了,IDE会给它分配一个默认的action id。在起始Fragment上点击右键,选择Set Start Destination,将它设置为起始位置,当宿主(Host)Activity启动的时候,它会做为默认的页面替换布局中的NavHostFragment。

导航图新增目的地默认是新增Fragment,可以指定启动模式,可以指定切换动画、可以指定参数及其类型。点击箭头可以更改这些参数。也通过使用安全类型插件来生成对应的代码来保证参数类型安全:
project gradle:

buildscript {
    repositories {
        jcenter()
        google()
    }
    dependencies {
        classpath 'android.arch.navigation:navigation-safe-args-gradle-plugin:1.0.0-alpha01'
    }
}

app gradle:

apply plugin: 'androidx.navigation.safeargs'

导航到目的地

使用NavController
来发起页面跳转,可以通过以下方法获取NavController:

获取到NavController后,就可以通过它的navigate()方法发起页面跳转,navigate()接受action id 或 fragment id 以及导航选项及Bundle参数等作为参数。

创建导航选项:

val options = NavOptions.Builder()
    .setEnterAnim(R.anim.slide_in_right)
    .setExitAnim(R.anim.slide_out_left)
    .setPopEnterAnim(R.anim.slide_in_left)
    .setPopExitAnim(R.anim.slide_out_right)
    .build()

通过指定的action来跳转页面:

Navigation.findNavController(view).navigate(R.id.viewTransactionsAction);

同一个导航图里可以有多个相同id的action。

创建一个跳转的OnClickListener:

button.setOnClickListener(Navigation.createNavigateOnClickListener(R.id.next_fragment, null));

绑定目的跳转和 Menu Item

要快捷的绑定Menu Item跳转和指定的页面,要保证目的地fragment id 和 item id 一致

// 导航图中的目的地
<fragment android:id="@+id/details_page_fragment"
     android:label="@string/details"
     android:name="com.example.android.myapp.DetailsFragment" />
// 目录项
<item
    android:id="@id/details_page_fragment"
    android:icon="@drawable/ic_details"
    android:title="@string/details" />
// 溢出菜单目录项
<item
    android:id="@id/details_page_fragment"
    android:icon="@drawable/ic_details"
    android:title="@string/details"
    android:menuCategory:"secondary" />

绑定NavigationView跳转

NavigationView navigationView = (NavigationView) findViewById(R.id.nav_view);
NavigationUI.setupWithNavController(navigationView, navController); 

其中navigationView 可以是NavigationView、BottomNavigationView等

绑定Menu Item跳转:

override fun onOptionsItemSelected(item: MenuItem): Boolean {
        // Have the NavHelper look for an action or destination matching the menu
        // item id and navigate there if found.
        // Otherwise, bubble up to the parent.
        return NavigationUI.onNavDestinationSelected(item,
                Navigation.findNavController(this, R.id.my_nav_host_fragment))
                || super.onOptionsItemSelected(item)
}

在目的地之前传输数据

先在导航图中创建要接收参数,然后在代码中用Bundle传数据:

Bundle bundle = new Bundle();
bundle.putString("amount", amount);
Navigation.findNavController(view).navigate(R.id.confirmationAction, bundle);

获取参数:

TextView tv = view.findViewById(R.id.textViewAmount);
tv.setText(getArguments().getString("amount"));

使用类型安全插件传送参数,这里假设你要从叫SpecifyAmountFragment跳转ConfirmationFragment并传送数据,同时跳转的action id为confirmationAction:

@Override
public void onClick(View view) {
   EditText amountTv = (EditText) getView().findViewById(R.id.editTextAmount);
   int amount = Integer.parseInt(amountTv.getText().toString());
   ConfirmationAction action =
           SpecifyAmountFragmentDirections.confirmationAction()
   action.setAmount(amount)
   Navigation.findNavController(view).navigate(action);
}

获取参数:

@Override
public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
    TextView tv = view.findViewById(R.id.textViewAmount);
    int amount = ConfirmationFragmentArgs.fromBundle(getArguments()).getAmount();
    tv.setText(amount + "")
}

将目的地组合到一个嵌套的导航图中

按住Shift点击多个目的地,在他们上面点击右键,选择Move to Nested Graph > New Graph

为目标分配深层链接

在导航图编辑器中选中目的地后在Attributes编辑器中添加,或者在xml文件中对应Fragment下添加:

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

推荐阅读更多精彩内容