Vue Demi是如何让你的库同时支持Vue2和Vue3的
qiyuwang 2024-10-06 12:21 24 浏览 0 评论
Vue Demi是什么
如果你想开发一个同时支持Vue2和Vue3的库可能想到以下两种方式:
1.创建两个分支,分别支持Vue2和Vue3
2.只使用Vue2和Vue3都支持的API
这两种方式都有缺点,第一种很麻烦,第二种无法使用Vue3新增的组合式 API,其实现在Vue2.7+版本已经内置支持组合式API,Vue2.6及之前的版本也可以使用@vue/composition-api插件来支持,所以完全可以只写一套代码同时支持Vue2和3。虽然如此,但是实际开发中,同一个API在不同的版本中可能导入的来源不一样,比如ref方法,在Vue2.7+中直接从vue中导入,但是在Vue2.6-中只能从@vue/composition-api中导入,那么必然会涉及到版本判断,Vue Demi就是用来解决这个问题,使用很简单,只要从Vue Demi中导出你需要的内容即可:
import { ref, reactive, defineComponent } from 'vue-demi'
Vue-demi会根据你的项目判断到底使用哪个版本的Vue,具体来说,它的策略如下:
- <=2.6: 从Vue和@vue/composition-api中导出
- 2.7: 从Vue中导出(组合式API内置于Vue 2.7中)
- >=3.0: 从Vue中导出,并且还polyfill了两个Vue 2版本的set和del API
接下来从源码角度来看一下它具体是如何实现的。
基本原理
当我们使用npm i vue-demi在我们的项目里安装完以后,它会自动执行一个脚本:
{
"scripts": {
"postinstall": "node ./scripts/postinstall.js"
}
}
// postinstall.js
const { switchVersion, loadModule } = require('./utils')
const Vue = loadModule('vue')
if (!Vue || typeof Vue.version !== 'string') {
console.warn('[vue-demi] Vue is not found. Please run "npm install vue" to install.')
}
else if (Vue.version.startsWith('2.7.')) {
switchVersion(2.7)
}
else if (Vue.version.startsWith('2.')) {
switchVersion(2)
}
else if (Vue.version.startsWith('3.')) {
switchVersion(3)
}
else {
console.warn(`[vue-demi] Vue version v${Vue.version} is not suppported.`)
}
导入我们项目里安装的vue,然后根据不同的版本分别调用switchVersion方法。
先看一下loadModule方法:
function loadModule(name) {
try {
return require(name)
} catch (e) {
return undefined
}
}
很简单,就是包装了一下require,防止报错阻塞代码。
然后看一下switchVersion方法:
function switchVersion(version, vue) {
copy('index.cjs', version, vue)
copy('index.mjs', version, vue)
copy('index.d.ts', version, vue)
if (version === 2)
updateVue2API()
}
执行了copy方法,从函数名可以大概知道是复制文件,三个文件的类型也很清晰,分别是commonjs版本的文件、ESM版本的文件、TS类型定义文件。
另外还针对Vue2.6及一下版本执行了updateVue2API方法。
updateVue2API方法我们后面再看,先看一下copy方法:
const dir = path.resolve(__dirname, '..', 'lib')
function copy(name, version, vue) {
vue = vue || 'vue'
const src = path.join(dir, `v${version}`, name)
const dest = path.join(dir, name)
let content = fs.readFileSync(src, 'utf-8')
content = content.replace(/'vue'/g, `'${vue}'`)
try {
fs.unlinkSync(dest)
} catch (error) { }
fs.writeFileSync(dest, content, 'utf-8')
}
其实就是从不同版本的目录里复制上述三个文件到外层目录,其中还支持替换vue的名称,这当你给vue设置了别名时需要用到。
到这里,Vue Demi安装完后自动执行的事情就做完了,其实就是根据用户项目中安装的Vue版本,分别从三个对应的目录中复制文件作为Vue Demi包的入口文件,Vue Demi支持三种模块语法:
{
"main": "lib/index.cjs",
"jsdelivr": "lib/index.iife.js",
"unpkg": "lib/index.iife.js",
"module": "lib/index.mjs",
"types": "lib/index.d.ts"
}
默认入口为commonjs模块cjs文件,支持ESM的可以使用mjs文件,同时还提供了可以直接在浏览器上使用的iife类型的文件。
接下来看一下分别针对三种版本的Vue具体都做了什么。
v2版本
Vue2.6版本只有一个默认导出:
我们只看mjs文件,cjs有兴趣的可以自行阅读:
import Vue from 'vue'
import VueCompositionAPI from '@vue/composition-api/dist/vue-composition-api.mjs'
function install(_vue) {
_vue = _vue || Vue
if (_vue && !_vue['__composition_api_installed__'])
_vue.use(VueCompositionAPI)
}
install(Vue)
// ...
导入Vue和VueCompositionAPI插件,并且自动调用Vue.use方法安装插件。
继续:
// ...
var isVue2 = true
var isVue3 = false
var Vue2 = Vue
var version = Vue.version
export {
isVue2,
isVue3,
Vue,
Vue2,
version,
install,
}
/**VCA-EXPORTS**/
export * from '@vue/composition-api/dist/vue-composition-api.mjs'
/**VCA-EXPORTS**/
首先导出了两个变量isVue2和isVue3,方便我们的库代码判断环境。
然后在导出Vue的同时,还通过Vue2的名称再导出了一遍,这是为啥呢,其实是因为Vue2的API都是挂载在Vue对象上,比如我要进行一些全局配置,那么只能这么操作:
import { Vue, isVue2 } from 'vue-demi'
if (isVue2) {
Vue.config.xxx
}
这样在Vue2的环境中没有啥问题,但是当我们的库处于Vue3的环境中时,其实是不需要导入Vue对象的,因为用不上,但是构建工具不知道,所以它会把Vue3的所有代码都打包进去,但是Vue3中很多我们没有用到的内容是不需要的,但是因为我们导入了包含所有API的Vue对象,所以无法进行去除,所以针对Vue2版本单独导出一个Vue2对象,我们就可以这么做:
import { Vue2 } from 'vue-demi'
if (Vue2) {
Vue2.config.xxx
}
然后后续你会看到在Vue3的导出中Vue2是undefined,这样就可以解决这个问题了。
接着导出了Vue的版本和install方法,意味着你可以手动安装VueCompositionAPI插件。
然后是导出VueCompositionAPI插件提供的API,也就是组合式API,但是可以看到前后有两行注释,还记得前面提到的switchVersion方法里针对Vue2版本还执行了updateVue2API方法,现在来看一看它做了什么事情:
function updateVue2API() {
const ignoreList = ['version', 'default']
// 检查是否安装了composition-api
const VCA = loadModule('@vue/composition-api')
if (!VCA) {
console.warn('[vue-demi] Composition API plugin is not found. Please run "npm install @vue/composition-api" to install.')
return
}
// 获取除了version、default之外的其他所有导出
const exports = Object.keys(VCA).filter(i => !ignoreList.includes(i))
// 读取ESM语法的入口文件
const esmPath = path.join(dir, 'index.mjs')
let content = fs.readFileSync(esmPath, 'utf-8')
// 将export * 替换成 export { xxx }的形式
content = content.replace(
/\/\*\*VCA-EXPORTS\*\*\/[\s\S]+\/\*\*VCA-EXPORTS\*\*\//m,
`/**VCA-EXPORTS**/
export { ${exports.join(', ')} } from '@vue/composition-api/dist/vue-composition-api.mjs'
/**VCA-EXPORTS**/`
)
// 重新写入文件
fs.writeFileSync(esmPath, content, 'utf-8')
}
主要做的事情就是检查是否安装了@vue/composition-api,然后过滤出了@vue/composition-api除了version和default之外的所有导出内容,最后将:
export * from '@vue/composition-api/dist/vue-composition-api.mjs'
的形式改写成:
export { EffectScope, ... } from '@vue/composition-api/dist/vue-composition-api.mjs'
为什么要过滤掉version和default呢,version是因为已经导出了Vue的version了,所以会冲突,本来也不需要,default即默认导出,@vue/composition-api的默认导出其实是一个包含它的install方法的对象,前面也看到了,可以默认导入@vue/composition-api,然后通过Vue.use来安装,这个其实也不需要从Vue Demi导出,不然像下面这样就显得很奇怪:
import VueCompositionAPI from 'vue-demi'
到这里,就导出所有内容了,然后我们就可以从vue-demi中导入各种需要的内容了,比如:
import { isVue2, Vue, ref, reactive, defineComponent } from 'vue-demi'
v2.7版本
接下来看一下是如何处理Vue2.7版本的导出的,和Vue2.6之前的版本相比,Vue2.7直接内置了@vue/composition-api,所以除了默认导出Vue对象外还导出了组合式API:
import Vue from 'vue'
var isVue2 = true
var isVue3 = false
var Vue2 = Vue
var warn = Vue.util.warn
function install() {}
export { Vue, Vue2, isVue2, isVue3, install, warn }
// ...
和v2相比,导出的内容是差不多的,因为不需要安装@vue/composition-api,所以install是个空函数,区别在于还导出了一个warn方法,这个文档里没有提到,但是可以从过往的issues中找到原因,大致就是Vue3导出了一个warn方法,而Vue2的warn方法在Vue.util对象上,所以为了统一手动导出,为什么V2版本不手动导出一个呢,原因很简单,因为这个方法在@vue/composition-api的导出里有。
继续:
// ...
export * from 'vue'
// ...
导出上图中Vue所有的导出,包括version、组合式API,但是要注意这种写法不会导出默认的Vue,所以如果你像下面这样使用默认导入是获取不到Vue对象的:
import Vue from 'vue-demi'
继续:
// ...
// createApp polyfill
export function createApp(rootComponent, rootProps) {
var vm
var provide = {}
var app = {
config: Vue.config,
use: Vue.use.bind(Vue),
mixin: Vue.mixin.bind(Vue),
component: Vue.component.bind(Vue),
provide: function (key, value) {
provide[key] = value
return this
},
directive: function (name, dir) {
if (dir) {
Vue.directive(name, dir)
return app
} else {
return Vue.directive(name)
}
},
mount: function (el, hydrating) {
if (!vm) {
vm = new Vue(Object.assign({ propsData: rootProps }, rootComponent, { provide: Object.assign(provide, rootComponent.provide) }))
vm.$mount(el, hydrating)
return vm
} else {
return vm
}
},
unmount: function () {
if (vm) {
vm.$destroy()
vm = undefined
}
},
}
return app
}
和Vue2的new Vue创建Vue实例不一样,Vue3是通过createApp方法,@vue/composition-api插件polyfill了这个方法,所以针对Vue2.7,Vue Demi手动进行了polyfill。
到这里,针对Vue2.7所做的事情就结束了。
v3版本
Vue3相比之前的版本,最大区别是不再提供一个单独的Vue导出:
import * as Vue from 'vue'
var isVue2 = false
var isVue3 = true
var Vue2 = undefined
function install() {}
export {
Vue,
Vue2,
isVue2,
isVue3,
install,
}
// ...
因为默认不导出Vue对象了,所以通过整体导入import * as Vue的方式把所有的导出都加载到Vue对象上,然后也可以看到导出的Vue2为undefined,install同样是个空函数。
继续:
// ...
export * from 'vue'
// ...
没啥好说的,直接导出Vue的所有导出内容。
继续:
// ...
export function set(target, key, val) {
if (Array.isArray(target)) {
target.length = Math.max(target.length, key)
target.splice(key, 1, val)
return val
}
target[key] = val
return val
}
export function del(target, key) {
if (Array.isArray(target)) {
target.splice(key, 1)
return
}
delete target[key]
}
最后polyfill了两个方法,这两个方法实际上是@vue/composition-api插件提供的,因为@vue/composition-api提供的响应性API实现上并没有使用Proxy代理,仍旧是基于Vue2的响应系统来实现的,所以Vue2中响应系统的限制仍旧还是存在的,所以需要提供两个类似Vue.set和Vue.delete方法用来给响应性数据添加或删除属性。
相关推荐
- 学习笔记-Linux JDK - 安装&配置
-
前提条件#检查是否存在JDKrpm-qa|grepjava#删除现存JDKyum-yremovejava*安装OracleJDK不分系统...
- Ubuntu16.04.1安装Java8(ubuntu安装java的命令)
-
上篇文章讲解了怎么在Windows下安装Java8《Windows10安装Java8》,这里讲解下怎么在Linux下安装Java。由于之前已经安装了Ubuntu16.04.1《...
- Ubuntu 下安装 JDK17(ubuntu安装jdk1.7)
-
JavaSE17Ubuntu下JDK的安装本文主要针对Ubuntu的环境进行Java17的JDK安装。下载地址:...
- Ubuntu安装JDK(ubuntu安装jdk报错)
-
在Ubuntu系统上安装JDK8u441版本,可以通过多种方式实现,包括使用官方JDK的PPA仓库、下载JDK的.tar.gz文件手动安装,或者使用第三方PPA仓库如WebUpd8。以下是通过JDK...
- 前端资源-实用的JS插件(前端浏览器插件)
-
现在前端资源越来越多,有创意十足的,有实用性高的,这些对于设计师和前端人员来说都是不错的灵感和资源,所以我们可多关注这些信息,对自己的专业技术有也会帮助的。今天设计达人网为大家分享有:页面进度条、图像...
- 图片延迟加载,你会使用吗?给你推荐几款插件,快来学习吧
-
图片延迟加载延迟加载就是当真正需要的时候,才执行加载操作。延迟加载作为Web前端性能优化的一种措施,已经越来越多的应用到各种程序中,而图片的延迟加载作为使用是最广泛的一种,更应该被我们掌握,今天我就给...
- 突发!Vite 插件惊现图片处理黑科技
-
【AlarmLevel】趣味【AlarmTitle】突发!Vite插件惊现图片处理黑科技【AlarmOverview】就在昨天,GitHub上一款名为vite-plugin-imagemi...
- 盘点前端程序员制作网站的常用工具
-
网站制作时,为了能够更快速、高效地完成任务,往往需要网站制作工具来进行辅助。尤其是前端程序员,五花八门的网站制作工具。今天就来盘点前端程序员一般开发网站程序时使用的那类网站制作工具。...
- MyBatis 插件原理与实战(mybatis好用的插件idea)
-
文章导读MyBatis插件原理与实战什么是插件?...
- VisBug:助力前端开发的浏览器插件
-
作为前端开发者相信肯定有遇到过以下场景:...
- 前端插件-unplugin-auto-import真的香香
-
没用这个插件前:你在Vue3中写了50个组件,每个文件开头都要重复这堆代码:import{ref,computed}from'vue'import{useRoute,...
- VSCode中值得推荐的常用的16个高效前端插件「主题篇」(一)
-
VSCode是我们前端开发的一个强大的IDE,所以选择趁手好用的插件是提高开发效率,然后剩下的时间用来摸鱼是很有必要滴。主题篇(16)VSCodeGreatIcons...
- 支持快速集成的前端网站反馈小插件
-
大家好,我是章鱼猫。...
- 很香的几款开源免费的流程设计器(开源流程图设计器)
-
1、LogicFlow(1)介绍:LogicFlow是一款流程图编辑框架,提供了一系列流程图交互、编辑所必需的功能和灵活的节点自定义、插件等拓展机制。LogicFlow支持前端研发自定义开发各种逻...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- navicat无法连接mysql服务器 (65)
- 下横线怎么打 (71)
- flash插件怎么安装 (60)
- lol体验服怎么进 (66)
- ae插件怎么安装 (62)
- yum卸载 (75)
- .key文件 (63)
- cad一打开就致命错误是怎么回事 (61)
- rpm文件怎么安装 (66)
- linux取消挂载 (81)
- ie代理配置错误 (61)
- ajax error (67)
- centos7 重启网络 (67)
- centos6下载 (58)
- mysql 外网访问权限 (69)
- centos查看内核版本 (61)
- ps错误16 (66)
- nodejs读取json文件 (64)
- centos7 1810 (59)
- 加载com加载项时运行错误 (67)
- php打乱数组顺序 (68)
- cad安装失败怎么解决 (58)
- 因文件头错误而不能打开怎么解决 (68)
- js判断字符串为空 (62)
- centos查看端口 (64)