const bindAll = require('lodash.bindall'); 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; const validate = require('../../lib/validate'); const JoinFlowStep = require('./join-flow-step.jsx'); const FormikInput = require('../../components/formik-forms/formik-input.jsx'); 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', 'validateForm', 'setCaptchaRef', 'captchaSolved', 'onCaptchaLoad', 'onCaptchaError' ]); this.state = { captchaIsLoading: true }; } componentDidMount () { // automatically start with focus on username field if (this.emailInput) this.emailInput.focus(); // If grecaptcha doesn't exist on window, we havent loaded the captcha js yet. Load it. if (!window.grecaptcha) { // 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.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; } handleSetEmailRef (emailInputRef) { this.emailInput = emailInputRef; } onCaptchaError () { // TODO send user to error step once we have one. } 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. // TODO: Put up the error screen when we have one. } // TODO: Add in error callback for render once we have an error screen. this.widgetId = this.grecaptcha.render(this.captchaRef, { callback: this.captchaSolved, sitekey: process.env.RECAPTCHA_SITE_KEY }, true); } validateEmail (email) { if (!email) return this.props.intl.formatMessage({id: 'general.required'}); const localResult = validate.validateEmailLocally(email); if (!localResult.valid) return this.props.intl.formatMessage({id: localResult.errMsgId}); return validate.validateEmailRemotely(email).then( remoteResult => { if (remoteResult.valid === true) { return null; } return this.props.intl.formatMessage({id: remoteResult.errMsgId}); } ); } 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) { // 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 ( {props => { const { errors, handleSubmit, isSubmitting, setFieldError, setFieldTouched, setFieldValue, validateField } = props; return ( ) }} /> )} headerImgSrc="/images/join-flow/email-header.png" innerClassName="join-flow-inner-email-step" nextButton={this.props.intl.formatMessage({id: 'registration.createAccount'})} title={this.props.intl.formatMessage({id: 'registration.emailStepTitle'})} waiting={this.props.waiting || isSubmitting || this.state.captchaIsLoading} onSubmit={handleSubmit} > validateField('email')} onChange={e => { setFieldValue('email', e.target.value); setFieldTouched('email'); setFieldError('email', null); }} /* eslint-enable react/jsx-no-bind */ onSetRef={this.handleSetEmailRef} />
); }} ); } } EmailStep.propTypes = { intl: intlShape, onNextStep: PropTypes.func, waiting: PropTypes.bool }; module.exports = injectIntl(EmailStep);