base.js 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. const { semver, loadModule } = require('@vue/cli-shared-utils')
  2. module.exports = (api, options) => {
  3. api.chainWebpack(webpackConfig => {
  4. const isLegacyBundle = process.env.VUE_CLI_MODERN_MODE && !process.env.VUE_CLI_MODERN_BUILD
  5. const resolveLocal = require('../util/resolveLocal')
  6. const getAssetPath = require('../util/getAssetPath')
  7. const inlineLimit = 4096
  8. const genAssetSubPath = dir => {
  9. return getAssetPath(
  10. options,
  11. `${dir}/[name]${options.filenameHashing ? '.[hash:8]' : ''}.[ext]`
  12. )
  13. }
  14. const genUrlLoaderOptions = dir => {
  15. return {
  16. limit: inlineLimit,
  17. // use explicit fallback to avoid regression in url-loader>=1.1.0
  18. fallback: {
  19. loader: require.resolve('file-loader'),
  20. options: {
  21. name: genAssetSubPath(dir)
  22. }
  23. }
  24. }
  25. }
  26. webpackConfig
  27. .mode('development')
  28. .context(api.service.context)
  29. .entry('app')
  30. .add('./src/main.js')
  31. .end()
  32. .output
  33. .path(api.resolve(options.outputDir))
  34. .filename(isLegacyBundle ? '[name]-legacy.js' : '[name].js')
  35. .publicPath(options.publicPath)
  36. webpackConfig.resolve
  37. // This plugin can be removed once we switch to Webpack 6
  38. .plugin('pnp')
  39. .use({ ...require('pnp-webpack-plugin') })
  40. .end()
  41. .extensions
  42. .merge(['.mjs', '.js', '.jsx', '.vue', '.json', '.wasm'])
  43. .end()
  44. .modules
  45. .add('node_modules')
  46. .add(api.resolve('node_modules'))
  47. .add(resolveLocal('node_modules'))
  48. .end()
  49. .alias
  50. .set('@', api.resolve('src'))
  51. webpackConfig.resolveLoader
  52. .plugin('pnp-loaders')
  53. .use({ ...require('pnp-webpack-plugin').topLevelLoader })
  54. .end()
  55. .modules
  56. .add('node_modules')
  57. .add(api.resolve('node_modules'))
  58. .add(resolveLocal('node_modules'))
  59. webpackConfig.module
  60. .noParse(/^(vue|vue-router|vuex|vuex-router-sync)$/)
  61. // js is handled by cli-plugin-babel ---------------------------------------
  62. // vue-loader --------------------------------------------------------------
  63. // try to load vue in the project
  64. // fallback to peer vue package in the instant prototyping environment
  65. const vue = loadModule('vue', api.service.context) || loadModule('vue', __dirname)
  66. if (vue && semver.major(vue.version) === 2) {
  67. // for Vue 2 projects
  68. const vueLoaderCacheConfig = api.genCacheConfig('vue-loader', {
  69. 'vue-loader': require('vue-loader/package.json').version,
  70. '@vue/component-compiler-utils': require('@vue/component-compiler-utils/package.json').version,
  71. 'vue-template-compiler': require('vue-template-compiler/package.json').version
  72. })
  73. webpackConfig.resolve
  74. .alias
  75. .set(
  76. 'vue$',
  77. options.runtimeCompiler
  78. ? 'vue/dist/vue.esm.js'
  79. : 'vue/dist/vue.runtime.esm.js'
  80. )
  81. webpackConfig.module
  82. .rule('vue')
  83. .test(/\.vue$/)
  84. .use('cache-loader')
  85. .loader(require.resolve('cache-loader'))
  86. .options(vueLoaderCacheConfig)
  87. .end()
  88. .use('vue-loader')
  89. .loader(require.resolve('vue-loader'))
  90. .options(Object.assign({
  91. compilerOptions: {
  92. whitespace: 'condense'
  93. }
  94. }, vueLoaderCacheConfig))
  95. webpackConfig
  96. .plugin('vue-loader')
  97. .use(require('vue-loader').VueLoaderPlugin)
  98. } else if (vue && semver.major(vue.version) === 3) {
  99. // for Vue 3 projects
  100. const vueLoaderCacheConfig = api.genCacheConfig('vue-loader', {
  101. 'vue-loader': require('vue-loader-v16/package.json').version,
  102. '@vue/compiler-sfc': require('@vue/compiler-sfc/package.json').version
  103. })
  104. webpackConfig.resolve
  105. .alias
  106. .set(
  107. 'vue$',
  108. options.runtimeCompiler
  109. ? 'vue/dist/vue.esm-bundler.js'
  110. : 'vue/dist/vue.runtime.esm-bundler.js'
  111. )
  112. webpackConfig.module
  113. .rule('vue')
  114. .test(/\.vue$/)
  115. .use('cache-loader')
  116. .loader(require.resolve('cache-loader'))
  117. .options(vueLoaderCacheConfig)
  118. .end()
  119. .use('vue-loader')
  120. .loader(require.resolve('vue-loader-v16'))
  121. .options({
  122. ...vueLoaderCacheConfig,
  123. babelParserPlugins: ['jsx', 'classProperties', 'decorators-legacy']
  124. })
  125. .end()
  126. .end()
  127. webpackConfig
  128. .plugin('vue-loader')
  129. .use(require('vue-loader-v16').VueLoaderPlugin)
  130. // feature flags <http://link.vuejs.org/feature-flags>
  131. webpackConfig
  132. .plugin('feature-flags')
  133. .use(require('webpack').DefinePlugin, [{
  134. __VUE_OPTIONS_API__: 'true',
  135. __VUE_PROD_DEVTOOLS__: 'false'
  136. }])
  137. }
  138. // static assets -----------------------------------------------------------
  139. webpackConfig.module
  140. .rule('images')
  141. .test(/\.(png|jpe?g|gif|webp)(\?.*)?$/)
  142. .use('url-loader')
  143. .loader(require.resolve('url-loader'))
  144. .options(genUrlLoaderOptions('img'))
  145. // do not base64-inline SVGs.
  146. // https://github.com/facebookincubator/create-react-app/pull/1180
  147. webpackConfig.module
  148. .rule('svg')
  149. .test(/\.(svg)(\?.*)?$/)
  150. .use('file-loader')
  151. .loader(require.resolve('file-loader'))
  152. .options({
  153. name: genAssetSubPath('img')
  154. })
  155. webpackConfig.module
  156. .rule('media')
  157. .test(/\.(mp4|webm|ogg|mp3|wav|flac|aac)(\?.*)?$/)
  158. .use('url-loader')
  159. .loader(require.resolve('url-loader'))
  160. .options(genUrlLoaderOptions('media'))
  161. webpackConfig.module
  162. .rule('fonts')
  163. .test(/\.(woff2?|eot|ttf|otf)(\?.*)?$/i)
  164. .use('url-loader')
  165. .loader(require.resolve('url-loader'))
  166. .options(genUrlLoaderOptions('fonts'))
  167. // Other common pre-processors ---------------------------------------------
  168. const maybeResolve = name => {
  169. try {
  170. return require.resolve(name)
  171. } catch (error) {
  172. return name
  173. }
  174. }
  175. webpackConfig.module
  176. .rule('pug')
  177. .test(/\.pug$/)
  178. .oneOf('pug-vue')
  179. .resourceQuery(/vue/)
  180. .use('pug-plain-loader')
  181. .loader(maybeResolve('pug-plain-loader'))
  182. .end()
  183. .end()
  184. .oneOf('pug-template')
  185. .use('raw')
  186. .loader(maybeResolve('raw-loader'))
  187. .end()
  188. .use('pug-plain-loader')
  189. .loader(maybeResolve('pug-plain-loader'))
  190. .end()
  191. .end()
  192. // shims
  193. webpackConfig.node
  194. .merge({
  195. // prevent webpack from injecting useless setImmediate polyfill because Vue
  196. // source contains it (although only uses it if it's native).
  197. setImmediate: false,
  198. // process is injected via DefinePlugin, although some 3rd party
  199. // libraries may require a mock to work properly (#934)
  200. process: 'mock',
  201. // prevent webpack from injecting mocks to Node native modules
  202. // that does not make sense for the client
  203. dgram: 'empty',
  204. fs: 'empty',
  205. net: 'empty',
  206. tls: 'empty',
  207. child_process: 'empty'
  208. })
  209. const resolveClientEnv = require('../util/resolveClientEnv')
  210. webpackConfig
  211. .plugin('define')
  212. .use(require('webpack').DefinePlugin, [
  213. resolveClientEnv(options)
  214. ])
  215. webpackConfig
  216. .plugin('case-sensitive-paths')
  217. .use(require('case-sensitive-paths-webpack-plugin'))
  218. // friendly error plugin displays very confusing errors when webpack
  219. // fails to resolve a loader, so we provide custom handlers to improve it
  220. const { transformer, formatter } = require('../util/resolveLoaderError')
  221. webpackConfig
  222. .plugin('friendly-errors')
  223. .use(require('@soda/friendly-errors-webpack-plugin'), [{
  224. additionalTransformers: [transformer],
  225. additionalFormatters: [formatter]
  226. }])
  227. const TerserPlugin = require('terser-webpack-plugin')
  228. const terserOptions = require('./terserOptions')
  229. webpackConfig.optimization
  230. .minimizer('terser')
  231. .use(TerserPlugin, [terserOptions(options)])
  232. })
  233. }