这是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的样式是放在nodemodules中的,不要在css的loader中添加exclude命令刨除掉nodemodules目录。

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

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

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

配置按需加载

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

配置按需加载需要使用另外一个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。

同步博客原文链接

使用webpack、babel、react、antdesign配置单页面应用开发环境的更多相关文章

  1. npm+webpack+babel+react安装

    npm+webpack+babel+react安装 1.首先要安装 Node.js, Node.js 自带了软件包管理器 npm 2.在项目文件目录下生成package.json # 进入项目目录$ ...

  2. 详解 Webpack+Babel+React 开发环境的搭建

    1.认识Webpack 构建应用前我们先来了解一下Webpack, Webpack是一个模块打包工具,能够把各种文件(例如:ReactJS.Babel.Coffeescript.Less/Sass等) ...

  3. webpack+babel+react+antd技术栈的基础配置

    webpack+babel+react+antd技术栈的基础配置 前段时间使用webpack+babel+react+antd做了一套后台管理系统,刚开始被一大堆的新知识压的喘不过气来,压力挺大的.还 ...

  4. react 基础语法复习1- 搭建开发环境

    之前有看过阮一峰老师的react教程跟着做了一遍,学习了一下.好久没看,有点忘记了,这次跟着脚手架工具系统的复习一遍.顺便学习学习 react-router 和 redux 首先,脚手架工具我使用的是 ...

  5. React Native入门教程 1 -- 开发环境搭建

    有人问我为啥很久不更新博客..我只能说在学校宿舍真的没有学习的环境..基本上在宿舍里面很颓废..不过要毕业找工作了,我要渐渐把这个心态调整过来,就从react-native第一篇博客开始.话说RN也出 ...

  6. webpack+babel+react操作小结

    最近学习了一下Webpack,个人感觉还是非常实用的,现在总结一下自己的学习笔记. 什么是 Webpack Webpack 是一个模块打包器.它将根据模块的依赖关系进行静态分析,然后将这些模块按照指定 ...

  7. 使用vue-cli快速搭建大型单页面应用开发环境

    工作环境:terminal,已经全局安装了vue(可使用npm install -g vue) 全局安装vue-cli npm install -g vue-cli 创建一个基于webpack模板的项 ...

  8. 优化单页面开发环境:webpack与react的运行时打包与热更新

    前面两篇文章介绍初步搭建单页面应用的开发环境: 第一篇:使用webpack.babel.react.antdesign配置单页面应用开发环境 第二篇:使用react-router实现单页面应用路由 这 ...

  9. 分离Webpack开发环境与生产环境的配置

    这是Webpack+React系列配置过程记录的第五篇.其他内容请参考: 第一篇:使用webpack.babel.react.antdesign配置单页面应用开发环境 第二篇:使用react-rout ...

随机推荐

  1. Codeforces Round #343 (Div. 2)-629A. Far Relative’s Birthday Cake 629B. Far Relative’s Problem

    A. Far Relative's Birthday Cake time limit per test 1 second memory limit per test 256 megabytes inp ...

  2. 2017 ECJTU ACM 程序设计竞赛

    大厦 Time Limit : 4000/2000ms (Java/Other)   Memory Limit : 65535/32768K (Java/Other) Total Submission ...

  3. NYoj_104最大和

    最大和 时间限制:1000 ms  |  内存限制:65535 KB 难度:5 描述 给定一个由整数组成二维矩阵(r*c),现在需要找出它的一个子矩阵,使得这个子矩阵内的所有元素之和最大,并把这个子矩 ...

  4. android企业级商城源码、360°全景图VR源码、全民直播源码等

    Android精选源码 [新版]Android技术博客精华汇总 开源了:乐乐音乐5.0-Android音乐播放器 android实现仿真水波纹效果源码 360°全景图VR,这是一个值得把玩的APP a ...

  5. Spring框架学习笔记(6)——bean的继承与依赖

    一.bean的继承 Spring通过bean标签的parent属性来实现bean的继承,这个继承的概念只存在于配置层面,也就是说bean只会继承其父bean的属性值. 父bean可以通过abstrac ...

  6. ASP.NET CORE中使用Cookie身份认证

    大家在使用ASP.NET的时候一定都用过FormsAuthentication做登录用户的身份认证,FormsAuthentication的核心就是Cookie,ASP.NET会将用户名存储在Cook ...

  7. 浅谈event.client、event.screen与event.offset

    每每看到event.client.event.screen与event.offset这几个,头都大了,今天又碰到了,特来总结下. 1.event.screenX与event.screenY. 首先,e ...

  8. 织梦dede在首页调用留言本

    织梦dedecms在首页调用留言本 . {dede:loop table=dede_guestbook sort=dtime row=10 titlelen=36 typeid=40 if=ische ...

  9. MySQL改写子查询成Join

    有时用别的方式而不是子查询可以获得更高的性能 : For example: SELECT * FROM t1 WHERE id IN (SELECT id FROM t2); 改写: SELECT D ...

  10. Executors线程池

    强烈建议程序员使用较为方便的 Executors 工厂方法 Executors.newCachedThreadPool()(无界线程池,可以进行自动线程回收).Executors.newFixedTh ...