2018-01-30 11:53:12 -05:00
|
|
|
const bindAll = require('lodash.bindall');
|
|
|
|
const connect = require('react-redux').connect;
|
|
|
|
const defaults = require('lodash.defaultsdeep');
|
|
|
|
const PropTypes = require('prop-types');
|
|
|
|
const React = require('react');
|
2016-07-21 16:56:01 -04:00
|
|
|
|
2018-01-30 11:53:12 -05:00
|
|
|
const api = require('../../lib/api');
|
|
|
|
const injectIntl = require('../../lib/intl.jsx').injectIntl;
|
|
|
|
const intlShape = require('../../lib/intl.jsx').intlShape;
|
|
|
|
const log = require('../../lib/log.js');
|
|
|
|
const sessionStatus = require('../../redux/session').Status;
|
2016-07-21 16:56:01 -04:00
|
|
|
|
2018-01-30 11:53:12 -05:00
|
|
|
const Deck = require('../../components/deck/deck.jsx');
|
|
|
|
const Progression = require('../../components/progression/progression.jsx');
|
|
|
|
const Spinner = require('../../components/spinner/spinner.jsx');
|
|
|
|
const Steps = require('../../components/registration/steps.jsx');
|
|
|
|
|
|
|
|
const render = require('../../lib/render.jsx');
|
2016-07-21 16:56:01 -04:00
|
|
|
|
|
|
|
require('./studentcompleteregistration.scss');
|
|
|
|
|
2018-01-30 11:53:12 -05:00
|
|
|
class StudentCompleteRegistration extends React.Component {
|
|
|
|
constructor (props) {
|
|
|
|
super(props);
|
|
|
|
bindAll(this, [
|
|
|
|
'handleAdvanceStep',
|
|
|
|
'handleLogOut',
|
|
|
|
'handleRegister',
|
|
|
|
'handleGoToClass'
|
|
|
|
]);
|
|
|
|
this.state = {
|
2016-07-21 16:56:01 -04:00
|
|
|
classroom: null,
|
|
|
|
formData: {},
|
2016-07-22 09:38:37 -04:00
|
|
|
registrationErrors: null,
|
2016-07-21 16:56:01 -04:00
|
|
|
step: 0,
|
|
|
|
waiting: false
|
|
|
|
};
|
2018-01-30 11:53:12 -05:00
|
|
|
}
|
|
|
|
componentDidUpdate (prevProps) {
|
2016-07-27 19:32:08 -04:00
|
|
|
if (prevProps.studentUsername !== this.props.studentUsername && this.props.newStudent) {
|
2018-01-30 11:53:12 -05:00
|
|
|
this.setState({waiting: true}); // eslint-disable-line react/no-did-update-set-state
|
2016-07-21 16:56:01 -04:00
|
|
|
api({
|
2018-01-30 11:53:12 -05:00
|
|
|
uri: `/classrooms/${this.props.classroomId}`
|
|
|
|
}, (err, body, res) => {
|
2016-07-25 11:41:10 -04:00
|
|
|
this.setState({waiting: false});
|
|
|
|
if (err || res.statusCode !== 200) {
|
2016-07-21 16:56:01 -04:00
|
|
|
return this.setState({
|
2016-07-22 09:38:37 -04:00
|
|
|
registrationErrors: {
|
2016-07-27 19:32:08 -04:00
|
|
|
__all__: this.props.intl.formatMessage({id: 'registration.classroomApiGeneralError'})
|
2016-07-22 09:38:37 -04:00
|
|
|
}
|
2016-07-21 16:56:01 -04:00
|
|
|
});
|
|
|
|
}
|
|
|
|
this.setState({classroom: body});
|
2018-01-30 11:53:12 -05:00
|
|
|
});
|
2016-07-21 16:56:01 -04:00
|
|
|
}
|
2018-01-30 11:53:12 -05:00
|
|
|
}
|
|
|
|
handleAdvanceStep (formData) {
|
|
|
|
formData = formData || {};
|
|
|
|
this.setState({
|
|
|
|
step: this.state.step + 1,
|
|
|
|
formData: defaults({}, formData, this.state.formData)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
handleLogOut (e) {
|
2016-07-27 10:24:37 -04:00
|
|
|
e.preventDefault();
|
|
|
|
api({
|
|
|
|
host: '',
|
|
|
|
method: 'post',
|
|
|
|
uri: '/accounts/logout/',
|
|
|
|
useCsrf: true
|
2018-01-30 11:53:12 -05:00
|
|
|
}, err => {
|
2016-07-27 10:24:37 -04:00
|
|
|
if (err) return log.error(err);
|
|
|
|
window.location = '/';
|
2018-01-30 11:53:12 -05:00
|
|
|
});
|
|
|
|
}
|
|
|
|
handleRegister (formData) {
|
2016-07-21 16:56:01 -04:00
|
|
|
this.setState({waiting: true});
|
2018-09-12 14:54:15 -04:00
|
|
|
|
2016-07-21 16:56:01 -04:00
|
|
|
formData = defaults({}, formData || {}, this.state.formData);
|
2018-01-30 11:53:12 -05:00
|
|
|
const submittedData = {
|
2016-07-21 16:56:01 -04:00
|
|
|
birth_month: formData.user.birth.month,
|
|
|
|
birth_year: formData.user.birth.year,
|
2018-01-30 11:53:12 -05:00
|
|
|
gender: (formData.user.gender === 'other' ? formData.user.genderOther : formData.user.gender),
|
2016-07-21 16:56:01 -04:00
|
|
|
country: formData.user.country,
|
|
|
|
is_robot: formData.user.isRobot
|
|
|
|
};
|
2016-07-27 19:32:08 -04:00
|
|
|
if (this.props.must_reset_password) {
|
2016-07-21 16:56:01 -04:00
|
|
|
submittedData.password = formData.user.password;
|
|
|
|
}
|
2018-09-12 14:54:15 -04:00
|
|
|
|
2016-07-21 16:56:01 -04:00
|
|
|
api({
|
|
|
|
host: '',
|
|
|
|
uri: '/classes/student_update_registration/',
|
|
|
|
method: 'post',
|
|
|
|
useCsrf: true,
|
|
|
|
formData: submittedData
|
2018-01-30 11:53:12 -05:00
|
|
|
}, (err, body, res) => {
|
2016-07-21 16:56:01 -04:00
|
|
|
this.setState({waiting: false});
|
|
|
|
if (err) return this.setState({registrationError: err});
|
2018-01-30 11:53:12 -05:00
|
|
|
if (body.success) return this.handleAdvanceStep(formData);
|
2016-12-13 12:12:23 -05:00
|
|
|
this.setState({
|
2018-01-30 11:53:12 -05:00
|
|
|
registrationErrors: body.errors || {
|
|
|
|
__all__:
|
|
|
|
`${this.props.intl.formatMessage({id: 'registration.generalError'})} (${res.statusCode})`
|
|
|
|
}
|
2016-12-13 12:12:23 -05:00
|
|
|
});
|
2018-01-30 11:53:12 -05:00
|
|
|
});
|
|
|
|
}
|
|
|
|
handleGoToClass () {
|
|
|
|
window.location = `/classes/${this.state.classroom.id}/`;
|
|
|
|
}
|
|
|
|
render () {
|
|
|
|
const demographicsDescription = this.props.intl.formatMessage({
|
|
|
|
id: 'registration.studentPersonalStepDescription'
|
|
|
|
});
|
|
|
|
let registrationErrors = this.state.registrationErrors;
|
2016-07-27 19:32:08 -04:00
|
|
|
if (!this.props.newStudent) {
|
2016-07-22 09:38:37 -04:00
|
|
|
registrationErrors = {
|
|
|
|
__all__: this.props.intl.formatMessage({id: 'registration.mustBeNewStudent'})
|
|
|
|
};
|
2016-07-21 16:56:01 -04:00
|
|
|
}
|
2018-01-30 11:53:12 -05:00
|
|
|
|
2016-07-21 16:56:01 -04:00
|
|
|
return (
|
|
|
|
<Deck className="student-registration">
|
2016-07-25 16:58:38 -04:00
|
|
|
{registrationErrors ? (
|
|
|
|
<Steps.RegistrationError>
|
|
|
|
<ul>
|
2018-01-30 11:53:12 -05:00
|
|
|
{Object.keys(registrationErrors).map(field => {
|
|
|
|
let label = `${field}: `;
|
2016-07-25 16:58:38 -04:00
|
|
|
if (field === '__all__') {
|
|
|
|
label = '';
|
|
|
|
}
|
2018-01-30 11:53:12 -05:00
|
|
|
return (
|
|
|
|
<li key={field}>
|
|
|
|
{label}{registrationErrors[field]}
|
|
|
|
</li>
|
|
|
|
);
|
2016-07-25 16:58:38 -04:00
|
|
|
})}
|
|
|
|
</ul>
|
|
|
|
</Steps.RegistrationError>
|
|
|
|
) : (
|
|
|
|
this.state.waiting || !this.state.classroom ? (
|
|
|
|
<Spinner />
|
|
|
|
) : (
|
2016-07-21 16:56:01 -04:00
|
|
|
<Progression {... this.state}>
|
2018-01-30 11:53:12 -05:00
|
|
|
<Steps.ClassInviteExistingStudentStep
|
|
|
|
classroom={this.state.classroom}
|
|
|
|
studentUsername={this.props.studentUsername}
|
|
|
|
waiting={this.state.waiting}
|
|
|
|
onHandleLogOut={this.handleLogOut}
|
|
|
|
onNextStep={this.handleAdvanceStep}
|
|
|
|
/>
|
2016-07-27 19:32:08 -04:00
|
|
|
{this.props.must_reset_password ?
|
2018-01-30 11:53:12 -05:00
|
|
|
<Steps.ChoosePasswordStep
|
|
|
|
showPassword
|
|
|
|
username={this.props.studentUsername}
|
|
|
|
waiting={this.state.waiting}
|
|
|
|
onNextStep={this.handleAdvanceStep}
|
|
|
|
/> : []
|
2016-07-21 16:56:01 -04:00
|
|
|
}
|
2018-01-30 11:53:12 -05:00
|
|
|
<Steps.DemographicsStep
|
|
|
|
description={demographicsDescription}
|
|
|
|
waiting={this.state.waiting}
|
|
|
|
onNextStep={this.handleRegister}
|
|
|
|
/>
|
|
|
|
<Steps.ClassWelcomeStep
|
|
|
|
classroom={this.state.classroom}
|
|
|
|
waiting={this.state.waiting}
|
|
|
|
onNextStep={this.handleGoToClass}
|
|
|
|
/>
|
2016-07-21 16:56:01 -04:00
|
|
|
</Progression>
|
|
|
|
)
|
2016-07-25 16:58:38 -04:00
|
|
|
)}
|
2016-07-21 16:56:01 -04:00
|
|
|
</Deck>
|
|
|
|
);
|
|
|
|
}
|
2018-01-30 11:53:12 -05:00
|
|
|
}
|
2016-07-21 16:56:01 -04:00
|
|
|
|
2018-01-30 11:53:12 -05:00
|
|
|
StudentCompleteRegistration.propTypes = {
|
|
|
|
classroomId: PropTypes.number.isRequired,
|
|
|
|
intl: intlShape,
|
|
|
|
must_reset_password: PropTypes.bool.isRequired,
|
|
|
|
newStudent: PropTypes.bool.isRequired,
|
|
|
|
sessionFetched: PropTypes.bool.isRequired,
|
|
|
|
studentUsername: PropTypes.string.isRequired
|
2016-07-21 16:56:01 -04:00
|
|
|
};
|
|
|
|
|
2018-01-30 11:53:12 -05:00
|
|
|
const IntlStudentCompleteRegistration = injectIntl(StudentCompleteRegistration);
|
|
|
|
|
|
|
|
const mapStateToProps = state => ({
|
|
|
|
classroomId: state.session.session.user && state.session.session.user.classroomId,
|
|
|
|
must_reset_password: state.session.session.flags && state.session.session.flags.must_reset_password,
|
|
|
|
newStudent: (
|
|
|
|
state.session.session.permissions &&
|
|
|
|
state.session.session.permissions.student &&
|
|
|
|
state.session.session.flags.must_complete_registration
|
|
|
|
),
|
|
|
|
sessionFetched: state.session.status === sessionStatus.FETCHED,
|
|
|
|
studentUsername: state.session.session.user && state.session.session.user.username
|
|
|
|
});
|
|
|
|
|
|
|
|
const ConnectedStudentCompleteRegistration = connect(mapStateToProps)(IntlStudentCompleteRegistration);
|
2016-07-21 16:56:01 -04:00
|
|
|
|
|
|
|
render(<ConnectedStudentCompleteRegistration />, document.getElementById('app'));
|