2019-06-24 13:35:01 -04:00
|
|
|
const bindAll = require('lodash.bindall');
|
2019-08-19 20:45:16 -04:00
|
|
|
const connect = require('react-redux').connect;
|
2019-06-24 13:35:01 -04:00
|
|
|
const defaults = require('lodash.defaultsdeep');
|
2019-06-24 11:31:16 -04:00
|
|
|
const PropTypes = require('prop-types');
|
|
|
|
const React = require('react');
|
|
|
|
|
2019-08-19 20:45:16 -04:00
|
|
|
const api = require('../../lib/api');
|
2019-06-24 11:31:16 -04:00
|
|
|
const injectIntl = require('../../lib/intl.jsx').injectIntl;
|
|
|
|
const intlShape = require('../../lib/intl.jsx').intlShape;
|
2019-08-19 20:45:16 -04:00
|
|
|
const sessionActions = require('../../redux/session.js');
|
2019-06-24 11:31:16 -04:00
|
|
|
|
2019-07-10 21:49:04 -04:00
|
|
|
const Progression = require('../progression/progression.jsx');
|
2019-07-18 20:39:16 -04:00
|
|
|
const UsernameStep = require('./username-step.jsx');
|
|
|
|
const BirthDateStep = require('./birthdate-step.jsx');
|
2019-07-23 21:21:16 -04:00
|
|
|
const GenderStep = require('./gender-step.jsx');
|
2019-08-08 00:26:17 -04:00
|
|
|
const CountryStep = require('./country-step.jsx');
|
2019-07-29 22:29:04 -04:00
|
|
|
const EmailStep = require('./email-step.jsx');
|
|
|
|
const WelcomeStep = require('./welcome-step.jsx');
|
2019-08-30 16:20:05 -04:00
|
|
|
const RegistrationErrorStep = require('./registration-error-step.jsx');
|
2019-06-24 13:35:01 -04:00
|
|
|
|
2019-06-24 11:31:16 -04:00
|
|
|
class JoinFlow extends React.Component {
|
|
|
|
constructor (props) {
|
|
|
|
super(props);
|
2019-06-24 13:35:01 -04:00
|
|
|
bindAll(this, [
|
2019-08-19 20:45:16 -04:00
|
|
|
'handleAdvanceStep',
|
2019-09-19 13:40:09 -04:00
|
|
|
'handleRegistrationError',
|
2019-08-30 16:20:26 -04:00
|
|
|
'handlePrepareToRegister',
|
2019-09-03 18:27:11 -04:00
|
|
|
'handleRegistrationResponse',
|
2019-08-30 16:20:26 -04:00
|
|
|
'handleSubmitRegistration'
|
2019-06-24 13:35:01 -04:00
|
|
|
]);
|
2019-07-10 21:49:04 -04:00
|
|
|
this.state = {
|
|
|
|
formData: {},
|
|
|
|
registrationError: null,
|
2019-08-19 20:45:16 -04:00
|
|
|
step: 0,
|
|
|
|
waiting: false
|
2019-07-10 21:49:04 -04:00
|
|
|
};
|
2019-06-24 13:35:01 -04:00
|
|
|
}
|
2019-09-19 13:40:09 -04:00
|
|
|
handleRegistrationError (message) {
|
2019-09-18 10:26:37 -04:00
|
|
|
if (!message) {
|
|
|
|
message = this.props.intl.formatMessage({
|
|
|
|
id: 'registration.generalError'
|
|
|
|
});
|
|
|
|
}
|
|
|
|
this.setState({registrationError: message});
|
|
|
|
}
|
2019-08-30 16:20:26 -04:00
|
|
|
handlePrepareToRegister (newFormData) {
|
|
|
|
newFormData = newFormData || {};
|
|
|
|
const newState = {
|
|
|
|
formData: defaults({}, newFormData, this.state.formData)
|
|
|
|
};
|
|
|
|
this.setState(newState, () => {
|
|
|
|
this.handleSubmitRegistration(this.state.formData);
|
|
|
|
});
|
|
|
|
}
|
2019-09-03 18:27:11 -04:00
|
|
|
handleRegistrationResponse (err, body, res) {
|
2019-09-04 23:34:17 -04:00
|
|
|
// example of failing response:
|
|
|
|
// [
|
|
|
|
// {
|
|
|
|
// "msg": "This field is required.",
|
|
|
|
// "errors": {
|
|
|
|
// "username": ["This field is required."],
|
|
|
|
// "recaptcha": ["Incorrect, please try again."]
|
|
|
|
// },
|
|
|
|
// "success": false
|
|
|
|
// }
|
|
|
|
// ]
|
2019-09-03 18:27:11 -04:00
|
|
|
this.setState({waiting: false}, () => {
|
|
|
|
let errStr = '';
|
|
|
|
if (!err && res.statusCode === 200) {
|
|
|
|
if (body && body[0]) {
|
|
|
|
if (body[0].success) {
|
|
|
|
this.props.refreshSession();
|
|
|
|
this.setState({
|
|
|
|
step: this.state.step + 1
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (body[0].errors) {
|
|
|
|
// body can include zero or more error objects, each
|
|
|
|
// with its own key and description. Here we assemble
|
|
|
|
// all of them into a single string, errStr.
|
|
|
|
const errorKeys = Object.keys(body[0].errors);
|
|
|
|
errorKeys.forEach(key => {
|
|
|
|
const val = body[0].errors[key];
|
|
|
|
if (val && val[0]) {
|
|
|
|
if (errStr.length) errStr += '; ';
|
|
|
|
errStr += `${key}: ${val[0]}`;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if (!errStr.length && body[0].msg) errStr = body[0].msg;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.setState({
|
|
|
|
registrationError: errStr ||
|
|
|
|
`${this.props.intl.formatMessage({
|
|
|
|
id: 'registration.generalError'
|
|
|
|
})} (${res.statusCode})`
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
2019-08-30 16:20:26 -04:00
|
|
|
handleSubmitRegistration (formData) {
|
2019-08-19 20:45:16 -04:00
|
|
|
this.setState({waiting: true}, () => {
|
|
|
|
api({
|
|
|
|
host: '',
|
|
|
|
uri: '/accounts/register_new_user/',
|
|
|
|
method: 'post',
|
|
|
|
useCsrf: true,
|
|
|
|
formData: {
|
2019-09-03 17:41:22 -04:00
|
|
|
'username': formData.username,
|
|
|
|
'email': formData.email,
|
|
|
|
'password': formData.password,
|
|
|
|
'birth_month': formData.birth_month,
|
|
|
|
'birth_year': formData.birth_year,
|
2019-09-11 12:36:56 -04:00
|
|
|
'g-recaptcha-response': formData['g-recaptcha-response'],
|
2019-09-03 17:41:22 -04:00
|
|
|
'gender': formData.gender,
|
|
|
|
'country': formData.country,
|
|
|
|
'subscribe': true,
|
|
|
|
'is_robot': formData.yesno
|
2019-08-30 16:19:27 -04:00
|
|
|
// no need to include csrfmiddlewaretoken; will be provided in
|
|
|
|
// X-CSRFToken header, which scratchr2 looks for in
|
|
|
|
// scratchr2/middleware/csrf.py line 237.
|
2019-08-19 20:45:16 -04:00
|
|
|
}
|
|
|
|
}, (err, body, res) => {
|
2019-09-03 18:27:11 -04:00
|
|
|
this.handleRegistrationResponse(err, body, res);
|
2019-08-19 20:45:16 -04:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
handleAdvanceStep (newFormData) {
|
|
|
|
newFormData = newFormData || {};
|
2019-06-24 13:35:01 -04:00
|
|
|
this.setState({
|
2019-08-30 16:20:26 -04:00
|
|
|
formData: defaults({}, newFormData, this.state.formData),
|
|
|
|
step: this.state.step + 1
|
2019-06-24 13:35:01 -04:00
|
|
|
});
|
2019-06-24 11:31:16 -04:00
|
|
|
}
|
|
|
|
render () {
|
|
|
|
return (
|
2019-06-24 13:35:01 -04:00
|
|
|
<React.Fragment>
|
2019-08-19 20:45:16 -04:00
|
|
|
{this.state.registrationError ? (
|
2019-08-30 16:20:05 -04:00
|
|
|
<RegistrationErrorStep
|
2019-08-19 20:45:16 -04:00
|
|
|
errorMsg={this.state.registrationError}
|
|
|
|
/* eslint-disable react/jsx-no-bind */
|
2019-08-30 16:20:26 -04:00
|
|
|
onTryAgain={() => this.handleSubmitRegistration(this.state.formData)}
|
2019-08-19 20:45:16 -04:00
|
|
|
/* eslint-enable react/jsx-no-bind */
|
2019-07-29 22:29:04 -04:00
|
|
|
/>
|
2019-08-19 20:45:16 -04:00
|
|
|
) : (
|
|
|
|
<Progression step={this.state.step}>
|
|
|
|
<UsernameStep onNextStep={this.handleAdvanceStep} />
|
2019-09-11 10:39:46 -04:00
|
|
|
<CountryStep onNextStep={this.handleAdvanceStep} />
|
2019-08-19 20:45:16 -04:00
|
|
|
<BirthDateStep onNextStep={this.handleAdvanceStep} />
|
|
|
|
<GenderStep onNextStep={this.handleAdvanceStep} />
|
|
|
|
<EmailStep
|
|
|
|
waiting={this.state.waiting}
|
2019-08-30 16:20:26 -04:00
|
|
|
onNextStep={this.handlePrepareToRegister}
|
2019-09-19 13:40:09 -04:00
|
|
|
onRegistrationError={this.handleRegistrationError}
|
2019-08-19 20:45:16 -04:00
|
|
|
/>
|
|
|
|
<WelcomeStep
|
2019-09-23 11:25:10 -04:00
|
|
|
createProjectOnComplete={this.props.createProjectOnComplete}
|
2019-08-19 20:45:16 -04:00
|
|
|
email={this.state.formData.email}
|
|
|
|
username={this.state.formData.username}
|
|
|
|
onNextStep={this.props.onCompleteRegistration}
|
|
|
|
/>
|
|
|
|
</Progression>
|
|
|
|
)}
|
2019-06-24 13:35:01 -04:00
|
|
|
</React.Fragment>
|
2019-06-24 11:31:16 -04:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
JoinFlow.propTypes = {
|
2019-09-23 11:25:10 -04:00
|
|
|
createProjectOnComplete: PropTypes.bool,
|
2019-06-24 11:31:16 -04:00
|
|
|
intl: intlShape,
|
2019-08-19 20:45:16 -04:00
|
|
|
onCompleteRegistration: PropTypes.func,
|
|
|
|
refreshSession: PropTypes.func
|
2019-06-24 11:31:16 -04:00
|
|
|
};
|
|
|
|
|
2019-08-19 20:45:16 -04:00
|
|
|
const IntlJoinFlow = injectIntl(JoinFlow);
|
|
|
|
|
|
|
|
const mapDispatchToProps = dispatch => ({
|
|
|
|
refreshSession: () => {
|
|
|
|
dispatch(sessionActions.refreshSession());
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2019-09-04 23:33:43 -04:00
|
|
|
// Allow incoming props to override redux-provided props. Used to mock in tests.
|
|
|
|
const mergeProps = (stateProps, dispatchProps, ownProps) => Object.assign(
|
|
|
|
{}, stateProps, dispatchProps, ownProps
|
|
|
|
);
|
|
|
|
|
2019-08-19 20:45:16 -04:00
|
|
|
const ConnectedJoinFlow = connect(
|
|
|
|
() => ({}),
|
2019-09-04 23:33:43 -04:00
|
|
|
mapDispatchToProps,
|
|
|
|
mergeProps
|
2019-08-19 20:45:16 -04:00
|
|
|
)(IntlJoinFlow);
|
2019-06-24 11:31:16 -04:00
|
|
|
|
2019-08-19 20:45:16 -04:00
|
|
|
module.exports = ConnectedJoinFlow;
|