记录 Android WebView内核更新,安全区 和 Insets 消费问题

大概2026刚过完年,线上 H5 页面中原本正常的弹窗,突然在底部多出了几十像素的透明空白。

第一反应是 Android 侧 Insets 的适配问题------但 Android 的 WebView 应该没有 iOS 那样原生支持 env(safe-area-inset-bottom) CSS 变量,按理说不应该有这个问题。

带着这个疑问开始排查。

排查过程

项目里使用的是 van-action-sheet 组件,通过 chrome://inspect 调试页面,在 DevTools 中检查样式,发现 padding-bottom: env(safe-area-inset-bottom); 是生效的, padding-bottom 的值不是 0

这说明 Android WebView 已经将 Window Insets 信息注入到了 CSS 环境变量中。

翻阅 Android 官方文档后找到了答案:

了解窗口边衬区 | Android Developers

Android WebView 内核在 144 版本中会将系统窗口的 Inset CSS 环境变量的形式透传给 WebView 内的页面。一旦 Native 层没有正确"消费"这些 Insets,WebView 就会感知到它们,并反映到 safe-area-inset-* 变量上。

问题根因

常见的 Insets 处理写法如下:

kotlin 复制代码
ViewCompat.setOnApplyWindowInsetsListener(findViewById(android.R.id.content)) { v, insets ->
    val barInsets = insets.getInsets(WindowInsetsCompat.Type.navigationBars())
    v.updatePadding(bottom = barInsets.bottom)
    insets // 直接返回原始 insets,未消费
}

这段代码虽然给 View 加上了底部 padding,但 将原始 insets 原封不动地返回,意味着这些 Insets 没有被消费(consume)。子 View(包括 WebView)仍然可以感知到完整的 Insets 值,进而影响 CSS 环境变量。

解决方案

参考官方文档:Avoid ghost padding by zeroing insets

在处理完 Insets 后,构建一个新的 WindowInsetsCompat,将已处理的 Insets 类型置为 NONE,从而告知系统这部分 Insets 已被当前层级消费:

kotlin 复制代码
ViewCompat.setOnApplyWindowInsetsListener(findViewById(android.R.id.content)) { v, insets ->
    val types = WindowInsetsCompat.Type.navigationBars() or WindowInsetsCompat.Type.ime()
    val barInsets = insets.getInsets(types)
    v.updatePadding(bottom = barInsets.bottom)

    // 消费 navigationBars 和 ime,避免 WebView 感知到这些 Insets
    WindowInsetsCompat.Builder(insets)
        .setInsets(types, Insets.NONE)
        .build()
}

改动后,WebView 内 safe-area-inset-bottom 恢复为 0,弹窗底部的透明空白消失。

总结

对比项 修复前 修复后
Insets 返回值 原始 insets(未消费) 构建新的 WindowInsetsCompat(已消费)
WebView CSS 变量 safe-area-inset-bottom 非零 safe-area-inset-bottom 为 0
页面表现 弹窗底部出现多余透明空白 正常显示

结论: 在 Android 中,如果 Native 层自己处理了导航栏/IME 的 Insets,必须在监听器中返回消费后的 WindowInsetsCompat,否则 WebView 会继承这些 Insets 并将其映射到 CSS 的 safe-area-inset-* 环境变量,导致 H5 页面布局异常。

相关推荐
ysh98882 小时前
2025年 Android Studio修仙传(kotlin版):基础篇
android·kotlin·android studio
XiaoLeisj2 小时前
Android 网络编程入门到实战:HttpURLConnection、JSON 处理、OkHttp 与 Retrofit2
android·网络·okhttp·json·gson·retrofit2·jsonobjecy
fengci.2 小时前
ctfshow36D杯
android
艾莉丝努力练剑3 小时前
【MYSQL】MYSQL学习的一大重点:MYSQL库的操作
android·linux·运维·数据库·人工智能·学习·mysql
胖大师3 小时前
Android 构建系统详解
android
安卓程序员_谢伟光3 小时前
安卓内存分析
android·jvm
符哥20089 小时前
新能源智能充电桩与 Android/iOS App 蓝牙通信协议
android·ios
JMchen1239 小时前
自定义View性能优化:从60fps到120fps的进阶之路
android·经验分享·性能优化·kotlin·自定义view
vistaup10 小时前
DevEco Studio 鸿蒙 HAR本地引入相互依赖问题解决
android·华为·harmonyos