webpack构建React、TSX项目(一)

时间:2022-08-17 01:07:49


此篇主要是记录下webpack构建react项目的过程。

前言:为什么想起来通过webpack去构建react的项目呢?现在可以通过cra或者vite或其他脚手架去构建比较省事。主要还是想要了解下webpack相关的知识,如果对webpack构建以不敢兴趣可以等后续笔者发布关于TSX的相关知识

1、webpack、webpack-cli

既然通过webpack构建项目,首先离不开两个包webpack webpack-cli

webpack 是一个JS程序静态模块化打包工具。通过一个或多个入口点,将所有模块打包成一个或多个 bundle。它本身可以通过插件或者loader支持所熟知的react、vue、Angular等。

webpack-cli是webpack的命令行工具,它支持在命令行中操作webpack。

他们两个都可以通过npm或yarn命令安装。

2、起步

初始化package.json文件

yarn init -y

其中-y参数是跳过初始化json文件时需要手动输入的项

webpack构建React、TSX项目(一)

安装webpack webpack-cli

yarn add -D webpack webpack-cli

3、webpack.config.js

在跟目录下创建webpack.config.js文件,该文件用于配置webpack。一份基础的webpack配置文件如下:

module.exports = {
    mode: 'production',
    entry: './src/index.js', //规定打包入口文件
    output: {               //打包出口
        path: __dirname + '/dist',
        // [contenthash:8] - hash值 会根据内容生成hash值,对应版本更新
        // 如果内容无变化 两次编译的js文件是一样的
        filename: '[name]-[contenthash:8].js',
        // 编译前清除目录
        clean: true,
    },
    module: {
        rules: [],      //loader集合
    },
    plugins: [],      //插件
};

如果对于以上webpack中的概念不太了解,建议先查阅webpack概念

4、支持react、配置react环境

首先支持react我们需要将安装react react-dom

yarn add react react-dom

安装完react后需要安装react配置包

  1. babel-loader
  2. @babel/core
  3. @babel/preset-env
  4. @babel/preset-react
  5. @babel/plugin-transform-runtime

@babel/core 是 Babel 的核心模块,它提供了 Babel 的转换引擎和 API,负责将代码转换成兼容多种浏览器的 JavaScript 代码。开发者需要通过配置 @babel/core 模块,来使用 Babel 完成对 JavaScript 代码的转换。

@babel/preset-env 是一个智能的预设,它根据你的目标环境自动确定需要的转换和插件,从而将最新的 JavaScript 语法转换为向后兼容的代码。它可以根据你的设置,自动检测目标浏览器或 Node.js 版本,并将代码转换为兼容这些环境的代码。这使得开发者可以使用最新的 ECMAScript 语法,同时兼容多个浏览器和环境,而不必担心兼容性问题。

@babel/preset-react 则是用于将 React 代码转换为兼容多种浏览器的 JavaScript 代码,其中包括支持 JSX 语法、React 的新特性等。它可以将 JSX 语法转换为普通的 JavaScript 代码,从而使得浏览器可以正确地解析和渲染 React 组件。

注意:@babel/preset-env* 和 **@babel/preset-react *的功能不同,前者是用于转换 JavaScript 语法,后者是用于转换 React 代码。在配置 Babel 时,一般需要同时使用这两个预设来完成对 JavaScript 和 React 的转换

@babel/plugin-transform-runtime 是一个 Babel 插件,它用于在编译过程中,将代码中的公共部分提取到一个单独的模块中,从而避免在每个模块中重复引入相同的代码。该插件可以减小编译后文件的体积,并提高代码的运行效率。

该插件会自动将代码中需要的辅助函数注入到单独的模块中,从而避免在每个模块中重复定义辅助函数。同时,它还会使用 @babel/runtime 模块中提供的 polyfill 来兼容一些新的 ECMAScript 特性,例如 Promise 和 Symbol 等。

安装:

yarn add -D babel-loader @babel/core @babel/preset-env @babel/preset-react @babel/plugin-transform-runtime

配置:

修改webpack.config.js 增加使用babel-loader对js或jsx转义

{
    test: /\.(js|jsx)$/,
    use: 'babel-loader',
},

.babelrc

在项目跟目录下增加**.babelrc**配置文件

{
    "presets": [
      "@babel/preset-env",
      "@babel/preset-react"
    ],
    "plugins": [
      [
        "@babel/plugin-transform-runtime",
        {
          //"regenerator"是.babelrc中的一个配置项,它指定了是否将async/await语法转换为ES5代码。
          // async/await是ES7中的新语法,它使异步编程变得更加容易和直观。
          // 然而,这种语法在旧版的浏览器和Node.js中并不支持,因此需要使用Babel将代码转换为向后兼容的代码。
          "regenerator": true 
        }
      ]
    ]
}

