Android实现底部导航栏方法(Navigation篇)

Navigation实现底部导航栏

  • 前言
  • 导入和基本使用
    • 导入
    • 基础使用
      • 创建nav文件
      • 编辑Nav文件
        • 添加页面(代码版)
        • 添加页面(图解版)
      • 创建导航动作 action
        • 创建action(代码版)
        • 创建action(图解版)
      • 编辑action参数
        • launchSingleTop
        • popUpTo
        • popUpToInclusive
        • popUpToSaveState
        • restoreState
      • 使用nav文件
      • 跳转Fragment
  • 底部导航栏实现方法
    • 创建nav文件
    • 点击导航
  • 结语

前言

底部导航栏一直是大部分App不可缺失的一部分
最近注意到Jetpack中的Navigation支持Fragment的切换操作
特此浅研究一下

导入和基本使用

选择性跳过

导入

此处使用Google开发者文档中介绍

dependencies {def nav_version = "2.5.3"// Java使用这两行导入implementation "androidx.navigation:navigation-fragment:$nav_version"implementation "androidx.navigation:navigation-ui:$nav_version"// Kotlin使用这两行导入implementation "androidx.navigation:navigation-fragment-ktx:$nav_version"implementation "androidx.navigation:navigation-ui-ktx:$nav_version"// 多模块使用implementation "androidx.navigation:navigation-dynamic-features-fragment:$nav_version"// 测试使用androidTestImplementation "androidx.navigation:navigation-testing:$nav_version"// Jetpack Compose使用implementation "androidx.navigation:navigation-compose:$nav_version"
}

基础使用

使用nav文件配合 FragmentContainerView组件 实现Fragment的切换操作

创建nav文件

导入后,在项目的res文件夹下,右键选择Android Resource File,弹出弹窗在这里插入图片描述Resource type下拉选择Navigation即可,剩下的就是填写文件名
完成后会在res文件下创建一个navigation文件夹 ,里面就存放着nav文件

编辑Nav文件

打开Nav文件,可以看到顶部有一排按钮
在这里插入图片描述

分别是

  • 添加页面 (Fragment、Activiry、include)
  • 创建分组 (选择一个或多个页面进行分组)
  • 设置初始页 (选择一个页面,设置为初始页,即默认页,设置为默认页的页面左上角会出现一个房子图标)
  • 创建depplink (选择一个页面创建深层链接)
  • 添加 action (选择一个页面 添加跳转动作)
  • 整理布局 (全部页面重排,优化布局)

首先使用添加页面 添加三个已经写好的Fragment
当然使用写代码的方式也是可以的

添加页面(代码版)
<?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"xmlns:tools="http://schemas.android.com/tools"app:startDestination="默认页id"><!--例--><fragmentandroid:id="@+id/标识此fragment的id"android:name="fragment类文件路径"android:label="fragment名称"tools:layout="fragment对应的layout" /></navigation>
添加页面(图解版)

在这里插入图片描述
可以通过搜索框搜索,点击需要添加的页面即可
此时在代码处会生成一个fragment标签

在这里插入图片描述
如图所示 成功添加了一个页面
如果需要添加页面预览 则在fragment处添加标签 layout 值为 fragment对应的layout
在这里插入图片描述

创建导航动作 action

action是跳转到fragment的关键要素

创建action(代码版)

最基本的写法

<actionandroid:id="@+id/标识此action的id"app:destination="@id/目的地的fragment Id" />
创建action(图解版)

在这里插入图片描述点击起始的fragment,右边有一个可以拖动的箭头,将箭头拖至目的地fragment即可
上述操作完成后会在 testFragment 中生成一段action标签
当然action的内容不止这些

编辑action参数

通过查看NavAction的源码参数 可以看到action有数个标签可以定义
(按住Ctrl+鼠标点击action中的destination属性即可)

 <declare-styleable name="NavAction"><attr name="android:id"/><!-- destination  目的地的id  --><attr format="reference" name="destination"/><attr format="boolean" name="launchSingleTop"/><attr format="boolean" name="restoreState"/><attr format="reference" name="popUpTo"/><attr format="boolean" name="popUpToInclusive"/><attr format="boolean" name="popUpToSaveState"/><attr format="reference" name="enterAnim"/><attr format="reference" name="exitAnim"/><attr format="reference" name="popEnterAnim"/><attr format="reference" name="popExitAnim"/></declare-styleable>

