uni-app 全局容器实战系列(一):全局容器的实现

uni-app 全局容器实战系列(一):全局容器的实现

系列总览

章节脉络

本系列围绕 全局容器 这一主题,从底层原理到上层应用,分为四个章节:

章节 主题 核心问题
第一章 全局容器的实现 如何让每个页面自动包裹布局组件?
第二章 Vite 虚拟模块 如何在运行时读取 pages.json 配置?
第三章 组件设计 如何设计可配置的 NavBar 和 TabBar?
第四章 动态调用设计 页面如何与全局容器进行通信?

整体架构

第三章
第二章
第一章
基础设施建设
第四章
await
useAppLayout Hook

页面通信入口
registerLayout

事件注册函数
promiseWithResolvers

异步等待机制
pages.json

页面配置
vite-plugin-uni-layout

编译时注入容器
vite-plugin-uni-pages-json

虚拟模块读取配置
AppLayout.vue

布局聚合层
Layout.vue

基础容器
NavBar 组件
TabBar 组件

依赖关系

  • 第一章和第二章是基础设施,可独立理解
  • 第三章依赖第一、二章
  • 第四章依赖第三章

一、设计背景

1.1 问题场景

在 uni-app 跨平台应用开发中,每个页面通常需要共享相同的布局结构,如导航栏、标签栏、页面容器等。传统做法是在每个页面组件中手动引入这些公共组件,导致:

  • 代码重复:每个页面都要写相同的容器结构
  • 维护困难:修改容器样式需要遍历所有页面
  • 一致性难以保证:各页面容器可能出现差异

1.2 uni-app 架构特点

uni-app 是一个基于 Vue.js 的跨平台开发框架,其架构从上到下分为四个层次:
原生渲染层
H5渲染引擎
微信小程序
App渲染引擎
框架层
WXS
生命周期
事件系统
平台适配层
view
scroll-view
text
image
业务层
index
list
detail
profile

关键特点 :uni-app 中,每个页面都是独立的 Vue 实例,页面之间没有天然的嵌套关系。

这意味着:

  1. 页面顶部区域(导航栏)需要每个页面自己处理
  2. 页面底部区域(标签栏)需要每个页面自己处理
  3. 页面容器包裹每个页面都需要手动添加

1.3 传统方案的问题

方案 问题
每个页面手动引入容器组件 代码重复、维护成本高
使用 Mixin 混入 侵入性强,隐式依赖
全局组件注册 无法精细控制哪些页面需要

核心矛盾:uni-app 框架层没有提供页面容器抽象,而业务层又需要统一的布局结构。


二、设计目标

设计一个编译时全局容器注入机制,实现:

目标 描述
自动包裹 编译时自动为所有页面包裹全局容器组件
零侵入 不修改业务代码,不影响开发体验
灵活配置 通过 pages.json 配置控制哪些页面需要容器
跨平台兼容 支持 uni-app 的所有平台(H5、App、小程序)

三、核心设计

3.1 设计理念

设计点 方案 为什么选择这个方案
注入时机 编译时(Transform Hook) uni-app 页面无父容器,运行时注入成本高
容器定位 编译时代码转换,非运行时逻辑 运行时注入需要修改 Vue 实例创建过程
页面识别 读取 pages.json 动态获取页面列表 精准控制,只为注册页面注入容器
模板处理 正则解析 <template> 内容 轻量无 AST 依赖,实现简洁
包裹方式 替换 <template> 内容区域 保留原有模板结构,用户无感知

关键洞察 :uni-app 的编译过程会处理每个 .vue 文件,我们在这个过程中拦截并注入容器,对业务代码完全透明。

3.2 注入流程



pages.json
读取页面配置
是否为注册页面?
Transform Hook
跳过
正则解析 template
注入 AppLayout 组件
输出转换结果

3.3 架构分层

注入容器后的架构层次:
原生渲染层
H5
微信小程序
App
平台适配层
uni-* 组件
业务层
index
list
detail
profile
全局容器层
AppLayout
Header
Content
Footer

关键变化 :通过编译时注入,在业务层和平台适配层之间新增了一层全局容器层,统一处理公共布局,而业务页面无需感知。


四、插件实现

4.1 核心代码

javascript 复制代码
// vite-plugin-uni-layout.js
import fs from 'fs';
import path from 'path';

