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)}
115 disabled={!validEmail(this.state.loginForm.username_or_email)}
116 onClick={linkEvent(this, this.handlePasswordReset)}
117 className="btn p-0 btn-link d-inline-block float-right text-muted small font-weight-bold"
119 {i18n.t('forgot_password')}
123 <div class="form-group row">
124 <div class="col-sm-10">
125 <button type="submit" class="btn btn-secondary">
126 {this.state.loginLoading ? (
127 <svg class="icon icon-spinner spin">
128 <use xlinkHref="#icon-spinner"></use>
142 <form onSubmit={linkEvent(this, this.handleRegisterSubmit)}>
143 <h5>{i18n.t('sign_up')}</h5>
145 <div class="form-group row">
146 <label class="col-sm-2 col-form-label" htmlFor="register-username">
150 <div class="col-sm-10">
153 id="register-username"
155 value={this.state.registerForm.username}
156 onInput={linkEvent(this, this.handleRegisterUsernameChange)}
160 pattern="[a-zA-Z0-9_]+"
165 <div class="form-group row">
166 <label class="col-sm-2 col-form-label" htmlFor="register-email">
169 <div class="col-sm-10">
174 placeholder={i18n.t('optional')}
175 value={this.state.registerForm.email}
176 onInput={linkEvent(this, this.handleRegisterEmailChange)}
182 <div class="form-group row">
183 <label class="col-sm-2 col-form-label" htmlFor="register-password">
186 <div class="col-sm-10">
189 id="register-password"
190 value={this.state.registerForm.password}
191 autoComplete="new-password"
192 onInput={linkEvent(this, this.handleRegisterPasswordChange)}
199 <div class="form-group row">
201 class="col-sm-2 col-form-label"
202 htmlFor="register-verify-password"
204 {i18n.t('verify_password')}
206 <div class="col-sm-10">
209 id="register-verify-password"
210 value={this.state.registerForm.password_verify}
211 autoComplete="new-password"
212 onInput={linkEvent(this, this.handleRegisterPasswordVerifyChange)}
218 {this.state.enable_nsfw && (
219 <div class="form-group row">
220 <div class="col-sm-10">
221 <div class="form-check">
223 class="form-check-input"
224 id="register-show-nsfw"
226 checked={this.state.registerForm.show_nsfw}
227 onChange={linkEvent(this, this.handleRegisterShowNsfwChange)}
229 <label class="form-check-label" htmlFor="register-show-nsfw">
230 {i18n.t('show_nsfw')}
236 <div class="form-group row">
237 <div class="col-sm-10">
238 <button type="submit" class="btn btn-secondary">
239 {this.state.registerLoading ? (
240 <svg class="icon icon-spinner spin">
241 <use xlinkHref="#icon-spinner"></use>
253 handleLoginSubmit(i: Login, event: any) {
254 event.preventDefault();
255 i.state.loginLoading = true;
257 WebSocketService.Instance.login(i.state.loginForm);
260 handleLoginUsernameChange(i: Login, event: any) {
261 i.state.loginForm.username_or_email = event.target.value;
265 handleLoginPasswordChange(i: Login, event: any) {
266 i.state.loginForm.password = event.target.value;
270 handleRegisterSubmit(i: Login, event: any) {
271 event.preventDefault();
272 i.state.registerLoading = true;
275 WebSocketService.Instance.register(i.state.registerForm);
278 handleRegisterUsernameChange(i: Login, event: any) {
279 i.state.registerForm.username = event.target.value;
283 handleRegisterEmailChange(i: Login, event: any) {
284 i.state.registerForm.email = event.target.value;
285 if (i.state.registerForm.email == '') {
286 i.state.registerForm.email = undefined;
291 handleRegisterPasswordChange(i: Login, event: any) {
292 i.state.registerForm.password = event.target.value;
296 handleRegisterPasswordVerifyChange(i: Login, event: any) {
297 i.state.registerForm.password_verify = event.target.value;
301 handleRegisterShowNsfwChange(i: Login, event: any) {
302 i.state.registerForm.show_nsfw = event.target.checked;
306 handlePasswordReset(i: Login) {
307 event.preventDefault();
308 let resetForm: PasswordResetForm = {
309 email: i.state.loginForm.username_or_email,
311 WebSocketService.Instance.passwordReset(resetForm);
314 parseMessage(msg: WebSocketJsonResponse) {
315 let res = wsJsonToRes(msg);
317 toast(i18n.t(msg.error), 'danger');
318 this.state = this.emptyState;
319 this.setState(this.state);
322 if (res.op == UserOperation.Login) {
323 let data = res.data as LoginResponse;
324 this.state = this.emptyState;
325 this.setState(this.state);
326 UserService.Instance.login(data);
327 WebSocketService.Instance.userJoin();
328 toast(i18n.t('logged_in'));
329 this.props.history.push('/');
330 } else if (res.op == UserOperation.Register) {
331 let data = res.data as LoginResponse;
332 this.state = this.emptyState;
333 this.setState(this.state);
334 UserService.Instance.login(data);
335 WebSocketService.Instance.userJoin();
336 this.props.history.push('/communities');
337 } else if (res.op == UserOperation.PasswordReset) {
338 toast(i18n.t('reset_password_mail_sent'));
339 } else if (res.op == UserOperation.GetSite) {
340 let data = res.data as GetSiteResponse;
341 this.state.enable_nsfw = data.site.enable_nsfw;
342 this.setState(this.state);
343 document.title = `${i18n.t('login')} - ${
344 WebSocketService.Instance.site.name