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

相关推荐
TheNextByte19 小时前
如何将Android短信导出到CSV/TEXT/Excel
android·excel
泡泡以安12 小时前
【Android逆向工程】第3章:Java 字节码与 Smali 语法基础
android·java·安卓逆向
一笑的小酒馆18 小时前
Android launcher3实现简单的负一屏功能
android
xuyin120418 小时前
【Android】Flow基础知识和使用
android
李新_20 小时前
基于Markwon封装Markdown组件
android·aigc·markdown
Non-existent9871 天前
Flutter + FastAPI 30天速成计划自用并实践-第10天-组件化开发实践
android·flutter·fastapi
@老蝴1 天前
MySQL数据库 - 约束和联合查询
android·数据库·mysql
ljt27249606611 天前
Compose笔记(六十一)--SelectionContainer
android·笔记·android jetpack
有位神秘人1 天前
Android中Compose系列之按钮Button
android
AI科技摆渡1 天前
GPT-5.2介绍+ 三步对接教程
android·java·gpt