vite.config.ts 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. import { resolve } from 'node:path'
  2. import type { ConfigEnv, UserConfig } from 'vite'
  3. import { loadEnv } from 'vite'
  4. import { format } from 'date-fns'
  5. import { wrapperEnv } from './build/utils'
  6. import { createVitePlugins } from './build/vite/plugin'
  7. import { OUTPUT_DIR } from './build/constant'
  8. import { createProxy } from './build/vite/proxy'
  9. import pkg from './package.json'
  10. const { dependencies, devDependencies, name, version } = pkg
  11. // 当使用文件系统路径的别名时,请始终使用绝对路径。相对路径的别名值会原封不动地被使用,因此无法被正常解析。
  12. // path.resolve () 方法用于将一系列路径段解析为绝对路径。它通过处理从右到左的路径序列来工作,在每个路径之前添加,直到创建绝对路径。
  13. function pathResolve(dir: string) {
  14. return resolve(process.cwd(), '.', dir)
  15. }
  16. const __APP_INFO__ = {
  17. // APP 后台管理信息
  18. pkg: { dependencies, devDependencies, name, version },
  19. // 最后编译时间
  20. lastBuildTime: format(new Date(), 'yyyy-MM-dd HH:mm:ss'),
  21. }
  22. /** @type {import('vite').UserConfig} */
  23. export default ({ command, mode }: ConfigEnv): UserConfig => {
  24. // process.cwd() 方法返回 Node.js 进程的当前工作目录
  25. // mode 返回应用的环境模式 development(开发环境) 或者 production(生产环境)
  26. // command 返回(dev/serve 或 build)命令模式,yarn dev 返回 dev/serve yarn build 返回 build
  27. const root = process.cwd()
  28. // loadEnv() 根据 mode 检查 root(项目根路径) 路径下 .env、.env.development 环境文件,输出 NODE_ENV 和 VITE_ 开头的键值队
  29. const env = loadEnv(mode, root)
  30. // 读取并处理所有环境变量配置文件 .env
  31. const viteEnv = wrapperEnv(env)
  32. const { VITE_PUBLIC_PATH, VITE_DROP_CONSOLE, VITE_PORT, VITE_PROXY, VITE_GLOB_PROD_MOCK }
  33. = viteEnv
  34. const prodMock = VITE_GLOB_PROD_MOCK
  35. const isBuild = command === 'build'
  36. // command === 'build'
  37. return {
  38. base: VITE_PUBLIC_PATH,
  39. root,
  40. // 别名
  41. resolve: {
  42. alias: [
  43. // @/xxxx => src/xxxx
  44. {
  45. find: /\@\//,
  46. replacement: `${pathResolve('src')}/`,
  47. },
  48. // #/xxxx => types/xxxx
  49. {
  50. find: /\#\//,
  51. replacement: `${pathResolve('types')}/`,
  52. },
  53. ],
  54. dedupe: ['vue'],
  55. },
  56. // 定义全局常量替换方式
  57. define: {
  58. // 在生产中 启用/禁用 intlify-devtools 和 vue-devtools 支持,默认值 false
  59. __INTLIFY_PROD_DEVTOOLS__: false,
  60. __APP_INFO__: JSON.stringify(__APP_INFO__),
  61. },
  62. esbuild: {
  63. // 使用 esbuild 压缩 剔除 console.log
  64. drop: VITE_DROP_CONSOLE ? ['debugger', 'console'] : [],
  65. // minify: true, // minify: true, 等于 minify: 'esbuild',
  66. },
  67. build: {
  68. // 设置最终构建的浏览器兼容目标
  69. target: 'es2015',
  70. minify: 'esbuild',
  71. // 构建后是否生成 source map 文件(用于线上报错代码报错映射对应代码)
  72. sourcemap: false,
  73. cssTarget: 'chrome80',
  74. // 指定输出路径(相对于 项目根目录)
  75. outDir: OUTPUT_DIR,
  76. // 只有 minify 为 terser 的时候, 本配置项才能起作用
  77. // terserOptions: {
  78. // compress: {
  79. // // 防止 Infinity 被压缩成 1/0,这可能会导致 Chrome 上的性能问题
  80. // keep_infinity: true,
  81. // // 打包是否自动删除 console
  82. // drop_console: VITE_DROP_CONSOLE,
  83. // },
  84. // },
  85. // 启用/禁用 gzip 压缩大小报告
  86. // 压缩大型输出文件可能会很慢,因此禁用该功能可能会提高大型项目的构建性能
  87. reportCompressedSize: true,
  88. // chunk 大小警告的限制(以 kbs 为单位)
  89. chunkSizeWarningLimit: 2000,
  90. // 自定义底层的 Rollup 打包配置
  91. rollupOptions: {
  92. // 静态资源分类打包
  93. output: {
  94. chunkFileNames: 'js/[name]-[hash].js', // 引入文件名的名称
  95. entryFileNames: 'js/[name]-[hash].js', // 包的入口文件名称
  96. assetFileNames: '[ext]/[name]-[hash].[ext]', // 资源文件像 字体,图片等
  97. // 将 node_modules 三方依赖包最小化拆分
  98. manualChunks(id) {
  99. if (id.includes('node_modules')) {
  100. const paths = id.toString().split('node_modules/')
  101. if (paths[2]) {
  102. return paths[2].split('/')[0].toString()
  103. }
  104. return paths[1].split('/')[0].toString()
  105. }
  106. },
  107. },
  108. },
  109. },
  110. css: {
  111. preprocessorOptions: {
  112. less: {
  113. modifyVars: {},
  114. javascriptEnabled: true,
  115. // 注入全局 less 变量
  116. additionalData: `@import "src/styles/var.less";`,
  117. },
  118. },
  119. },
  120. server: {
  121. host: true,
  122. // 服务启动时是否自动打开浏览器
  123. open: true,
  124. // 服务端口号
  125. port: Number(VITE_PORT),
  126. proxy: createProxy(VITE_PROXY),
  127. // 预热文件以降低启动期间的初始页面加载时长
  128. warmup: {
  129. // 预热的客户端文件:首页、views、 components
  130. clientFiles: ['./index.html', './src/{views,components}/*'],
  131. },
  132. // proxy: {
  133. // '/api': {
  134. // target: '',
  135. // changeOrigin: true,
  136. // rewrite: (path) => path.replace(/^\/api/, '/api/v1')
  137. // }
  138. // }
  139. },
  140. optimizeDeps: {
  141. /**
  142. * 依赖预构建,vite 启动时会将下面 include 里的模块,编译成 esm 格式并缓存到 node_modules/.vite 文件夹,
  143. * 页面加载到对应模块时如果浏览器有缓存就读取浏览器缓存,如果没有会读取本地缓存并按需加载
  144. * 尤其当您禁用浏览器缓存时(这种情况只应该发生在调试阶段)必须将对应模块加入到 include 里,
  145. * 否则会遇到开发环境切换页面卡顿的问题(vite 会认为它是一个新的依赖包会重新加载并强制刷新页面),
  146. * 因为它既无法使用浏览器缓存,又没有在本地 node_modules/.vite 里缓存
  147. * 温馨提示:如果你使用的第三方库是全局引入,也就是引入到 src/main.ts 文件里,
  148. * 就不需要再添加到 include 里了,因为 vite 会自动将它们缓存到 node_modules/.vite
  149. */
  150. include: [
  151. 'pinia',
  152. 'lodash-es',
  153. 'axios',
  154. ],
  155. // 打包时强制排除的依赖项
  156. exclude: [
  157. // https://www.mulingyuer.com/archives/928/
  158. 'vant',
  159. '@vant/use',
  160. ],
  161. },
  162. // 加载插件
  163. plugins: createVitePlugins(viteEnv, isBuild, prodMock),
  164. }
  165. }