Мои this.props не определены для handleSubmit и других функций - PullRequest
0 голосов
/ 30 июня 2019

Я не знаю, почему мои this.props показывают неопределенное на handleSubmit.Я сделал все возможное.Когда я регистрирую состояние в mapStateToProps, я вижу, что отправлено действие, но this.props не принимает его.

Это то, что я получаю (https://imgur.com/a/AXixWn9) в логгере, когда я ввожу неправильные данныеЯ искал в Интернете, но понятия не имел. Я отлаживал это часами

Login.js

import React from 'react';
import { NavLink } from 'react-router-dom';
import { Redirect } from 'react-router';
import qs from 'qs';
import { connect } from 'react-redux';
import Outer from '../templates/outer';
import history from '../../_helpers/history';
import routes from '../../services/urls';
import apiRequest from '../../services/api';
import loginUser from '../../redux/actions/authentication.action';
import { LOGIN_FAILURE, LOGIN_SUCCESS } from '../../redux/constants/authentication.constants';
import alertActions from '../../redux/actions/alert.actions';


class LoginComponent extends React.Component {
    constructor(props) {
        super(props);
        this.state = {
            fields: {},
            errors: {},
            loginError: '',
            submitted: false,
        };
        const { dispatch } = this.props;
        dispatch(alertActions.clear());
        this.handleSubmit = this.handleSubmit.bind(this);
    }

    handleChange(field, e) {
        const { fields } = this.state;
        fields[field] = e.target.value;
        this.setState({ fields });
    }

    handleValidation() {
        const { fields } = this.state;
        const errors = {};
        let formIsValid = true;

        // Login code
        if (!fields.code) {
            formIsValid = false;
            errors.code = 'This field is required';
        }

        // Password
        if (!fields.password) {
            formIsValid = false;
            errors.password = 'This field is required';
        }

        this.setState({ errors });
        return formIsValid;
    }

    handleSubmit(e) {
        e.preventDefault();
        const { code, password } = this.state.fields;
        this.setState({ submitted: true });
        if (this.handleValidation()) {
            this.setState({ submitted: true });
            const payload = {
                email: code,
                // level: 'SYSTEM',
                password,
            };
            const { dispatch } = this.props;
            dispatch(loginUser(payload));
        } else {
            this.setState({ submitted: false });
        }
    }

    render() {
        const { loginError, submitted } = this.state;
        return (
            <Outer>
                <form onSubmit={this.handleSubmit}>
                    <div className="mt-5" />
                    {loginError
                    && <div className="alert alert-danger">{loginError}</div>
                    }
                    <label>Login Code</label>
                    <input type="text" className="form-control" onChange={this.handleChange.bind(this, 'code')} value={this.state.fields.code || ''} />
                    <div className="text-danger mt-1 mb-1">{this.state.errors.code}</div>
                    <br />

                    <label>Password</label>
                    <input type="password" className="form-control" onChange={this.handleChange.bind(this, 'password')} value={this.state.fields.password || ''} />
                    <div className="text-danger mt-1 mb-1">{this.state.errors.password}</div>
                    <br />
                    <button
                        className="btn btn-primary btn-block text-uppercase"
                        type="submit"
                        disabled={submitted}
                    >
                        { !submitted ? 'Login to manage my cards' : 'loading...' }
                    </button>
                    <div className="row no-gutters mt-4">
                        <div className="col-md-12">
                            <NavLink to="/reset-password" className="grey">I have forgotten my password</NavLink>
                        </div>
                    </div>
                </form>
            </Outer>
        );
    }
}

function mapStateToProps(state) {
    console.error(state);
    const { auth } = state.authentication;
    const { alert } = state.alert;
    return { auth, alert };
}

export default connect(mapStateToProps)(LoginComponent);

store.js


export const store = createStore(
    rootReducer,
    applyMiddleware(
        thunkMiddleware,
        loggerMiddleware,
    ),
);

authentication.reducer.js

function loginReducer(state = {}, action) {
    switch (action.type) {
    case LOGIN_REQUEST:
        return {
            user_status: LOGIN_REQUEST,
            user_data: action,
        };
    case LOGIN_SUCCESS:
        return {
            user_status: LOGIN_SUCCESS,
            user_data: action,
        };
    case LOGIN_FAILURE:
        return {
            user_status: LOGIN_FAILURE,
            user_data: action,
        };
    default:
        return state;
    }
}

export default loginReducer;

authentication.action.js

function loginUser(payload) {
    function request(user) { return { type: LOGIN_REQUEST, user }; }
    function success(response) { return { type: LOGIN_SUCCESS, response }; }
    function failure(error) { return { type: LOGIN_FAILURE, error }; }

    return (dispatch) => {
        // const request = apiRequest(routes.LOGIN, 'POST', qs.stringify(payload));
        // const ah = loginUser(qs.stringify(payload));
        // console.log(ah);
        dispatch(request(payload));
        const fetch = apiRequest(routes.LOGIN, 'POST', qs.stringify(payload));
        return fetch.then((response) => {
            switch (response.status) {
            case 400:
                dispatch(failure(response));
                dispatch(alertActions.error(response.data.description));
                break;
            case 200:
                if (response.data.code === 0) {
                    localStorage.setItem('qwplx44', JSON.stringify(response.data));
                    dispatch(success(response.data));
                    history.push('/dashboard');
                    break;
                }
                dispatch(failure(response.data.description));
                dispatch(alertActions.error(response.data.description));
                break;
            default:
                return {};
            }
        }).catch((error) => {
            dispatch(failure(error.response.data.message));
            dispatch(alertActions.error(error.response.data.message.toString()));
            // return false;
        });

root-reducer


const rootReducer = combineReducers({
    authentication: loginReducer,
    alert: alertReducer,
});

export default rootReducer;

1 Ответ

1 голос
/ 30 июня 2019

Вы неправильно подключаете свой компонент.Вы пытаетесь уничтожить ключи из состояния, которых там нет.Я не вижу ключ с именем auth в вашем loginReducer.Таким образом, эта строка const { auth } = state.authentication; вернет неопределенное значение.Вот почему при ведении журнала реквизиты auth не определены.

Вместо этого просто вынимайте то, что вы хотите из состояния, и вы можете псевдоним authentication как auth при деструктурировании:)

const mapStateToProps = ({authentication: auth, alert}) => ({
    auth,
    alert
})

Если вы пытаетесь использовать данные, хранящиеся в вашем редукторе, после выполнения запроса, вам следует использовать componentDidUpdate метод жизненного цикла

componentDidUpdate(prevProps) {
  const { auth: prevAuth } = prevProps
  const { auth } = this.props

  if (auth.user_status && ((!prevAuth.user_status && auth.user_status) || prevAuth.user_status !== auth.user_status)) {
    // TODO handle response from api here
    /* ex
      if (auth.user_status === LOGIN_FAILURE) {
        this.setState({loginFailure: auth.user_data})
      }
    */
  }
}

Несколько других вещей.

Вы не являетесьпривязывает handleValidation к классу, но пытается получить доступ к состоянию.Вы звоните this.setState({ submitted: true }); дважды в handleSubmit.Второй излишний и не нужен.Я бы реорганизовал ваш apiRequest для обработки qs.stringify и обработки статуса ошибки / аутентификации / ответа, так что вам не нужно записывать один и тот же материал для каждого вызова API:)

...