Android-jetpack-navigation工作原理剖析(一)
Navigation框架创建
如果我们要创建一个使用Navigation框架的项目,那么我们可以直接在Android studio的New Project中直接选中Bottom Navigation Activity选项 如图所示
这样我们就可以直接创建出来一个带有三个bottom的新项目了 并且相关Navigation的依赖已经在build.gradle里面自动配好了,就不用我们手动去添加了,并且这样也有利于我们顺藤摸瓜去了解navigation是如何工作的。
Navigation工作原理
在我们创建好项目后,我们点击底部三个按钮时,可以实现fragment之间来回切换,具体Navigation框架是如何实现的,让我们从MainActivity开始看
- MainActivity
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
BottomNavigationView navView = findViewById(R.id.nav_view);
// Passing each menu ID as a set of Ids because each
// menu should be considered as top level destinations.
AppBarConfiguration appBarConfiguration = new AppBarConfiguration.Builder(
R.id.navigation_home, R.id.navigation_dashboard, R.id.navigation_notifications)
.build();
NavController navController = Navigation.findNavController(this, R.id.nav_host_fragment);
NavigationUI.setupActionBarWithNavController(this, navController, appBarConfiguration);
NavigationUI.setupWithNavController(navView, navController);
}
}
整个MainActivity里实现的主要功能其实是非常明显的,拿到BottomNavigationView布局ID,通过建造者的方式来创建AppBarConfiguration,最后新建一个控制器NavController将BottomNavigationView和AppBarConfiguration配置进去,这样就可以实现点击来回切换的效果了
- R.layout.activity_main
<?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"
android:id="@+id/container"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingTop="?attr/actionBarSize">
<com.google.android.material.bottomnavigation.BottomNavigationView
android:id="@+id/nav_view"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginStart="0dp"
android:layout_marginEnd="0dp"
android:background="?android:attr/windowBackground"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:menu="@menu/bottom_nav_menu" />
<fragment
android:id="@+id/nav_host_fragment"
android:name="androidx.navigation.fragment.NavHostFragment"
android:layout_width="match_parent"
android:layout_height="match_parent"
app:defaultNavHost="true"
app:layout_constraintBottom_toTopOf="@id/nav_view"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:navGraph="@navigation/mobile_navigation" />
在activity_main布局文件里我们可以看到由两部分组成,第一部分就是BottomNavigationView,第二部分就是fragment。 第一部分需要重点留意的就是app:menu属性,这里面包含了一些图标配置显示信息等等,如果要增删改底部图标需要在里面操作,第二部分需要重点留意的就是name属性,defaultNavHost属性以及navGraph属性
name: 代表当前fragment标签真正的实现类
defaultNavHost: 是否和系统的返回键相关联,点击返回键时会判断回退栈里是否还有Fragement存在,如果有就会拦截返回键事件,如果没有,就正常执行返回键操作。
navGraph: 页面具体路由结构
- mobile_navigation
<?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"
android:id="@+id/mobile_navigation"
app:startDestination="@+id/navigation_home">
<fragment
android:id="@+id/navigation_home"
android:name="com.gcp.juejintest.ui.home.HomeFragment"
android:label="@string/title_home"
tools:layout="@layout/fragment_home" />
<fragment
android:id="@+id/navigation_dashboard"
android:name="com.gcp.juejintest.ui.dashboard.DashboardFragment"
android:label="@string/title_dashboard"
tools:layout="@layout/fragment_dashboard" />
<fragment
android:id="@+id/navigation_notifications"
android:name="com.gcp.juejintest.ui.notifications.NotificationsFragment"
android:label="@string/title_notifications"
tools:layout="@layout/fragment_notifications" />
startDestination: 默认显示页面
在navigation框架下的fragment标签里,可以添加deepLink标签,具体用法如下:
<fragment
android:id="@+id/navigation_home"
android:name="com.gcp.juejintest.ui.home.HomeFragment"
android:label="@string/title_home"
tools:layout="@layout/fragment_home">
<deepLink
android:id="@+id/deepLink"
app:uri="www.jj.com" />
</fragment>
如果有浏览器和三方应用需要拉起本项目并且具体到某个fragment或者类似的场景,那么我们就可以使用deepLink作为解决方案 在xml里面配置好deepLink标签,并且写好uri的值,在调用时直接在代码里用handleDeepLink方法调用即可
navController.handleDeepLink(getIntent());