文章目录
前言
近期在项目中引入了一个支持多格式(PDF、Video、Word等)的文件预览组件,发现该组件库未经过打包构建,而是直接将源码发布到 npm。用户通过 import 引入后,组件会直接使用用户项目中的 Vue 进行构建。
调整打包策略:UMD 格式 + CDN 引入(独立运行,不依赖用户构建工具)
当用户项目的 Vue 版本与组件开发者使用的版本不一致时(例如用户用 Vue 2.6,组件依赖 Vue 2.7 的特性),会导致兼容性错误等问题
一、 原始方案:直接发布 npm 组件
package.json
{
"name": "file-preview",
"version": "1.0.0",
"description": "文件预览组件",
"main": "/src/components/filePreview/index.vue",
"files": [
"src/components/filePreview" // 只允许列出的文件/目录被打包发布
]
}
vue 相关依赖作为开发依赖不进行打包,依赖于用户进行提供,对于vue 版本不适配影响严重
二、升级为独立 SDK:支持 CDN 引入
(1) 改造目标
- 输出 UMD 格式 的包,支持
<script>
标签引入。 - 内置 Vue 依赖,避免与用户项目冲突。
- 提供简洁的 API(如 mount/destroy)。
(2) SDK 核心代码(main.js)
import Vue from 'vue';
import FilePreview from './components/filePreview/index.vue';
export class FilePreviewSDK {
constructor(id, options) {
const {
fileUrl,
fileExt,
word2 = false,
showDownload = false,
controls = false,
showPageNum = false,
excel2 = false,
pdfPriority = false,
convertUrl,
onError = () => {},
} = options;
this.app = new Vue({
render: (h) =>
h(FilePreview, {
props: {
src: fileUrl,
fileExt: fileExt,
word2,
showDownload,
controls,
showPageNum,
excel2,
pdfPriority,
convertUrl,
},
on: {
error: onError,
},
}),
}).$mount(id);
}
destroy() {
if (this.app) {
this.app.$destroy();
}
}
}
(3)打包配置
核心配置修改: "build:lib": "vue-cli-service build --target lib --inline-vue --entry ./src/main.js"
"scripts": {
"dev": "vue-cli-service serve",
"build": "vue-cli-service build",
"lint": "vue-cli-service lint",
"build:lib": "vue-cli-service build --target lib --inline-vue --entry ./src/main.js",
},
关键参数详解
(1) --target lib(库模式)
作用:告诉 Vue CLI 你正在构建一个可复用的库,而不是一个完整的 SPA 应用。
输出文件:
默认生成以下格式(在 dist/ 目录):
umd.js:通用模块定义(浏览器直接使用 <script>
引入)。
common.js:CommonJS 格式(Node.js 或 Webpack 1)。
esm.js:ES Module 格式(现代打包工具如 Vite、Webpack 2+)。
可能还有 .css 文件(如果组件有样式)。
(2) --inline-vue(内联 Vue)
作用:将 Vue 直接打包进最终文件,避免依赖用户项目的 Vue 版本。
为什么需要?
如果不加此参数,库会默认依赖 peerDependencies 中的 Vue,要求用户自己安装 Vue,可能导致版本冲突。
加上后,库会自带 Vue,但会增加体积(适合小型工具库)。
(3) --entry ./src/main.js(入口文件)
作用:指定库的入口文件
(即 main.js,导出 FilePreviewSDK
类)。
注意:
如果你的库有多个组件,可以改为入口目录(如 --entry ./src/components)。
(4)用户使用方式
<script src="https://unpkg.com/file-preview/dist/file-preview.umd.js"></script>
<script>
const preview = new FilePreviewSDK('#app', { fileUrl: '...' });
</script>
三、核心要点总结
1. 独立 Vue 实例
- SDK 内部通过 new Vue() 创建了一个全新的 Vue 根实例,与用户项目的 Vue 环境完全隔离。
这意味着:
不会继承用户项目的 Vue 插件(如 Vuex、Vue Router)。
不会混入用户项目的 全局组件/指令。
不共享用户项目的 Vue 配置(如 errorHandler)。
2. 动态渲染组件
通过render 函数 + h(createElement)
动态渲染 SDK 内部的 FilePreview 组件。
相当于在运行时生成一个类似这样的模板:
<HTML>
<FilePreview
:src="fileUrl"
@error="onError"
/>
3. 手动挂载到 DOM
- 通过
.$mount(id)
将组件挂载到用户指定的 DOM 节点(如 #preview-container)。 - 挂载后:
SDK 完全控制该组件的生命周期(可随时通过 this.app.$destroy() 销毁)。
用户只需关心 mount 和 destroy,无需了解内部实现。
4. 与用户环境的关系
- 无依赖:即使用户项目没有 Vue,只要 SDK 的打包配置正确(如 UMD 包含 Vue 依赖),也能运行。
- 无冲突:多个 SDK 实例或多个 Vue 版本可以共存(各自独立)。