React 教程第十四篇 —— Redux 跨组件通信高级篇之中间件

前端之家收集整理的这篇文章主要介绍了React 教程第十四篇 —— Redux 跨组件通信高级篇之中间件前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

Redux 复习总结

在前面三篇 Redux 的教程中已详细提到 Redux 的实现,大概可可以总结以下几点

  • Redux

    • ActionsReducerStore这三层
    • 通过createStore(reducer)得到store,换名话说store包含了reducer的逻辑实现
    • 通过store.dispath(action)调用reducer,从而改变state
    • 通过store.getState()获取reducer改变的state
    • Redux 本身与 React 没有并没有半毛线关系
  • React

    • Componentstateprops三大关键要素
    • 本身通过setState()改变state从而触发render,更新component
  • react-redux

    • 这是一个第三方模块,它的作用就是本来没有半毛钱关系的 Redux 和 React 关联在一起
    • 它有组件Provier方法connect
    • connect将 React 的state和 Redux 的actions合并成一个对象props,再将这个对象和component生成一个新的组件
    • Provider负责将 Redux 的store属性connect的新组件,从面将 React 和 Redux 关联到了一起
    • 当新组件调用action的时候,Provider.store就会映射调用reducer从而改变state,当state发生改变,就会触发新组件的render,重新更新组件。

中间件

上面是 React 依赖 Redux 的实现过程,但问题来了,如果项目中有异步请求,根据 Redux 的规则是:

  • Action 必须返回一个带有属性type的对象
  • Reducer 必须是一个纯函数,负责改变state

这样一来,这个异步请求就变得无处安放了,这个时候的解决方案就是需要一个模块,在这个模块中发起 ajax 请求,然后在请求的回调函数中去手动调用reducer,而这个发起 ajax 请求的模块就称之为中间件

实现

该案例是以中间件调用 nodejs 的公共接口,实现一个数据列表。

源码下载:https://github.com/dk-lan/rea...

效果预览

源码下载后执行下面步骤例可查看效果

  • npm install
  • npm start

结构

| src
|——| components
|——|——| datagrid
|——|——|——| datagridcomponent.js
|——|——|——| datagridaction.js
|——|——|——| datagridconstants.js
|——|——|——| datagridreducer.js
|——|——| cnode
|——|——|——| cnode.js
|——|——| spinner
|——|——|——| spinner.js
|——|——|——| spinner.scss
|——| redux
|——|——| store.js
|——|——| middleware.js
|——|——| rootReducer.js
|——| utils
|——|——| httpclient.js
|——| app.js

datagridcomponent.js
  1. import React from 'react'
  2. import {connect} from 'react-redux'
  3.  
  4. import SpinnerComponent from '../spinner/spinner'
  5.  
  6. import * as actions from './datagridaction'
  7.  
  8. class DatagridComponent extends React.Component{
  9. getKeys(item){
  10. let cols = item ? Object.keys(item) : [];
  11. return this.props.config.cols || cols;
  12. }
  13. componentWillMount(){
  14. this.props.refresh(this.props.config)
  15. }
  16. render(){
  17. return (
  18. <div>
  19. <table className="table">
  20. <thead>
  21. <tr>
  22. {
  23. this.getKeys(this.props.dataset[0]).map((key) => {
  24. return <th key={Math.random()}>{key}</th>
  25. })
  26. }
  27. </tr>
  28. </thead>
  29. <tbody>
  30. {
  31. this.props.dataset.map((item) => {
  32. return (
  33. <tr key={item.id || item.indexid} onDoubleClick={this.selectTr.bind(this,item)}>
  34. {
  35. this.getKeys(item).map((key) => {
  36. return <td key={Math.random()}>{item[key]}</td>
  37. })
  38. }
  39. </tr>
  40. )
  41. })
  42. }
  43. <tr></tr>
  44. </tbody>
  45. </table>
  46. <SpinnerComponent show={this.props.show}/>
  47. </div>
  48. )
  49. }
  50. }
  51.  
  52. const mapStateToProps = (state) => {
  53. return {
  54. dataset: state.datagrid.dataset || [],show: state.datagrid.show,error: state.datagrid.error
  55. }
  56. }
  57.  
  58. export default connect(mapStateToProps,actions)(DatagridComponent);

datagridconstants.js

  1. export const Requesting = 'Requesting'
  2. export const Requested = 'Requested'
  3. export const RequestError = 'RequestError'

datagridaction.js

  1. import * as constants from './datagridconstants'
  2.  
  3. export function refresh(_config){
  4. return {
  5. types: [constants.Requesting,constants.Requested,constants.RequestError],url: _config.url,method: _config.method || 'get',data: _config.data || {},name: _config.name
  6. }
  7. }

datagridreducer.js

  1. import * as constants from './datagridconstants'
  2.  
  3. export default function datagrid(state = {},action){
  4. let _state = JSON.parse(JSON.stringify(state));
  5. switch(action.type){
  6. case constants.Requesting:
  7. _state.show = true;
  8. break;
  9. case constants.Requested:
  10. _state.show = false;
  11. if(action.name){
  12. _state[action.name] = _state[action.name] || {};
  13. _state[action.name].dataset = action.result;
  14. } else {
  15. _state.dataset = action.result;
  16. }
  17. break;
  18. case constants.RequestError:
  19. _state.show =false;
  20. _state.error = action.error;
  21. break
  22. }
  23. return _state;
  24. }

