?前言
大家好,我是yma16,本文分享 前端vite+vue3——利用環境變量和路由對前端區分h5和pc模塊打包(從0到1)。
背景:
前端本地開發pc和h5的項目,發布時需要區分開h5和pc的頁面
vite
Vite 通過在一開始將應用中的模塊區分為 依賴 和 源碼 兩類,改進了開發服務器啟動時間。
- 依賴 大多為在開發時不會變動的純 JavaScript。一些較大的依賴(例如有上百個模塊的組件庫)處理的代價也很高。依賴也通常會存在多種模塊化格式(例如 ESM 或者
CommonJS)。Vite 將會使用 esbuild 預構建依賴。esbuild 使用 Go 編寫,并且比以 JavaScript 編寫的打包器預構建依賴快
10-100 倍。
- 源碼 通常包含一些并非直接是 JavaScript 的文件,需要轉換(例如 JSX,CSS 或者 Vue/Svelte 組件),時常會被編輯。同時,并不是所有的源碼都需要同時被加載(例如基于路由拆分的代碼模塊)。
Vite 以 原生 ESM 方式提供源碼。這實際上是讓瀏覽器接管了打包程序的部分工作:Vite
只需要在瀏覽器請求源碼時進行轉換并按需提供源碼。根據情景動態導入代碼,即只在當前屏幕上實際使用時才會被處理。
node系列往期文章
node_windows環境變量配置
node_npm發布包
linux_配置node
node_nvm安裝配置
node筆記_http服務搭建(渲染html、json)
node筆記_讀文件
node筆記_寫文件
node筆記_連接mysql實現crud
node筆記_formidable實現前后端聯調的文件上傳
node筆記_koa框架介紹
node_koa路由
node_生成目錄
node_讀寫excel
node筆記_讀取目錄的文件
node筆記——調用免費qq的smtp發送html格式郵箱
node實戰——搭建帶swagger接口文檔的后端koa項目(node后端就業儲備知識)
node實戰——后端koa結合jwt連接mysql實現權限登錄(node后端就業儲備知識)
node實戰——koa給郵件發送驗證碼并緩存到redis服務(node后端儲備知識)
koa系列項目文章
前端vite+vue3結合后端node+koa——實現代碼模板展示平臺(支持模糊搜索+分頁查詢)
node+vue3+mysql前后分離開發范式——實現對數據庫表的增刪改查
node+vue3+mysql前后分離開發范式——實現視頻文件上傳并渲染
koa-vue性能監控到封裝sdk系列文章
性能監控系統搭建——node_koa實現性能監控數據上報(第一章)
性能監控系統搭建——vue3實現性能監控數據展示(第二章)
性能監控計算——封裝帶性能計算并上報的npm包(第三章)
?初始化前端項目到打包
思路:利用打包時候的變量去區分是否打包需要的路由,即需要區分pc端和h5的路由。
example:這里我使用element-plus模擬pc端頁面,vant模擬移動端頁面。
對比組件:時間選擇
基本的package.json
{"name": "vue-with-vite","version": "0.0.0","scripts": {"dev": "vite","build": "vite build","build-h5": "vite build --mode h5","build-pc": "vite build --mode pc","preview": "vite preview --port 4173"},"dependencies": {"element-plus": "^2.7.3","vant": "^4.9.1","vue": "^3.2.37","vue-router": "^4.4.0"},"devDependencies": {"@types/node": "^18.19.6","@vitejs/plugin-vue": "^4.0.0","prettier": "^3.2.2","typescript": "^4.9.3","vite": "^4.0.0","vue-tsc": "^1.0.11"}
}
💖配置vite變量文件
配置env文件
env
# .env
VITE_APP_TITLE=My App (staging)
VITE_BUILD_MODE=dev# .env.h5
NODE_ENV=production
VITE_APP_TITLE=My App (staging)
VITE_BUILD_MODE=h5# .env.pc
NODE_ENV=production
VITE_APP_TITLE=My App (staging)
VITE_BUILD_MODE=PC
對應文件位置如下
💖配置vite打包時的區分路徑
vite.config.tsn內容配置如下
// @ts-ignore
import { defineConfig,loadEnv } from 'vite'
import { resolve } from "path";
// @ts-ignore
import vue from '@vitejs/plugin-vue'
// https://vitejs.dev/config/
// @ts-ignore
export default defineConfig(({mode})=>{// 運行模式console.log('mode',mode)// 當前路徑console.log('process.cwd()',process.cwd())// @ts-ignoreconst env=loadEnv(mode,process.cwd())console.log('env',env)return {// 打包相對路徑base: '/',build:{// 輸出的目錄 h5 pcoutDir:'dist/'+env.VITE_BUILD_MODE},server: {host:true,},resolve: {alias: {"@": resolve(__dirname, "src"),},},plugins: [vue()],}
});
💖h5頁面
MobilePage.vue主要顯示移動端時間選擇
<template><van-cell title="選擇單個日期" :value="date" @click="show = true" /><van-calendar v-model:show="show" @confirm="onConfirm" />
</template><script setup>import { ref } from 'vue';const date = ref('');const show = ref(false);const formatDate = (date) => `${date.getMonth() + 1}/${date.getDate()}`;const onConfirm = (value) => {show.value = false;date.value = formatDate(value);};
</script>
頁面效果:
💖pc頁面
WebPage.vue主要顯示網頁選擇時間
<template><div class="demo-date-picker"><el-date-pickerv-model="value"type="date"placeholder="Pick a day"format="YYYY/MM/DD"value-format="YYYY-MM-DD"><template #default="cell"><div class="cell" :class="{ current: cell.isCurrent }"><span class="text">{{ cell.text }}</span><span v-if="isHoliday(cell)" class="holiday" /></div></template></el-date-picker></div>
</template><script setup>import { ref } from 'vue'const value = ref('2021-10-29')const holidays = ['2021-10-01','2021-10-02','2021-10-03','2021-10-04','2021-10-05','2021-10-06','2021-10-07',]const isHoliday = ({ dayjs }) => {return holidays.includes(dayjs.format('YYYY-MM-DD'))}
</script><style scoped>.cell {height: 30px;padding: 3px 0;box-sizing: border-box;}.cell .text {width: 24px;height: 24px;display: block;margin: 0 auto;line-height: 24px;position: absolute;left: 50%;transform: translateX(-50%);border-radius: 50%;}.cell.current .text {background: #626aef;color: #fff;}.cell .holiday {position: absolute;width: 6px;height: 6px;background: var(--el-color-danger);border-radius: 50%;bottom: 0px;left: 50%;transform: translateX(-50%);}
</style>
頁面效果
💖關鍵router配置:區分打包路由
main.js的入口文件區分打包
// @ts-ignore
import { createApp } from 'vue'
// @ts-ignore
import App from './App.vue'
// 路由
// @ts-ignore
import router from "./router/index.ts";// pc 端 element-plus
// @ts-ignore
import ElementPlus from 'element-plus'
import 'element-plus/dist/index.css'// 移動端 vant
// 1. 引入你需要的組件
// @ts-ignore
import { Calendar,Cell } from 'vant';
// 2. 引入組件樣式
import 'vant/lib/index.css';const app=createApp(App)
app.use(router)// 環境變量
// @ts-ignore
const env = import.meta.env
console.log('env', env)
const { VITE_BUILD_MODE } = env
if (VITE_BUILD_MODE === 'h5') {
// h5 組件app.use(Calendar)app.use(Cell)
}
else if(VITE_BUILD_MODE === 'pc'){
//pc 組件app.use(ElementPlus)
}
else if(VITE_BUILD_MODE === 'dev'){
// dev開發模式 h5 組件 pc組件都引入app.use(Calendar)app.use(Cell)app.use(ElementPlus)
}app.mount('#app')
router/index.ts 路由配置,讀取目錄modules下的h5和pc路由
// @ts-ignore
import * as VueRouter from "vue-router";
// @ts-ignore
let modules = []
// @ts-ignore
const routeModuleList = []
// @ts-ignore
console.log('modules', modules)
// 環境變量
// @ts-ignore
const env = import.meta.env
console.log('env', env)
const { VITE_BUILD_MODE } = env
if (VITE_BUILD_MODE === 'h5') {// import.meta.glob() 直接引入所有的模塊 Vite 獨有的功能// @ts-ignoremodules = import.meta.glob('./modules/h5/*.ts', { eager: true });
}
else if(VITE_BUILD_MODE === 'pc'){// import.meta.glob() 直接引入所有的模塊 Vite 獨有的功能// @ts-ignoremodules = import.meta.glob('./modules/pc/*.ts', { eager: true });
}
else if(VITE_BUILD_MODE === 'dev'){// dev 開發模式pc和后所有路徑都加載// @ts-ignoremodules = import.meta.glob('./modules/**/*.ts', { eager: true });
}// 加入到路由集合中
// @ts-ignore
Object.keys(modules).forEach((key) => {
// @ts-ignoreconst mod = modules[key].default || {};const modList = Array.isArray(mod) ? [...mod] : [mod];console.log('modList', modList)// @ts-ignorerouteModuleList.push(...modList);
});// @ts-ignore
console.log('routeModuleList', routeModuleList)
const router = VueRouter.createRouter({// 4. 內部提供了 history 模式的實現。為了簡單起見,我們在這里使用 hash 模式。history: VueRouter.createWebHashHistory(),// @ts-ignoreroutes: routeModuleList,
});
// 路由權限 beforeResolve
// @ts-ignore
router.beforeResolve(async (to, from, next) => {next()
});
export default router;
h5/h5.ts 移動端路由
export default {path: '/h5',name: 'MobilePage',component: () => {// @ts-ignorereturn import('@/components/h5/MobilePage.vue')},meta: {title: 'MobilePage',},children: [],
};
pc/pc.ts 網頁路由
export default {path: '/pc',name: 'WebPage',component: () => {// @ts-ignorereturn import('@/components/pc/WebPage.vue')},meta: {title: 'WebPage',},children: [],
};
💖vite build:h5頁面
運行 build-h5 會拉起 .env.h5變量
# npm run build-h5
打包文件在dist/h5目錄
打開build文件 打包成功!
💖vite build:pc頁面
運行 build-pc 會拉起 .env.pc變量
# npm run build-pc
打包文件在dist/pc目錄
打開打包build index頁面
?結束
項目地址:https://gitcode.com/qq_38870145/vue-build-pc-h5
本文分享到這結束,如有錯誤或者不足之處歡迎指出!
👍 點贊,是我創作的動力!
?? 收藏,是我努力的方向!
?? 評論,是我進步的財富!
💖 最后,感謝你的閱讀!