提示:.babelrc是一个用于配置Babel编译器的文件,它可以帮助开发者将现代JavaScript代码转换为可在旧版浏览器或环境中运行的向后兼容代码。 Babel是一个JavaScript编译器,它可以将新的JavaScript语法转换为旧的语法,以便在旧版浏览器中运行。例如,如果你想使用ES6的箭头函数,但某些浏览器不支持它,你可以使用Babel将代码转换为ES5语法。 .babelrc文件是一个JSON格式的配置文件,它包含了Babel编译器的配置选项,包括预设和插件。预设(presets)是一组插件的集合,它们一起工作以转换代码,而插件(plugins)是单个功能的小组件,它们可以添加到编译器中以提供额外的功能。

index.jsx:

在项目中增加src目录,并在该目录下新建index.jsx文件

import React from "react";
import ReactDOM from "react-dom";

const App = () => {
  return (
    <div>Hello word</div>
  )
}

ReactDOM.render(<App />, document.getElementById("root"));

修改webpack.config.js中入口为index.jsx如下:

module.exports = {
    mode: 'production',
    entry: './src/index.jsx',
    output: {
        path: __dirname + '/dist',
        // [contenthash:8] - hash值 对应版本更新
        filename: '[name]-[contenthash:8].js',
        // 编译前清除目录
        clean: true,
    },
    module: {
        rules: [
            {
                test: /\.(js|jsx)$/,
                use: 'babel-loader',
            },
        ],
    },
    plugins: [],
};

测试打包:

在package.json中增加script对象如下:

"scripts": {
    "build": "webpack"
  },

运行yarn run build 可以看到在dist下面已经将包打出来了

webpack构建React、TSX项目(一)

至此,一个支持打包jsx的webpack可以正常进行了。

5、webpack-dev服务

前面四个模块就实现了webpack支持jsx打包。但是开发过程中,不能每次修改之后重现编译在,再在浏览器进行刷新查看,影响效率。

引入webpack-dev

yarn add -D webpack-dev-server

Webpack-dev是Webpack的一个开发服务器,它提供了一些特殊的功能,比如:

自动重载、Hot Module Replacement(HMR)、代理服务器、Source Map

使用Webpack-dev,可以提高开发效率,节省开发时间。

package.json中增加如下:

"scripts": {
    "start": "webpack serve", //增加start
    "build": "webpack"
  },

在webpack.config.js中配置webpack-dev配置项

devServer: {
    //是否启用 HTML5 历史记录模式,默认为 false。
    // 如果启用,所有请求都会返回 index.html 页面,适用于单页应用
    // 也可以用Object针对特定的路由返回设定的html页面
    historyApiFallback: true,
    open: true, // 自动打开页面
    hot: true,  // 是否启用热更新 默认为false
    // 是否开启代码压缩
    compress: false,
    // 启动的端口
    port: 8888,
    proxy: {
        '/api': 'http://localhost:3000',
      },
    client: {
        overlay: false, //当出现编译错误或警告时,在浏览器中显示全屏覆盖。
        progress: true, //在浏览器中以百分比显示编译进度。
    },
},

devServer更多配置详见此处

前面在index.js中挂在所需的root节点还未提供。需要新增index.html文件

index.html

在项目根目录下新增public文件夹,并新增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>myapp</title>
  </head>
  <body>
    <div ></div>
  </body>
</html>

正常运行项目 yarn start

webpack构建React、TSX项目(一)


发现页面是空白的,这是引入我们的js文件没有在html中引入。那如何让webpack自动将相关资源挂载到html中呢?

html-webpack-plugin

介绍:

它可以将生成的js文件自动注入到html文件中,生成一个包含所有资源的html文件。

使用HTML webpack插件可以避免手动管理生成的HTML文件和打包后的JS文件之间的关系,同时也可以自定义生成的HTML文件,例如添加自定义的meta信息、favicon图标等。


安装:

yarn add -D html-webpack-plugin

配置:

// 顶部引入
const htmlWebpackPlugin = require('html-webpack-plugin');

plugins: [
    // htmlWebpackPlugin 插件会将相关资源自动挂在到html中
    new htmlWebpackPlugin({
        filename: 'index.html', //生成的html文件名
        // 根据指定模板生成
        template: path.resolve(__dirname, './public/index.html'),
    }),
],

运行测试:

webpack构建React、TSX项目(一)

打包测试:

打包后发现dist下多了index.html 并且改文件中自动引入了js文件

webpack构建React、TSX项目(一)

webpack构建React、TSX项目(一)


至此,一基础的个可运行、可编译的jsx的react项目就完成后。

6、CSS、LESS

安装:

yarn add -D style-loader css-loader less-loader less
  • style-loader: 把 CSS 插入到 DOM 中。
  • css-loader: 会对 @import 和 url() 进行处理。将css打包到js中
  • less-lodaer: 将.less文件转换为CSS文件

在webpack.config.js中配置:

{
    test: /\.(le|c)ss$/,
    use: [
        'style-loader',
        'css-loader',
        'less-loader'
    ],
},

新建index.less文件。在index.jsx中引入该样式文件。运行之后

webpack构建React、TSX项目(一)


后续还会在配置css优化、字体引入、tsx、js打包优化等等。敬请关注哟,谢谢!