spinner.js

  1. import React,{Component} from 'react'
  2. import './spinner.scss'
  3.  
  4. class SpinnerComponent extends React.Component{
  5. render(){
  6. let html = (
  7. <div>
  8. <div className="dk-spinner-mask"></div>
  9. <div className="dk-spinner dk-spinner-three-bounce">
  10. <div className="dk-bounce1"></div>
  11. <div className="dk-bounce2"></div>
  12. <div className="dk-bounce3"></div>
  13. </div>
  14. </div>
  15. )
  16. return this.props.show ? html : null;
  17. }
  18. }
  19.  
  20. export default SpinnerComponent

spinner.scss

  1. .dk-spinner-mask {
  2. position: fixed;
  3. top: 0;
  4. right: 0;
  5. bottom: 0;
  6. left: 0;
  7. background-color: #fff;
  8. opacity: .4;
  9. z-index: 2090;
  10. }
  11.  
  12. .dk-spinner-three-bounce.dk-spinner {
  13. position: absolute;
  14. top: 53%;
  15. left: 47%;
  16. background-color: none!important;
  17. z-index: 2099;
  18. margin: 0 auto;
  19. width: 70px;
  20. text-align: center;
  21. }
  22.  
  23. .dk-spinner-three-bounce div {
  24. width: 18px;
  25. height: 18px;
  26. background-color: #1ab394;
  27. border-radius: 100%;
  28. display: inline-block;
  29. -webkit-animation: dk-threeBounceDelay 1.4s infinite ease-in-out;
  30. animation: dk-threeBounceDelay 1.4s infinite ease-in-out;
  31. -webkit-animation-fill-mode: both;
  32. animation-fill-mode: both;
  33. }
  34.  
  35. .dk-spinner-three-bounce .dk-bounce1 {
  36. -webkit-animation-delay: -.32s;
  37. animation-delay: -.32s;
  38. }
  39.  
  40. .dk-spinner-three-bounce .dk-bounce2 {
  41. -webkit-animation-delay: -.16s;
  42. animation-delay: -.16s;
  43. }
  44.  
  45. @-webkit-keyframes dk-threeBounceDelay{
  46. 0%,100%,80%{-webkit-transform:scale(0); transform:scale(0)}
  47. 40%{-webkit-transform:scale(1);transform:scale(1)}
  48. }
  49. @keyframes dk-threeBounceDelay{
  50. 0%,80%{-webkit-transform:scale(0);transform:scale(0)}
  51. 40%{-webkit-transform:scale(1);transform:scale(1)}
  52. }

cnode.js

  1. import React from 'react'
  2. import Datagrid from '../../components/datagrid/datagridcomponent'
  3.  
  4. export default class CNodeComponent extends React.Component{
  5. static defaultProps = {
  6. config: {
  7. url: 'https://cnodejs.org/api/v1/topics',data: {page: 1,limit: 10},cols: ['title','reply_count','top','create_at','last_reply_at']
  8. }
  9. }
  10. render(){
  11. return (
  12. <div>
  13. <Datagrid config={this.props.config}/>
  14. </div>
  15. )
  16. }
  17. }

rootReducer.js

  1. import React from 'react'
  2. import {combineReducers} from 'redux'
  3.  
  4. import datagrid from '../components/datagrid/datagridreducer'
  5.  
  6. export default combineReducers({
  7. datagrid
  8. })

middleware.js

  1. import http from '../utils/httpclient'
  2. import * as constants from '../components/datagrid/datagridconstants'
  3.  
  4. export default function(api){
  5. return function(dispatch){
  6. return function(action){
  7. let {type,types,url,data,method = 'get',name} = action;
  8. if(!url){
  9. return dispatch(action)
  10. }
  11.  
  12. dispatch({type: constants.Requesting})
  13.  
  14. http[method](url,data).then((res) => {
  15. let _action = {
  16. type: constants.Requested,name,result: res.data
  17. }
  18. dispatch(_action)
  19. }).catch((error) => {
  20. dispatch({type: constants.RequestError})
  21. })
  22. }
  23. }
  24. }

store.js

  1. import React from 'react'
  2. import {createStore,applyMiddleware} from 'redux'
  3.  
  4. import rootReducer from './rootReducer'
  5.  
  6. import middleware from './middleware'
  7.  
  8. const store = createStore(rootReducer,applyMiddleware(middleware));
  9.  
  10. export default store;

app.js

  1. import './libs/bootstrap/css/bootstrap.min.css'
  2. import './libs/font-awesome/css/font-awesome.min.css'
  3.  
  4. import React from 'react'
  5. import ReactDOM from 'react-dom'
  6. import {Provider} from 'react-redux'
  7.  
  8. import store from './redux/configStore'
  9. import CNodeComponent from './components/cnode/cnode'
  10.  
  11. ReactDOM.render(
  12. <Provider store={store}>
  13. <CNodeComponent/>
  14. </Provider>,document.getElementById('app')
  15. )

小结

  • 组件 datagrid 支持动态配置
  • ajax请求放到了中间件执行
  • ajax请求为了统一处理,分为三个状态,就是constants.js文件中的三个变量,分别代表请求前,请求中,请求后
  • 请求前显示加载组件spinner,请求结束后移除加载组件spinner

猜你在找的React相关文章