Android Jetpack Navigation组件(七):扩展知识

一、返回按键

1.物理返回按键

如果通过app:defaultNavHost="true"或者FragmentManager.setPrimaryNavigationFragment()Navigation组件监听物理返回按键事件,那么物理返回按键事件的传递如下:

ComponentActivity.onBackPressed() -> OnBackPressedDispatcher.onBackPressed() -> OnBackPressedCallback.handleOnBackPressed() -> NavController.popBackStack()

关键源码:

ComponentActivity

public void onBackPressed() {
   
    mOnBackPressedDispatcher.onBackPressed();
}

OnBackPressedDispatcher

public void onBackPressed() {
   
    Iterator<OnBackPressedCallback> iterator = mOnBackPressedCallbacks.descendingIterator();
    // 通过降序寻找 最新的且enable的OnBackPressedCallback 去处理物理返回按键事件
    while (iterator.hasNext()) {
   
        OnBackPressedCallback callback = iterator.next();
        if (callback.isEnabled()) {
   
            callback.handleOnBackPressed();
            return;
        }
    }
    // 如果没有OnBackPressedCallback处理本次物理返回按键事件,则默认调用Activity的onBackPressed()方法(Activity.finish())
    if (mFallbackOnBackPressed != null) {
   
        mFallbackOnBackPressed.run();
    }
}

NavController

// 实现onBackPressedCallback
private val onBackPressedCallback: OnBackPressedCallback =
    object : OnBackPressedCallback(false) {
   
        override fun handleOnBackPressed() {
   
        // 将栈顶的目的地弹出
            popBackStack()
        }
    }

private fun updateOnBackPressedCallbackEnabled() {
   
	// 1.enableOnBackPressedCallback由NavHostFragment是否为PrimaryNavigationFragment决定
	// 2.destinationCountOnBackStack > 1 表示NavController返回栈中的目的地大于1
	// 满足上述两个条件,NavController才会处理物理返回按键事件
    onBackPressedCallback.isEnabled = (enableOnBackPressedCallback && destinationCountOnBackStack > 1)
}

由上面的内容可以推出

  • 嵌套NavHost内只有一个目的地时,物理返回按键事件会由其父NavHost处理,弹出NavHost所在Fragment
  • 通过ComponentActivity.getOnBackPressedDispatcher().addCallback()可以拦截Navigation组件对物理返回按键事件的处理
  • 从其他应用通过隐式DeepLink导航到本应用的某个目的地,按物理返回按键会finish activity并回到其他应用的任务栈

2.Toolbar导航按钮

通过NavigationUI绑定Toolbar后,Toolbar的导航按钮事件会由Navigation组件处理。

关键源码:

NavigationUI

public fun setupWithNavController(toolbar: Toolbar,navController: NavController,
    configuration: AppBarConfiguration =
        AppBarConfiguration.Builder(navController.graph).build()
) {
   
	// 变换目的地时变换Toolbar内容
    navController.addOnDestinationChangedListener(
        ToolbarOnDestinationChangedListener(toolbar, configuration)
    )
    // 给Toolbar设置导航按钮事件
    toolbar.setNavigationOnClickListener {
    navigateUp(navController, configuration) }
}


public fun navigateUp(navController: NavController, configuration: AppBarConfiguration): Boolean {
   
    val openableLayout = configuration.openableLayout
    val currentDestination = navContr
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值