178 lines
6.0 KiB
TypeScript
178 lines
6.0 KiB
TypeScript
import { fileURLToPath, URL } from 'node:url'
|
||
import UnoCSS from '@unocss/vite'
|
||
import vue from '@vitejs/plugin-vue'
|
||
import vueJsx from '@vitejs/plugin-vue-jsx'
|
||
import { internalIpV4 } from 'internal-ip'
|
||
import os from 'os'
|
||
import postcsspxtorem from 'postcss-pxtorem'
|
||
import AutoImport from 'unplugin-auto-import/vite' //自动导入
|
||
import { NaiveUiResolver, VantResolver } from 'unplugin-vue-components/resolvers'
|
||
import Components from 'unplugin-vue-components/vite' //组件注册
|
||
import { type ConfigEnv, defineConfig, loadEnv } from 'vite'
|
||
import VueSetupExtend from 'vite-plugin-vue-setup-extend'
|
||
import { getComponentsDirs, getComponentsDtsPath } from './build/config/components'
|
||
import { createManualChunks } from './build/config/chunks'
|
||
import { atStartup } from './build/config/console'
|
||
import packageJson from './package.json'
|
||
|
||
function getLocalIP() {
|
||
const interfaces = os.networkInterfaces()
|
||
for (const name of Object.keys(interfaces)) {
|
||
for (const iface of interfaces[name] || []) {
|
||
// 只要 IPv4、非内网回环地址
|
||
if (iface.family === 'IPv4' && !iface.internal) {
|
||
return iface.address
|
||
}
|
||
}
|
||
}
|
||
return void 0
|
||
}
|
||
|
||
// 读取 package.json 依赖
|
||
const dependencies = Object.keys(packageJson.dependencies || {})
|
||
|
||
// 预先获取本地IP
|
||
const rawIP = getLocalIP() || (await internalIpV4())
|
||
|
||
// https://vitejs.dev/config/
|
||
/**! 不需要优化前端打包(如开启gzip) */
|
||
export default defineConfig(({ mode }: ConfigEnv) => {
|
||
// 获取当前环境的配置,如何设置第三个参数则加载所有变量,而不是以"VITE_"前缀的变量
|
||
const config = loadEnv(mode, process.cwd(), '')
|
||
const currentPlatform = config.TAURI_ENV_PLATFORM
|
||
const isPC = currentPlatform === 'windows' || currentPlatform === 'darwin' || currentPlatform === 'linux'
|
||
|
||
const serverPort = isPC ? 6130 : 5210
|
||
const componentsDirs = getComponentsDirs(currentPlatform)
|
||
const componentsDtsPath = getComponentsDtsPath(currentPlatform)
|
||
|
||
// 根据平台决定host地址
|
||
const host = (() => {
|
||
if (isPC) {
|
||
return '127.0.0.1'
|
||
}
|
||
|
||
// 移动端逻辑:检查是否为有效的内网IP地址
|
||
if (rawIP && !rawIP.endsWith('.0') && !rawIP.endsWith('.255')) {
|
||
return rawIP // 有效IP且非网段/广播地址
|
||
}
|
||
|
||
// 无效IP或特殊地址的情况
|
||
return config.TAURI_ENV_PLATFORM === 'ios'
|
||
? (rawIP ?? '127.0.0.1')
|
||
: config.TAURI_ENV_PLATFORM === 'android'
|
||
? '0.0.0.0'
|
||
: '127.0.0.1'
|
||
})()
|
||
|
||
// 是否开启启动时打印信息
|
||
atStartup(config, mode, host)()
|
||
|
||
return {
|
||
resolve: {
|
||
alias: {
|
||
// 配置主路径别名@
|
||
'@': fileURLToPath(new URL('./src', import.meta.url)),
|
||
// 配置移动端路径别名#
|
||
'#': fileURLToPath(new URL('./src/mobile', import.meta.url)),
|
||
// 配置路径别名~(根路径)
|
||
'~': fileURLToPath(new URL('.', import.meta.url))
|
||
}
|
||
},
|
||
css: {
|
||
preprocessorOptions: {
|
||
scss: {
|
||
api: 'modern-compiler',
|
||
additionalData: '@use "@/styles/scss/global/variable.scss" as *;' // 加载全局样式,使用scss特性
|
||
}
|
||
},
|
||
postcss: {
|
||
plugins: [
|
||
postcsspxtorem({
|
||
rootValue: 16, // 1rem = 16px,可根据设计稿调整
|
||
propList: ['*'], // 所有属性都转换
|
||
unitPrecision: 5, // 保留小数位数
|
||
selectorBlackList: [], // 不转换的类名(可选)
|
||
replace: true, // 替换原来的 px
|
||
mediaQuery: false, // 是否在媒体查询中转换
|
||
minPixelValue: 0 // 最小转换单位
|
||
})
|
||
]
|
||
}
|
||
},
|
||
plugins: [
|
||
/**
|
||
* vue3.5.0已支持解构并具有响应式
|
||
* */
|
||
vue(),
|
||
VueSetupExtend(), // setup 中给组件命名(keepAlive需要)
|
||
vueJsx(), // 开启jsx功能
|
||
UnoCSS(), // 开启UnoCSS
|
||
AutoImport({
|
||
imports: [
|
||
'vue',
|
||
'vue-router',
|
||
'pinia',
|
||
{ 'naive-ui': ['useDialog', 'useMessage', 'useNotification', 'useLoadingBar', 'useModal'] }
|
||
],
|
||
dts: 'src/typings/auto-imports.d.ts'
|
||
}),
|
||
/**自动导入组件,但是不会自动导入jsx和tsx*/
|
||
Components({
|
||
dirs: componentsDirs, // 根据平台加载对应组件目录
|
||
resolvers: [NaiveUiResolver(), VantResolver()],
|
||
dts: componentsDtsPath
|
||
})
|
||
],
|
||
worker: {
|
||
format: 'es' as const
|
||
},
|
||
build: {
|
||
// 设置兼容低版本浏览器的目标
|
||
target: ['chrome87', 'edge88', 'firefox78', 'safari14'],
|
||
cssCodeSplit: true, // 启用 CSS 代码拆分
|
||
minify: 'esbuild' as const, // 指定使用哪种混淆器
|
||
// chunk 大小警告的限制(kb)
|
||
chunkSizeWarningLimit: 1200,
|
||
// esbuild配置,解决低版本浏览器兼容性问题
|
||
esbuild: {
|
||
supported: {
|
||
'top-level-await': false
|
||
},
|
||
// 生产环境移除 console.log、debugger(默认移除注释)
|
||
drop: mode === 'production' ? ['console', 'debugger'] : []
|
||
},
|
||
sourcemap: false, // 关闭源码映射
|
||
// 分包配置
|
||
rollupOptions: {
|
||
output: {
|
||
chunkFileNames: 'static/js/[name]-[hash].js', // 引入文件名的名称
|
||
entryFileNames: 'static/js/[name]-[hash].js', // 包的入口文件名称
|
||
assetFileNames: 'static/[ext]/[name]-[hash].[ext]', // 资源文件像 字体,图片等
|
||
manualChunks: createManualChunks(dependencies)
|
||
}
|
||
}
|
||
},
|
||
// Vite options tailored for Tauri development and only applied in `tauri dev` or `tauri build`
|
||
//
|
||
// 1. prevent vite from obscuring rust errors
|
||
clearScreen: false,
|
||
// 2. tauri expects a fixed port, fail if that port is not available
|
||
server: {
|
||
hmr: {
|
||
protocol: 'ws',
|
||
host: host,
|
||
port: serverPort
|
||
},
|
||
cors: true, // 配置 CORS
|
||
host: '0.0.0.0',
|
||
port: serverPort,
|
||
strictPort: true,
|
||
watch: {
|
||
// 3. tell vite to ignore watching `src-tauri`
|
||
ignored: ['**/src-tauri/**']
|
||
}
|
||
}
|
||
}
|
||
})
|