沿着上一部分 Get Started Redux,在利用 redux 构建了 store reducer 之后,现在需要给应用提供个 view 了。我们用当下最流行的 react。
安装 react
$ npm install --save react react-dom react-redux其中 react react-dom 是 react 的原生包,react-redux 提供了一些方便的方法用来将 redux 和 react 一起使用。后面的代码示例会着重讲解。
添加 babel 对 jsx 的支持。
jsx 是 react 支持的一种特殊的语法,这种语法用于将 html 写到 js 中,例如
return (
<p>
Click {value} times.
{ ' ' }
<button onClick={onIncrement}>+</button>
{ ' ' }
<button onClick={onDecrement}>-</button>
</p>
)目前可以认为 jsx 是写 react 的标配,那么就需要在 babel 中添加一个新的 react 翻译器。
$ npm install --save-dev babel-preset-react然后修改 .babelrc
{
"presets": ["es2015", "react"]
}这样,我们就可以开始写 react 了。
在 redux 的官方文档 Use With React 解释了一种 redux 和 react 结合的模式:将 react 的组件分为两种,一种是不与 redux 交互的 presentational 组件,它是一种通用的组件,任何提供给它所需要的 func 或者是 prop 的框架都可以使用它。另一种是 container 组件,是和 redux 的 action 以及 store 绑定的组件。通常都是会先写一个 presentational 组件,然后再创建一个 container 组件对 presentational 组件进行包装后使用。后面的 full example 展示了 Counter 与 Visible Counter 两个不同类型的组件。当然在官方也提供了非常好的例子。
首先展示一下目录结构
.
├── actions
│ └── index.js
├── components
│ ├── App.js
│ └── Counter.js
├── containers
│ └── VisibleCounter.js
├── dist
│ ├── bundle.js
│ ├── index.html
│ └── styles.css
├── entry.js
├── package.json
├── reducers
│ └── counter.js
├── styles
│ ├── index.scss
│ └── theme.scss
├── test
│ └── reducers
│ └── counter.spec.js
└── webpack.config.js相对于之前的目录结构多了两个目录 containers components 分别对应了 container 和 presentational 的组件。
首先我们定义一个 presentational 的组件 Counter
components/Counter.js:
import React, { Component, PropTypes } from 'react';
class Counter extends Component {
render() {
const { value, onIncrement, onDecrement } = this.props;
return (
<p>
Click {value} times.
{ ' ' }
<button onClick={onIncrement}>+</button>
{ ' ' }
<button onClick={onDecrement}>-</button>
</p>
)
}
}
Counter.PropTypes = {
value: PropTypes.number.isRequired,
onIncrement: PropTypes.func.isRequired,
onDecrement: PropTypes.func.isRequired
};
export default Counter;注意 PropTypes 有点像是一个函数的参数说明,它明确的定义了在使用这个组件时需要提供什么样子的东西,react 官方建议一定要对每一个组件都明确的定义这样的参数说明,它也起到了文档的作用,方便与其他人协作。这里我们需要三个东西:
value 当前的计数onIncrement 当点击 + 时的动作onDecrement 当点击 - 时的动作然后对 Counter 进行包装
containers/VisibleCounter.js:
import { connect } from 'react-redux';
import Counter from "../components/Counter";
import { increment, decrement } from '../actions/index';
function mapStatToProps(state) {
return {
value: state
}
}
function mapDispatchToProps(dispatch) {
return {
onIncrement: () => {
dispatch(increment())
},
onDecrement: () => {
dispatch(decrement())
}
}
}
const VisibleCounter = connect(
mapStatToProps,
mapDispatchToProps
)(Counter);
export default VisibleCounter;这里用到了一个 react-redux 提供的方法 connect 它和下文提到的 Provider 配合使用,用于为 react 传递全局的 store。connect 需要两个函数 mapStatToProps 与 mapDispatchToProps 分别将 store 里的属性和 store 的 dispatch 动作传递给 presentational 组件。上面的代码就分别将 store 的 state 对应给组件的 value 属性,将两个 action 的 dispatch 对应到 onIncrement 与 onDecrement。
然后还有一个 App 组件,用于包装 VisibleCounter,它没有任何依赖的属性。
components/App.js:
import React, { Component } from 'react';
import VisibleCounter from '../containers/VisibleCounter';
class App extends Component {
render() {
return (
<div>
<VisibleCounter />
</div>
)
}
}
export default App;最后我们修改 entry.js 将 store 与我们的视图绑定起来。
entry.js:
require('./styles/index.scss');
import React from "react";
import ReactDOM from 'react-dom';
import counter from './reducers/counter';
import { Provider } from 'react-redux';
import App from './components/App';
import { createStore } from 'redux';
const store = createStore(counter);
const rootEl = document.querySelector("#root");
function render() {
ReactDOM.render(
<Provider store={store}>
<App />
</Provider>,
rootEl
)
}
render();
store.subscribe(render);这里就用到了 Provider 方法,将 store 传递给 App,然后用 ReactDOM 在页面上生成 react 组件。