React和Vue项目的webpack配置基本思路

本篇文章是自己通过搜集资料使用webpack4+react16+ts4搭建的一个React开发环境,目的主要是学习webpack整体搭建流程,以及各模块负责内容,然后准备使用这套环境使用react+ts+redux技术栈实现自己的移动端博客。

本文不是使用webpack搭建ts+react环境的开发教程,是搭建的一些前期规划与后期总结。

前期准备

首先使用npm init初始化一个项目在项目中创建一个config文件夹用来保存配置文件。

第一步区分环境
我的需求简单只是需要一个开发环境一个生产环境。因此在config文件夹下分别创建webpack.dev.jswebpack.prod.js两个文件。

第二步增加常量配置文件
webpack.dev.jswebpack.prod.js文件中一些经常使用的变量,比如判断是否是dev环境的标志变量IS_DEV,还有项目根路径PROJECT_PATH,以及devServer里面要用的HOST,PORT等一些变量统一放到config.js文件中。

第三步独立BASE配置文件
第一步我们区分开了开发环境和生产环境,但是生产环境和开发环境有很多相同的配置项,因此将相同配置项抽出来放到同目录下的webpack.base.js文件中。最后使用webpack-merge插件将webpack.base.js文件和webpack-dev.js合并,还有将webpack.base.jswebpack.prod.js合并。

第四步独立build文件
为了在 build 时候方便做一些多余的处理,比如在build的时候在控制台使用一些提示插件提示正在打包文字提升开发体验等等,当打包完成后取消打包中显示。因此独立出来一个build.js文件,专门用来执行打包。

看下图结构:


config文件夹配置

通过上面四步,就在config文件夹下创建了build.jsconfig.jswebpack.base.jswebpack.dev.jswebpack.prod.js五个文件。
这五个文件分别是干什么的上面也说清楚了,然后说一下其他文件以及文件夹的作用吧。

config文件夹用来存放webpack配置文件;
dist是构建输出目录;
node_modules存放下载的node包;
public文件夹存放一些静态文件,以及html模板文件;
src文件夹用来存放开发时的代码;
.babelrc是babel的配置项;
.npmrc是对node包下载源的源配置,比如想用淘宝源,就在该文件中配置npm config set registry https://registry.npm.taobao.org;这样之后使用npm下载依赖包的时候,就默认使用淘宝源了。不用手动切换。
package.json存储项目信息;
README.md文件是我用来记录项目中遇到的一些问题一些解决方法等等。
tsconfig.json文件是typescript的配置文件。

好,整个文件目录介绍完了,现在就主要看webpack配置文件吧。

wepack配置规划图

配置文件内容

config.js
先从config文件夹下的config.js说起吧,直接看代码吧。

const path = require('path');
const IS_DEV = process.env.NODE_ENV !== 'production';
module.exports = {
    PROJECT_PATH: path.resolve(__dirname, "../"),
    IS_DEV,
    PORT: 8000,
    HOST: 'localhost'
}

这就是config.js中所有内容了,主要利用node的path模块对外暴露了项目根目录PROJECT_PATH,还通过process.env.NODE_ENV获取环境变量然后判断是否是development环境,并导出IS_DEV变量,这个环境变量问题稍后说。还导出了devServer要用的PORTHOST,这里是自定义的一些东西,不必非得写在这里。假如devServer中的port和host完全可以就在devServer里面直接写死,不必再从这个文件中获取。

然后接着说一下环境变量的问题,环境变量我这里使用了cross-env插件,然后在package.js中配置run执行脚本命令的时候传递参数通过process.env.NODE_ENV动态获取参数内容,来做的。

npm i cross-env -D

package.json文件中:

{
    // ...
    "scripts":{
        "start": "cross-env NODE_ENV=development webpack-dev-server --config ./config/webpack.dev.js",
        "build": "cross-env NODE_ENV=production node ./config/build.js"
    }
    // ...
}

