一、前言
前不久,对ant-design-vue-pro项目进行了迁移,参考文章:全网最硬核的Ant-Design-Vue从Vue-cli迁移至Vite(一),迁移后的项目地址:GitHub - Seals-Studio/ant-design-vue-pro-vite
本期针对Element-UI库进行迁移,以vue-element-admin项目为例进行迁移,element-ui版本为2.13.2。同时,提供了迁移后的仓库,欢迎Star~
GitHub - Seals-Studio/vue-element-admin-vite
二、背景
众所周知,Vite作为下一代前端开发与构建工具,就是一个字:快。并且Vite已经作为Vue3默认的构建工具。通过实验表明,项目迁移后,从Vue-cli的近2分钟,到Vite的5秒(项目大小不同,时间也不同),提升了几十倍甚至上百倍的速度。
迁移前后对比(参考)
构建工具 | 服务器启动耗时 | 页面首次加载速度 (无缓存) | 第二次加载速度 (有缓存) | 热更新 HMR | 打包 |
---|---|---|---|---|---|
Webpack | 83s | 4.78s | 3.35s | 4.78s | 3mins 37s |
Vite | 4.72s (第二次 0.72s) | 1.71s | 1.33s | 瞬间 | 51.45s |
三、删除package.json相关依赖
删除@vue和babel相关
{ "@vue/cli-plugin-babel": "4.4.4", "@vue/cli-plugin-eslint": "4.4.4", "@vue/cli-plugin-unit-jest": "4.4.4", "@vue/cli-service": "4.4.4", "@vue/test-utils": "1.0.0-beta.29", "babel-eslint": "10.1.0", "babel-jest": "23.6.0", "babel-plugin-dynamic-import-node": "2.3.3", }
删除loader(webpack插件)和webpack
{ "html-webpack-plugin": "3.2.0", "script-ext-html-webpack-plugin": "2.1.3", "sass-loader": "8.0.2", "svg-sprite-loader": "4.1.3", }
- 删除babel.conf.js和jsconfig.json
- 安装pnpm工具
pnpm是快速的,节省磁盘空间的包管理工具
npm i -g pnpm
# 淘宝源
pnpm config set registry https://registry.npm.taobao.org
pnpm config set disturl https://npm.taobao.org/dist
pnpm config set NVM_NODEJS_ORG_MIRROR http://npm.taobao.org/mirrors/node
pnpm config set NVM_IOJS_ORG_MIRROR http://npm.taobao.org/mirrors/iojs
pnpm config set PHANTOMJS_CDNURL https://npm.taobao.org/dist/phantomjs
pnpm config set ELECTRON_MIRROR http://npm.taobao.org/mirrors/electron/
pnpm config set SASS_BINARY_SITE http://npm.taobao.org/mirrors/node-sass
pnpm config set SQLITE3_BINARY_SITE http://npm.taobao.org/mirrors/sqlite3
pnpm config set PYTHON_MIRROR http://npm.taobao.org/mirrors/python
四、安装最新版vite和vite-plugin-vue2
pnpm add vite vite-plugin-vue2 -D
五、在根目录下新建vite.conf.js
import { defineConfig } from 'vite'
// vue2的vite插件
import { createVuePlugin } from 'vite-plugin-vue2'
export default ({ mode }) => {
return defineConfig({
plugins: [
createVuePlugin({
jsx: true
})
]
})
})
六、index.html修改
- 移动public/index.html到代码根目录(和package.json同级)
在body标签中新增如下:
替换htmlWebpackPlugin插件注入的变量
htmlWebpackPlugin是webpack插件,所以不能再使用了,vite提供了vite-plugin-html插件来向index.html注入变量
安装vite-plugin-html
pnpm add vite-plugin-html -D
- 修改vite.config.js,添加配置
plugins: [ // ... createHtmlPlugin({ minify: true, inject: { data: { title: 'vue Element Admin', cdn: { css: [], js: [ '//cdn.jsdelivr.net/npm/[email protected]/dist/vue.min.js', '//cdn.jsdelivr.net/npm/[email protected]/dist/vue-router.min.js', '//cdn.jsdelivr.net/npm/[email protected]/dist/vuex.min.js', '//cdn.jsdelivr.net/npm/[email protected]/dist/axios.min.js' ] } } } }), // ... ]
修改index.html
修改title
<%= title %> 修改css和js引入
<% for (var i in cdn.css) { %> <% } %> <% for (var i in cdn.js) { %> <% } %>
七、环境变量更换
出于安全考虑,vite只能识别以VITE_
开头的环境变量了,原VUE_环境变量不生效了,同时,也不能使用process.env.xxx
来读取环境变量了。需要修改vite.conf.js配置,手动添加process.env.xxx
环境变量
修改vite.conf.js配置,添加环境变量
import { defineConfig, loadEnv } from 'vite' export default ({ mode }) => { const env = loadEnv(mode, process.cwd()) return defineConfig({ define: { 'process.env': { ...env } }, }) })
- 将所有开头的
VUE_
环境变量全部替换为VITE_
- 将所有的
process.env.NODE_ENV
更改为import.meta.env.MODE
- 将所有开头为
process.env.
全部更改为import.meta.env.
八、Element-UI按需引入
# 注意本插件必须采用1.4.1版本,不能采用最新版2.0.0
pnpm add vite-plugin-style-import@^1.4.1 -D
增加vite.conf.js配置
plugins: [ // ... styleImport({ libs: [ { libraryName: 'element-ui', // styleLibraryName: 'theme-chalk', esModule: true, resolveStyle: (name) => { return `theme-chalk/${name}.css` } } ], }), // ... ]
九、添加代理
安装path-browserify
pnpm add path-browserify -D
添加vite.conf.js配置
plugin: [], // ... server: { port: 8000, //proxy: { // '/api': { // target: 'https://mock.ihx.me/mock/5baf3052f7da7e07e04a5116/antd-pro', // changeOrigin: true, // ws: false, // rewrite: (path) => path.replace(/^\/api/, ''), // } //}, },
十、package.json脚本命令修改
将脚本命令修改为如下:
"scripts": {
"dev": "vite",
"build": "vite build",
"preview": "vite preview"
},
十一、postcss配置
安装插件
pnpm add postcss autoprefixer -D
十二、添加eslint插件
安装插件
pnpm remove eslint eslint-plugin-html eslint-plugin-vue pnpm add eslint eslint-plugin-html eslint-plugin-vue eslint-config-prettier eslint-plugin-prettier prettier -D # vite-eslint插件 pnpm add vite-plugin-eslint -D
添加vite.conf.js配置
import eslintPlugin from 'vite-plugin-eslint' export default ({ mode }) => { return defineConfig({ plugins: [ // ... eslintPlugin(), // ... ] }) }
十三、在写有jsx语法的文件中添加lang="jsx"
十四、添加@别名
修改vite.conf.js配置
export default ({ mode }) => {
return defineConfig({
resolve: {
// ...
alias: [
{
find: /@\/.+/,
replacement: (val) => {
return val.replace(/^@/, path.resolve(__dirname, './src/'))
},
},
{
// this is required for the SCSS modules
find: /^~(.*)$/,
replacement: '$1'
}
]
},
)
}
十五、静态文件引入
动态组件引入
const modules = import.meta.glob('../views/**/*.vue') const currentRouter { ... // component: constantRouterComponents[item.component || item.key] || (() => import(`/src/views/${item.component}`)), component: constantRouterComponents[item.component || item.key] || modules[`../views/${item.component}.vue`], ... }
静态图片引入
直接
import
图片采用
import.meta.globEager
-
- 图片加载
export default { methods: { getImg(path) { const modules = import.meta.globEager('../../assets/img/*.svg') return modules[path].default } } }
* 2. require.context替换
// 修改前
// const req = require.context('./svg', false, /.svg$/)
// const requireAll = requireContext => requireContext.keys().map(requireContext)// 修改后
const req = import.meta.globEager('./svg/*.svg')
const requireAll = (requireContext) => Object.keys(requireContext).map((key) => requireContext[key].default)-
### 十六、雪碧图加载svg-sprite-loader替换
pnpm add vite-plugin-svg-icons -D
配置vite.config.js
import { createSvgIconsPlugin } from 'vite-plugin-svg-icons' defineConfig({ plugins: [ // ... // 雪碧图 createSvgIconsPlugin({ // 指定需要缓存的图标文件夹 iconDirs: [path.resolve(__dirname, './src/icons/svg')], // 指定symbolId格式 symbolId: 'icon-[dir]-[name]', /** * 自定义插入位置 * @default: body-last */ inject: 'body-last' | 'body-first', /** * custom dom id * @default: __svg__icons__dom__ */ customDomId: '__svg__icons__dom__' }) ] })
在main.js中添加