同时可以通过右侧Attributes 页进行参数查看

launchSingleTop

默认false,类似Activity的singleTop
设为true后 activity的singleTop会判断顶部的activity是否为当前activity,是则复用,否则新建
navigation的singleTop会判断顶部的fragment是的为目的地fragment ,是则销毁顶部,重新创建放置在顶部
可见图 唯一的区别就是 执行了action动作后有无删除旧fragment
在这里插入图片描述
此处使用了以下action

 <actionandroid:id="@+id/action_testFragment_self_singleTop"app:launchSingleTop="true"app:destination="@id/testFragment" /><actionandroid:id="@+id/action_testFragment_self"app:destination="@id/testFragment" />
popUpTo

默认为空
设为某个fragment的id后 执行此action 会挨个出栈 直到出栈的fragment为popUpTo指定的fragment (此fragment不出)然后再创建 目的地fragment
以下为图解在这里插入图片描述主要看右下角处 2->1 popUpTo = 1
当popUpTo指定1后 会把所有不是 1 的fragment出栈,再在旧的1上面入栈新的1
如下图,即使多个1存在,只会弹出最上层的1之上的fragment
在这里插入图片描述

popUpToInclusive

默认false
结合popUpTo使用,当popUpToInclusive为true的时候,会把旧的1也出栈
如下图,区别与上图 本次连黄色的1都出栈了
在这里插入图片描述

popUpToSaveState

默认false
结合popUpTo使用
设为true后 popUpTo操作弹出的fragment 都会保存状态
以便restoreState 恢复操作

restoreState

默认false
结合popUpToSaveState使用
设为true后 还原目的地fragment的状态
如果之前没有保存状态 此参数不起效
在这里插入图片描述
如图 当popUpToSaveState为true后 弹出的fragment会保存到一个Map内
之后再调用action
action中restoreState为true
action的目的地为2
此时就会取出map中ID为2的fragment 重新放进栈中
取出顺序为 先popup的后入栈 也就先显示 顺序和popup前一样
需要注意的是 这样子的状态保存实际上需要view model配合使用 ,当fragment销毁(onDestroy)后,fragment绑定的viewmodel没有跟着销毁
此时恢复状态,fragment依旧会onCreate,就需要从view model中获取数据,所以数据需要保存在viewmodel才是最优选

  • 当 popUpTo和起始idfragment不同时,会发生不同情况
  • popUpToInclusive 会影响状态恢复
    在这里插入图片描述
    如图,当fragment3跳转至fragment4时 弹出2以上的所有fragment,此时两个fragment3都会保存状态,直到fragment4跳至fragment2,并使用restoreState=true属性后,会把两个fragment3恢复,这时就与之前冲突了。
    这时再把popUpToInclusive改为true,就会发生以下情况
    在这里插入图片描述
    可以看到,把popUpToInclusive设为true后,弹出了fragment2以上所有页面,包括fragment2自己,在随后的恢复中,原先的fragment3被置顶,明明是4->2却显示3!
    原因未知,如果上面描述有错误,或者有更好的见解,欢迎评论区讨论。

使用nav文件

这里需要使用官方的组件进行fragment的显示,具体步骤如下 在activity的layout中添加

   <androidx.fragment.app.FragmentContainerViewandroid:id="@+id/main_fragment_container"android:name="androidx.navigation.fragment.NavHostFragment"android:layout_width="match_parent"android:layout_height="match_parent"app:defaultNavHost="true"app:navGraph="@navigation/nav文件名" />

这里有两个属性需要说明

app:defaultNavHost:设为false时,返回就退出Activity 设为true时,返回就是fragment出栈
app:navGraph:设置nav文件

跳转Fragment

引用官方文档中的话

使用 FragmentContainerView 创建 NavHostFragment,或通过 FragmentTransaction 手动将 NavHostFragment 添加到您的 Activity 时,尝试通过 Navigation.findNavController(Activity, @IdRes int) 检索 Activity 的 onCreate() 中的 NavController 将失败。您应改为直接从 NavHostFragment 检索 NavController。

