webView 的canGoBack/goBack 回退栈

使用

Kotlin 复制代码
        // 系统返回键
        onBackPressedDispatcher.addCallback(this, object : OnBackPressedCallback(true) {
            override fun handleOnBackPressed() {
                if (dwebView?.canGoBack() == true) dwebView?.goBack()
                else {
                    isEnabled = false
                    onBackPressedDispatcher.onBackPressed()
                    isEnabled = true
                }
            }
        })

这样就可以了 ✅

但要实现"真正的回退功能",还需要注意几个关键点:

历史栈构建要点

  1. 必须确保历史栈能正常生成
  2. shouldOverrideUrlLoading 方法中对 http/https 请求要返回 false,否则无法记录历史,导致 canGoBack() 始终为 false
  3. 遇到 target="_blank"window.open 的页面,需要在 WebChromeClient.onCreateWindow 中将新窗口 URL 重定向到当前 WebView,否则不会生成历史记录

返回键处理的优化建议

  • 在 Activity 中,"没有历史"时直接调用 finish() 即可
  • 使用 dispatcher.onBackPressed() 递归处理也可以,但 finish() 更直观

推荐实现代码

kotlin 复制代码
onBackPressedDispatcher.addCallback(this, object : OnBackPressedCallback(true) {
    override fun handleOnBackPressed() {
        val wv = dwebView
        if (wv != null && wv.canGoBack()) {
            wv.goBack()
        } else {
            // Activity 处理
            finish()
            // Fragment 处理
            // findNavController().popBackStack() 或 parentFragmentManager.popBackStack()
        }
    }
})

常见问题排查

  1. 前端是 SPA 应用且使用了 history.replaceState(不增加历史记录)导致回退失效 - 需要与前端协调
  2. 拦截了 http/https 请求并返回 true 自行处理加载 - 历史记录不会增加
  3. 未处理 target="_blank" - 历史记录不增加
  4. 错误页面/重定向导致停留在同一页面(可在 onReceivedError 中跳转自定义错误页并允许后退)

Fragment 使用注意

  • 使用 addCallback(viewLifecycleOwner, ...) 绑定到视图生命周期,防止内存泄漏
  • 没有历史记录时使用 popBackStack(),不要调用 finish()

X5(TBS)内核适配

只需将类名替换为 com.tencent.smtt.sdk.*,逻辑完全一致

总结

基础框架"canGoBack → goBack;否则执行系统返回"的思路是正确的。只需处理好"同页打开链接"和"新窗口重定向到当前页"这两点,回退功能就能稳定工作。

相关推荐
石山岭16 小时前
自己动手写了一个 Android 虚拟定位 App:GPSSimulate 技术实
android·前端
杉氧18 小时前
副作用 (Side Effects) 全攻略:如何像大师一样掌控 Composable 的生命周期?
android·架构·android jetpack
Kapaseker1 天前
Kotlin Toolchain 0.11 发布:主要是把 Amper 干没了
android·kotlin
三少爷的鞋1 天前
Android 现代架构不需要事件总线进阶篇
android
杉氧2 天前
深入理解 Compose 重组机制:快照系统如何驱动 UI 精准刷新?
android·架构·android jetpack
召钱熏2 天前
状态枚举正确≠渲染正确:一个语音按钮的状态机边界修复实录
android·前端
杉氧2 天前
深度解析:Jetpack Compose 核心架构与底层原理 —— 十年安卓老兵的“破茧重生”
android·架构·android jetpack
通玄2 天前
Jetpack Compose 入门系列(七):ViewModel 与界面状态管理
android
落魄Android在线炒饭2 天前
Android Framework 开发技巧:android.jar 生成与系统快速编译验证
android
如此风景2 天前
Kotlin Flow操作符学习
android·kotlin