2020-01-17 23:37:52 +03:00
|
|
|
import React, { ReactNode } from 'react';
|
2016-05-02 20:32:03 +03:00
|
|
|
import { FormattedMessage as Message } from 'react-intl';
|
2019-12-30 09:29:39 +02:00
|
|
|
import { Helmet } from 'react-helmet-async';
|
2019-12-07 21:02:00 +02:00
|
|
|
import { SlideMotion } from 'app/components/ui/motion';
|
|
|
|
import { ScrollIntoView } from 'app/components/ui/scroll';
|
|
|
|
import {
|
|
|
|
Input,
|
|
|
|
Button,
|
|
|
|
Form,
|
|
|
|
FormModel,
|
|
|
|
FormError,
|
|
|
|
} from 'app/components/ui/form';
|
|
|
|
import { BackButton } from 'app/components/profile/ProfileForm';
|
|
|
|
import styles from 'app/components/profile/profileForm.scss';
|
|
|
|
import helpLinks from 'app/components/auth/helpLinks.scss';
|
|
|
|
import Stepper from 'app/components/ui/stepper';
|
2016-05-02 20:32:03 +03:00
|
|
|
|
|
|
|
import changeEmail from './changeEmail.scss';
|
2016-05-22 10:53:40 +03:00
|
|
|
import messages from './ChangeEmail.intl.json';
|
2016-05-02 20:32:03 +03:00
|
|
|
|
|
|
|
const STEPS_TOTAL = 3;
|
|
|
|
|
2019-12-10 09:47:32 +02:00
|
|
|
export type ChangeEmailStep = 0 | 1 | 2;
|
|
|
|
|
|
|
|
interface Props {
|
|
|
|
onChangeStep: (step: ChangeEmailStep) => void;
|
|
|
|
lang: string;
|
|
|
|
email: string;
|
2020-01-17 23:37:52 +03:00
|
|
|
stepForms: Array<FormModel>;
|
2019-12-10 09:47:32 +02:00
|
|
|
onSubmit: (step: ChangeEmailStep, form: FormModel) => Promise<void>;
|
|
|
|
step: ChangeEmailStep;
|
|
|
|
code?: string;
|
|
|
|
}
|
2019-12-07 13:28:52 +02:00
|
|
|
|
2020-01-17 23:37:52 +03:00
|
|
|
interface State {
|
2019-12-10 09:47:32 +02:00
|
|
|
newEmail: string | null;
|
|
|
|
activeStep: ChangeEmailStep;
|
|
|
|
code: string;
|
|
|
|
}
|
2019-11-27 11:03:32 +02:00
|
|
|
|
2020-01-17 23:37:52 +03:00
|
|
|
interface FormStepParams {
|
|
|
|
form: FormModel;
|
|
|
|
isActiveStep: boolean;
|
|
|
|
isCodeSpecified: boolean;
|
|
|
|
email: string;
|
|
|
|
code?: string;
|
|
|
|
}
|
|
|
|
|
2019-12-10 09:47:32 +02:00
|
|
|
export default class ChangeEmail extends React.Component<Props, State> {
|
|
|
|
static get defaultProps(): Partial<Props> {
|
2019-11-27 11:03:32 +02:00
|
|
|
return {
|
|
|
|
stepForms: [new FormModel(), new FormModel(), new FormModel()],
|
|
|
|
onChangeStep() {},
|
|
|
|
step: 0,
|
2016-05-02 20:32:03 +03:00
|
|
|
};
|
2019-11-27 11:03:32 +02:00
|
|
|
}
|
|
|
|
|
2019-12-10 09:47:32 +02:00
|
|
|
state: State = {
|
|
|
|
newEmail: null,
|
2019-11-27 11:03:32 +02:00
|
|
|
activeStep: this.props.step,
|
|
|
|
code: this.props.code || '',
|
|
|
|
};
|
|
|
|
|
2019-12-10 09:47:32 +02:00
|
|
|
static getDerivedStateFromProps(props: Props, state: State) {
|
|
|
|
return {
|
2019-11-27 11:03:32 +02:00
|
|
|
activeStep:
|
2019-12-10 09:47:32 +02:00
|
|
|
typeof props.step === 'number' ? props.step : state.activeStep,
|
2019-12-29 15:26:07 +02:00
|
|
|
code: props.code || state.code,
|
2019-12-10 09:47:32 +02:00
|
|
|
};
|
2019-11-27 11:03:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
const { activeStep } = this.state;
|
|
|
|
const form = this.props.stepForms[activeStep];
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Form
|
|
|
|
form={form}
|
|
|
|
onSubmit={this.onFormSubmit}
|
|
|
|
onInvalid={() => this.forceUpdate()}
|
|
|
|
>
|
2019-12-29 15:26:07 +02:00
|
|
|
<div
|
|
|
|
className={styles.contentWithBackButton}
|
|
|
|
data-testid="change-email"
|
|
|
|
>
|
2019-11-27 11:03:32 +02:00
|
|
|
<BackButton />
|
|
|
|
|
|
|
|
<div className={styles.form}>
|
|
|
|
<div className={styles.formBody}>
|
|
|
|
<Message {...messages.changeEmailTitle}>
|
|
|
|
{pageTitle => (
|
|
|
|
<h3 className={styles.violetTitle}>
|
2019-12-30 10:47:29 +02:00
|
|
|
<Helmet title={pageTitle as string} />
|
2019-11-27 11:03:32 +02:00
|
|
|
{pageTitle}
|
|
|
|
</h3>
|
|
|
|
)}
|
|
|
|
</Message>
|
|
|
|
|
|
|
|
<div className={styles.formRow}>
|
|
|
|
<p className={styles.description}>
|
|
|
|
<Message {...messages.changeEmailDescription} />
|
|
|
|
</p>
|
|
|
|
</div>
|
2016-05-22 10:53:40 +03:00
|
|
|
</div>
|
2019-11-27 11:03:32 +02:00
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={styles.stepper}>
|
|
|
|
<Stepper
|
|
|
|
color="violet"
|
|
|
|
totalSteps={STEPS_TOTAL}
|
|
|
|
activeStep={activeStep}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={styles.form}>
|
|
|
|
{activeStep > 0 ? <ScrollIntoView /> : null}
|
|
|
|
|
|
|
|
{this.renderStepForms()}
|
|
|
|
|
|
|
|
<Button
|
|
|
|
color="violet"
|
|
|
|
type="submit"
|
|
|
|
block
|
|
|
|
label={
|
|
|
|
this.isLastStep()
|
|
|
|
? messages.changeEmailButton
|
|
|
|
: messages.sendEmailButton
|
|
|
|
}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={helpLinks.helpLinks}>
|
|
|
|
{this.isLastStep() ? null : (
|
|
|
|
<a href="#" onClick={this.onSwitchStep}>
|
|
|
|
<Message {...messages.alreadyReceivedCode} />
|
|
|
|
</a>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</Form>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
renderStepForms() {
|
|
|
|
const { email } = this.props;
|
|
|
|
const { activeStep, code } = this.state;
|
|
|
|
const isCodeSpecified = !!this.props.code;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<SlideMotion activeStep={activeStep}>
|
|
|
|
{new Array(STEPS_TOTAL).fill(0).map((_, step) => {
|
2020-01-17 23:37:52 +03:00
|
|
|
const formParams: FormStepParams = {
|
|
|
|
form: this.props.stepForms[step],
|
|
|
|
isActiveStep: step === activeStep,
|
|
|
|
isCodeSpecified,
|
2019-11-27 11:03:32 +02:00
|
|
|
email,
|
|
|
|
code,
|
2020-01-17 23:37:52 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
switch (step) {
|
|
|
|
case 0:
|
|
|
|
return this.renderStep0(formParams);
|
|
|
|
case 1:
|
|
|
|
return this.renderStep1(formParams);
|
|
|
|
case 2:
|
|
|
|
return this.renderStep2(formParams);
|
|
|
|
}
|
2019-11-27 11:03:32 +02:00
|
|
|
})}
|
|
|
|
</SlideMotion>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-01-17 23:37:52 +03:00
|
|
|
renderStep0({ email, form }: FormStepParams): ReactNode {
|
2019-11-27 11:03:32 +02:00
|
|
|
return (
|
2019-12-29 15:26:07 +02:00
|
|
|
<div key="step0" data-testid="step1" className={styles.formBody}>
|
2019-11-27 11:03:32 +02:00
|
|
|
<div className={styles.formRow}>
|
|
|
|
<p className={styles.description}>
|
|
|
|
<Message {...messages.currentAccountEmail} />
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={styles.formRow}>
|
|
|
|
<h2 className={changeEmail.currentAccountEmail}>{email}</h2>
|
|
|
|
|
|
|
|
<FormError error={form.getError('email')} />
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={styles.formRow}>
|
|
|
|
<p className={styles.description}>
|
|
|
|
<Message {...messages.pressButtonToStart} />
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-01-17 23:37:52 +03:00
|
|
|
renderStep1({
|
|
|
|
email,
|
|
|
|
form,
|
|
|
|
code,
|
|
|
|
isCodeSpecified,
|
|
|
|
isActiveStep,
|
|
|
|
}: FormStepParams): ReactNode {
|
2019-11-27 11:03:32 +02:00
|
|
|
return (
|
2019-12-29 15:26:07 +02:00
|
|
|
<div key="step1" data-testid="step2" className={styles.formBody}>
|
2019-11-27 11:03:32 +02:00
|
|
|
<div className={styles.formRow}>
|
|
|
|
<p className={styles.description}>
|
|
|
|
<Message
|
|
|
|
{...messages.enterInitializationCode}
|
|
|
|
values={{
|
|
|
|
email: <b>{email}</b>,
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={styles.formRow}>
|
|
|
|
<Input
|
|
|
|
{...form.bindField('key')}
|
|
|
|
required={isActiveStep}
|
|
|
|
disabled={isCodeSpecified}
|
|
|
|
value={code}
|
|
|
|
onChange={this.onCodeInput}
|
|
|
|
autoComplete="off"
|
|
|
|
skin="light"
|
|
|
|
color="violet"
|
|
|
|
placeholder={messages.codePlaceholder}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={styles.formRow}>
|
|
|
|
<p className={styles.description}>
|
|
|
|
<Message {...messages.enterNewEmail} />
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={styles.formRow}>
|
|
|
|
<Input
|
|
|
|
{...form.bindField('email')}
|
|
|
|
required={isActiveStep}
|
|
|
|
skin="light"
|
|
|
|
color="violet"
|
|
|
|
placeholder={messages.newEmailPlaceholder}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-01-17 23:37:52 +03:00
|
|
|
renderStep2({
|
|
|
|
form,
|
|
|
|
code,
|
|
|
|
isCodeSpecified,
|
|
|
|
isActiveStep,
|
|
|
|
}: FormStepParams): ReactNode {
|
2019-11-27 11:03:32 +02:00
|
|
|
const { newEmail } = this.state;
|
|
|
|
|
|
|
|
return (
|
2019-12-29 15:26:07 +02:00
|
|
|
<div key="step2" data-testid="step3" className={styles.formBody}>
|
2019-11-27 11:03:32 +02:00
|
|
|
<div className={styles.formRow}>
|
|
|
|
<p className={styles.description}>
|
|
|
|
{newEmail ? (
|
|
|
|
<span>
|
|
|
|
<Message
|
|
|
|
{...messages.finalizationCodeWasSentToEmail}
|
|
|
|
values={{
|
|
|
|
email: <b>{newEmail}</b>,
|
|
|
|
}}
|
|
|
|
/>{' '}
|
|
|
|
</span>
|
|
|
|
) : null}
|
|
|
|
<Message {...messages.enterFinalizationCode} />
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={styles.formRow}>
|
|
|
|
<Input
|
|
|
|
{...form.bindField('key')}
|
|
|
|
required={isActiveStep}
|
|
|
|
disabled={isCodeSpecified}
|
|
|
|
value={code}
|
|
|
|
onChange={this.onCodeInput}
|
|
|
|
autoComplete="off"
|
|
|
|
skin="light"
|
|
|
|
color="violet"
|
|
|
|
placeholder={messages.codePlaceholder}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
nextStep() {
|
|
|
|
const { activeStep } = this.state;
|
|
|
|
const nextStep = activeStep + 1;
|
|
|
|
let newEmail = null;
|
|
|
|
|
|
|
|
if (activeStep === 1) {
|
|
|
|
newEmail = this.props.stepForms[1].value('email');
|
2016-05-22 10:53:40 +03:00
|
|
|
}
|
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
if (nextStep < STEPS_TOTAL) {
|
|
|
|
this.setState({
|
2019-12-10 09:47:32 +02:00
|
|
|
activeStep: nextStep as ChangeEmailStep,
|
2019-11-27 11:03:32 +02:00
|
|
|
newEmail,
|
|
|
|
});
|
2016-05-22 10:53:40 +03:00
|
|
|
|
2019-12-10 09:47:32 +02:00
|
|
|
this.props.onChangeStep(nextStep as ChangeEmailStep);
|
2016-05-22 10:53:40 +03:00
|
|
|
}
|
2019-11-27 11:03:32 +02:00
|
|
|
}
|
2016-05-22 10:53:40 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
isLastStep() {
|
|
|
|
return this.state.activeStep + 1 === STEPS_TOTAL;
|
|
|
|
}
|
2016-05-22 10:53:40 +03:00
|
|
|
|
2019-12-29 15:26:07 +02:00
|
|
|
onSwitchStep = (event: React.MouseEvent) => {
|
2019-11-27 11:03:32 +02:00
|
|
|
event.preventDefault();
|
2016-05-22 10:53:40 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
this.nextStep();
|
|
|
|
};
|
2016-05-22 10:53:40 +03:00
|
|
|
|
2019-12-29 15:26:07 +02:00
|
|
|
onCodeInput = (event: React.ChangeEvent<HTMLInputElement>) => {
|
2019-11-27 11:03:32 +02:00
|
|
|
const { value } = event.target;
|
2016-05-02 20:32:03 +03:00
|
|
|
|
2019-12-29 15:26:07 +02:00
|
|
|
if (this.props.code) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
this.setState({
|
2019-12-29 15:26:07 +02:00
|
|
|
code: value,
|
2019-11-27 11:03:32 +02:00
|
|
|
});
|
|
|
|
};
|
2016-05-20 08:14:14 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
onFormSubmit = () => {
|
|
|
|
const { activeStep } = this.state;
|
|
|
|
const form = this.props.stepForms[activeStep];
|
|
|
|
const promise = this.props.onSubmit(activeStep, form);
|
2016-05-20 08:14:14 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
if (!promise || !promise.then) {
|
|
|
|
throw new Error('Expecting promise from onSubmit');
|
|
|
|
}
|
2016-05-22 10:53:40 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
promise.then(
|
2019-12-29 15:26:07 +02:00
|
|
|
() => {
|
|
|
|
this.nextStep();
|
|
|
|
this.setState({
|
|
|
|
code: '',
|
|
|
|
});
|
|
|
|
},
|
2019-11-27 11:03:32 +02:00
|
|
|
resp => {
|
|
|
|
if (resp.errors) {
|
|
|
|
form.setErrors(resp.errors);
|
|
|
|
this.forceUpdate();
|
|
|
|
} else {
|
|
|
|
return Promise.reject(resp);
|
2016-05-22 10:53:40 +03:00
|
|
|
}
|
2019-11-27 11:03:32 +02:00
|
|
|
},
|
|
|
|
);
|
|
|
|
};
|
2016-05-02 20:32:03 +03:00
|
|
|
}
|