记录 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 页面布局异常。

相关推荐
imuliuliang2 小时前
存储过程(SQL)
android·数据库·sql
AgCl234 小时前
MYSQL-6-函数与约束-3/17
android·数据库·mysql
zzb15805 小时前
Fragment 生命周期深度图解:从 onAttach 到 onDetach 完整流程(面试必备)
android·java·面试·安卓
众少成多积小致巨5 小时前
Android 源码查看笔记
android·源码
angerdream5 小时前
Android手把手编写儿童手机远程监控App之前台服务
android
敲代码的瓦龙7 小时前
Android?Activity!!!
android
重生之我在安卓搞音频8 小时前
二、Android 音频框架
android·音视频
willhuo8 小时前
# 自动化数据采集技术研究与实现:基于Playwright的抖音网页自动化方案
运维·selenium·c#·自动化·chrome devtools·webview
studyForMokey9 小时前
【Android面试】Java专题 todo
android·java·面试
代码改善世界9 小时前
【MATLAB初阶】矩阵操作(二):矩阵的运算
android·matlab·矩阵