98 lines
3.2 KiB
TypeScript
Raw Normal View History

2019-12-07 13:28:52 +02:00
import React from 'react';
import { connect } from 'react-redux';
2019-12-10 09:47:32 +02:00
import { RouteComponentProps, Redirect } from 'react-router-dom';
import FormModel from 'app/components/ui/form/FormModel';
2020-05-24 02:08:24 +03:00
import ChangeEmail, { ChangeEmailStep } from 'app/components/profile/changeEmail/ChangeEmail';
import { requestEmailChange, setNewEmail, confirmNewEmail } from 'app/services/api/accounts';
import { RootState } from 'app/reducers';
2019-12-12 09:26:23 +02:00
import Context from 'app/components/profile/Context';
2019-12-07 13:28:52 +02:00
interface RouteParams {
2020-05-24 02:08:24 +03:00
step: 'step1' | 'step2' | 'step3';
code: string;
2019-12-07 13:28:52 +02:00
}
2019-12-07 13:28:52 +02:00
interface Props extends RouteComponentProps<RouteParams> {
2020-05-24 02:08:24 +03:00
email: string;
2019-12-07 13:28:52 +02:00
}
2019-12-07 13:28:52 +02:00
class ChangeEmailPage extends React.Component<Props> {
2020-05-24 02:08:24 +03:00
static contextType = Context;
/* TODO: use declare */ context: React.ContextType<typeof Context>;
2020-05-24 02:08:24 +03:00
render() {
const { step = 'step1', code } = this.props.match.params;
2020-05-24 02:08:24 +03:00
if (step && !/^step[123]$/.test(step)) {
// wrong param value
return <Redirect to="/404" />;
}
2020-05-24 02:08:24 +03:00
return (
<ChangeEmail
onSubmit={this.onSubmit}
email={this.props.email}
step={(Number(step.slice(-1)) - 1) as ChangeEmailStep}
onChangeStep={this.onChangeStep}
code={code}
/>
);
}
2020-05-24 02:08:24 +03:00
onChangeStep = (step: number) => {
this.props.history.push(`/profile/change-email/step${++step}`);
};
2020-05-24 02:08:24 +03:00
onSubmit = (step: number, form: FormModel): Promise<void> => {
return this.context
.onSubmit({
form,
sendData: () => {
const { userId } = this.context;
const data = form.serialize();
2020-05-24 02:08:24 +03:00
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();
});
};
}
2020-05-24 02:08:24 +03:00
function handleErrors(repeatUrl?: string): <T extends { errors: Record<string, any> }>(resp: T) => Promise<T> {
return (resp) => {
if (resp.errors) {
if (resp.errors.key) {
resp.errors.key = {
type: resp.errors.key,
payload: {},
};
2020-05-24 02:08:24 +03:00
if (['error.key_not_exists', 'error.key_expire'].includes(resp.errors.key.type) && repeatUrl) {
Object.assign(resp.errors.key.payload, {
repeatUrl,
});
}
}
}
2020-05-24 02:08:24 +03:00
return Promise.reject(resp);
};
}
2019-12-07 13:28:52 +02:00
export default connect((state: RootState) => ({
2020-05-24 02:08:24 +03:00
email: state.user.email,
}))(ChangeEmailPage);