背景介绍
在 uni-app 开发中,弹窗、抽屉、下拉菜单等覆盖型组件是非常常见的交互元素。这些组件通常需要相对于视口定位,不受父元素影响。然而,在某些场景下 Position: Fixed 会有不符合预期的表现题。而且 uni-app 一次编码,多端报错的特性,导致这个问题十分棘手,因此今天我们就要探讨一个跨端的解决方案来处理这些定位问题。
CSS Position: Fixed 的失效问题
问题描述
根据 CSS 规范,position: fixed
元素的定位上下文默认是相对于视口(viewport)的。但在以下情况下,定位上下文会发生改变:
- Transform 属性 :当祖先元素应用了
transform
属性时 - Filter 效果 :当祖先元素设置了
filter
或backdrop-filter
属性时 - 3D 渲染上下文 :当祖先元素设置了
perspective
属性时 - will-change :当祖先元素的
will-change
属性设置为上述值时
这种行为在 MDN 文档中有明确说明:
"当元素祖先的 transform、perspective、filter 或 backdrop-filter 属性非 none 时,容器由视口改为该祖先。"
------ MDN - position: fixed
实际开发中的影响
这个问题在实际开发中经常会带来以下困扰:
-
模态框定位异常
- 在带有变换效果的容器中,模态框无法相对于视口居中
- 弹窗位置会随父容器滚动而改变
-
固定导航失效
- 使用 CSS transform 实现动画效果的页面中,固定导航栏会失去固定效果
- 在滚动时导航栏可能会跟随内容移动
-
交互组件错位
- 下拉菜单、提示框等定位不准确
- 遮罩层无法完全覆盖视口
问题示例
html
<div style="transform: scale(1);">
<!-- fixed 定位将相对于这个 div,而不是视口 -->
<div style="position: fixed; top: 0; left: 0;">
这个元素不会固定在视口顶部
</div>
</div>
uni-app Vue3 中的跨平台解决方案
我们期望针对 uni-app Vue3 提供了一个优雅的跨平台解决方案。通过条件编译和平台特定的实现,组件能够在不同端完美运行。核心思路是将内容传送到应用根节点,从而避免中间层级的 CSS 上下文影响。
条件编译实现
使用 uni-app 的条件编译特性,我们可以为不同平台提供最优的实现方案:
vue
<!-- #ifdef H5 -->
<teleport to="body">
<slot />
</teleport>
<!-- #endif -->
<!-- #ifdef MP-WEIXIN || MP-ALIPAY -->
<root-portal>
<slot />
</root-portal>
<!-- #endif -->
各端实现原理与细节
1. H5 环境 - Teleport 实现原理
Vue3 的 teleport
组件实现了一个传送门的概念,其工作原理如下:
- 在组件的虚拟 DOM 树中正常渲染内容
- 但在实际 DOM 操作时,将内容移动到指定目标
ts
// teleport 在 uni-app H5 端的工作方式
// 1. 组件逻辑
const show = ref(false)
// 2. 模板中使用
<teleport to="body">
<view v-if="show" class="popup">
<slot />
</view>
</teleport>
优点:
- 完全复用 Vue3 的能力
- 支持动态目标节点
- 保持组件状态和事件绑定
2. 小程序环境 - root-portal 实现原理
小程序的 root-portal
组件可以使整个子树从页面中脱离出来。
html
<root-portal>
<view class="popup">
<slot />
</view>
</root-portal>
3. App 环境 - renderjs 实现原理
App 端使用 renderjs
实现节点操作,这是一个强大的跨平台解决方案:
- 直接运行在视图层(Webview)中
- 可以访问完整的浏览器 API
- 支持直接 DOM 操作
ts
// App 端的实现
<script module="render" lang="renderjs">
export default {
mounted() {
// 获取根节点
const root = document.querySelector('uni-app') || document.body
if (this.$ownerInstance.$el) {
root.appendChild(this.$ownerInstance.$el)
}
}
}
</script>
优点:
- 直接 DOM 操作,灵活控制
- 可以精确控制节点的生命周期
- 保持完整的事件系统
统一封装实现
为了统一管理这三种实现方式,我们会将其封装一个统一的组件,在 WotUI 组件库中提供。
这个统一封装:
- 使用条件编译区分平台
- 保持一致的 API 和使用方式
- 解决了跨平台兼容性问题
- 支持微信小程序、支付宝小程序、APP和h5
扩展阅读
CSS 规范与文档
框架与工具
相关技术文章
最后
关注公众号【阿鱼聊前端】,爱摸鱼,不迷路。