最新公告
  • 欢迎您光临起源地模板网,本站秉承服务宗旨 履行“站长”责任,销售只是起点 服务永无止境!立即加入钻石VIP
  • Webpack5 搭建 react 脚手架

    正文概述 掘金(JasonJiang)   2021-02-28   734

    Webpack 5 发布也有挺长时间了,现在在公司用的脚手架还是 v4 版的,听说 v5 版本更新的重点主要是针对构型性能的优化。听起来感觉挺香的,就想去用 Webpack 5 搭建一个架手架 看看整体感觉如何,因为我现在主要是用 React 开发 所以就以 React 为例搭建,对于 Webpack 5 不是很熟的朋友可以 看看这篇文章 Webpack5 上手测评,其实配置内容跟 Webpack 4 变化不大。主要还是性能上的提升,那么废话不多说,开始了。

    项目初始化

    mkdir webpack5-demo
    cd webpack5-demo
    npm init -y
    

    安装基础的依赖文件

    刚刚开始需要安装的包,我直接贴出来吧,可以复制到 package.json 文件上 执行 npm i 或 yarn

    package.json

    "devDependencies": {
        "@babel/core": "^7.13.8",
        "@babel/preset-env": "^7.13.8",
        "@babel/preset-react": "^7.12.13",
        "babel-loader": "^8.2.2",
        "chalk": "^4.1.0",
        "clean-webpack-plugin": "^3.0.0",
        "css-loader": "^5.1.0",
        "html-webpack-plugin": "^5.2.0",
        "ip": "^1.1.5",
        "style-loader": "^2.0.0",
        "progress-bar-webpack-plugin": "^2.1.0",
        "speed-measure-webpack-plugin": "^1.4.2",
        "webpack": "^5.24.2",
        "webpack-cli": "^4.5.0",
        "webpack-dev-server": "^3.11.2"
      },
      "dependencies": {
        "react": "^17.0.1",
        "react-dom": "^17.0.1"
      }
    

    此时文件目录情况

    webpack5-demo
    ├─ node_modules
    ├─ package-lock.json
    └─ package.json
    

    配置 webpack

    为了区分开发生产环境,方便维护和特殊处理,就不把所有内容配置到 webpack.config.js 这一个文件里面了。

    # 这里我们新建一个 config 目录用来专门存放 webpack 配置文件
    mkdir config
    cd config
    touch webpack.common.js # 开发环境 和 生产环境 公共配置 存放在这个文件里面
    touch webpack.dev.js # 需要针对开发环境特殊处理的配置存放在这里
    touch webpack.prod.js # 需要针对生产环境特殊处理的配置存放在这里
    

    在配置 Webpack 的时候少不了会与 文件路径打交道,避免路径零乱,单独创建一个路径调用的配置文件,也是参考 CRA 的

    touch paths.js
    

    paths.js

    const path = require("path");
    const fs = require("fs");
    // 获取当前工作目录
    const appDirectory = fs.realpathSync(process.cwd());
    // 从相对路径中解析绝对路径
    const resolveApp = (relativePath) => path.resolve(appDirectory, relativePath);
    // 默认的模块扩展名
    const moduleFileExtensions = ["js", "jsx", "ts", "tsx", "json"];
    // 解析模块路径
    const resolveModule = (resolveFn, filePath) => {
        // 查看文件存不存在
        const extension = moduleFileExtensions.find((extension) =>
            fs.existsSync(resolveFn(`${filePath}.${extension}`))
        );
        if (extension) {
            return resolveFn(`${filePath}.${extension}`);
        }
        return resolveFn(`${filePath}.js`); // 如果没有默认就是js
    };
    
    module.exports = {
        appBuild: resolveApp("build"), // 打包路径
        appPublic: resolveApp("public"), // 静态资源路径
        appHtml: resolveApp("public/index.html"), // html 模板路径
        appIndexJs: resolveModule(resolveApp, "src/index"), // 打包入口路径
        appNodeModules: resolveApp("node_modules"), // node_modules 路径
        appSrc: resolveApp("src"), // 主文件入口路径
        moduleFileExtensions, // 模块扩展名
    };
    

    此时文件目录情况

    demo
    ├─ config
    ├─ paths.js
    │  ├─ webpack.common.js
    │  ├─ webpack.dev.js
    │  └─ webpack.prod.js
    ├─ node_modules
    ├─ package-lock.json
    └─ package.json
    

    webpack.common.js

    const paths = require("./paths");
    const HtmlWebpackPlugin = require("html-webpack-plugin");
    module.exports = function (options) {
        return {
            mode: options.mode,
            entry: paths.appSrc,
            output: {
                path: paths.appBuild,
                publicPath: "/",
            },
            cache: {
                // 使用持久化缓存
                type: "filesystem", //memory:使用内容缓存 filesystem:使用文件缓存
            },
            devtool: false,
            module: {
                rules: [
                    {
                        test: /\.js$/,
                        exclude: /node_modules/,
                        use: [
                            {
                                loader: "babel-loader",
                                options: {
                                    presets: [
                                        "@babel/preset-env",
                                        "@babel/preset-react",
                                    ],
                                },
                            },
                        ],
                    },
                ],
            },
            devServer: {},
            plugins: [
                new HtmlWebpackPlugin({
                    template: "./public/index.html",
                }),
                ...options.plugins,
            ],
            stats: options.stats, // 打包日志发生错误和新的编译时输出
        };
    };
    

    entry 是 入口,默认寻找路径就是我们写的这个,这里我们需要提供一个入口文件

    # 回到webpack5-demo 根目录
    # 创建一个src 目录开发文件
    
    mkdir src
    touch index.js # 创建一个入口文件
    
    import React from "react";
    import ReactDOM from "react-dom";
    
    const App = () => {
        return <div> App入口 </div>;
    };
    
    ReactDOM.render(<App />, document.querySelector("#root"));
    

    在打包的时候,使用了HtmlWebpackPlugin,所以得去创建一个 html 模板提供足够插件使用

    # 回到webpack5-demo 根目录
    # 创建一个public 目录专门存放 静态资源
    
    mkdir public
    touch index.html # 创建一个html 模板
    

    index.html 模板

    <!DOCTYPE html>
    <html lang="en">
        <head>
            <meta charset="UTF-8" />
            <meta http-equiv="X-UA-Compatible" content="IE=edge" />
            <meta name="viewport" content="width=device-width, initial-scale=1.0" />
            <title>Webpack-React-Cli</title>
        </head>
        <body>
            <div id="root"></div>
        </body>
    </html>
    

    webpack.dev.js 开发配置

    module.exports = require("./webpack.common")({
        mode: "development",
        plugins: [],
        stats: "errors-only", //只在发生错误或有新的编译时输出
    });
    

    webpack.prod.js 生产配置

    const { CleanWebpackPlugin } = require("clean-webpack-plugin"); //打包前清空build目录文件
    const ProgressBarPlugin = require("progress-bar-webpack-plugin"); // 打包进度条美化
    const chalk = require("chalk");
    
    module.exports = require("./webpack.common")({
        mode: "production",
        devtool: "source-map",
        plugins: [
            new CleanWebpackPlugin(),
            new ProgressBarPlugin({
                format:
                    `${chalk.green.bold("build[:bar]")} ` +
                    chalk.green.bold(":percent") +
                    " (:elapsed seconds)",
                clear: false,
                width: 60,
            }),
        ],
        stats: "normal", //标准输出
    });
    

    此时文件目录情况

    webpack5-demo
    ├─ config
    │  ├─ paths.js
    │  ├─ webpack.common.js
    │  ├─ webpack.dev.js
    │  └─ webpack.prod.js
    ├─ node_modules
    ├─ public
    │  └─ index.html
    ├─ src
    │  └─ index.js
    ├─ package-lock.json
    └─ package.json
    

    启动服务

    现在我们的 Webpack 基础配置已经 OK 了,就差启动了

    配置 pageage.json 脚本

    {
      "name": "webpack5-demo",
      "version": "1.0.0",
      "description": "基于webpack5 的React架手架",
      "main": "index.js",
      "scripts": {
        "build": "webpack --config config/webpack.prod.js",
        "start": "webpack serve --config config/webpack.dev.js", // v5
        "dev": "webpack-dev-server --config config/webpack.dev.js" //v4
      },
      "keywords": [],
      "author": "Jason",
      "license": "ISC",
      ...
    }
    

    这里有点小改动:

    Webpack 4 里面通过 webpack-dev-server起服务 在 Webpack 5 里面启动服务里通过 webpack serve

    Webpack5 搭建 react 脚手架

    现在已经成功启动了,但是这个启动日志不是很喜欢,我得改造一下(实现自定义输出编译内容),需要重新配置一下开发服务器(webpack-dev-server)

    配置开发服务器

    虽然建议通过 CLI 运行 webpack-dev-server,但我们也可以选择通过 API 启动服务器。

    # 回到webpack5-demo 根目录
    # 创建一个server目录
    
    mkdir server
    touch index.js # 服务入口
    touch appConfig.js # 基础服务配置 自定义服务端口,ip, 代理地址
    touch logger.js # 控制台输出的日志
    

    appConfig.js

    module.exports = {
        deployUrl: "127.0.0.0:8080", // 本地代码推推送到指定服务器
        proxyUrlMap: {
            "/api": "localtion:3000", // 代理的接口
            "/api2": "localtion:4000", // 代理的接口
        },
        port: 9000, //端口号,
        host: "localhost", //主机号
    };
    

    logger.js

    const ip = require("ip");
    
    const divider = chalk.gray("\n-----------------------------------");
    
    const logger = {
        error: (err) => {
            console.error(chalk.red(err));
        },
        appStarted: (port, host, tunnelStarted) => {
            console.log(`Server started ! ${chalk.green("✓")}`);
    
            if (tunnelStarted) {
                console.log(`Tunnel initialised ${chalk.green("✓")}`);
            }
            console.log(`
    ${chalk.bold("Access URLs:")}${divider}
    Localhost: ${chalk.magenta(`http://${host}:${port}`)}
    LAN: ${
                chalk.magenta(`http://${ip.address()}:${port}`) +
                (tunnelStarted
                    ? `\n    Proxy: ${chalk.magenta(tunnelStarted)}`
                    : "")
            }${divider}
    ${chalk.blue(`Press ${chalk.italic("CTRL-C")} to stop`)}
        `);
        },
    };
    
    module.exports = logger;
    

    index.js

    const Webpack = require("webpack");
    const WebpackDevServer = require("webpack-dev-server");
    // webpack开发 配置文件
    const webpackConfig = require("../config/webpack.dev");
    // 自定义日志输出
    const logger = require("./logger");
    // 服务配置
    const appConfig = require("./appConfig");
    
    const { port, host } = appConfig; // 监听的端口号
    //编译器
    const compiler = Webpack(webpackConfig);
    //  devServer 参数
    const devServerOptions = Object.assign({}, webpackConfig.devServer, {
        // open: true, // 自动打开浏览器
        compress: true, // gzip 压缩
        stats: "minimal",
    });
    const server = new WebpackDevServer(compiler, devServerOptions);
    
    server.listen(port, host, async (err) => {
        if (err) {
            return logger.error(err.message);
        }
        logger.appStarted(port, "localhost");
    });
    

    package.json 配置启动命令

    "scripts": {
        "build": "webpack --config config/webpack.prod.js",
        "start": "node server"
      },
    

    Webpack5 搭建 react 脚手架

    此时文件目录情况

    webpack5-demo
    ├─ config
    │  ├─ paths.js
    │  ├─ webpack.common.js
    │  ├─ webpack.dev.js
    │  └─ webpack.prod.js
    ├─ node_modules
    ├─ public
    │  └─ index.html
    ├─ server
    │  ├─ appConfig.js
    │  ├─ index.js
    │  └─ logger.js
    ├─ src
    │  └─ index.js
    ├─ package-lock.json
    └─ package.json
    

    resolve 解析

    resolve 配置以下这几个就可以了,其他使用默认就行

    • modules: 使用第三模块 第一反应去 根目录下的 node_modules 寻找
    • extensions : 在 import 的时候不加文件扩展名,会依次遍历extensions 添加扩展名进行匹配
    • alias: 创建别名, 在importrequire 的别名,来确保模块引入变得更简单
     resolve: {
        modules: [paths.appNodeModules],
        extensions: ['.js', '.jsx', '.css'],
        alias: {
            moment$: 'moment/moment.js',
            '@src': paths.appSrc,
            '@public': paths.appPublic,
        },
    },
    

    基础 loader 配置

    css 和 sass

    安装

    npm i -D style-loader css-loader
    npm i -D node-sass sass-loader postcss postcss-loader postcss-preset-env
    

    css 和 sass 的 loader 配置很简单 考虑到兼容性问题,还需要 postcss-loader 添加浏览器厂家标识头

    用到的插件单独放在根目录的 postcss.config.js 配置文件里头

    postcss.config.js

    module.exports = {
        plugins: {
            "postcss-preset-env": {},
        },
    };
    

    模块资源

    Webpack 4 在处理图片或文本类文件的时候用的都是 file-loader 或者 url-loader 现在对于 Webpack 5 来说可以用 Asset Modules (资源模块),就不需要配置 loader 了

    • asset/source 导出资源的源代码 (相当于 raw-loader)
    • asset/resource 发送一个单独的文件并导出 URL(相当于 file-loader)
    • asset/inline 导出一个资源的 data URI(相当于 url-loader)
    • asset 在导出一个 data URI 和发送一个单独的文件之间自动选择,之前通过使用 url-loader,并且配置资源体积限制实现
    // 设置 常量
    const cssRegex = /\.css$/;
    const cssModuleRegex = /\.module\.css$/;
    const sassRegex = /\.(scss|sass)$/;
    const sassModuleRegex = /\.module\.(scss|sass)$/;
    const imageInlineSizeLimit = 4 * 1024;
    
    module.exports = function (options) {
    
        return {
    
            ...
    
            module: {
                rules: [
                    {
                        oneOf: [
    
                            ...
    
                            {
                                test: cssRegex,
                                exclude: cssModuleRegex,
                                use: ['style-loader', {
                                    loader: 'css-loader',
                                    options: {
                                        importLoaders: 1 // 0 => 无 loader(默认); 1 => postcss-loader; 2 => postcss-loader, sass-loader
                                    }
                                },'postcss-loader'],
                            },
                            {
                                test: sassRegex,
                                exclude: sassModuleRegex,
                                use: ['style-loader', {
                                    loader: 'css-loader',
                                    options: {
                                        importLoaders: 1 // 查询参数 importLoaders,用于配置「css-loader 作用于 @import 的资源之前」有多少个 loader
                                    }
                                }, 'postcss-loader', 'sass-loader'],
                            },
                            {
                                test: [/\.bmp$/, /\.gif$/, /\.jpe?g$/, /\.png$/],
                                type: 'asset',
                                parser: {
                                    dataUrlCondition: {
                                        maxSize: imageInlineSizeLimit // 4kb
                                    }
                                }
                            },
                            {
                                test: /\.(eot|svg|ttf|woff|woff2?)$/,
                                type: 'asset/resource'
                            },
                        ]
                    }
    
                ]
            },
        }
    
    

    测试一下 css 和图片

    # 在src 目录下新一个 style.scss 文件
    
    touch style.scss
    

    先写入点内容

    * {
        margin: 0;
        padding: 0;
    }
    
    div {
        color: red;
    }
    

    引入样式和图片

    import React from "react";
    import ReactDOM from "react-dom";
    
    import npm from "@public/assets/imgs/npm.png";
    
    import "./style.scss";
    
    const App = () => {
        return (
            <div>
                App入口
                <img src={npm} />
            </div>
        );
    };
    
    ReactDOM.render(<App />, document.querySelector("#root"));
    

    Webpack5 搭建 react 脚手架

    再看看打包情况

    Webpack5 搭建 react 脚手架

    由于设置了持久化缓存,第二次速度就很快

    此时文件目录情况

    webpack5-demo
    ├─ build
    ├─ config
    │  ├─ paths.js
    │  ├─ webpack.common.js
    │  ├─ webpack.dev.js
    │  └─ webpack.prod.js
    ├─ node_modules
    ├─ public
    │  ├─ assets
    │  │  └─ imgs
    │  │     └─ npm.png
    │  └─ index.html
    ├─ server
    │  ├─ appConfig.js
    │  ├─ index.js
    │  └─ logger.js
    ├─ src
    │  ├─ index.js
    │  └─ style.css
    ├─ package-lock.json
    └─ package.json
    

    小结

    内容还是比较多的打算分两篇来写,第一篇还是写的比较简单。 第二篇更多的是优化的,针对 webpack 5 新特性来优化、代码规范配置 eslint、一些插件的使用, 也是也想尝试写写文章,看看第一稿怎么样吧,超过 50 赞写下一篇 (哈哈哈 O(∩_∩)O)


    起源地下载网 » Webpack5 搭建 react 脚手架

    常见问题FAQ

    免费下载或者VIP会员专享资源能否直接商用?
    本站所有资源版权均属于原作者所有,这里所提供资源均只能用于参考学习用,请勿直接商用。若由于商用引起版权纠纷,一切责任均由使用者承担。更多说明请参考 VIP介绍。
    提示下载完但解压或打开不了?
    最常见的情况是下载不完整: 可对比下载完压缩包的与网盘上的容量,若小于网盘提示的容量则是这个原因。这是浏览器下载的bug,建议用百度网盘软件或迅雷下载。若排除这种情况,可在对应资源底部留言,或 联络我们.。
    找不到素材资源介绍文章里的示例图片?
    对于PPT,KEY,Mockups,APP,网页模版等类型的素材,文章内用于介绍的图片通常并不包含在对应可供下载素材包内。这些相关商业图片需另外购买,且本站不负责(也没有办法)找到出处。 同样地一些字体文件也是这种情况,但部分素材会在素材包内有一份字体下载链接清单。
    模板不会安装或需要功能定制以及二次开发?
    请QQ联系我们

    发表评论

    还没有评论,快来抢沙发吧!

    如需帝国cms功能定制以及二次开发请联系我们

    联系作者

    请选择支付方式

    ×
    迅虎支付宝
    迅虎微信
    支付宝当面付
    余额支付
    ×
    微信扫码支付 0 元