Обновление компонента State после получения Redux - PullRequest
0 голосов
/ 23 ноября 2018

Во-первых, извините за мой английский :)

Я изучаю React и Redux.Я просто хочу знать, почему я не могу обновить состояние одного контейнера с помощью Redux.Единственный способ решить эту проблему - это написать функцию времени ожидания (поскольку JS является асинхронным).Но я уверен, что есть лучший способ кодировать то, что я хочу сделать.Если кто-то может мне помочь, пожалуйста.Моя загрузка работает без проблем.

Контейнер FormAmdmin.js

import React, {Component} from 'react';
import {Container, Row, Col} from 'react-bootstrap';
import TextField from '@material-ui/core/TextField';
import Button from '@material-ui/core/Button';
import {getFormAdmin} from '../Actions/PostFormAdmin';
import { bindActionCreators } from 'redux';
import { connect } from 'react-redux';
import './FormAdmin.css'


const MapDispatchToProps = dispatch => {
    return bindActionCreators({getFormAdmin}, dispatch)
}

const MapStateToProps = state => {
    return {Auth : state.Auth}
}

class FormAdmin extends Component {
    constructor(props) {
        super(props);
        this.state = {
            email: '',
            password: '',
            isSuccess: false,
            response : {}
        };

    this.handleChange = this.handleChange.bind(this)
    this.handleSubmit = this.handleSubmit.bind(this)
    }


    componentWillReceiveProps(nextProps){
        console.log(nextProps)
        let response = nextProps.Auth.response
        let isSuccess = nextProps.Auth.isSuccess
        this.setState({
            response,
            isSuccess
        })
        this.onNaviguateAdmin()
   }

    // there 's no change to the state of my component
       onNaviguateAdmin = () => {
       console.log(this.state)
    }

    // It's working thanks to a setTimeOut Fonction
    //    onNaviguateAdmin = () => {
    //     setTimeout(() => {
    //         console.log('succes', this.state)
    //     }, 1000)

    // }


    handleChange(e){
       this.setState({
           [e.target.name] : e.target.value
       })
    }

    handleSubmit(e){
        e.preventDefault(); 
        const body = {
            email : this.state.email,
            password : this.state.password
        }
       this.props.getFormAdmin(body)
     }

    render() {
    console.log('state render', this.state) // The state in the render is OK
     const { Auth } = this.props
     let divResult = <div></div>
       if (this.state.response.flash ==='ok') {
        divResult = <div>Connexion réussie</div>
       }
       else if (this.state.response.flash ==='Not a yet a Success') {
           divResult = <div>Identifiants ou mot de passe invalides</div>
       }

        return (
            <div className="ContainerFormAdmin">
                <Container fluid>
                    <h3>Administration du site</h3>
                    <Row>
                        <Col xs={12}>
                        <form onSubmit={this.handleSubmit}>
                        <h5>Connexion à l\'Administration</h5>
                            <div>
                            <TextField 

                            name="email"
                            id="email"
                            label="email"
                            margin="normal"
                            onChange={this.handleChange}
                            />
                            </div>
                           <div>
                           <TextField 

                           name ="password"
                           id="password"
                           label="password"
                           margin="normal"
                           onChange={this.handleChange}
                           />
                           </div> 
                           <Button
                           type="submit"
                           >Connexion</Button>

                        </form>

                        </Col>
                    </Row>
                </Container>
                <div>{divResult}</div>

        </div>

        );
    }
}


export default connect(MapStateToProps, MapDispatchToProps)(FormAdmin);

Я использую MapDispatchToProps для получения действия: выборка.Я хотел бы получить ответ в состоянии моего компонента.Вот почему я использую "componentWillReceiveProps" для обновления моего состояния.Но это не работает.

AuthReducer.js

const initialState = {
    error: null,
    response: {},
    isSuccess: false
}

const FormAdmin = (state = initialState, action) => {
    console.log('je rentre dans auth reducers')
    switch (action.type) {
        case 'FORM_ADMIN_SUCCESS':
            if (action.payload.json.flash === 'ok') {
                return {
                    ...state,
                    response: action.payload.json,
                    isSuccess: true
                }
            } else if
                (action.payload.json.flash === 'Not a yet a Success'){
                    return {
                        ...state,
                        response: action.payload.json,
                        isSuccess: false
                    }
                }
        break   
        case 'FORM_ADMIN_ERROR':
            return {
                ...state,
                error: true,
                isSuccess: false
            }

        default:
            return state;
    }
}

export default FormAdmin

Действие PostFormAdmin.js

export const postFormAdminSuccess = (json) => ({
    type : 'FORM_ADMIN_SUCCESS', 
    payload : {json}
})

export const postFormAdminError = (error) => ({
    type : 'FORM_ADMIN_ERROR', 
    payload : {error}
})



function handleError(response){

    if (!response.ok){
        throw Error(response.statusText)
    }
   return response
}


export function getFormAdmin(formAdmin){
   return(dispatch) => {
        return fetch('http://localhost:4000/admin', {
            method : 'POST', 
            headers: {
                'Accept': 'application/json',
                'Content-Type': 'application/json'
              },
            body : JSON.stringify(formAdmin)
        })
        .then(handleError)
        .then(res => res.json())
        .then(
            res => {
                dispatch(postFormAdminSuccess(res));
            },
            err => {
                dispatch(postFormAdminError(err)); 
                console.log('erreur de response', err)
            }
        )
    }
}

Я использую Redx Thunk.

Index.js

import React from 'react';
import ReactDOM from 'react-dom';
import './index.css';
import App from './App';
import * as serviceWorker from './serviceWorker';
import 'bootstrap/dist/css/bootstrap.min.css';
import { Provider } from 'react-redux'
import { createStore, applyMiddleware } from 'redux'
import allReducers from './Reducers/Index';
import reduxThunk from 'redux-thunk';


const createStoreWithMiddleware = applyMiddleware(reduxThunk)(createStore); 
const store = createStoreWithMiddleware(
    allReducers
)


ReactDOM.render(
    <Provider store={store}>
    <App />
    </Provider>, document.getElementById('root'));

Я действительно ценю, если кто-то может помочь.Я знаю, что это работает с функцией setTimeout, но я уверен, что есть другой способ сделать это.

Спасибо за все!hollowspy

1 Ответ

0 голосов
/ 23 ноября 2018

Попробуйте передать обратный вызов setState , чтобы onNaviguateAdmin вызывался после фактического изменения состояния:

this.setState({
    response,
    isSuccess
}, this.onNaviguateAdmin)

Обратите внимание, что componentWillReceiveProps равно устарело и может быть причиной ваших проблем.Вы должны использовать componentDidUpdate вместо:

componentDidUpdate(prevProps){
    console.log(prevProps, this.props)
    if (this.props.Auth !== prevProps.Auth) {
        let response = this.props.Auth.response
        let isSuccess = this.props.Auth.isSuccess
        this.setState({
            response,
            isSuccess
        }, this.onNaviguateAdmin)
    }
}
...