以上就是通过cross-env指定NODE_ENV的值之后在配置文件中可以通过process.env.NODE_ENV来获取。

这个环境变量还要其他的设置方式,比如在本地配置环境变量文件,可以参考creat-react-app配置。

webpack.base.js

webpack.base.js文件中主要做了webpack的基础配置项,代码如下:

// base 
const {resolve} = require('path');
const {PROJECT_PATH, IS_DEV} = require('./config');
const HtmlWebpackPlugin = require('html-webpack-plugin');
const CopyPlugin = require('copy-webpack-plugin');
const WebpackBar = require('webpackbar');
const ForkTsCheckerWebpackPlugin = require('fork-ts-checker-webpack-plugin');
const MiniCssExtractPlugin = require('mini-css-extract-plugin');
const getCssLoaders = () => {
    return [
        IS_DEV ? 'style-loader' : MiniCssExtractPlugin.loader,
        {
            loader: 'css-loader',
            options: {
                sourceMap: IS_DEV,
            }
        },
        {
            loader: 'postcss-loader',
            options: {
                ident: 'postcss',
                plugins: [
                    require('postcss-flexbugs-fixes'),
                    require('postcss-preset-env')({
                        autoprefixer:{
                            grid: true,
                            flexbox: 'no-2009'
                        },
                        stage: 3
                    }),
                    require('postcss-normalize')
                ],
                sourceMap: IS_DEV
            }
        }
    ]
}
module.exports = {
    entry: {
        app: resolve(PROJECT_PATH, './src/index.tsx')
    },
    output: {
        filename: `js/[name]${IS_DEV ? '' :'.[hash:8]'}.js`,
        path: resolve(PROJECT_PATH, './dist')
    },
    module: {
        rules: [
            {
                test: /\.(tsx|js)$/,
                loader: 'babel-loader',
                options: {cacheDirectory: true},
                exclude: /node_modules/
            },
            {
                test: /\.css$/,
                use: getCssLoaders(),
            },
            {
                test: /\.less$/,
                use: [
                    ... getCssLoaders(),
                    {
                        loader: 'less-loader',
                        options: {
                            sourceMap: IS_DEV
                        }
                    }
                ]
            },
            {
               test: /\.scss$/,
               use: [
                   ...getCssLoaders(),
                   {
                       loader: 'sass-loader',
                       options: {
                           sourceMap: IS_DEV
                       }
                   }
               ] 
            },
            {
                test: /\.(png|jpg|jpeg|gif|svg)$/i,
                use: [
                    {
                        loader: 'url-loader',
                        options: {
                            limit: 10 * 1024,
                            name: '[name].[contenthash:8].[ext]',
                            outputPath: 'images'
                        }
                    }
                ]
            },
            {
                test: /\.(ttf|woff|woff2|eot|otf)$/,
                use: [
                    {
                        loader: 'url-loader',
                        options: {
                            name: '[name].[contenthash:8].[ext]',
                            outputPath: 'fonts'
                        }
                    }
                ]
            }
        ]
    },
    plugins: [
        new HtmlWebpackPlugin({
            template: resolve(PROJECT_PATH, './public/index.html'),
            filename: 'index.html',
            cache: false,
            minify: IS_DEV ? false : {
                removeAttributeQuotes: true,
                collapseWhitespace: true,
                removeComments: true,
                collapseBooleanAttributes: true,
                collapseInlineTagWhitespace: true,
                removeRedundantAttributes: true,
                removeScriptTypeAttributes: true,
                removeStyleLinkTypeAttributes: true,
                minifyCSS: true,
                minifyJS: true,
                minifyURLs: true,
                useShortDoctype: true,
            }
        }),
        new CopyPlugin({ // 拷贝静态资源
            patterns: [
                {
                    context: resolve(PROJECT_PATH, './public'),
                    from: '*',
                    to: resolve(PROJECT_PATH, './dist'),
                    toType: 'dir'
                },
                {
                    context: resolve(PROJECT_PATH, './public/static'),
                    from: '*',
                    to: resolve(PROJECT_PATH, './dist/static'),
                    toType: 'dir'
                }
            ]
        }),
        new WebpackBar({ // 显示启动进度
            name: IS_DEV ? '正在启动' : '正在打包'
        }),
        new ForkTsCheckerWebpackPlugin(), // 编译时typescript类型检查
    ],
    resolve: {
        extensions: ['.tsx', '.ts', '.js', '.json'],
        alias: {
            'Src': resolve(PROJECT_PATH, './src'),
            'Components': resolve(PROJECT_PATH, './src/components'),
            'Containers': resolve(PROJECT_PATH, './src/containers')
        }
    },
    devtool: IS_DEV ? 'cheap-module-eval-source-map' : 'cheap-module-source-map'
}

