从npm库 Vue 组件到独立SDK:打包与 CDN 引入的最佳实践
文章目录
- 前言
- 一、 原始方案:直接发布 npm 组件
- 二、升级为独立 SDK:支持 CDN 引入
- 三、核心要点总结
- 1. 独立 Vue 实例
- 2. 动态渲染组件
- 3. 手动挂载到 DOM
- 4. 与用户环境的关系
前言
近期在项目中引入了一个支持多格式(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 版本可以共存(各自独立)。