根本原因是 transform、opacity 或 will-change 等属性意外触发层叠上下文,导致子菜单 z-index 仅在局部生效;iPad Safari 对层叠上下文创建更敏感,需检查并移除相关属性或显式重置父容器层叠行为。
HTML5 页面在 iPad 上菜单层级错乱,根本原因几乎总是 z-index 计算被 transform、opacity 或 will-change 等属性意外触发的层叠上下文(stacking context)打断,而非 z-index 值本身写错了。
iPad(尤其是 iOS 15+)Safari 对层叠上下文的创建更敏感,且渲染管线对合成层(composited layer)的判断逻辑与桌面不同。一个看似无害的 transform: translateZ(0) 或 opacity: 0.99 就会让父元素提前建立新的层叠上下文,导致其子菜单的 z-index 只在该局部上下文中生效,无法盖过页面其他区域的元素。
transform(哪怕只是 translate3d(0,0,0))、opacity、filter、will-change
z-index 是否显示为 auto,并留意「Stacking Context」提示z-index 数值有效得多常见于使用 position: fixed 或 position: absolute 的下拉菜单,实际渲染时被轮播图、导航栏或模态框的某个祖先节点截断了层叠范围。
z-index: 0 并设 position: relative,强制它成为「中性」层叠容器backface-visibility: hidden —— 它在 iOS Safari 中会隐式触发层叠上下文 底部,而非嵌套在某个低 z-index 的模块内部别依赖 @media (max-width: 1024px) 或 screen and (orientation: portrait) 来单独修复 iPad 层级问题——这些规则在桌面浏览器缩放时也会命中,造成误判;而且 iPad Pro(12.9")的 viewport 宽度可能达 1366px,根本不会进规则。
userAgent + platform,但仅用于加载轻量级修复 CSS(不用于核心布局)z-index: 9999 级别给所有可交互浮层(菜单、tooltip、toast),并在全局重置关键父容器的层叠行为body > * { transform: none !important; opacity: 1 !important; filter: none !important; },再针对性放开需要动效的元素,能快速定位是否是某处全局样式污染所致最常被忽略的是:iOS Safari 对 z-index 的解析严格依赖 DOM 顺序 + 层叠上下文链。即使数值再大,一旦被包进一个意外创建的上下文里,就彻底出局。动手前先打开 Web Inspector 看「Layers」面