export default function uniLayout() {
    let pagesPaths = new Set();    // 存储 pages.json 中的页面路径
    let pagesJsonPath = '';        // pages.json 文件路径

    // 从 pages.json 读取页面路径列表
    function updatePages() {
        const content = fs.readFileSync(pagesJsonPath, 'utf-8');
        const config = JSON.parse(content);
        pagesPaths.clear();
        (config.pages || []).forEach(page => {
            pagesPaths.add(page.path);
        });
    }

    return {
        name: 'vite-plugin-uni-layout',
        enforce: 'pre',  // 🔍 pre 模式:确保在 @dcloudio/vite-plugin-uni 之前执行
                          //    否则 uni-app 的编译插件会先处理 .vue 文件,
                          //    我们的注入代码会被其内部流程覆盖或丢失

        // Vite 配置解析完成时调用,初始化 pages.json 路径
        configResolved(config) {
            // config.root 为项目根目录,pages.json 通常在 src/ 下
            pagesJsonPath = path.resolve(config.root, 'src/pages.json');
            updatePages();
        },

        // Transform 钩子:转换每个 .vue 文件
        // id 参数格式:'/path/to/src/pages/index/index.vue?xxx=xxx'
        //                                    ^^^ Query String
        transform(code, id) {
            // 1️⃣ 去除 Query String(如 ?ts=1700000000)
            //    Vite 热更新时会在 id 追加时间戳,必须清除
            const cleanId = id.split('?')[0];

            // 2️⃣ 仅处理 .vue 文件
            if (!cleanId.endsWith('.vue')) return;

            // 3️⃣ 计算文件相对于 pages.json 目录的路径
            //    示例:
            //    pagesJsonPath  = '/project/src/pages.json'
            //    cleanId        = '/project/src/pages/index/index.vue'
            //    dirname        = '/project/src'
            //    relative       = 'pages/index/index'  (Unix 风格路径)
            const relativePath = path.relative(
                path.dirname(pagesJsonPath), cleanId
            ).replace(/\\/g, '/');  // Windows 反斜杠转正斜杠

            // 4️⃣ 转为路由路径:去掉 .vue 后缀
            //    'pages/index/index' 匹配 pages.json 中的 path
            const routePath = relativePath.replace(/\.vue$/, '');

            // 5️⃣ 仅对 pages.json 中注册的页面注入容器
            if (!pagesPaths.has(routePath)) return;

            // 6️⃣ 核心:包裹 <template> 内容区域
            //    找到 <template> 标签的开始位置和结束位置
            const templateOpenRegex = /<template[^>]*>/;
            const templateOpenMatch = code.match(templateOpenRegex);

            if (templateOpenMatch) {
                const openTag = templateOpenMatch[0];                    // '<template>'
                const openIndex = templateOpenMatch.index + openTag.length; // 内容起始位置
                const closeIndex = code.lastIndexOf('</template>');       // 内容结束位置

                if (closeIndex > openIndex) {
                    const before = code.substring(0, openIndex);   // '<template>' 之前
                    const content = code.substring(openIndex, closeIndex);  // 模板内容
                    const after = code.substring(closeIndex);      // '</template>' 之后

                    // 注入 AppLayout 组件
                    return `${before}\n<AppLayout>\n${content}\n</AppLayout>\n${after}`;
                }
            }
        }
    };
}

4.2 转换效果

转换前(业务页面)

vue 复制代码
<template>
  <view class="content">
    <text>Hello World</text>
  </view>
</template>

转换后(编译产物)

vue 复制代码
<template>
<AppLayout>
  <view class="content">
    <text>Hello World</text>
  </view>
</AppLayout>
</template>

4.3 数据流



pages.json 配置
updatePages
configResolved
transform
路径匹配检查
正则替换模板
跳过
输出转换结果

4.4 正则解析的局限性与注意事项

当前实现的局限性

本方案使用正则表达式解析 Vue 模板,具有轻量、无 AST 依赖的优点,但存在以下局限:

