2016-05-02 23:02:03 +05:30
|
|
|
import React, { Component, PropTypes } from 'react';
|
|
|
|
|
|
|
|
import { FormattedMessage as Message } from 'react-intl';
|
|
|
|
import { Link } from 'react-router';
|
|
|
|
import classNames from 'classnames';
|
|
|
|
import Helmet from 'react-helmet';
|
|
|
|
import { Motion, spring } from 'react-motion';
|
|
|
|
|
|
|
|
import { Input, Button, Form, FormModel } from 'components/ui/form';
|
|
|
|
import styles from 'components/profile/profileForm.scss';
|
|
|
|
import helpLinks from 'components/auth/helpLinks.scss';
|
|
|
|
import MeasureHeight from 'components/MeasureHeight';
|
|
|
|
|
|
|
|
import changeEmail from './changeEmail.scss';
|
|
|
|
import messages from './ChangeEmail.messages';
|
|
|
|
|
|
|
|
const STEPS_TOTAL = 3;
|
|
|
|
|
|
|
|
// TODO: disable code field, if the code was passed through url
|
|
|
|
|
|
|
|
export default class ChangeEmail extends Component {
|
|
|
|
static displayName = 'ChangeEmail';
|
|
|
|
|
|
|
|
static propTypes = {
|
2016-05-20 10:44:14 +05:30
|
|
|
onChangeStep: PropTypes.func,
|
2016-05-02 23:02:03 +05:30
|
|
|
email: PropTypes.string.isRequired,
|
|
|
|
form: PropTypes.instanceOf(FormModel),
|
2016-05-20 10:44:14 +05:30
|
|
|
onSubmit: PropTypes.func.isRequired,
|
|
|
|
step: PropTypes.oneOf([0, 1, 2]),
|
|
|
|
code: PropTypes.string
|
2016-05-02 23:02:03 +05:30
|
|
|
};
|
|
|
|
|
|
|
|
static get defaultProps() {
|
|
|
|
return {
|
2016-05-20 10:44:14 +05:30
|
|
|
form: new FormModel(),
|
|
|
|
onChangeStep() {},
|
|
|
|
step: 0
|
2016-05-02 23:02:03 +05:30
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
state = {
|
2016-05-20 10:44:14 +05:30
|
|
|
activeStep: this.props.step,
|
|
|
|
code: this.props.code || ''
|
2016-05-02 23:02:03 +05:30
|
|
|
};
|
|
|
|
|
2016-05-20 10:44:14 +05:30
|
|
|
componentWillReceiveProps(nextProps) {
|
|
|
|
this.setState({
|
|
|
|
activeStep: nextProps.step || this.state.activeStep,
|
|
|
|
code: nextProps.code || ''
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2016-05-02 23:02:03 +05:30
|
|
|
render() {
|
|
|
|
const {form} = this.props;
|
|
|
|
const {activeStep} = this.state;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Form onSubmit={this.onFormSubmit}
|
|
|
|
form={form}
|
|
|
|
>
|
|
|
|
<div className={styles.contentWithBackButton}>
|
|
|
|
<Link className={styles.backButton} to="/" />
|
|
|
|
|
|
|
|
<div className={styles.form}>
|
|
|
|
<div className={styles.formBody}>
|
|
|
|
<Message {...messages.changeEmailTitle}>
|
|
|
|
{(pageTitle) => (
|
|
|
|
<h3 className={styles.violetTitle}>
|
|
|
|
<Helmet title={pageTitle} />
|
|
|
|
{pageTitle}
|
|
|
|
</h3>
|
|
|
|
)}
|
|
|
|
</Message>
|
|
|
|
|
|
|
|
<div className={styles.formRow}>
|
|
|
|
<p className={styles.description}>
|
|
|
|
<Message {...messages.changeEmailDescription} />
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={changeEmail.steps}>
|
|
|
|
{(new Array(STEPS_TOTAL)).fill(0).map((_, step) => (
|
|
|
|
<div className={classNames(changeEmail.step, {
|
|
|
|
[changeEmail.activeStep]: step <= activeStep
|
|
|
|
})} key={step} />
|
|
|
|
))}
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={styles.form}>
|
|
|
|
{this.renderStepForms()}
|
|
|
|
|
|
|
|
<Button
|
|
|
|
color="violet"
|
|
|
|
block
|
|
|
|
label={this.isLastStep() ? messages.changeEmailButton : messages.sendEmailButton}
|
|
|
|
onClick={this.onSwitchStep}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={helpLinks.helpLinks}>
|
|
|
|
{this.isLastStep() ? null : (
|
|
|
|
<a href="#" onClick={this.onSwitchStep}>
|
|
|
|
<Message {...messages.alreadyReceivedCode} />
|
|
|
|
</a>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</Form>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
renderStepForms() {
|
|
|
|
const {form, email} = this.props;
|
2016-05-20 10:44:14 +05:30
|
|
|
const {activeStep, code} = this.state;
|
|
|
|
const isCodeEntered = !!this.props.code;
|
2016-05-02 23:02:03 +05:30
|
|
|
|
|
|
|
const activeStepHeight = this.state[`step${activeStep}Height`] || 0;
|
|
|
|
|
|
|
|
// a hack to disable height animation on first render
|
|
|
|
const isHeightMeasured = this.isHeightMeasured;
|
2016-05-02 23:22:37 +05:30
|
|
|
this.isHeightMeasured = isHeightMeasured || activeStepHeight > 0;
|
2016-05-02 23:02:03 +05:30
|
|
|
|
|
|
|
return (
|
|
|
|
<Motion
|
|
|
|
style={{
|
|
|
|
transform: spring(activeStep * 100, {stiffness: 500, damping: 50, precision: 0.5}),
|
|
|
|
height: isHeightMeasured ? spring(activeStepHeight, {stiffness: 500, damping: 20, precision: 0.5}) : activeStepHeight
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
{(interpolatingStyle) => (
|
|
|
|
<div style={{
|
|
|
|
overflow: 'hidden',
|
|
|
|
height: `${interpolatingStyle.height}px`
|
|
|
|
}}>
|
|
|
|
<div className={changeEmail.stepForms} style={{
|
|
|
|
WebkitTransform: `translateX(-${interpolatingStyle.transform}%)`,
|
|
|
|
transform: `translateX(-${interpolatingStyle.transform}%)`
|
|
|
|
}}>
|
|
|
|
<MeasureHeight className={changeEmail.stepForm} onMeasure={this.onStepMeasure(0)}>
|
|
|
|
<div className={styles.formBody}>
|
|
|
|
<div className={styles.formRow}>
|
|
|
|
<p className={styles.description}>
|
|
|
|
<Message {...messages.currentAccountEmail} />
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={styles.formRow}>
|
|
|
|
<h2 className={changeEmail.currentAccountEmail}>
|
|
|
|
{email}
|
|
|
|
</h2>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={styles.formRow}>
|
|
|
|
<p className={styles.description}>
|
|
|
|
<Message {...messages.pressButtonToStart} />
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</MeasureHeight>
|
|
|
|
|
|
|
|
<MeasureHeight className={changeEmail.stepForm} onMeasure={this.onStepMeasure(1)}>
|
|
|
|
<div className={styles.formBody}>
|
|
|
|
<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('initializationCode')}
|
|
|
|
required
|
2016-05-20 10:44:14 +05:30
|
|
|
disabled={isCodeEntered}
|
|
|
|
value={code}
|
|
|
|
onChange={this.onCodeInput}
|
2016-05-02 23:02:03 +05:30
|
|
|
skin="light"
|
2016-05-02 23:06:05 +05:30
|
|
|
color="violet"
|
2016-05-02 23:02:03 +05:30
|
|
|
placeholder={messages.codePlaceholder}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={styles.formRow}>
|
|
|
|
<p className={styles.description}>
|
|
|
|
<Message {...messages.enterNewEmail} />
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={styles.formRow}>
|
|
|
|
<Input {...form.bindField('newEmail')}
|
|
|
|
required
|
|
|
|
skin="light"
|
2016-05-02 23:06:05 +05:30
|
|
|
color="violet"
|
2016-05-02 23:02:03 +05:30
|
|
|
placeholder={messages.newEmailPlaceholder}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</MeasureHeight>
|
|
|
|
|
|
|
|
<MeasureHeight className={changeEmail.stepForm} onMeasure={this.onStepMeasure(2)}>
|
|
|
|
<div className={styles.formBody}>
|
|
|
|
<div className={styles.formRow}>
|
|
|
|
<p className={styles.description}>
|
|
|
|
<Message {...messages.enterFinalizationCode} values={{
|
|
|
|
email: (<b>{form.value('newEmail')}</b>)
|
|
|
|
}} />
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={styles.formRow}>
|
|
|
|
<Input {...form.bindField('finalizationCode')}
|
|
|
|
required
|
2016-05-20 10:44:14 +05:30
|
|
|
disabled={isCodeEntered}
|
|
|
|
value={code}
|
|
|
|
onChange={this.onCodeInput}
|
2016-05-02 23:02:03 +05:30
|
|
|
skin="light"
|
2016-05-02 23:06:05 +05:30
|
|
|
color="violet"
|
2016-05-02 23:02:03 +05:30
|
|
|
placeholder={messages.codePlaceholder}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</MeasureHeight>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</Motion>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
onStepMeasure(step) {
|
|
|
|
return (height) => this.setState({
|
|
|
|
[`step${step}Height`]: height
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
onSwitchStep = (event) => {
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
|
|
const {activeStep} = this.state;
|
|
|
|
const nextStep = activeStep + 1;
|
|
|
|
|
|
|
|
if (nextStep < STEPS_TOTAL) {
|
|
|
|
this.setState({
|
|
|
|
activeStep: nextStep
|
|
|
|
});
|
2016-05-20 10:44:14 +05:30
|
|
|
|
|
|
|
this.props.onChangeStep(nextStep);
|
2016-05-02 23:02:03 +05:30
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-05-20 10:44:14 +05:30
|
|
|
onCodeInput = (event) => {
|
|
|
|
const {value} = event.target;
|
|
|
|
|
|
|
|
this.setState({
|
|
|
|
code: this.props.code || value
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2016-05-02 23:02:03 +05:30
|
|
|
isLastStep() {
|
|
|
|
return this.state.activeStep + 1 === STEPS_TOTAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
onFormSubmit = () => {
|
|
|
|
this.props.onSubmit(this.props.form);
|
|
|
|
};
|
|
|
|
}
|