动手实现 React-redux(一) connect 和 mapStateToProps

react

1.结合 context 和 store

import React, { Component } from 'react';

import PropTypes from 'prop-types'

function createStore (reducer) {

let state = null

const listeners = []

const subscribe = (listener) => listeners.push(listener)

const getState = () => state

const dispatch = (action) => {

state = reducer(state, action)

listeners.forEach((listener) => listener())

}

dispatch({}) // 初始化 state

return { getState, dispatch, subscribe }

}

const themeReducer = (state, action) => {

if (!state) return {

themeColor: 'red'

}

switch (action.type) {

case 'CHANGE_COLOR':

return { ...state, themeColor: action.themeColor }

default:

return state

}

}

const store = createStore(themeReducer)

class Header extends Component {

static contextTypes = {

store: PropTypes.object

}

constructor () {

super()

this.state = { themeColor: '' }

}

componentWillMount () {

const { store } = this.context

this._updateThemeColor()

store.subscribe(() => this._updateThemeColor())

}

_updateThemeColor () {

const { store } = this.context

const state = store.getState()

this.setState({ themeColor: state.themeColor })

}

render () {

return (

<div>

<h1 style={{ color: this.state.themeColor }}>xutongbao</h1>

<ThemeSwitch />

</div>

)

}

}

class ThemeSwitch extends Component {

static contextTypes = {

store: PropTypes.object

}

// dispatch action 去改变颜色

handleSwitchColor (color) {

const { store } = this.context

store.dispatch({

type: 'CHANGE_COLOR',

themeColor: color

})

}

render () {

return (

<div>

<button onClick={this.handleSwitchColor.bind(this, 'red')}>Red</button>

<button onClick={this.handleSwitchColor.bind(this, 'blue')}>Blue</button>

</div>

)

}

}

class App extends Component {

static childContextTypes = {

store: PropTypes.object

}

getChildContext () {

return { store }

}

render() {

return (

<div>

<Header />

</div>

);

}

}

export default App;

2.connect 和 mapStateToProps

App.js:

import React, { Component } from 'react';

import PropTypes from 'prop-types'

import Header from './Header'

function createStore (reducer) {

let state = null

const listeners = []

const subscribe = (listener) => listeners.push(listener)

const getState = () => state

const dispatch = (action) => {

state = reducer(state, action)

listeners.forEach((listener) => listener())

}

dispatch({}) // 初始化 state

return { getState, dispatch, subscribe }

}

const themeReducer = (state, action) => {

if (!state) return {

themeColor: 'red'

}

switch (action.type) {

case 'CHANGE_COLOR':

return { ...state, themeColor: action.themeColor }

default:

return state

}

}

const store = createStore(themeReducer)

class App extends Component {

static childContextTypes = {

store: PropTypes.object

}

getChildContext () {

return { store }

}

render() {

return (

<div>

<Header />

</div>

);

}

}

export default App;

Header.js:

import React, { Component } from 'react';

import PropTypes from 'prop-types'

import { connect } from './react-redux'

import ThemeSwitch from './ThemeSwitch'

class Header extends Component {

static propTypes = {

themeColor: PropTypes.string

}

render () {

return (

<div>

<h1 style={{ color: this.props.themeColor }}>xutongbao</h1>

<ThemeSwitch/>

</div>

)

}

}

const mapStateToProps = (state) => {

return {

themeColor: state.themeColor

}

}

Header = connect(mapStateToProps)(Header)

export default Header

react-redux.js:

import React, { Component } from 'react'

import PropTypes from 'prop-types'

export const connect = (mapStateToProps) => (WrappedComponent) => {

class Connect extends Component {

static contextTypes = {

store: PropTypes.object

}

constructor () {

super()

this.state = { allProps: {} }

}

componentWillMount () {

const { store } = this.context

this._updateProps()

store.subscribe(() => this._updateProps())

}

_updateProps () {

const { store } = this.context

let stateProps = mapStateToProps(store.getState(), this.props) // 额外传入 props,让获取数据更加灵活方便

this.setState({

allProps: { // 整合普通的 props 和从 state 生成的 props

...stateProps,

...this.props

}

})

}

render () {

return <WrappedComponent {...this.state.allProps} />

}

}

return Connect

}

ThemeSwitch.js

import React, { Component } from 'react';

import PropTypes from 'prop-types'

class ThemeSwitch extends Component {

static contextTypes = {

store: PropTypes.object

}

// dispatch action 去改变颜色

handleSwitchColor (color) {

const { store } = this.context

store.dispatch({

type: 'CHANGE_COLOR',

themeColor: color

})

}

render () {

return (

<div>

<button onClick={this.handleSwitchColor.bind(this, 'red')}>Red</button>

<button onClick={this.handleSwitchColor.bind(this, 'blue')}>Blue</button>

</div>

)

}

}

export default ThemeSwitch

以上是 动手实现 React-redux(一) connect 和 mapStateToProps 的全部内容, 来源链接: utcz.com/z/383684.html

回到顶部