web-dev-qa-db-fra.com

Redux Router - "La répartition n'est pas définie"

J'ai un composant simple qui appelle une action lorsqu'un utilisateur charge une page, et à l'intérieur de cette action, j'essaie d'envoyer une autre action pour définir l'état loggedIn du magasin sur vrai ou faux:

import React, { Component } from 'react'
import { Link, browserHistory } from 'react-router'
import $ from 'jquery'

class Login extends Component {

  constructor(props) {
    super(props)
  }
  componentDidMount() {
    this.props.actions.guestLoginRequest()
  }
  render() {
    return (
      <div>
        <div classNameName="container">
          <div className="row">
            We are signing you in as a guest
          </div>
        </div>
      </div>
    )
  }
}

export default Login

Je peux obtenir les informations de connexion lorsque l'action guestLoginRequest est appelée, mais lorsque j'essaie d'envoyer une autre action à l'intérieur, rien ne se passe:

guestLoginRequest: function(){
    var ref = new Firebase("https://penguinradio.firebaseio.com");
    ref.authAnonymously(function(error, authData) {
      if (error) {
        console.log("Login Failed!", error);
      } else {
        console.log("Authenticated successfully with payload:", authData);
        return dispatch => {
          dispatch(actions.setLoginStatus(true, authData))
          console.log("dispatched");
        };
      }
    });
  }

J'obtiens une erreur de Uncaught ReferenceError: dispatch is not defined lorsque je supprime le return dispatch => { } déclaration. Dans mon magasin, j'utilise redux-thunk, donc je peux répartir à l'intérieur des actions:

// Store.js
import { applyMiddleware, compose, createStore } from 'redux'
import rootReducer from './reducers'
import logger from 'redux-logger'
import thunk from 'redux-thunk'

let finalCreateStore = compose(
  applyMiddleware(thunk, logger())
)(createStore)


export default function configureStore(initialState = { loggedIn: false }) {
  return finalCreateStore(rootReducer, initialState)
}

Je mappe également la répartition aux accessoires dans mon app.js:

function mapStateToProps(state) {
  return state
}
function mapDispatchToProps(dispatch) {
  return {
      actions: bindActionCreators(actions, dispatch)
  }
}

export default connect(mapStateToProps, mapDispatchToProps)(App)

Juste au cas où cela pourrait être utile, voici mes fichiers client.js et réducteur:

// client.js
import React from 'react'
import { render } from 'react-dom'
import App from '../components/App'
import configureStore from '../redux/store'
import { Provider } from 'react-redux'


let initialState = {
  loggedIn: false
}

let store = configureStore(initialState)

render(
  <Provider store={store}>
    <App />
  </Provider>,
  document.getElementById('app')
)
// Reducer.js
import { combineReducers } from 'redux'

let LoginStatusReducer = function reducer(loggedIn = false, action) {
  switch (action.type) {
    case 'UPDATE_LOGIN_STATUS':
      return loggedIn = action.boolean
    default:
      return loggedIn
  }
}
export default LoginStatusReducer

const rootReducer = combineReducers({
  loggedIn: LoginStatusReducer
})

export default rootReducer

Des idées pourquoi ma fonction de répartition ne fonctionne pas? Je suis confus depuis que j'ai configuré redux-thunk avec mon magasin, et j'utilise un code similaire aux documents lorsque j'appelle return dispatch => { }. Y a-t-il quelque chose qui me manque? Merci d'avance pour tout conseil!

11
Mike

Vous avez besoin de votre action pour renvoyer une fonction pour utiliser le middleware thunk, puis redux y injectera le répartiteur. Vous avez mélangé votre invocation de répartiteur avec les détails d'implémentation. L'extrait de code suivant corrige les deux défauts.

guestLoginRequest: function(){
  return function (dispatch) {
    var ref = new Firebase("https://penguinradio.firebaseio.com");
    ref.authAnonymously(function(error, authData) {
      if (error) {
        console.log("Login Failed!", error);
      } else {
        console.log("Authenticated successfully with payload:", authData);
        dispatch(actions.setLoginStatus(true, authData))
        console.log("dispatched");
      }
    });
  }
}

De plus, vous devez répartir correctement votre action sur la classe Login.

dispatch(this.props.actions.guestLoginRequest())

L'invocation de votre action se fait toujours en appelant dispatch. Le flux devrait être quelque chose comme ceci:

React component --> dispatch ---> API call (thunk middleware) --> dispatch ---> reducer
9
Matt Lo