常用的导入以及配置细节就不多说了,都是遵循webpack配置规则,有配置疑问的可以在webpack官网或者npm官网查资料,看一下里面有个getCssLoaders函数,如下:

const getCssLoaders = () => {
    return [
        IS_DEV ? 'style-loader' : MiniCssExtractPlugin.loader,
        {
            loader: 'css-loader',
            options: {
                sourceMap: IS_DEV,
            }
        },
        {
            loader: 'postcss-loader',
            options: {
                ident: 'postcss',
                plugins: [
                    require('postcss-flexbugs-fixes'),
                    require('postcss-preset-env')({
                        autoprefixer:{
                            grid: true,
                            flexbox: 'no-2009'
                        },
                        stage: 3
                    }),
                    require('postcss-normalize')
                ],
                sourceMap: IS_DEV
            }
        }
    ]
}

这个函数的作用主要是将loader配置中的公共部分 style-loadercss-loaderpostcss-loader这些配置提取出来,这样的话就可以将css,less,sass中多余的配置项都抽取出来,也可以将该函数放到外面的config.js文件中去维护,尤其是base.js文件特别多的时候,目前我就暂且这么放了。

webpack.dev.js
在dev环境中最重要的就是devServer了,因此在该文件中主要做了devServer的配置还有热更新的配置。

const {merge} = require('webpack-merge');
const baseConfig = require('./webpack.base');
const {PORT, HOST} = require('./config');
const webpack = require('webpack');
module.exports = merge(baseConfig, {
    mode: 'development',
    devServer: {
        host: HOST,
        port: PORT,
        open: true,
        hot: true,
        stats: 'errors-only', // 终端仅打印 error
    },
    plugins: [
        new webpack.HotModuleReplacementPlugin()
    ]
})

当该文件配置内容完成时,需要将最终结果导出的时候,这个时候需要将base的配置和当前dev配置进行merge然后再导出。这里使用了webpack.HotModuleReplacementPlugin插件。同时在使用热更新插件时,需要在项目入口文件(也就是src/index.tsx)中添加如下代码:

if ((module as any) && (module as any).hot) {
  // 热更新设置 as any解决 Property 'hot' does not exist on type 'NodeModule'.
  (module as any).hot.accept();
}

webpack.prod.js
在生产环境比较重要的就是代码体积压缩,分包优化,缓存处理。更好的支持tree-shaking等。

