Пользовательские данные возвращаются неопределенными с Redux React App - PullRequest
0 голосов
/ 03 октября 2018

В настоящее время я пытаюсь передать пользовательские данные через приложение реагирования с Redux.Я создал пользовательский API с бэкэндом django, который определенно работает, так как я могу перейти по URL-адресу и увидеть весь json, который выходит из него.Тем не менее, когда я пытаюсь передать его в компонент, я получаю неопределенным.Вот мой код:

userActions.js:

import Axios from "axios";

    export function getUser() {
      const id = this.params.match.id
      return dispatch => {
        dispatch(fetchUserBegin());
        return Axios.get(`/api/user/${id}`)
        .then((res) => {
          this.setState({
              user: res.data,

          })
        })
      }
    }

    export const FETCH_USER_BEGIN   = 'FETCH_USER_BEGIN';
    export const FETCH_USER_SUCCESS = 'FETCH_USER_SUCCESS';
    export const FETCH_USER_FAILURE = 'FETCH_USER_FAILURE';

    export const fetchUserBegin = () => ({
      type: FETCH_USER_BEGIN
    });

    export const fetchUserSuccess = user => ({
      type: FETCH_USER_SUCCESS,
      payload: { user }
    });

    export const fetchUserFailure = error => ({
      type: FETCH_USER_FAILURE,
      payload: { error }
    });

userReducer.js

import {  FETCH_USER_BEGIN, FETCH_USER_SUCCESS, FETCH_USER_FAILURE  } from '../actions/actionTypes'


const initialState = {
    user: {},
    loading: false,
    error: null
  };

  export default function productReducer(state = initialState, action) {
    switch(action.type) {
      case FETCH_USER_BEGIN:
        // Mark the state as "loading" so we can show a spinner or something
        // Also, reset any errors. We're starting fresh.
        return {
          ...state,
          loading: true,
          error: null
        };

      case FETCH_USER_SUCCESS:
        // All done: set loading "false".
        // Also, replace the items with the ones from the server
        return {
          ...state,
          loading: false,
          user: action.user
        };

      case FETCH_USER_FAILURE:
        // The request failed, but it did stop, so set loading to "false".
        // Save the error, and we can display it somewhere
        // Since it failed, we don't have items to display anymore, so set it empty.
        // This is up to you and your app though: maybe you want to keep the items
        // around! Do whatever seems right.
        return {
          ...state,
          loading: false,
          error: action.payload.error,
          user: {}
        };

      default:
        // ALWAYS have a default case in a reducer
        return state;
    }
  }

И компонент дисплея:

UserInformation.js:

import React from "react";
import { connect } from "react-redux";
import { getUser } from "../store/actions/userActions";

class UserDetailView extends React.Component {
  componentDidMount() {
    this.props.dispatch(getUser());
  }

  render() {
    const { user } = this.props;
    console.log(user)

    return (
      <ul>
        {user.map(user =>
          <li key={user.id}>{user.username}</li>
        )}
      </ul>
    );
  }
}

const mapStateToProps = state => ({
  user: state.user,

});

export default connect(mapStateToProps)(UserDetailView);

Index.js

import React from 'react';
import ReactDOM from 'react-dom';
import App from './App';
import registerServiceWorker from './registerServiceWorker';
import { createStore, compose, applyMiddleware } from 'redux';
import { Provider } from 'react-redux';
import thunk from 'redux-thunk';

import reducer from './store/reducers/auth';


const composeEnhances = window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__ || compose

const store  = createStore(reducer, composeEnhances(
    applyMiddleware(thunk)
))

const app = (
    <Provider store={store}>
        <App />
    </Provider>

)

ReactDOM.render(app, document.getElementById('root'));
registerServiceWorker();

У кого-нибудь есть идеи, почему это не работает?

1 Ответ

0 голосов
/ 03 октября 2018

Вы не должны setState() в этом создателе действия:

 this.setState({
    user: res.data,
 })

вы должны отправить действие вместо

Попробуйте это:

export function getUser() {
      const id = this.params.match.id
      return dispatch => {
         dispatch(fetchUserBegin());
         return Axios.get(`/api/user/${id}`)
          .then( res => { 
            dispatch(fetchUserSuccess(res.data);
         })
       }
     }

Вы должны передать функцию mapDispatchToProps методу connect() в качестве второго аргумента, например:

import React from "react";
import { connect } from "react-redux";
import { getUser } from "../store/actions/userActions";

class UserDetailView extends React.Component {
  componentDidMount() {
    this.props.getUser()  //fixed
  }

  render() {
    const { user } = this.props;
    console.log(user)

    return (
      <ul>
        {user.map(user =>
          <li key={user.id}>{user.username}</li>
        )}
      </ul>
    );
  }
}

const mapStateToProps = state => ({
  user: state.user,
});
const mapDispatchToProps = dispatch => ({   //added
  getUser: dispatch(getUser())
})

export default connect(mapStateToProps,mapDispatchToProps)(UserDetailView);  //fixed

А также исправить это:

   case FETCH_USER_SUCCESS:
        // All done: set loading "false".
        // Also, replace the items with the ones from the server
        return {
          ...state,
          loading: false,
          user: action.payload.user  //fixed
        };
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...