[Vue] 專案升級之路 — Vite 篇

施慶銓
11 min readDec 26, 2022

--

專案升級之路的系列文章是紀錄我將Composition API與Vite 導入到公司專案的過程. 記錄的範圍只包含升級Vue 以及Vue 相關的libraries (Vuex/Router/i18n/Jest)與解決相容性錯誤的問題. 這些升級的事項發生在2022/08, 日後在做這類升級時, 或許官方會出更多相容性版本, 就可能不會發生以下的問題

這系列文章包含這些章節

[Vue] 專案升級之路 — Vite

[Vue] 專案升級之路 — Composition API

[Vue] 專案升級之路 — Vuex/Vue Router

[Vue] 專案升級之路 — Vue I18n

[Vue] 專案升級之路 — Unit Test

這篇文章是用來記錄把Vite 導入到公司專案的過程, 原本的專案是用VueCLI 建立專案的架構, 預設的local server是用Webpack, 為了能夠提升啟動速度, 所以決定試試看是否能夠導入Vite, 並將Webpack 取代

最後測試結果

取代前:

使用Vue CLI內建的Webpack 啟動local server

啟動時間需要約: 1minue

取代後:

使用Vite 起動local server

啟動時間需要約: 2–3sencods

更換紀錄

Vite 官方文件

https://vitejs.dev/guide/

Node.js 最低支援版本

  • 14.18+
  • 16+

Vite可支援的Vue 版本

Vite 目前提供了幾個plugin給Vue 的各個版本, 讓Vue 的開發者可以使用Vite

Vue 3 SFC support via @vitejs/plugin-vue
Vue 3 JSX support via @vitejs/plugin-vue-jsx
Vue 2.7 support via @vitejs/plugin-vue2
Vue <2.7 support via vite-plugin-vue2

可以依照自己Vue的版本選擇對應的plugin 安裝

Installation

npm i -D vite

因為之後我為了將我的專案升級到Vue2.7 所以我另外安裝相容的plugin

npm i -D @vitejs/plugin-vue2

Package.json


{
"scripts": {
"dev": "vite", // start dev server, aliases: `vite dev`, `vite serve`
"build": "vite build", // build for production
"preview": "vite preview" // locally preview production build
}
}

Breaking Changes

Upgrade the plugins if installed

- stylelint(`^12.0.0` -> `^14.9.1`)
- stylelint-config-recommended-scss (`^4.0.0` -> `^7.0.0`)
- stylelint-config-standard (`^19.0.0` -> `^26.0.0`)
- stylelint-scss (`^3.12.1` -> `^4.3.0`)
- node-sass(`^4.13.1` -> `^7.0.1`)

SFC 內所有import 的檔案都要加入file extension

webpack 可以幫忙處理parse import的檔案, 因此可以省略副檔名, 但Vite 沒有, 若要在Vite server 也可以省略不寫副檔名, 依然仍可內有設定

官網教學: https://vitejs.dev/config/shared-options.html#resolve-extensions

不過官方已經不推薦這種寫法, 所以建議還是補上去.

快速補上去的方式可參考先前的文章

啟動有HTTPS的local server

如果用webpack 啟動local server時, 若要使用HTTPS, 可在vue-cli-service serve的指令加入 “--https“, 這樣server 啟動的時候, 就是https://.x.x.x.x.

如果是Vite的話, 則需要安裝@vitejs/plugin-basic-ssl.

// vite.config.js
import basicSsl from '@vitejs/plugin-basic-ssl'

export default {
plugins: [
basicSsl()
]
}

使用.env.[mode]內的變數

官網文件教學: https://vitejs.dev/guide/env-and-mode.html#env-variables-and-modes

原本在VueCLI 的專案中, 若在.env 內宣告自定義的變數, 如

// .env.test

VUE_APP_CUSTOM_VERSION=1.0

現在如果要取得VUE_APP_CUSTOM_VERSION, 則需要改成這樣

// .env.test

VITE_APP_CUSTOM_VERSION=1.0

另外原本取得變數的方式是

process.env.VUE_APP_CUSTOM_VERSION

在Vite裡面也需要改成

import.meta.env.VITE_APP_CUSTOM_VERSION

動態載入圖片

當原本在VueCLI 的專案時, 因為使用webpack 來pre-build 專案程式碼, 而webpack 可以將使用CommonJS的語法編譯成browser 可以看得懂的程式碼, 所以可以用 require(xxx) 這樣的方式來動態顯示圖片

<img
:src="require(`@/assets/images/image.svg`)"
>

而Vite 預設是使用esbuild來pre-build檔案, 而esbuild並不會處理CommonJS的語法, 除非再另外引入plugin:vite-plugin-commonjs(reference). 而官方有提供新的方式: import.meta.url + URL constructor,

image = new URL(`/src/assets/images/image.svg`, import.meta.url).href;

官網教學文件: https://vitejs.dev/guide/assets.html#new-url-url-import-meta-url