简单来说就是,先尝试下列方法获取控制器
Kotlin:

Fragment.findNavController()
View.findNavController()
Activity.findNavController(viewId: Int)

Java:

NavHostFragment.findNavController(Fragment)
Navigation.findNavController(Activity, @IdRes int viewId)
Navigation.findNavController(View)

如果报错,获取不了,应该改为

   val navHostFragment =supportFragmentManager.findFragmentById(R.id.main_fragment_container) asNavHostFragmentval controller = navHostFragment.navController
      NavHostFragment f = (NavHostFragment) getSupportFragmentManager().findFragmentById(R.id.main_fragment_container);NavController controller;if (f != null) {controller = f.getNavController();}

Activity所继承的必须为 AppCompatActivity
拿到控制器后,只需要

 controller.navigate( action的ID )

即可完成页面跳转,例:

    <fragmentandroid:id="@+id/testFragment"android:name="com.a.demo.ui.nav.TestFragment"android:label="TestFragment"tools:layout="@layout/fragment_test"><actionandroid:id="@+id/action_testFragment_to_test2Fragment"app:destination="@id/test2Fragment" /></fragment><fragmentandroid:id="@+id/test2Fragment"android:name="com.a.demo.ui.nav.Test2Fragment"android:label="Test2Fragment"tools:layout="@layout/fragment_test2"></fragment>
controller.navigate(R.id.action_testFragment_to_test2Fragment)

这样就完成了从testFragment跳转至test2Fragment的操作

底部导航栏实现方法

假如现在底部导航栏有五个按钮和五个fragment

创建nav文件

	<fragmentandroid:id="@+id/mainFragment1"android:name="com.a.demo.ui.activity.main.MainFragment1"android:label="MainFragment1"tools:layout="@layout/fragment_1" /><fragmentandroid:id="@+id/mainFragment2"android:name="com.a.demo.ui.activity.main.MainFragment2"android:label="MainFragment2" /><fragmentandroid:id="@+id/mainFragment3"android:name="com.a.demo.ui.activity.main.MainFragment3"android:label="MainFragment3" /><fragmentandroid:id="@+id/mainFragment4"android:name="com.a.demo.ui.activity.main.MainFragment4"android:label="MainFragment4" /><fragmentandroid:id="@+id/mainFragment5"android:name="com.a.demo.ui.activity.main.MainFragment5"android:label="MainFragment5" />

点击导航

在activity处,设置五个点击事件 分别对应五个按钮(此处不展示详细代码)

//获取控制器val navHostFragment = supportFragmentManager.findFragmentById(R.id.main_fragment_container) as NavHostFragmentval controller = navHostFragment.navController//设置导航配置val builder = NavOptions.Builder().setLaunchSingleTop(true).setRestoreState(true)builder.setPopUpTo(controller.graph.findStartDestination().id,inclusive = false,saveState = true)//设置点击事件vb.but1.setOnClickListener {controller.navigate(R.id.mainFragment1,null,builder.build())}vb.but2.setOnClickListener {controller.navigate(R.id.mainFragment2,null,builder.build())}vb.but3.setOnClickListener {controller.navigate(R.id.mainFragment3,null,builder.build())}vb.but4.setOnClickListener {controller.navigate(R.id.mainFragment4,null,builder.build())}vb.but5.setOnClickListener {controller.navigate(R.id.mainFragment5,null,builder.build())}

解释一下上面代码

navigate方法可以传fragment的id直接跳转,而不使用action ID,这时等同于 当前fragment->传递的fragment 在这里插入图片描述
NavOptions.Builder是导航配置,等同于action中其他参数 ,但有更高的自定义程度,相当于动态控制
controller.graph.findStartDestination().id 可以拿到当前当前fragment ID

此时的activity的布局需要修改

app:defaultNavHost=“false”

这种导航栏方式,fragment1始终被压在栈底,如果将返回键交予fragment分发,就会出现先退到fragment1再退出activity的情况

  <androidx.fragment.app.FragmentContainerViewandroid:id="@+id/main_fragment_container"android:name="androidx.navigation.fragment.NavHostFragment"android:layout_width="match_parent"android:layout_height="match_parent"app:defaultNavHost="false"app:navGraph="@navigation/nav_main" />

