美文网首页前段react+webpackReact
使用webpack、babel、react、antdesign配

使用webpack、babel、react、antdesign配

作者: chardlau | 来源:发表于2017-04-28 16:54 被阅读7848次

    这是Webpack+React系列配置过程记录的第一篇。其他内容请参考:

    本文内容将记录使用webpack、babel、react、antdesign配置单页面应用开发环境的过程。这是我首次使用前端框架开发Web应用,在此之前,我写Web代码使用最原生的HTML、CSS、JS,即使是敲这些文字的这一刻,我还没能熟练使用jQuery库。最近想跨出这个怪圈,希望使用一些“前沿”一些的框架重构之前写的书籍分享网站的管理后台后台,因此才选择使用webpack、babel、react、ant design作为突破口。

    主要依赖库的版本如下(PS. 不同版本配置不一定一样):

    • webpack: 2.4.1
    • react: 15.4.2
    • react-dom: 15.4.2
    • babel-core: 6.24.1
    • babel-loader: 7.0.0
    • antd: 2.5.2

    准备静态Web服务器

    静态Web服务器可以使用Nginx、Apache等现成的服务器软件,仅用于接下来对配置结果的测试和校验。

    为了节省安装这些服务器软件的精力,我直接使用node.js配合express写了个简单的静态服务器。也把这一步骤记录下来。

    1. 创建项目目录demo,并使用npm init命令初始化node.js空项目,生成package.json文件。
    2. 使用npm安装express。
    3. 创建Web的启动脚本index.js。
    4. 创建存放静态资源的目录public
    5. 在public中创建页面文件index.html

    index.js文件的内容如下:

    var express = require('express');
    var app = express();
    app.use('/', express.static('public'));
    var server = app.listen(2000, function() {
      var port = server.address().port;
      console.log('Open http://localhost:%s', port);
    });
    

    index.html文件的内容如下:

    <html>
      <head></head>
      <body>
        <p>Hello world</p>
      </body>
    </html>
    

    使用node启动index.js,就可以在浏览器中输入 http://localhost:2000访问到index.html中的内容了。

    接下来才是真正开始本文的真正内容。我的目的很明确——开发一个网页,因此我将从React开始。

    使用React开发页面

    使用React框架编写页面的方法有两种,第一种是直接在html文件中直接引用react.js和react-dom.js,然后在后续的脚本中使用React带来的能力。

    仅改动前文提到的index.html文件即可做到:

    <html>
      <head>
        <script src="https://unpkg.com/react@15/dist/react.js"></script>
        <script src="https://unpkg.com/react-dom@15/dist/react-dom.js"></script>
      </head>
      <body>
        <p>Hello world</p>
        <div id='main'/>
        <script>
          var e = React.createElement('p', null, 'This is React');
          ReactDOM.render(e, document.getElementById('main'));
        </script>
      </body>
    </html>
    

    在浏览器中访问index.html,你将看到页面使用React渲染了一行文字。

    但是通过这种方式使用React并不能很好发挥其作用。我们要使用ES6、JSX、UI重用等特性,因此我们将使用下面的第二种方法。

    使用Webpack、Babel和React开发页面

    这种方法实际上你所写的代码并不是最终在用户浏览器上执行的代码。源代码将经过Babel和Webpack进行转换处理,然后生成可在用户浏览器上执行的代码。这个过程有点像C或者C++的编译。

    安装React

    使用下面命令安装React(参考:https://facebook.github.io/react/docs/installation.html)

    npm install react@15.4.2 react-dom@15.4.2 --save
    

    由于在写这篇文章之前做过了一次测试,预料到如果使用最新版本的react,后面引入的ant design将会报兼容性错误,故指定了react的版本。

    因为浏览器并没有提供给我们引入其他模块的能力,我们也就无法直接使用React模块,这时需要Webpack的帮助。

    安装与配置Webpack

    使用下面命令安装Webpack

    npm install webpack --save-dev
    

    配置Webpack的方法有好几种,我使用的是配置文件的方式。在项目根目录下生成webpack.config.js文件。内容如下:

    var path = require('path');
    module.exports = {
      entry: './public/index.js',
      output: {
        filename: 'out.js',
        path: path.resolve(__dirname, 'public')
      }
    };
    

    这个文件配置了webpack的入口为public目录下的index.js,转换后生成的文件为public/out.js。

    接下来在package.json配置一个build命令,内容为:

    ...
    "scripts": {
        "build": "webpack --config webpack.config.js",
        "start": "node index.js",
      },
    ...
    

    这样就可以使用npm run build调用webpack对js文件进行转换。

    再看一下此刻的public目录的index.js文件内容:

    import React from 'react';
    import ReactDOM from 'react-dom';
    var e = React.createElement('p', null, 'This is React');
    ReactDOM.render(e, document.getElementById('main'));
    

    而index.html文件的内容:

    <html>
      <head>
      </head>
      <body>
        <p>Hello world</p>
        <div id='main'></div>
        <script src="out.js"></script>
      </body>
    </html>
    

    注意到html中script标签引用的js文件是out.js,这个文件是webpack生成的,是最终在用户浏览器上执行的脚本。

    现在我们执行npm run build命令,项目文件布局内容如下图所示:

    项目文件布局
    确实生成了我们想要的out.js文件了。

    Webpack有一个优点就是针对JS文件,它能够做到按需加载。后面你将会发现对于其他类型的文件,如:css、html、jade等,只要使用了适当的插件也能做到按需加载。

    到这里,你会发现不需要使用Babel,我们就可以使用React开发页面内容了。但是你会注意到,除了import语句是webpack做了兼容性处理的,我所使用的其他语法都是ES5的语法。

    如果在index.js中使用JSX语法,webpack构建的时候就会报错。同样地,如果在index.js中添加ES6的语法,尽管webpack构建时不会报错,但生成的out.js文件末尾部分依旧可以找到这部分ES6语法的代码,这样的代码被用户加载到浏览器中执行,是否能被浏览器支持是无法得到保证的。

    为此,我们需要引入Babel,让其把所有浏览器可能不支持的语法转换成ES5的语法。

    安装与配置Babel

    使用Babel的方法也有好多种,官网的帮助文档可以根据环境提供对应的帮助。本文是基于Webpack使用Babel,因此无论是增加的依赖库还是配置流程均只对Webpack的场景有效。我们将使用Webpack关于Babel的一个扩展加载器babel-loader关联这两者。而且关于Babel的配置也集合到webpack.config.js中,通过webpack传递给babel,这样可以省去独立配置babel步骤。

    安装基于Webpack的Babel使用下面命令:

    npm install --save-dev babel-loader
    npm install --save-dev babel-core
    

    上文提到,Babel用于把新版本的JS代码翻译成大多数浏览器都支持的ES5版本的代码。要做到这些翻译,只需要使用对应的扩展即可。例如:

    • babel-preset-env根据配置环境计算babel对代码填充何种等级的polyfill,已包括es2015的配置,但没有包括stage-x。官方首推。
    • babel-preset-[stage-0、stage-1、stage-2、stage-3]与ES7相关的配置;
    • 除此之外,Babel还可以有其他扩展,比如我们希望使用JSX语法,这时需要引入babel-preset-react处理器。

    根据官方的推荐,我们暂时先使用这几个:

    npm install --save-dev babel-preset-react
    npm install --save-dev babel-preset-stage-0
    npm install --save-dev babel-preset-env
    

    在webpack.config.js中配置babel,内容更新为:

    var path = require('path');
    module.exports = {
      entry: './public/index.js',
      output: {
        filename: 'out.js',
        path: path.resolve(__dirname, 'public')
      },
      module: {
        rules: [
          {
            test: /\.js$/,
            exclude: /node_modules/,
            use: {
              loader: 'babel-loader',
              options: {
                presets: ['env', 'stage-0', 'react'],
                plugins: []
              }
            }
          }
        ],
      },
    };
    

    注意到我们在module.rules中添加了一个规则。这个规则只要匹配到node_modules以外的js后缀的文件就会使用babel-loader进行转换,转换过程依次使用了plugins和presets制定的扩展。注意presets的处理顺序比较特别,是从右到左顺序执行的。

    到这里就配置完babel了。我们修改index.js文件:

    import React from 'react';
    import ReactDOM from 'react-dom';
    class Text extends React.Component {
      render() {
        return (
          <p>This is a react Component</p>
        );
      }
    }
    ReactDOM.render(<Text/>, document.getElementById('main'));
    

    使用命令编译并启动服务器,没有再因为JSX语法报错。

    npm run build
    npm start
    

    在浏览器访问index.html可以看到“This is a react Component”,这说明Babel很好地工作了。

    到这里如果没有其他需求就可以开始开发了。

    使用Ant Design作为React的UI库

    由于我已经怕自己造轮子了,这里我要引入基于React的一个Ant Design规范UI库antd。

    安装Ant Design
    npm install --save antd
    

    好了可以直接使用Ant Design的UI了。修改index.js:

    import React from 'react';
    import ReactDOM from 'react-dom';
    import { DatePicker } from 'antd';
    class Text extends React.Component {
      render() {
        return (
          <div>
            <p>This is a react Component</p>
            <DatePicker/>
          </div>
        );
      }
    }
    ReactDOM.render(<Text/>, document.getElementById('main'));
    

    重新构建后执行,在浏览器访问index.html可以看到多了个日期选择器,但是没有相应的样式。

    加载样式表

    想要在js中引入css样式表,需要添加webpack的加载器,这样在js中引入的css样式将会被webpack构建成以动态方式插入到html文件中。

    针对css样式,需要下面两个加载器:

    npm install --save-dev css-loader
    npm install --save-dev style-loader
    

    然后在webpack.config.js中配置一个新的规则:

    var path = require('path');
    module.exports = {
      entry: './public/index.js',
      output: {
        filename: 'out.js',
        path: path.resolve(__dirname, 'public')
      },
      module: {
        rules: [
          {
            test: /\.js$/,
            exclude: /node_modules/,
            use: {
              loader: 'babel-loader',
              options: {
                presets: ['env', 'stage-0', 'react'],
                plugins: []
              }
            }
          },
          {
            test: /\.css$/,
            use: ['style-loader', 'css-loader']
          }
        ],
      },
    };
    

    注意不要犯我的一个错误:因为antd的样式是放在node_modules中的,不要在css的loader中添加exclude命令刨除掉node_modules目录。

    这时候可以在index.js中添加下面一行代码:

    ...
    import { DatePicker } from 'antd';
    import 'antd/dist/antd.css';  // Add
    ...
    

    重新构建运行就可以看到日期选择器有样式了。

    配置按需加载

    上述方法实际上加载整个antd包到最终生成的out.js文件中了。不信的话,在浏览器访问index.html可以在console中看到下面的提示:


    Ant Design警告

    配置按需加载需要使用另外一个Babel的插件babel-plugin-import,安装命令如下:

    npm install --save-dev babel-plugin-import
    

    因为这是Babel的插件,所以它的配置要在babel-loader的plugins节点中配置。修改后的webpack.config.js内容如下:

    var path = require('path');
    module.exports = {
      entry: './public/index.js',
      output: {
        filename: 'out.js',
        path: path.resolve(__dirname, 'public')
      },
      module: {
        rules: [
          {
            test: /\.js$/,
            exclude: /node_modules/,
            use: {
              loader: 'babel-loader',
              options: {
                presets: ['env', 'stage-0', 'react'],
                plugins: [['import', {"libraryName": "antd", "style": "css"}]]
              }
            }
          },
          {
            test: /\.css$/,
            use: ['style-loader', 'css-loader']
          }
        ],
      },
    };
    

    这样就完成了按需加载的配置了。

    使用antd的组件的时候也无需再增加import样式文件的语句了。去除掉刚刚在index.js中增加的那一行语句吧。重新构建运行,你会看到警告没有了。out.js中的行数从原来的13万多行减少到了5万多。

    到此配置完毕。

    注:所有内容均参考自React、Babel、Webpack、AntDesign、第三方插件等的官方网站或对应Github。

    本文来自作者同步博客

    相关文章

      网友评论

      • 86d4576e2306:还有就是少了自动编译的配置,可以依据webpack-dev-server模块搞
      • 86d4576e2306:webpack.config.js中的导出文件路径path是相对当前文件的根目录创建的;npm start命令失效或者报错可能是全局未安装core-js模块,使用淘宝镜像重新安装,命令cnpm install -g core-js;还有就是package.json中的script中默认命令路径是相对当前目录的路径(start:controller/main,js)
      • 星南樱射:一步一步跟下来,对webpack和这些配置,终于又更明白一点了,感谢!
      • 48026991aba9:你好,能帮我看下吗,我这边配置了之后,antd一直不能加载
        6b1a0a11a7aa:@chardlau 大神能帮我看一下配置文件吗
        48026991aba9:@chardlau 可以,怎么私信给你,
        chardlau:你碰到的问题的错误提示或者错误日志能告诉我吗?
      • 花卷菌:楼主 npm start不能启动项目 为啥
        chardlau:今天收到了某位简友的邮件,发现有部分小伙伴的环境里面缺少了cross-env这个小工具,所以导致项目无法启动。
        使用“npm install --save-dev cross-env”安装这个工具,然后重新启动。
        不知道你碰到的是不是这个问题。
      • e51a6af457cb:谢谢大神提供教程:kissing_heart:
      • Atrbear:你好,请问配置完了babel-plugin-import之后仍然会有"You are using a whole package of antd, please use https://www.npmjs.com/package/babel-plugin-import to reduce app bundle size."这样子的错误应该怎么解决?
        0e5faaba8d10:@Atrbear 你问题怎么解决的?我的也是配置完成后,依旧报错
        Atrbear:@chardlau :stuck_out_tongue_closed_eyes: 问题已经解决了,谢谢。
        chardlau:你好,能给我私信发一下你的配置吗?
      • 9c2b2388a02b:请问下,有github源码地址吗?
        chardlau:@NAN_5793 已加,不过发消息给你没收到你的回应
        6b1a0a11a7aa:@chardlau 可以帮我看一下吗 我已经2天没解决了 麻烦大神加一下我qq 1907915060 我给你看一下 不胜感激
        chardlau:本来是没有的,不过当我看到你的评论时马上上github建了一个:https://github.com/chardlau/webpack-react-staging
        没怎么整理,代码依旧有点挫

      本文标题:使用webpack、babel、react、antdesign配

      本文链接:https://www.haomeiwen.com/subject/lskazttx.html