根據官網敘述, 利用esbuild 原生的語法import.meta.url可以取得image 在專案內的相對位置, 再搭配瀏覽器環境原生的語法URL constructor, 取得可以讓JavaScript 解析的完整路徑

另外, 在production 模式下, Vite 也會自動轉換成編譯後的路徑, 所以無需再多做設定

安裝SASS

在原本使用Webpack情況下, 我們可以透過loader 將sass/scss轉譯成css 檔案, 但當我們改用Vite的時候, 我們就必須自己另外安裝css preprocessor

官網教學文件: https://vitejs.dev/guide/features.html#css-pre-processors

# .scss and .sass
npm add -D sass

# .less
npm add -D less

# .styl and .stylus
npm add -D stylus

Note: 如果發現在安裝新版的sass後,發現map-get()語法出現錯誤時, 可參考這個解決方法

Sass map.get() doesn’t work. map-get() does. What gives?

主要在.sass/.scss 檔案內的最上方加上這一段

@use "sass:map";

並將map-get()改成map.get() 即可

移除CSS import 路徑的prefix”~”

當import path 上有“~“, 例如: import “~/a/b/c”, 原本是要讓Webpack內的css-loade知道該路徑需要去node的相依模組內找到符合”/a/b/c”的檔案或模組.

官網說明文件: https://webpack.docschina.org/loaders/css-loader/#import

而Vite 已經不使用Webpack, 所以也就不會使用css loader來解析”~”, 因此可能會出現這樣的錯誤訊息

In vite it is simply @import 'npm_package/path/file.scss' without any extra prefix like ~.

要解決這個問題只要將這個prefix”~” 移除即可

vite.config.js adjustment

大部分的設定都可以沿用vue.config.js

不過使用 Vite 的時候, Vite預設不會將打包的檔案分類, dist資料夾只會長這樣

// /dist/

/dist/
- /assets/ //All static files are put here
- index.html
- favicon.png

如果需要做檔案分類的話, 可以在vite.config.js 加入”build設定

// vite.config.js

build: {
...,
+ rollupOptions: {
+ output: {
+ chunkFileNames: 'js/[name]-[hash].js',
+ entryFileNames: 'js/[name]-[hash].js',
+ assetFileNames: ({ name }) => {
+ if (/\\.(gif|jpe?g|png|svg)$/.test(name ?? '')) {
+ return 'images/[name]-[hash][extname]';
+ }
+
+ if (/\\.css$/.test(name ?? '')) {
+ return 'css/[name]-[hash][extname]';
+ }
+
+ if (/\\.(woff|ttf|eot)$/.test(name ?? '')) {
+ return 'fonts/[name]-[hash][extname]';
+ }
+
+ // default value
+ // ref: <https://rollupjs.org/guide/en/#outputassetfilenames>
+ return '[name]-[hash][extname]';
+ },
+ },
+ },
},

其他障礙排除

Stylelint 發生錯誤 “Unknown word CssSyntaxError”

解決方式: 安裝 “stylelint-config-recommended-vue“ 並在 “stylelintrc.js” 的extends 引用該套件

// stylelintrc.js

module.exports = {
...,
extends: [
...,
"stylelint-config-recommended-vue/scss",
],
...
}

Unable to resolve the path to the module

解決方式: 在eslintrc.js內加入 moduleDirectory

參考來源: https://www.codegrepper.com/code-examples/javascript/unable+to+resolve+path+to+module+eslint(import%2Fno-unresolved)+absoute+path

module.exports = {
settings: {
'import/resolver': {
node: {
moduleDirectory: ['src/'],
},
},
},
}

當用Jest 跑unit test cases時, 出現此錯誤“SyntaxError: Cannot use ‘import.meta’ outside a module”

解決方式:

  1. Install babel-plugin-transform-import-meta: Replace import.meta.url for Node.js environments.
  2. Install babel-preset-vite: Replace import.meta.glob for Node.js environments.
// babel.config.js

module.exports = {
env: {
test: {
presets: ['babel-preset-vite'],
plugins: [
'babel-plugin-transform-import-meta',
]
}
}
}

CLI Engine is not a constructor

解決方式: 更新eslinteslint-plugin-vue 版本到以下版本

參考來源: https://github.com/eslint/eslint/issues/15175#issuecomment-1016552619

   "eslint": "^6.8.0",
"eslint-plugin-vue": "^6.2.2"

無法建立WebSocket 連現時

修改vite.config.js 內websocket的proxy設定

'/ws': {
target: [URL],
ws: true, // Add "ws" property
},

Sign up to discover human stories that deepen your understanding of the world.

Free

Distraction-free reading. No ads.

Organize your knowledge with lists and highlights.

Tell your story. Find your audience.

Membership

Read member-only stories

Support writers you read most

Earn money for your writing

Listen to audio narrations

Read offline with the Medium app

--

--

施慶銓
施慶銓

No responses yet

Write a response