场景 问题 示例
多根节点模板 v-if/v-else-if/v-else 会产生多个根节点 vue\n<template v-if="ok"><view>A</view></template>\n<template v-else><view>B</view></template>\n
动态组件 <component :is="..."> 包裹的内容无法正确提取 vue\n<template><component :is="componentName"/></template>\n
深度嵌套 template 多层 <template> 标签匹配混乱 vue\n<template><template v-for="...">...</template></template>\n
特殊属性值 > 在字符串中出现导致标签匹配错误 vue\n<template><text>{``{ '<template>' }}</text></template>\n
为什么选择正则而非 AST?
方案 优点 缺点
正则解析 轻量、无依赖、实现简单 边界 case 较多
AST 解析(@vue/compiler-sfc) 精确、可预测 体积增大、API 复杂

实际项目中,如果你的模板满足以下条件,正则方案足够稳定:

  • 使用单根节点模板
  • 不使用深度嵌套的 <template> 标签
  • 模板内容不包含 </template> 字符串
生产环境建议

对于复杂模板场景,推荐升级方案:

javascript 复制代码
// 使用 @vue/compiler-sfc 精确解析
import { parse } from '@vue/compiler-sfc'

const { descriptor } = parse(code)
const templateContent = descriptor.template.content
// 然后精确操作 AST

4.5 插件执行时机与 Vite 钩子顺序

Vite 插件钩子执行顺序

Vite 插件的钩子按以下顺序执行:
watchChange build ssrGenerate config CLI / Dev Server watchChange build ssrGenerate config CLI / Dev Server 单次执行 loop [Transform files] loop [Bundle] 监听文件变化 configResolved buildStart transform closeBundle buildStart transform generateBundle closeBundle handleHotUpdate transform

enforce: 'pre' vs 'post'
javascript 复制代码
export default function myPlugin() {
    return {
        name: 'my-plugin',
        enforce: 'pre',  // ✅ 在默认插件之前执行(如 @dcloudio/vite-plugin-uni)
        // enforce: 'post', // 后置执行
    }
}
模式 执行时机 适用场景
默认 介于 prepost 之间 普通转换逻辑
pre 最先执行 需要在其他插件之前处理,如本方案的容器注入
post 最后执行 清理工作、最终修改
本插件使用 'pre' 的原因

