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;否则执行系统返回"的思路是正确的。只需处理好"同页打开链接"和"新窗口重定向到当前页"这两点,回退功能就能稳定工作。

相关推荐
SHEN_ZIYUAN4 小时前
Flow 责任链模式图解
android
沐怡旸6 小时前
【底层机制】LeakCanary深度解析:从对象监控到内存泄漏分析的完整技术体系
android·面试
又菜又爱coding6 小时前
Android + Flutter打包出来的APK体积太大
android·flutter
LiuYaoheng6 小时前
【Android】Drawable 基础
android·java
Jerry8 小时前
构建 Compose 界面
android
Y多了个想法9 小时前
Linux驱动开发与Android驱动开发
android·linux·驱动开发
2501_9160074711 小时前
从零开始学习iOS App开发:Xcode、Swift和发布到App Store完整教程
android·学习·ios·小程序·uni-app·iphone·xcode
姝然_952711 小时前
ConstraintLayout属性详解
android
2501_9160088912 小时前
前端工具全景实战指南,从开发到调试的效率闭环
android·前端·小程序·https·uni-app·iphone·webview