1 import { Component, linkEvent } from 'inferno';
2 import { Subscription } from 'rxjs';
3 import { retryWhen, delay, take } from 'rxjs/operators';
11 WebSocketJsonResponse,
12 } from '../interfaces';
13 import { WebSocketService, UserService } from '../services';
14 import { wsJsonToRes, validEmail, toast } from '../utils';
15 import { i18n } from '../i18next';
19 registerForm: RegisterForm;
20 loginLoading: boolean;
21 registerLoading: boolean;
25 export class Login extends Component<any, State> {
26 private subscription: Subscription;
30 username_or_email: undefined,
36 password_verify: undefined,
41 registerLoading: false,
42 enable_nsfw: undefined,
45 constructor(props: any, context: any) {
46 super(props, context);
48 this.state = this.emptyState;
50 this.subscription = WebSocketService.Instance.subject
51 .pipe(retryWhen(errors => errors.pipe(delay(3000), take(10))))
53 msg => this.parseMessage(msg),
54 err => console.error(err),
55 () => console.log('complete')
58 WebSocketService.Instance.getSite();
61 componentWillUnmount() {
62 this.subscription.unsubscribe();
67 <div class="container">
69 <div class="col-12 col-lg-6 mb-4">{this.loginForm()}</div>
70 <div class="col-12 col-lg-6">{this.registerForm()}</div>
79 <form onSubmit={linkEvent(this, this.handleLoginSubmit)}>
80 <h5>{i18n.t('login')}</h5>
81 <div class="form-group row">
83 class="col-sm-2 col-form-label"
84 htmlFor="login-email-or-username"
86 {i18n.t('email_or_username')}
88 <div class="col-sm-10">
92 id="login-email-or-username"
93 value={this.state.loginForm.username_or_email}
94 onInput={linkEvent(this, this.handleLoginUsernameChange)}
100 <div class="form-group row">
101 <label class="col-sm-2 col-form-label" htmlFor="login-password">
104 <div class="col-sm-10">
108 value={this.state.loginForm.password}
109 onInput={linkEvent(this, this.handleLoginPasswordChange)}
114 disabled={!validEmail(this.state.loginForm.username_or_email)}
115 onClick={linkEvent(this, this.handlePasswordReset)}
116 className="btn p-0 btn-link d-inline-block float-right text-muted small font-weight-bold"
118 {i18n.t('forgot_password')}
122 <div class="form-group row">
123 <div class="col-sm-10">
124 <button type="submit" class="btn btn-secondary">
125 {this.state.loginLoading ? (
126 <svg class="icon icon-spinner spin">
127 <use xlinkHref="#icon-spinner"></use>
141 <form onSubmit={linkEvent(this, this.handleRegisterSubmit)}>
142 <h5>{i18n.t('sign_up')}</h5>
144 <div class="form-group row">
145 <label class="col-sm-2 col-form-label" htmlFor="register-username">
149 <div class="col-sm-10">
152 id="register-username"
154 value={this.state.registerForm.username}
155 onInput={linkEvent(this, this.handleRegisterUsernameChange)}
159 pattern="[a-zA-Z0-9_]+"
164 <div class="form-group row">
165 <label class="col-sm-2 col-form-label" htmlFor="register-email">
168 <div class="col-sm-10">
173 placeholder={i18n.t('optional')}
174 value={this.state.registerForm.email}
175 onInput={linkEvent(this, this.handleRegisterEmailChange)}
181 <div class="form-group row">
182 <label class="col-sm-2 col-form-label" htmlFor="register-password">
185 <div class="col-sm-10">
188 id="register-password"
189 value={this.state.registerForm.password}
190 autoComplete="new-password"
191 onInput={linkEvent(this, this.handleRegisterPasswordChange)}
198 <div class="form-group row">
200 class="col-sm-2 col-form-label"
201 htmlFor="register-verify-password"
203 {i18n.t('verify_password')}
205 <div class="col-sm-10">
208 id="register-verify-password"
209 value={this.state.registerForm.password_verify}
210 autoComplete="new-password"
211 onInput={linkEvent(this, this.handleRegisterPasswordVerifyChange)}
217 {this.state.enable_nsfw && (
218 <div class="form-group row">
219 <div class="col-sm-10">
220 <div class="form-check">
222 class="form-check-input"
223 id="register-show-nsfw"
225 checked={this.state.registerForm.show_nsfw}
226 onChange={linkEvent(this, this.handleRegisterShowNsfwChange)}
228 <label class="form-check-label" htmlFor="register-show-nsfw">
229 {i18n.t('show_nsfw')}
235 <div class="form-group row">
236 <div class="col-sm-10">
237 <button type="submit" class="btn btn-secondary">
238 {this.state.registerLoading ? (
239 <svg class="icon icon-spinner spin">
240 <use xlinkHref="#icon-spinner"></use>
252 handleLoginSubmit(i: Login, event: any) {
253 event.preventDefault();
254 i.state.loginLoading = true;
256 WebSocketService.Instance.login(i.state.loginForm);
259 handleLoginUsernameChange(i: Login, event: any) {
260 i.state.loginForm.username_or_email = event.target.value;
264 handleLoginPasswordChange(i: Login, event: any) {
265 i.state.loginForm.password = event.target.value;
269 handleRegisterSubmit(i: Login, event: any) {
270 event.preventDefault();
271 i.state.registerLoading = true;
274 WebSocketService.Instance.register(i.state.registerForm);
277 handleRegisterUsernameChange(i: Login, event: any) {
278 i.state.registerForm.username = event.target.value;
282 handleRegisterEmailChange(i: Login, event: any) {
283 i.state.registerForm.email = event.target.value;
284 if (i.state.registerForm.email == '') {
285 i.state.registerForm.email = undefined;
290 handleRegisterPasswordChange(i: Login, event: any) {
291 i.state.registerForm.password = event.target.value;
295 handleRegisterPasswordVerifyChange(i: Login, event: any) {
296 i.state.registerForm.password_verify = event.target.value;
300 handleRegisterShowNsfwChange(i: Login, event: any) {
301 i.state.registerForm.show_nsfw = event.target.checked;
305 handlePasswordReset(i: Login) {
306 event.preventDefault();
307 let resetForm: PasswordResetForm = {
308 email: i.state.loginForm.username_or_email,
310 WebSocketService.Instance.passwordReset(resetForm);
313 parseMessage(msg: WebSocketJsonResponse) {
314 let res = wsJsonToRes(msg);
316 toast(i18n.t(msg.error), 'danger');
317 this.state = this.emptyState;
318 this.setState(this.state);
321 if (res.op == UserOperation.Login) {
322 let data = res.data as LoginResponse;
323 this.state = this.emptyState;
324 this.setState(this.state);
325 UserService.Instance.login(data);
326 WebSocketService.Instance.userJoin();
327 toast(i18n.t('logged_in'));
328 this.props.history.push('/');
329 } else if (res.op == UserOperation.Register) {
330 let data = res.data as LoginResponse;
331 this.state = this.emptyState;
332 this.setState(this.state);
333 UserService.Instance.login(data);
334 WebSocketService.Instance.userJoin();
335 this.props.history.push('/communities');
336 } else if (res.op == UserOperation.PasswordReset) {
337 toast(i18n.t('reset_password_mail_sent'));
338 } else if (res.op == UserOperation.GetSite) {
339 let data = res.data as GetSiteResponse;
340 this.state.enable_nsfw = data.site.enable_nsfw;
341 this.setState(this.state);
342 document.title = `${i18n.t('login')} - ${
343 WebSocketService.Instance.site.name