结语

到此,Nav自定义导航栏已经实现,基本使用的模块来源日常使用经验。
至于底部导航栏,网上大部分人都推荐使用 BottomNavigationView 配合使用

  <com.google.android.material.bottomnavigation.BottomNavigationViewandroid:id="@+id/main_bottomNavigationView"android:layout_width="match_parent"android:layout_height="@dimen/dp_50"app:menu="@menu/menu_main" />

好用,但自定义样式比较难,然后就只能翻BottomNavigationView的源码,看它是怎么实现切换页面而不销毁fragment
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/682642.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

「C++ 类和对象篇 12」static成员

目录 一、static成员是什么&#xff1f; 二、为什么需要static成员&#xff1f; 三、怎么使用static成员&#xff1f; 1. 定义static成员变量 2. 定义static成员函数 3. 访问static成员 四、特性 【面试题】 【总结】 一、static成员是什么&#xff1f; 被static修饰的类成…

品牌之门:概率与潜力的无限延伸

在品牌的世界里&#xff0c;每一个成功的推广都像是打开一扇门&#xff0c;从未知走向已知&#xff0c;从潜在走向显现。这扇门&#xff0c;既是品牌的起点&#xff0c;也是品牌发展的无限可能。 品牌&#xff0c;就像一扇紧闭的门&#xff0c;它静静地矗立在那里&#xff0c;…

微信强制分享红包裂变系统源码

这是一款新型的微信裂变引流系统源码&#xff0c;支持试看、直播、朋友圈转发、分享任务、 邀请入群、群聊、红包等多种裂变引流方式&#xff0c;让你的广告流量引流、吸粉变现更加高效。 该系统源码还优化了整个页面&#xff0c;减少了繁重多余的代码&#xff0c;让访问速度…

【51单片机】直流电机驱动(PWM)(江科大)

1.直流电机介绍 直流电机是一种将电能转换为机械能的装置。一般的直流电机有两个电极,当电极正接时,电机正转,当电极反接时,电机反转 直流电机主要由永磁体(定子)、线圈(转子)和换向器组成 除直流电机外,常见的电机还有步进电机、舵机、无刷电机、空心杯电机等 2.电机驱动…

对(一维)数组与指针的深入理解(1)

目录 1.数组名的理解2.使用指针访问&#xff08;一维&#xff09;数组3.&#xff08;一维&#xff09;数组传参的本质 1.数组名的理解 以前我们在使用指针访问数组内容时&#xff0c;有这样的代码&#xff1a; #include <stdio.h>int main() {int arr[10] { 1,2,3,4,5…

详解Qt多线程(包含:什么是CPU,单核处理器和多核处理器,举餐厅和QQ音乐的例子详解进程和线程,Qt多线程案例)

目录 一.什么是CPU&#xff1f;二.单核处理器与多核处理器三.什么是进程和线程&#xff1f;3.1 定义3.2 以餐厅为例子解释进程和线程3.2 以QQ音乐为例子&#xff0c;解释QQ音乐里面的进程和线程 四.Qt中的多线程五.Qt多线程案例任务描述案例演示设置显示内容的字体大小和位置运…

pands常用操作

1.导入库和文件读取和文件分信息分析 import pandas as pd import numpy as np csvf pd.read_csv(D:/各个站程序版本说明.csv) csvf.info() <class pandas.core.frame.DataFrame> RangeIndex: 51 entries, 0 to 50 Data columns (total 6 columns):# Column Non-Nul…

java面试题整理

2023.2.14&#xff08;第二天&#xff09; 数组是不是对象&#xff1f; 在Java中&#xff0c;数组是对象。数组是一种引用类型&#xff0c;它可以存储固定大小的相同类型的元素序列。在Java中&#xff0c;数组是通过new关键字创建的&#xff0c;它们在内存中被分配为对象&…

「数据结构」MapSet

&#x1f387;个人主页&#xff1a;Ice_Sugar_7 &#x1f387;所属专栏&#xff1a;Java数据结构 &#x1f387;欢迎点赞收藏加关注哦&#xff01; Map&Set &#x1f349;概念&#x1f349;模型&#x1f349;Map&#x1f34c;TreeMap和HashMap的区别&#x1f34c;Map常用方…

