mirror of
https://github.com/nuxt/nuxt.git
synced 2024-11-24 06:35:10 +00:00
308 lines
8.4 KiB
JavaScript
308 lines
8.4 KiB
JavaScript
import path from 'path'
|
|
|
|
import TimeFixPlugin from 'time-fix-plugin'
|
|
import _ from 'lodash'
|
|
import VueLoader from 'vue-loader'
|
|
import MiniCssExtractPlugin from 'mini-css-extract-plugin'
|
|
import WebpackBar from 'webpackbar'
|
|
|
|
import { isUrl, urlJoin } from '../../common/utils'
|
|
import StyleLoader from './utils/style-loader'
|
|
import WarnFixPlugin from './plugins/warnfix'
|
|
import StatsPlugin from './plugins/stats'
|
|
|
|
export default class WebpackBaseConfig {
|
|
constructor(builder, options) {
|
|
this.name = options.name
|
|
this.isServer = options.isServer
|
|
this.builder = builder
|
|
this.nuxt = this.builder.nuxt
|
|
this.isStatic = builder.isStatic
|
|
this.options = builder.options
|
|
this.spinner = builder.spinner
|
|
}
|
|
|
|
getBabelOptions() {
|
|
const options = _.clone(this.options.build.babel)
|
|
|
|
if (typeof options.presets === 'function') {
|
|
options.presets = options.presets({ isServer: this.isServer })
|
|
}
|
|
|
|
if (!options.babelrc && !options.presets) {
|
|
options.presets = [
|
|
[
|
|
require.resolve('@nuxtjs/babel-preset-app'),
|
|
{
|
|
buildTarget: this.isServer ? 'server' : 'client'
|
|
}
|
|
]
|
|
]
|
|
}
|
|
|
|
return options
|
|
}
|
|
|
|
getFileName(name) {
|
|
let fileName = this.options.build.filenames[name]
|
|
|
|
// Don't use hashes when watching
|
|
// https://github.com/webpack/webpack/issues/1914#issuecomment-174171709
|
|
if (this.options.dev) {
|
|
fileName = fileName.replace(/\[(chunkhash|contenthash|hash)(?::(\d+))?\]\./g, '')
|
|
}
|
|
|
|
// Don't use [name] for production assets
|
|
if (!this.options.dev && this.options.build.optimization.splitChunks.name !== true) {
|
|
fileName = fileName.replace(/\[name\]\./g, '')
|
|
}
|
|
|
|
return fileName
|
|
}
|
|
|
|
env() {
|
|
const env = {
|
|
'process.mode': JSON.stringify(this.options.mode),
|
|
'process.static': this.isStatic
|
|
}
|
|
_.each(this.options.env, (value, key) => {
|
|
env['process.env.' + key] =
|
|
['boolean', 'number'].indexOf(typeof value) !== -1
|
|
? value
|
|
: JSON.stringify(value)
|
|
})
|
|
return env
|
|
}
|
|
|
|
output() {
|
|
return {
|
|
path: path.resolve(this.options.buildDir, 'dist', this.isServer ? 'server' : 'client'),
|
|
filename: this.getFileName('app'),
|
|
chunkFilename: this.getFileName('chunk'),
|
|
publicPath: isUrl(this.options.build.publicPath)
|
|
? this.options.build.publicPath
|
|
: urlJoin(this.options.router.base, this.options.build.publicPath)
|
|
}
|
|
}
|
|
|
|
alias() {
|
|
return {
|
|
'~': path.join(this.options.srcDir),
|
|
'~~': path.join(this.options.rootDir),
|
|
'@': path.join(this.options.srcDir),
|
|
'@@': path.join(this.options.rootDir),
|
|
[this.options.dir.assets]: path.join(
|
|
this.options.srcDir,
|
|
this.options.dir.assets
|
|
),
|
|
[this.options.dir.static]: path.join(
|
|
this.options.srcDir,
|
|
this.options.dir.static
|
|
)
|
|
}
|
|
}
|
|
|
|
rules() {
|
|
const styleLoader = new StyleLoader(
|
|
this.options,
|
|
this.builder.nuxt,
|
|
{ isServer: this.isServer }
|
|
)
|
|
|
|
const perfLoader = this.builder.perfLoader
|
|
|
|
return [
|
|
{
|
|
test: /\.vue$/,
|
|
loader: 'vue-loader',
|
|
options: Object.assign({
|
|
productionMode: !this.options.dev,
|
|
transformAssetUrls: {
|
|
video: 'src',
|
|
source: 'src',
|
|
object: 'src',
|
|
embed: 'src'
|
|
}
|
|
}, this.options.build.vueLoader)
|
|
},
|
|
{
|
|
test: /\.pug$/,
|
|
oneOf: [
|
|
{
|
|
resourceQuery: /^\?vue/,
|
|
use: ['pug-plain-loader']
|
|
},
|
|
{
|
|
use: ['raw-loader', 'pug-plain-loader']
|
|
}
|
|
]
|
|
},
|
|
{
|
|
test: /\.jsx?$/,
|
|
exclude: (file) => {
|
|
// not exclude files outside node_modules
|
|
if (/node_modules/.test(file)) {
|
|
for (const module of [/\.vue\.js/].concat(this.options.build.transpile)) {
|
|
// item in transpile can be string or regex object
|
|
if (module.test(file)) {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
},
|
|
use: perfLoader.pool('js', {
|
|
loader: 'babel-loader',
|
|
options: this.getBabelOptions()
|
|
})
|
|
},
|
|
{
|
|
test: /\.css$/,
|
|
oneOf: perfLoader.poolOneOf('css', styleLoader.apply('css'))
|
|
},
|
|
{
|
|
test: /\.less$/,
|
|
oneOf: perfLoader.poolOneOf('css', styleLoader.apply('less', 'less-loader'))
|
|
},
|
|
{
|
|
test: /\.sass$/,
|
|
oneOf: perfLoader.poolOneOf('css', styleLoader.apply('sass', {
|
|
loader: 'sass-loader',
|
|
options: { indentedSyntax: true }
|
|
}))
|
|
},
|
|
{
|
|
test: /\.scss$/,
|
|
oneOf: perfLoader.poolOneOf('css', styleLoader.apply('scss', 'sass-loader'))
|
|
},
|
|
{
|
|
test: /\.styl(us)?$/,
|
|
oneOf: perfLoader.poolOneOf('css', styleLoader.apply('stylus', 'stylus-loader'))
|
|
},
|
|
{
|
|
test: /\.(png|jpe?g|gif|svg)$/,
|
|
use: perfLoader.pool('assets', {
|
|
loader: 'url-loader',
|
|
options: {
|
|
limit: 1000, // 1KO
|
|
name: this.getFileName('img')
|
|
}
|
|
})
|
|
},
|
|
{
|
|
test: /\.(woff2?|eot|ttf|otf)(\?.*)?$/,
|
|
use: perfLoader.pool('assets', {
|
|
loader: 'url-loader',
|
|
options: {
|
|
limit: 1000, // 1 KO
|
|
name: this.getFileName('font')
|
|
}
|
|
})
|
|
},
|
|
{
|
|
test: /\.(webm|mp4|ogv)$/,
|
|
use: perfLoader.pool('assets', {
|
|
loader: 'file-loader',
|
|
options: {
|
|
name: this.getFileName('video')
|
|
}
|
|
})
|
|
}
|
|
]
|
|
}
|
|
|
|
plugins() {
|
|
const plugins = [ new VueLoader.VueLoaderPlugin() ]
|
|
|
|
Array.prototype.push.apply(plugins, this.options.build.plugins || [])
|
|
|
|
// Add timefix-plugin before others plugins
|
|
if (this.options.dev) {
|
|
plugins.unshift(new TimeFixPlugin())
|
|
}
|
|
|
|
// Hide warnings about plugins without a default export (#1179)
|
|
plugins.push(new WarnFixPlugin())
|
|
|
|
// Build progress indicator
|
|
plugins.push(new WebpackBar({
|
|
profile: this.options.build.profile,
|
|
name: this.isServer ? 'server' : 'client',
|
|
color: this.isServer ? 'orange' : 'green',
|
|
compiledIn: false,
|
|
done: (states) => {
|
|
if (this.options.dev) {
|
|
const hasErrors = Object.values(states).some(state => state.stats.hasErrors())
|
|
|
|
if (!hasErrors) {
|
|
this.nuxt.showReady(false)
|
|
}
|
|
}
|
|
}
|
|
}))
|
|
|
|
// Add stats plugin
|
|
if (!this.options.dev && this.options.build.stats) {
|
|
plugins.push(new StatsPlugin(this.options.build.stats))
|
|
}
|
|
|
|
// CSS extraction
|
|
// MiniCssExtractPlugin does not currently supports SSR
|
|
// https://github.com/webpack-contrib/mini-css-extract-plugin/issues/48
|
|
// So we use css-loader/locals as a fallback (utils/style-loader)
|
|
if (this.options.build.extractCSS && !this.isServer) {
|
|
plugins.push(new MiniCssExtractPlugin(Object.assign({
|
|
filename: this.getFileName('css'),
|
|
chunkFilename: this.getFileName('css')
|
|
}, this.options.build.extractCSS)))
|
|
}
|
|
|
|
return plugins
|
|
}
|
|
|
|
customize(config) {
|
|
if (typeof this.options.build.extend === 'function') {
|
|
const extendedConfig = this.options.build.extend.call(this.builder, config, {
|
|
isDev: this.options.dev,
|
|
isServer: this.isServer,
|
|
isClient: !this.isServer
|
|
})
|
|
// Only overwrite config when something is returned for backwards compatibility
|
|
if (extendedConfig !== undefined) {
|
|
return extendedConfig
|
|
}
|
|
}
|
|
return config
|
|
}
|
|
|
|
config() {
|
|
// Prioritize nested node_modules in webpack search path (#2558)
|
|
const webpackModulesDir = ['node_modules'].concat(this.options.modulesDir)
|
|
const config = {
|
|
name: this.name,
|
|
mode: this.options.dev ? 'development' : 'production',
|
|
optimization: {},
|
|
output: this.output(),
|
|
performance: {
|
|
maxEntrypointSize: 1000 * 1024,
|
|
hints: this.options.dev ? false : 'warning'
|
|
},
|
|
resolve: {
|
|
extensions: ['.wasm', '.mjs', '.js', '.json', '.vue', '.jsx'],
|
|
alias: this.alias(),
|
|
modules: webpackModulesDir
|
|
},
|
|
resolveLoader: {
|
|
modules: webpackModulesDir
|
|
},
|
|
module: {
|
|
rules: this.rules()
|
|
},
|
|
plugins: this.plugins()
|
|
}
|
|
|
|
// Clone deep avoid leaking config between Client and Server
|
|
return _.cloneDeep(config)
|
|
}
|
|
}
|