122 lines
4.1 KiB
TypeScript
122 lines
4.1 KiB
TypeScript
|
import React, { ChangeEvent, FormEvent } from 'react';
|
||
|
import { RouteComponentProps, Redirect } from 'react-router';
|
||
|
|
||
|
import ErrorBox from './error-box';
|
||
|
import { Link } from 'react-router-dom';
|
||
|
|
||
|
interface PasswordResetState {
|
||
|
error: string;
|
||
|
success: boolean;
|
||
|
token: string;
|
||
|
password: string;
|
||
|
confirmPassword: string;
|
||
|
}
|
||
|
|
||
|
export default class PasswordReset extends React.Component<RouteComponentProps, PasswordResetState> {
|
||
|
|
||
|
static tokenMissingMessage = 'Password reset token is missing. Make sure to follow the link from a password reset email!';
|
||
|
|
||
|
constructor(props) {
|
||
|
super(props);
|
||
|
|
||
|
this.state = {
|
||
|
error: undefined,
|
||
|
success: undefined,
|
||
|
token: undefined,
|
||
|
password: undefined,
|
||
|
confirmPassword: undefined
|
||
|
};
|
||
|
}
|
||
|
|
||
|
componentDidMount() {
|
||
|
const queryParams = new URLSearchParams(this.props.location.search);
|
||
|
const token = queryParams.get('token');
|
||
|
if(token == undefined) {
|
||
|
this.setState({ error: PasswordReset.tokenMissingMessage });
|
||
|
} else {
|
||
|
this.setState({ token: token });
|
||
|
}
|
||
|
}
|
||
|
|
||
|
handleChange(event: FormEvent<HTMLInputElement>) {
|
||
|
const { name, value } = event.currentTarget;
|
||
|
this.setState({
|
||
|
[name]: value,
|
||
|
} as any);
|
||
|
}
|
||
|
|
||
|
async handleSubmit(event: FormEvent) {
|
||
|
event.preventDefault();
|
||
|
|
||
|
this.setState({ error: undefined });
|
||
|
|
||
|
if(this.state.token == undefined) {
|
||
|
this.setState({ error: PasswordReset.tokenMissingMessage });
|
||
|
return;
|
||
|
}
|
||
|
if(this.state.password !== this.state.confirmPassword) {
|
||
|
this.setState({ error: 'Passwords do not match' });
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
const requestData = {
|
||
|
token: this.state.token,
|
||
|
password: this.state.password
|
||
|
};
|
||
|
|
||
|
const res = await fetch('/api/users/password', {
|
||
|
method: 'PUT',
|
||
|
body: JSON.stringify(requestData),
|
||
|
headers: {
|
||
|
'Content-Type': 'application/json'
|
||
|
}
|
||
|
});
|
||
|
const data = await res.json();
|
||
|
|
||
|
if (data.error != undefined) {
|
||
|
this.setState({ error: data.error });
|
||
|
} else if (data.success === true) {
|
||
|
this.setState({ success: true });
|
||
|
} else {
|
||
|
this.setState({ error: 'Unexpected result.' });
|
||
|
}
|
||
|
}
|
||
|
|
||
|
render() {
|
||
|
if (this.state.success) {
|
||
|
return <Redirect to="/my-account.html" />;
|
||
|
}
|
||
|
return (
|
||
|
|
||
|
<article>
|
||
|
<section className="main-col">
|
||
|
<h1 className="h2">Forgotten password</h1>
|
||
|
<p>Please input the new password you want to set for your account. If your password reset token does not work, go back to the <Link to="/forgotten-password.html">forgotten password page</Link> to request a new link.</p>
|
||
|
<ErrorBox msg={this.state.error} />
|
||
|
<form onSubmit={e => this.handleSubmit(e)}>
|
||
|
<label htmlFor="email">New password</label>
|
||
|
<input name="password"
|
||
|
className="form-control" type="password"
|
||
|
placeholder="New password" required
|
||
|
onChange={e => this.handleChange(e)}
|
||
|
/>
|
||
|
|
||
|
<label htmlFor="email">Confirm new password</label>
|
||
|
<input name="confirmPassword"
|
||
|
className="form-control" type="password"
|
||
|
placeholder="Confirm new password" required
|
||
|
onChange={e => this.handleChange(e)}
|
||
|
/>
|
||
|
|
||
|
<input name="token" id="token" type="hidden" value={this.state.token} />
|
||
|
|
||
|
<div className="buttons-container">
|
||
|
<input type="submit" value="Change password" className="btn btn-primary" />
|
||
|
</div>
|
||
|
</form>
|
||
|
</section>
|
||
|
</article>
|
||
|
)
|
||
|
}
|
||
|
}
|