import React from 'react'; import PropTypes from 'prop-types'; import { connect } from 'react-redux'; import { RouteComponentProps } from 'react-router-dom'; import FormModel from 'components/ui/form/FormModel'; import ChangeEmail from 'components/profile/changeEmail/ChangeEmail'; import { requestEmailChange, setNewEmail, confirmNewEmail, } from 'services/api/accounts'; import { RootState } from 'reducers'; interface RouteParams { step: 'step1' | 'step2' | 'step3'; code: string; } interface Props extends RouteComponentProps { lang: string; email: string; } class ChangeEmailPage extends React.Component { static contextTypes = { userId: PropTypes.number.isRequired, onSubmit: PropTypes.func.isRequired, goToProfile: PropTypes.func.isRequired, }; componentWillMount() { const { step } = this.props.match.params; if (step && !/^step[123]$/.test(step)) { // wrong param value this.props.history.push('/404'); } } render() { const { step = 'step1', code } = this.props.match.params; return ( ); } onChangeStep = (step: number) => { this.props.history.push(`/profile/change-email/step${++step}`); }; onSubmit = (step: number, form: FormModel) => { return this.context .onSubmit({ form, sendData: () => { const { userId } = this.context; const data = form.serialize(); switch (step) { case 0: return requestEmailChange(userId, data.password).catch( handleErrors(), ); case 1: return setNewEmail(userId, data.email, data.key).catch( handleErrors('/profile/change-email'), ); case 2: return confirmNewEmail(userId, data.key).catch( handleErrors('/profile/change-email'), ); default: throw new Error(`Unsupported step ${step}`); } }, }) .then(() => { step > 1 && this.context.goToProfile(); }); }; } function handleErrors(repeatUrl: string | void) { return resp => { if (resp.errors) { if (resp.errors.key) { resp.errors.key = { type: resp.errors.key, payload: {}, }; if ( ['error.key_not_exists', 'error.key_expire'].includes( resp.errors.key.type, ) && repeatUrl ) { Object.assign(resp.errors.key.payload, { repeatUrl, }); } } } return Promise.reject(resp); }; } export default connect((state: RootState) => ({ email: state.user.email, lang: state.user.lang, }))(ChangeEmailPage);