const {merge} = require('webpack-merge');
const {resolve} = require('path');
const {PROJECT_PATH} = require('./config');
const baseConfig = require('./webpack.base');
const {CleanWebpackPlugin} = require('clean-webpack-plugin');
const MiniCssExtractPlugin = require('mini-css-extract-plugin');
const PurgeCSSPlugin = require('purgecss-webpack-plugin');
const glob = require('glob');
const TerserWebpackPlugin = require('terser-webpack-plugin');
const OptimizeCSSAssetsPlugin = require('optimize-css-assets-webpack-plugin');
module.exports = merge(baseConfig, {
    mode: 'production',
    plugins: [
        new CleanWebpackPlugin(), // 清理构建产物
        new MiniCssExtractPlugin({
            filename: 'css/[name].[contenthash:8].css',
            chunkFilename: 'css/[name].[contenthash:8].css',
            ignoreOrder: false
        }),
        new PurgeCSSPlugin({ // 剔除没有用到的css样式
            paths: glob.sync(`${resolve(PROJECT_PATH, './src')}/**/*.{tsx,scss,less,css}`, {nodir: true}),
            whitelist: ['html', 'body']
        })
    ],
    optimization: {
        minimize: true,
        minimizer: [
            new TerserWebpackPlugin({ // js压缩
                extractComments: false,
                terserOptions: {
                    compress: {
                        pure_funcs: ['console.log']
                    }
                }
            }),
            new OptimizeCSSAssetsPlugin() // css压缩整合
        ].filter(Boolean),
        splitChunks:{ // 分包优化
            chunks: 'async',
            minSize: 30000,
            minChunks: 1,
            maxAsyncRequests: 5,
            maxInitialRequests: 3,
            name: false,
            cacheGroups: {
              vendor: {
                name: 'vendor',
                chunks: 'initial',
                priority: -10,
                reuseExistingChunk: false,
                test: /node_modules\/(.*)\.js/
              },
              styles: {
                name: 'styles',
                test: /\.(scss|css)$/,
                chunks: 'all',
                minChunks: 1,
                reuseExistingChunk: true,
                enforce: true
              }
            }
        }
    }
})

以上文件中常用的生产环境内容简单的说一下。
clean-webpack-plugin插件用来在每次build之前自动清理构建产物。
mini-css-extract-plugin插件用来将css内容提取出来到一个.css文件中(使用style-loader的css样式文件是默认被插入到html文件的style标签里的,这样不利于做缓存),然后加入文件指纹(hash,chunkHash,contentHash)可以用来配合浏览器做样式缓存。
purgecss-webpack-plugin插件主要用来剔除在文件中没有用到的样式内容,类似于tree-shaking将死代码剔除掉。可以减小代码体积。
terser-webpack-plugin插件主要是webpack4中用来替换UglifyJs插件,更好支持ES6语法压缩,也可以额外配置多进程压缩。
optimize-css-assets-webpack-plugin插件主要是对css样式文件进行压缩处理。
splitChunks中的一些配置项主要是针对被多次引用文件,体积较大文件进行分包单独抽离,减小文件打包体积。

build.js
将build的文件单独拎出来,在执行webpack函数时可以做一些额外操作。

const ora = require('ora')
const webpack = require('webpack');
const webpackConfig = require('./webpack.prod.js');
const spinner = ora('building for production...')
spinner.start()
webpack(webpackConfig, (err, stats) => {
    spinner.stop()
})

比如在build的时候,执行提示,执行完成之后提示消失。

还有就是配置文件中的mode,它不仅仅是用来区分开发环境与生产环境的,而是在不同的模式下webpack会默认根据不同模式执行不同的内置函数。执行内置函数对项目进行优化等操作。
设置mode可以自动触发webpack中的某些函数

Mode的内置函数功能

选项 描述
development 设置 process.env.NODE_ENV的值为development.
开启NameChunksPluginNameModulesPlugin.
production 设置process.env.NODE_ENV的值为production.
开启FlagDependencyUsagePlugin, FlagIncludeChunksPlugin
ModileConcatentationPluginNoEmitOnErrorsPlugin
OccurrenceOrderPluginSideEffectsFlagPluginTerserPlugin.
none 不开启任何优化选项

到这里整个webpack里面的配置文件就完成了,该文章提供一个webpack配置思路,可以用来配置Vue应用配置,也可以用来配置React应用配置。

我会用这套配置自己写一下个人移动端小博客,技术栈使用webpack+react+redux+typescript

webpack配置代码库 https://github.com/Mstian/webpack-config-react

你可能感兴趣的:(React和Vue项目的webpack配置基本思路)