2048游戏C++板来啦!

个人主页&#xff1a;PingdiGuo_guo 收录专栏&#xff1a;C干货专栏 大家好呀&#xff0c;我是PingdiGuo_guo&#xff0c;今天我们来学习如何用C编写一个2048小游戏。 文章目录 1.2048的规则 2.步骤实现 2.1: 初始化游戏界面 2.1.1知识点 2.1.2: 创建游戏界面 2.2: 随机…

Days 31 ElfBoard 自启脚本中打开看门狗

1.在开机自启脚本中打开看门狗 rootELF1:~# vi /etc/rc.local 2.在自启脚本中添加上之后&#xff0c;然后在咱们的QT界面中找到看门狗应用&#xff0c; 发现显示打开看门狗失败&#xff1a; 3.修改看门狗源码&#xff0c;设置了超时时间后&#xff0c;关闭/dev/dev/watchdog节…

【Tomcat】:One or more listeners failed to start.报错解决方案

报错信息:One or more listeners failed to start. Full details will be found in the appropriate container log file. 具体就是web.xml此配置报错: 服务器启动错误Tomcat:One or more listeners failed to start.报错解决方案 IDEA:在使用IDEA运行SSM项目的时候 , Tomcat运…

error MSB8008: 指定的平台工具集(v143)未安装或无效。请确保选择受支持的 PlatformToolset 值解决办法

右击解决方案&#xff0c;选择属性 将工具集为143的修改为其他&#xff0c;如图 重新编译即可运行

网络原理(3)--以太网协议,DNS

&#x1f495;"Echo"&#x1f495; 作者&#xff1a;Mylvzi 文章主要内容&#xff1a;网络原理(3)–以太网协议,DNS 在网络原理(2)中介绍了网络层中的一个重要的协议–ip协议,网络层关注的通信时的起点和终点,而数据链路层更加"底层"一些,关注的是传输过程…

【Effective Objective - C 2.0】——读书笔记(四)

文章目录 二十三、通过委托与数据源协议进行对象间通信二十四、将类的实现代码分散到便于管理的数个分类之中二十五、总是为第三方的分类名称加前缀二十六、切勿在分类里面声明属性二十七、使用“class-continuation分类”隐藏实现细节二十八、通过协议提供匿名对象 二十三、通…

springboot187社区养老服务平台的设计与实现

简介 【毕设源码推荐 javaweb 项目】基于springbootvue 的 适用于计算机类毕业设计&#xff0c;课程设计参考与学习用途。仅供学习参考&#xff0c; 不得用于商业或者非法用途&#xff0c;否则&#xff0c;一切后果请用户自负。 看运行截图看 第五章 第四章 获取资料方式 **项…

MySQL 基础知识(六)之数据查询(一)

目录 1 基本查询 1.1 查询相关列 (select * / 列名) 1.2 别名 (as) 1.3 去重 (distinct) 1.4 对列中的数据进行运算 (、-、*、/) 2 条件查询 (where) 2.1 等值查询 () 2.2 非等值查询 (>、<、>、<、!、><) 2.3 逻辑判断 (and、or、not) 2.4 区间判…

源码推荐:hello-algo @ github

github https://github.com/krahets/hello-algo 本项目旨在创建一本开源、免费、对新手友好的数据结构与算法入门教程。全书采用动画图解&#xff0c;结构化地讲解数据结构与算法知识&#xff0c;内容清晰易懂&#xff0c;学习曲线平滑。算法源代码皆可一键运行&#xff0c;支…

上位机图像处理和嵌入式模块部署(借鉴与学习)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 对于很多学院派的同学来说&#xff0c;他们对市场的感觉一般是比较弱的。如果写一个软件的话&#xff0c;或者说开发一个项目的话&#xff0c;他们…

(一)【Jmeter】JDK及Jmeter的安装部署及简单配置

JDK的安装和环境变量配置 对于Linux、Mac和Windows系统&#xff0c;JDK的安装和环境变量配置方法略有不同。以下是针对这三种系统的详细步骤&#xff1a; 对于Linux系统&#xff1a; 下载适合Linux系统的JDK安装包&#xff0c;可以选择32位或64位的版本。 将JDK的安装包放置…