scratch-www/src/components/join-flow/email-step.jsx

249 lines
10 KiB
React
Raw Normal View History

const bindAll = require('lodash.bindall');
2019-08-01 15:40:46 -04:00
const classNames = require('classnames');
const React = require('react');
const PropTypes = require('prop-types');
import {Formik} from 'formik';
const {injectIntl, intlShape} = require('react-intl');
const FormattedMessage = require('react-intl').FormattedMessage;
2019-08-25 11:31:07 -04:00
const validate = require('../../lib/validate');
const JoinFlowStep = require('./join-flow-step.jsx');
2019-08-01 15:40:46 -04:00
const FormikInput = require('../../components/formik-forms/formik-input.jsx');
2019-08-16 17:13:24 -04:00
const FormikCheckbox = require('../../components/formik-forms/formik-checkbox.jsx');
const InfoButton = require('../info-button/info-button.jsx');
require('./join-flow-steps.scss');
class EmailStep extends React.Component {
constructor (props) {
super(props);
bindAll(this, [
'handleSetEmailRef',
'handleValidSubmit',
'validateEmail',
'validateEmailRemotelyWithCache',
'validateForm',
'setCaptchaRef',
'captchaSolved',
'onCaptchaLoad'
]);
this.state = {
captchaIsLoading: true
};
// simple object to memoize remote requests for email addresses.
// keeps us from submitting multiple requests for same data.
this.emailRemoteCache = {};
}
componentDidMount () {
this.props.sendAnalytics('join-email');
// automatically start with focus on username field
if (this.emailInput) this.emailInput.focus();
2019-08-29 11:33:50 -04:00
2019-10-17 22:22:44 -04:00
if (window.grecaptcha) {
this.onCaptchaLoad();
} else {
// If grecaptcha doesn't exist on window, we havent loaded the captcha js yet. Load it.
// ReCaptcha calls a callback when the grecatpcha object is usable. That callback
// needs to be global so set it on the window.
window.grecaptchaOnLoad = this.onCaptchaLoad;
// Load Google ReCaptcha script.
const script = document.createElement('script');
script.async = true;
script.onerror = this.props.onCaptchaError;
script.src = `https://www.recaptcha.net/recaptcha/api.js?onload=grecaptchaOnLoad&render=explicit&hl=${window._locale}`;
document.body.appendChild(script);
}
}
componentWillUnmount () {
window.grecaptchaOnLoad = null;
}
2019-08-29 11:33:50 -04:00
handleSetEmailRef (emailInputRef) {
this.emailInput = emailInputRef;
}
onCaptchaLoad () {
this.setState({captchaIsLoading: false});
this.grecaptcha = window.grecaptcha;
if (!this.grecaptcha) {
// According to the reCaptcha documentation, this callback shouldn't get
// called unless window.grecaptcha exists. This is just here to be extra defensive.
this.props.onCaptchaError();
return;
}
this.widgetId = this.grecaptcha.render(this.captchaRef,
{
callback: this.captchaSolved,
sitekey: process.env.RECAPTCHA_SITE_KEY
},
true);
}
// simple function to memoize remote requests for usernames
validateEmailRemotelyWithCache (email) {
if (this.emailRemoteCache.hasOwnProperty(email)) {
return Promise.resolve(this.emailRemoteCache[email]);
}
// email is not in our cache
return validate.validateEmailRemotely(email).then(
remoteResult => {
// cache result, if it successfully heard back from server
if (remoteResult.requestSucceeded) {
this.emailRemoteCache[email] = remoteResult;
}
return remoteResult;
}
);
}
2019-08-16 16:22:33 -04:00
validateEmail (email) {
if (!email) return this.props.intl.formatMessage({id: 'general.required'});
2019-08-25 11:31:07 -04:00
const localResult = validate.validateEmailLocally(email);
if (!localResult.valid) return this.props.intl.formatMessage({id: localResult.errMsgId});
return this.validateEmailRemotelyWithCache(email).then(
2019-08-25 11:31:07 -04:00
remoteResult => {
if (remoteResult.valid === true) {
return null;
}
return this.props.intl.formatMessage({id: remoteResult.errMsgId});
}
);
2019-08-01 15:40:46 -04:00
}
validateForm () {
return {};
}
handleValidSubmit (formData, formikBag) {
this.formData = formData;
this.formikBag = formikBag;
// Change set submitting to false so that if the user clicks out of
// the captcha, the button is clickable again (instead of a disabled button with a spinner).
this.formikBag.setSubmitting(false);
this.grecaptcha.execute(this.widgetId);
}
captchaSolved (token) {
2019-08-29 11:33:50 -04:00
// Now thatcaptcha is done, we can tell Formik we're submitting.
this.formikBag.setSubmitting(true);
this.formData['g-recaptcha-response'] = token;
this.props.onNextStep(this.formData);
}
setCaptchaRef (ref) {
this.captchaRef = ref;
}
render () {
return (
<Formik
initialValues={{
email: '',
subscribe: false
}}
validate={this.validateForm}
validateOnBlur={false}
validateOnChange={false}
onSubmit={this.handleValidSubmit}
>
{props => {
const {
2019-08-01 15:40:46 -04:00
errors,
handleSubmit,
2019-08-01 15:40:46 -04:00
isSubmitting,
2019-08-16 16:22:33 -04:00
setFieldError,
setFieldTouched,
setFieldValue,
2019-08-01 15:40:46 -04:00
validateField
} = props;
return (
<JoinFlowStep
footerContent={(
<FormattedMessage
id="registration.acceptTermsOfUse"
values={{
touLink: (
<a
className="join-flow-link"
href="/terms_of_use"
target="_blank"
>
<FormattedMessage id="general.termsOfUse" />
</a>
)
}}
/>
)}
headerImgClass="email-step-image"
2019-08-16 15:05:15 -04:00
headerImgSrc="/images/join-flow/email-header.png"
2019-08-17 00:52:52 -04:00
innerClassName="join-flow-inner-email-step"
nextButton={this.props.intl.formatMessage({id: 'registration.createAccount'})}
title={this.props.intl.formatMessage({id: 'registration.emailStepTitle'})}
titleClassName="join-flow-email-title"
waiting={this.props.waiting || isSubmitting || this.state.captchaIsLoading}
onSubmit={handleSubmit}
2019-08-01 15:40:46 -04:00
>
<FormikInput
autoCapitalize="off"
autoComplete="off"
autoCorrect="off"
2019-08-01 15:40:46 -04:00
className={classNames(
'join-flow-input',
'join-flow-input-tall',
{fail: errors.email}
)}
error={errors.email}
id="email"
name="email"
2019-08-08 11:19:18 -04:00
placeholder={this.props.intl.formatMessage({id: 'general.emailAddress'})}
type="email"
2019-08-16 16:22:33 -04:00
validate={this.validateEmail}
2019-08-01 15:40:46 -04:00
validationClassName="validation-full-width-input"
2019-08-16 16:22:33 -04:00
/* eslint-disable react/jsx-no-bind */
onBlur={() => validateField('email')}
onChange={e => {
setFieldValue('email', e.target.value.substring(0, 254));
setFieldTouched('email');
setFieldError('email', null);
}}
2019-08-16 16:22:33 -04:00
/* eslint-enable react/jsx-no-bind */
onSetRef={this.handleSetEmailRef}
2019-08-01 15:40:46 -04:00
/>
<div className="join-flow-privacy-message join-flow-email-privacy">
<FormattedMessage id="registration.private" />
<InfoButton
message={this.props.intl.formatMessage({id: 'registration.emailStepInfo'})}
/>
</div>
2019-08-16 17:13:24 -04:00
<div className="join-flow-email-checkbox-row">
<FormikCheckbox
id="subscribeCheckbox"
label={this.props.intl.formatMessage({id: 'registration.receiveEmails'})}
name="subscribe"
/>
</div>
<div
className="g-recaptcha"
data-badge="bottomright"
data-sitekey={process.env.RECAPTCHA_SITE_KEY}
data-size="invisible"
ref={this.setCaptchaRef}
/>
2019-08-01 15:40:46 -04:00
</JoinFlowStep>
);
}}
</Formik>
);
}
}
EmailStep.propTypes = {
intl: intlShape,
onCaptchaError: PropTypes.func,
onNextStep: PropTypes.func,
<<<<<<< HEAD
=======
onRegistrationError: PropTypes.func,
<<<<<<< HEAD
sendAnalytics: PropTypes.func,
>>>>>>> Add analytics logging to join flow. Adding page views for each step in the flow.
=======
sendAnalytics: PropTypes.func.isRequired,
>>>>>>> Set sendAnalytics to be required and send the right props to the error step. Also add a test for the error step.
waiting: PropTypes.bool
};
module.exports = injectIntl(EmailStep);