渲染错误: Mermaid 渲染失败: Parse error on line 3: ...i-layout] B --> C[默认插件
@dcloudio ----------------------^ Expecting 'AMP', 'COLON', 'PIPE', 'TESTSTR', 'DOWN', 'DEFAULT', 'NUM', 'COMMA', 'NODE_STRING', 'BRKT', 'MINUS', 'MULT', 'UNICODE_TEXT', got 'LINK_ID'

  1. uni-app 编译流程复杂@dcloudio/vite-plugin-uni 内部会多次转换 .vue 文件
  2. 注入代码必须优先 :如果在其之后执行,注入的 <AppLayout> 会被其内部处理流程覆盖
  3. 单向保证:pre 可以看到"最原始"的代码,而 post 可以看到"处理后"的代码
configResolved 钩子详解

configResolved 在 Vite 解析完配置后调用,此时:

  • config.root 已确定
  • config.plugins 已确定顺序
  • 可以读取最终的配置进行初始化
javascript 复制代码
configResolved(config) {
    // config.root: string - 项目根目录
    // config.srcDir?: string - src 目录(如果有)
    // config.plugins: Plugin[] - 按执行顺序排列的插件列表
    pagesJsonPath = path.resolve(config.root, 'src/pages.json');
}
transform 钩子详解

每个文件在被读取、转换时都会触发 transform

javascript 复制代码
transform(code, id) {
    // code: string - 文件原始内容
    // id: string   - 文件绝对路径 + query string
    //
    // 返回值类型:
    // - null/undefined: 不处理,交由后续插件
    // - string: 替换后的内容
    // - { code, map }: 替换内容 + Source Map
}

id 参数的 Query String 示例

触发场景 id 示例
初始加载 /src/pages/index/index.vue
热更新 /src/pages/index/index.vue?import&t=1700000000
SSR 构建 /src/pages/index/index.vue?ssr=true

五、插件注册

5.1 vite.config.js 配置

javascript 复制代码
import { defineConfig } from 'vite';
import uniLayout from './vite/plugins/vite-plugin-uni-layout';
import uni from '@dcloudio/vite-plugin-uni';

export default defineConfig({
    plugins: [
        uniLayout(),  // 全局容器插件
        uni(),
    ],
});

5.2 AppLayout 基础组件

vue 复制代码
<template>
  <view class="app-layout">
    <!-- 顶部导航栏 -->
    <header class="layout-header">
      <slot name="header">
        <text class="header-title">{{ title }}</text>
      </slot>
    </header>

    <!-- 页面内容 -->
    <main class="layout-content">
      <slot></slot>
    </main>

    <!-- 底部标签栏 -->
    <footer class="layout-footer" v-if="showTabBar">
      <slot name="footer"></slot>
    </footer>
  </view>
</template>

<script>
export default {
    name: 'AppLayout',
    props: {
        title: {
            type: String,
            default: 'App Name'
        },
        showTabBar: {
            type: Boolean,
            default: true
        }
    }
};
</script>

<style scoped>
.app-layout {
    display: flex;
    flex-direction: column;
    height: 100vh;
    background-color: #f5f5f5;
}

.layout-header {
    height: 88rpx;
    background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
    color: #fff;
    display: flex;
    align-items: center;
    padding: 0 30rpx;
}

.layout-content {
    flex: 1;
    overflow-y: auto;
}

.layout-footer {
    height: 100rpx;
    background: #fff;
    box-shadow: 0 -2rpx 10rpx rgba(0, 0, 0, 0.1);
}
</style>

5.3 pages.json 配置

json 复制代码
{
    "pages": [
        {
            "path": "pages/index/index",
            "style": { "navigationBarTitleText": "首页" }
        },
        {
            "path": "pages/list/list",
            "style": { "navigationBarTitleText": "列表" }
        }
    ],
    "globalStyle": {
        "navigationBarTextStyle": "white",
        "navigationBarTitleText": "应用名称",
        "navigationBarBackgroundColor": "#667eea"
    }
}

六、设计价值

维度 传统方式 本方案
开发效率 每个页面手动引入容器 一次配置,全局生效
可维护性 分散管理 集中管理
性能 运行时动态创建 编译时零开销
一致性 各页面可能存在差异 所有页面完全一致
侵入性 业务代码需要修改 零侵入,无感知

七、总结

本方案通过 Vite 编译时注入 机制,基于 uni-app 架构实现了全局容器的优雅设计:

核心要点 说明
架构契合 适配 uni-app 页面平级、无父容器的特点
编译时注入 利用 Transform Hook,在构建期完成容器包裹
配置驱动 依托 pages.json 天然管控,无需额外配置
零侵入 业务代码完全无感知,保持简洁
可扩展 容器内部支持 slot 插槽,可灵活定制各页面

这一设计模式可推广至以下场景:

  • 全局错误边界注入
  • 页面切换动画包装
  • 状态管理自动初始化
  • 性能监控埋点注入

下章预告

第一章解决了「如何让每个页面自动包裹布局组件」的问题。

下一章我们将探讨:如何让组件在运行时读取 pages.json 配置

相关推荐
安生生申4 小时前
uni-app 连接 JDY-31 蓝牙串口模块实践
c语言·前端·javascript·stm32·单片机·嵌入式硬件·uni-app
小离a_a4 小时前
uniapp小程序封装圆环显示比例数据
android·小程序·uni-app
__zRainy__4 小时前
uni-app 全局容器实战系列(三):全局 NavBar 和 TabBar 组件设计
uni-app
发现一只大呆瓜14 小时前
Vite凭什么这么快?3分钟带你彻底搞懂 Vite 热更新的幕后黑手
前端·面试·vite
一颗小青松18 小时前
uniapp输入框fixed定位,导致页面顶起解决方案
前端·uni-app
2501_915106321 天前
深入解析无源码iOS加固原理与方案,保护应用安全
android·安全·ios·小程序·uni-app·cocoa·iphone
Hello--_--World1 天前
利用CDN进行首屏优化。能不能看CDN与本地服务器谁快用谁?
运维·服务器·前端·javascript·vite
万能小林子1 天前
2026 AI开发新范式:Vibe Coding生成网页 + 3分钟打包成App,非技术人也能独立发布自己的App!
人工智能·uni-app·ai编程·web app·vibecoding
Hello--_--World1 天前
为什么 用vite进行分包后,可以通过 浏览器强制缓存 提高性能?路由懒加载进行的分包与 vite进行的分包有什么不同?
前端·javascript·缓存·vite