valtech-components 2.0.554 → 2.0.557
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/components/molecules/docs-callout/docs-callout.component.mjs +60 -0
- package/esm2022/lib/components/organisms/login/login.component.mjs +721 -0
- package/esm2022/lib/components/organisms/login/types.mjs +10 -0
- package/esm2022/public-api.mjs +4 -1
- package/fesm2022/valtech-components.mjs +4482 -3707
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/molecules/docs-callout/docs-callout.component.d.ts +17 -0
- package/lib/components/organisms/login/login.component.d.ts +73 -0
- package/lib/components/organisms/login/types.d.ts +68 -0
- package/package.json +1 -1
- package/public-api.d.ts +3 -0
|
@@ -0,0 +1,721 @@
|
|
|
1
|
+
import { Component, EventEmitter, inject, Input, Output, } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import { Validators } from '@angular/forms';
|
|
4
|
+
import { Router } from '@angular/router';
|
|
5
|
+
import { IonButton, IonButtons, IonContent, IonHeader, IonIcon, IonModal, IonText, IonToolbar, } from '@ionic/angular/standalone';
|
|
6
|
+
import { AuthService } from '../../../services/auth';
|
|
7
|
+
import { ToastService } from '../../../services/toast.service';
|
|
8
|
+
import { FormComponent } from '../form/form.component';
|
|
9
|
+
import { ImageComponent } from '../../atoms/image/image.component';
|
|
10
|
+
import { InputType, ComponentStates, } from '../../types';
|
|
11
|
+
import { SolidDefaultBlock } from '../../atoms/button/factory';
|
|
12
|
+
import { LOGIN_DEFAULTS, } from './types';
|
|
13
|
+
import * as i0 from "@angular/core";
|
|
14
|
+
export class LoginComponent {
|
|
15
|
+
constructor() {
|
|
16
|
+
this.props = {};
|
|
17
|
+
this.onSuccess = new EventEmitter();
|
|
18
|
+
this.onError = new EventEmitter();
|
|
19
|
+
this.onMFARequired = new EventEmitter();
|
|
20
|
+
// Services
|
|
21
|
+
this.authService = inject(AuthService);
|
|
22
|
+
this.toastService = inject(ToastService);
|
|
23
|
+
this.router = inject(Router);
|
|
24
|
+
// Timers
|
|
25
|
+
this.resendTimer = null;
|
|
26
|
+
this.resetResendTimer = null;
|
|
27
|
+
// State
|
|
28
|
+
this.isOAuthLoading = false;
|
|
29
|
+
this.isRegisterModalOpen = false;
|
|
30
|
+
this.isVerifyModalOpen = false;
|
|
31
|
+
this.isForgotPasswordModalOpen = false;
|
|
32
|
+
this.isResetPasswordModalOpen = false;
|
|
33
|
+
this.isMFAVerifyModalOpen = false;
|
|
34
|
+
this.pendingVerificationEmail = '';
|
|
35
|
+
this.pendingResetEmail = '';
|
|
36
|
+
this.resendCooldown = 0;
|
|
37
|
+
this.resetResendCooldown = 0;
|
|
38
|
+
// ==========================================
|
|
39
|
+
// LOGIN FORM
|
|
40
|
+
// ==========================================
|
|
41
|
+
this.emailInput = {
|
|
42
|
+
type: InputType.EMAIL,
|
|
43
|
+
label: 'Correo electrónico',
|
|
44
|
+
name: 'email',
|
|
45
|
+
token: 'login-email',
|
|
46
|
+
hint: '',
|
|
47
|
+
placeholder: 'tu@email.com',
|
|
48
|
+
errors: {
|
|
49
|
+
required: 'El correo es requerido',
|
|
50
|
+
email: 'Ingresa un correo válido',
|
|
51
|
+
},
|
|
52
|
+
validators: [Validators.required, Validators.email],
|
|
53
|
+
order: 0,
|
|
54
|
+
state: ComponentStates.ENABLED,
|
|
55
|
+
control: undefined,
|
|
56
|
+
};
|
|
57
|
+
this.passwordInput = {
|
|
58
|
+
type: InputType.PASSWORD,
|
|
59
|
+
label: 'Contraseña',
|
|
60
|
+
name: 'password',
|
|
61
|
+
token: 'login-password',
|
|
62
|
+
hint: '',
|
|
63
|
+
placeholder: '••••••••',
|
|
64
|
+
errors: {
|
|
65
|
+
required: 'La contraseña es requerida',
|
|
66
|
+
},
|
|
67
|
+
validators: [Validators.required],
|
|
68
|
+
order: 1,
|
|
69
|
+
state: ComponentStates.ENABLED,
|
|
70
|
+
control: undefined,
|
|
71
|
+
};
|
|
72
|
+
this.loginFormProps = {
|
|
73
|
+
name: 'Iniciar sesión',
|
|
74
|
+
sections: [
|
|
75
|
+
{
|
|
76
|
+
name: '',
|
|
77
|
+
order: 0,
|
|
78
|
+
fields: [this.emailInput, this.passwordInput],
|
|
79
|
+
},
|
|
80
|
+
],
|
|
81
|
+
actions: {
|
|
82
|
+
...SolidDefaultBlock('Iniciar sesión', 'submit'),
|
|
83
|
+
token: 'login-submit',
|
|
84
|
+
},
|
|
85
|
+
state: ComponentStates.ENABLED,
|
|
86
|
+
};
|
|
87
|
+
// ==========================================
|
|
88
|
+
// REGISTER FORM
|
|
89
|
+
// ==========================================
|
|
90
|
+
this.nameInput = {
|
|
91
|
+
type: InputType.TEXT,
|
|
92
|
+
label: 'Nombre completo',
|
|
93
|
+
name: 'name',
|
|
94
|
+
token: 'register-name',
|
|
95
|
+
hint: '',
|
|
96
|
+
placeholder: 'Tu nombre',
|
|
97
|
+
errors: {
|
|
98
|
+
required: 'El nombre es requerido',
|
|
99
|
+
minlength: 'Mínimo 2 caracteres',
|
|
100
|
+
},
|
|
101
|
+
validators: [Validators.required, Validators.minLength(2)],
|
|
102
|
+
order: 0,
|
|
103
|
+
state: ComponentStates.ENABLED,
|
|
104
|
+
control: undefined,
|
|
105
|
+
};
|
|
106
|
+
this.registerEmailInput = {
|
|
107
|
+
type: InputType.EMAIL,
|
|
108
|
+
label: 'Correo electrónico',
|
|
109
|
+
name: 'email',
|
|
110
|
+
token: 'register-email',
|
|
111
|
+
hint: '',
|
|
112
|
+
placeholder: 'tu@email.com',
|
|
113
|
+
errors: {
|
|
114
|
+
required: 'El correo es requerido',
|
|
115
|
+
email: 'Ingresa un correo válido',
|
|
116
|
+
},
|
|
117
|
+
validators: [Validators.required, Validators.email],
|
|
118
|
+
order: 1,
|
|
119
|
+
state: ComponentStates.ENABLED,
|
|
120
|
+
control: undefined,
|
|
121
|
+
};
|
|
122
|
+
this.registerPasswordInput = {
|
|
123
|
+
type: InputType.PASSWORD,
|
|
124
|
+
label: 'Contraseña',
|
|
125
|
+
name: 'password',
|
|
126
|
+
token: 'register-password',
|
|
127
|
+
hint: '',
|
|
128
|
+
placeholder: '••••••••',
|
|
129
|
+
errors: {
|
|
130
|
+
required: 'La contraseña es requerida',
|
|
131
|
+
minlength: 'Mínimo 8 caracteres',
|
|
132
|
+
},
|
|
133
|
+
validators: [Validators.required, Validators.minLength(8)],
|
|
134
|
+
order: 2,
|
|
135
|
+
state: ComponentStates.ENABLED,
|
|
136
|
+
control: undefined,
|
|
137
|
+
};
|
|
138
|
+
this.registerFormProps = {
|
|
139
|
+
name: 'Crear cuenta',
|
|
140
|
+
sections: [
|
|
141
|
+
{
|
|
142
|
+
name: '',
|
|
143
|
+
order: 0,
|
|
144
|
+
fields: [this.nameInput, this.registerEmailInput, this.registerPasswordInput],
|
|
145
|
+
},
|
|
146
|
+
],
|
|
147
|
+
actions: {
|
|
148
|
+
...SolidDefaultBlock('Registrarse', 'submit'),
|
|
149
|
+
token: 'register-submit',
|
|
150
|
+
},
|
|
151
|
+
state: ComponentStates.ENABLED,
|
|
152
|
+
};
|
|
153
|
+
// ==========================================
|
|
154
|
+
// VERIFY EMAIL FORM
|
|
155
|
+
// ==========================================
|
|
156
|
+
this.verifyPinInput = {
|
|
157
|
+
type: InputType.PIN_CODE,
|
|
158
|
+
label: '',
|
|
159
|
+
name: 'code',
|
|
160
|
+
token: 'verify-pin',
|
|
161
|
+
hint: '',
|
|
162
|
+
placeholder: '',
|
|
163
|
+
errors: {
|
|
164
|
+
required: 'El código es requerido',
|
|
165
|
+
minlength: 'Ingresa los 6 dígitos',
|
|
166
|
+
},
|
|
167
|
+
validators: [Validators.required, Validators.minLength(6)],
|
|
168
|
+
order: 0,
|
|
169
|
+
state: ComponentStates.ENABLED,
|
|
170
|
+
control: undefined,
|
|
171
|
+
length: 6,
|
|
172
|
+
allowNumbersOnly: true,
|
|
173
|
+
autoFocus: true,
|
|
174
|
+
};
|
|
175
|
+
this.verifyFormProps = {
|
|
176
|
+
name: 'Verificar correo',
|
|
177
|
+
sections: [
|
|
178
|
+
{
|
|
179
|
+
name: 'Ingresa el código de verificación enviado a tu correo.',
|
|
180
|
+
order: 0,
|
|
181
|
+
fields: [this.verifyPinInput],
|
|
182
|
+
},
|
|
183
|
+
],
|
|
184
|
+
actions: {
|
|
185
|
+
...SolidDefaultBlock('Verificar', 'submit'),
|
|
186
|
+
token: 'verify-submit',
|
|
187
|
+
},
|
|
188
|
+
state: ComponentStates.ENABLED,
|
|
189
|
+
};
|
|
190
|
+
// ==========================================
|
|
191
|
+
// MFA VERIFY FORM
|
|
192
|
+
// ==========================================
|
|
193
|
+
this.mfaPinInput = {
|
|
194
|
+
type: InputType.PIN_CODE,
|
|
195
|
+
label: '',
|
|
196
|
+
name: 'code',
|
|
197
|
+
token: 'mfa-pin',
|
|
198
|
+
hint: '',
|
|
199
|
+
placeholder: '',
|
|
200
|
+
errors: {
|
|
201
|
+
required: 'El código es requerido',
|
|
202
|
+
minlength: 'Ingresa los 6 dígitos',
|
|
203
|
+
},
|
|
204
|
+
validators: [Validators.required, Validators.minLength(6)],
|
|
205
|
+
order: 0,
|
|
206
|
+
state: ComponentStates.ENABLED,
|
|
207
|
+
control: undefined,
|
|
208
|
+
length: 6,
|
|
209
|
+
allowNumbersOnly: true,
|
|
210
|
+
autoFocus: true,
|
|
211
|
+
};
|
|
212
|
+
this.mfaVerifyFormProps = {
|
|
213
|
+
name: 'Verificación MFA',
|
|
214
|
+
sections: [
|
|
215
|
+
{
|
|
216
|
+
name: '',
|
|
217
|
+
order: 0,
|
|
218
|
+
fields: [this.mfaPinInput],
|
|
219
|
+
},
|
|
220
|
+
],
|
|
221
|
+
actions: {
|
|
222
|
+
...SolidDefaultBlock('Verificar', 'submit'),
|
|
223
|
+
token: 'mfa-verify-submit',
|
|
224
|
+
},
|
|
225
|
+
state: ComponentStates.ENABLED,
|
|
226
|
+
};
|
|
227
|
+
// ==========================================
|
|
228
|
+
// FORGOT PASSWORD FORM
|
|
229
|
+
// ==========================================
|
|
230
|
+
this.forgotEmailInput = {
|
|
231
|
+
type: InputType.EMAIL,
|
|
232
|
+
label: '',
|
|
233
|
+
name: 'email',
|
|
234
|
+
token: 'forgot-email',
|
|
235
|
+
hint: '',
|
|
236
|
+
placeholder: 'tu@email.com',
|
|
237
|
+
errors: {
|
|
238
|
+
required: 'El correo es requerido',
|
|
239
|
+
email: 'Ingresa un correo válido',
|
|
240
|
+
},
|
|
241
|
+
validators: [Validators.required, Validators.email],
|
|
242
|
+
order: 0,
|
|
243
|
+
state: ComponentStates.ENABLED,
|
|
244
|
+
control: undefined,
|
|
245
|
+
};
|
|
246
|
+
this.forgotPasswordFormProps = {
|
|
247
|
+
name: 'Recuperar contraseña',
|
|
248
|
+
sections: [
|
|
249
|
+
{
|
|
250
|
+
name: 'Ingresa tu correo electrónico y te enviaremos un código para restablecer tu contraseña.',
|
|
251
|
+
order: 0,
|
|
252
|
+
fields: [this.forgotEmailInput],
|
|
253
|
+
},
|
|
254
|
+
],
|
|
255
|
+
actions: {
|
|
256
|
+
...SolidDefaultBlock('Enviar código', 'submit'),
|
|
257
|
+
token: 'forgot-submit',
|
|
258
|
+
},
|
|
259
|
+
state: ComponentStates.ENABLED,
|
|
260
|
+
};
|
|
261
|
+
// ==========================================
|
|
262
|
+
// RESET PASSWORD FORM
|
|
263
|
+
// ==========================================
|
|
264
|
+
this.resetPinInput = {
|
|
265
|
+
type: InputType.PIN_CODE,
|
|
266
|
+
label: '',
|
|
267
|
+
name: 'code',
|
|
268
|
+
token: 'reset-pin',
|
|
269
|
+
hint: '',
|
|
270
|
+
placeholder: '',
|
|
271
|
+
errors: {
|
|
272
|
+
required: 'El código es requerido',
|
|
273
|
+
minlength: 'Ingresa los 6 dígitos',
|
|
274
|
+
},
|
|
275
|
+
validators: [Validators.required, Validators.minLength(6)],
|
|
276
|
+
order: 0,
|
|
277
|
+
state: ComponentStates.ENABLED,
|
|
278
|
+
control: undefined,
|
|
279
|
+
length: 6,
|
|
280
|
+
allowNumbersOnly: true,
|
|
281
|
+
autoFocus: true,
|
|
282
|
+
};
|
|
283
|
+
this.newPasswordInput = {
|
|
284
|
+
type: InputType.PASSWORD,
|
|
285
|
+
label: 'Nueva contraseña',
|
|
286
|
+
name: 'newPassword',
|
|
287
|
+
token: 'reset-new-password',
|
|
288
|
+
hint: 'Mínimo 8 caracteres',
|
|
289
|
+
placeholder: '••••••••',
|
|
290
|
+
errors: {
|
|
291
|
+
required: 'La contraseña es requerida',
|
|
292
|
+
minlength: 'Mínimo 8 caracteres',
|
|
293
|
+
},
|
|
294
|
+
validators: [Validators.required, Validators.minLength(8)],
|
|
295
|
+
order: 1,
|
|
296
|
+
state: ComponentStates.ENABLED,
|
|
297
|
+
control: undefined,
|
|
298
|
+
};
|
|
299
|
+
this.resetPasswordFormProps = {
|
|
300
|
+
name: 'Restablecer contraseña',
|
|
301
|
+
sections: [
|
|
302
|
+
{
|
|
303
|
+
name: 'Hemos enviado un código de verificación a tu correo.',
|
|
304
|
+
order: 0,
|
|
305
|
+
fields: [this.resetPinInput, this.newPasswordInput],
|
|
306
|
+
},
|
|
307
|
+
],
|
|
308
|
+
actions: {
|
|
309
|
+
...SolidDefaultBlock('Cambiar contraseña', 'submit'),
|
|
310
|
+
token: 'reset-submit',
|
|
311
|
+
},
|
|
312
|
+
state: ComponentStates.ENABLED,
|
|
313
|
+
};
|
|
314
|
+
}
|
|
315
|
+
// Resolved props with defaults
|
|
316
|
+
get config() {
|
|
317
|
+
return { ...LOGIN_DEFAULTS, ...this.props };
|
|
318
|
+
}
|
|
319
|
+
// ==========================================
|
|
320
|
+
// HANDLERS
|
|
321
|
+
// ==========================================
|
|
322
|
+
loginHandler(event) {
|
|
323
|
+
const email = event.fields['email'];
|
|
324
|
+
const password = event.fields['password'];
|
|
325
|
+
if (!email || !password) {
|
|
326
|
+
this.showToast('Completa todos los campos.');
|
|
327
|
+
return;
|
|
328
|
+
}
|
|
329
|
+
this.loginFormProps.state = ComponentStates.WORKING;
|
|
330
|
+
this.authService.signin({ email, password }).subscribe({
|
|
331
|
+
next: () => {
|
|
332
|
+
this.loginFormProps.state = ComponentStates.ENABLED;
|
|
333
|
+
if (this.authService.mfaPending().required) {
|
|
334
|
+
this.openMFAVerifyModal();
|
|
335
|
+
return;
|
|
336
|
+
}
|
|
337
|
+
this.handleLoginSuccess();
|
|
338
|
+
},
|
|
339
|
+
error: (err) => {
|
|
340
|
+
this.loginFormProps.state = ComponentStates.ENABLED;
|
|
341
|
+
const errorCode = err?.code;
|
|
342
|
+
if (errorCode === 'AUTHV2_EMAIL_NOT_VERIFIED') {
|
|
343
|
+
this.openVerifyModal(email);
|
|
344
|
+
return;
|
|
345
|
+
}
|
|
346
|
+
this.handleError(err, 'signin');
|
|
347
|
+
},
|
|
348
|
+
});
|
|
349
|
+
}
|
|
350
|
+
loginWithOAuth(provider) {
|
|
351
|
+
this.isOAuthLoading = true;
|
|
352
|
+
this.authService.signinWithOAuth(provider).subscribe({
|
|
353
|
+
next: () => {
|
|
354
|
+
this.isOAuthLoading = false;
|
|
355
|
+
if (this.authService.mfaPending().required) {
|
|
356
|
+
this.openMFAVerifyModal();
|
|
357
|
+
return;
|
|
358
|
+
}
|
|
359
|
+
this.handleLoginSuccess(provider);
|
|
360
|
+
},
|
|
361
|
+
error: (err) => {
|
|
362
|
+
this.isOAuthLoading = false;
|
|
363
|
+
const errorCode = err?.code;
|
|
364
|
+
if (errorCode === 'POPUP_CLOSED') {
|
|
365
|
+
return;
|
|
366
|
+
}
|
|
367
|
+
this.handleError(err, 'oauth');
|
|
368
|
+
},
|
|
369
|
+
});
|
|
370
|
+
}
|
|
371
|
+
// ==========================================
|
|
372
|
+
// REGISTER HANDLERS
|
|
373
|
+
// ==========================================
|
|
374
|
+
openRegisterModal() {
|
|
375
|
+
this.isRegisterModalOpen = true;
|
|
376
|
+
}
|
|
377
|
+
closeRegisterModal() {
|
|
378
|
+
this.isRegisterModalOpen = false;
|
|
379
|
+
}
|
|
380
|
+
registerHandler(event) {
|
|
381
|
+
const name = event.fields['name'];
|
|
382
|
+
const email = event.fields['email'];
|
|
383
|
+
const password = event.fields['password'];
|
|
384
|
+
if (!name || !email || !password) {
|
|
385
|
+
this.showToast('Completa todos los campos.');
|
|
386
|
+
return;
|
|
387
|
+
}
|
|
388
|
+
this.registerFormProps.state = ComponentStates.WORKING;
|
|
389
|
+
this.authService.signup({ name, email, password }).subscribe({
|
|
390
|
+
next: () => {
|
|
391
|
+
this.registerFormProps.state = ComponentStates.ENABLED;
|
|
392
|
+
this.closeRegisterModal();
|
|
393
|
+
this.openVerifyModal(email);
|
|
394
|
+
},
|
|
395
|
+
error: (err) => {
|
|
396
|
+
this.registerFormProps.state = ComponentStates.ENABLED;
|
|
397
|
+
this.handleError(err, 'signup');
|
|
398
|
+
},
|
|
399
|
+
});
|
|
400
|
+
}
|
|
401
|
+
// ==========================================
|
|
402
|
+
// VERIFY EMAIL HANDLERS
|
|
403
|
+
// ==========================================
|
|
404
|
+
openVerifyModal(email) {
|
|
405
|
+
this.pendingVerificationEmail = email;
|
|
406
|
+
this.verifyFormProps.sections[0].name = `Ingresa el código enviado a ${email}`;
|
|
407
|
+
this.verifyFormProps.state = ComponentStates.ENABLED;
|
|
408
|
+
this.isVerifyModalOpen = true;
|
|
409
|
+
this.startResendCooldown();
|
|
410
|
+
}
|
|
411
|
+
closeVerifyModal() {
|
|
412
|
+
this.isVerifyModalOpen = false;
|
|
413
|
+
this.pendingVerificationEmail = '';
|
|
414
|
+
this.stopResendCooldown();
|
|
415
|
+
}
|
|
416
|
+
verifyHandler(event) {
|
|
417
|
+
const code = event.fields['code'];
|
|
418
|
+
this.verifyFormProps.state = ComponentStates.WORKING;
|
|
419
|
+
this.authService
|
|
420
|
+
.verifyEmail({
|
|
421
|
+
email: this.pendingVerificationEmail,
|
|
422
|
+
code,
|
|
423
|
+
})
|
|
424
|
+
.subscribe({
|
|
425
|
+
next: () => {
|
|
426
|
+
this.verifyFormProps.state = ComponentStates.ENABLED;
|
|
427
|
+
this.showToast('¡Email verificado! Bienvenido.');
|
|
428
|
+
this.closeVerifyModal();
|
|
429
|
+
this.handleLoginSuccess();
|
|
430
|
+
},
|
|
431
|
+
error: (err) => {
|
|
432
|
+
this.verifyFormProps.state = ComponentStates.ENABLED;
|
|
433
|
+
this.handleError(err, 'verify');
|
|
434
|
+
},
|
|
435
|
+
});
|
|
436
|
+
}
|
|
437
|
+
resendCode() {
|
|
438
|
+
if (this.resendCooldown > 0)
|
|
439
|
+
return;
|
|
440
|
+
this.authService
|
|
441
|
+
.resendCode({
|
|
442
|
+
email: this.pendingVerificationEmail,
|
|
443
|
+
type: 'EMAIL_VERIFY',
|
|
444
|
+
})
|
|
445
|
+
.subscribe({
|
|
446
|
+
next: () => {
|
|
447
|
+
this.showToast('Código reenviado. Revisa tu email.');
|
|
448
|
+
this.startResendCooldown();
|
|
449
|
+
},
|
|
450
|
+
error: (err) => {
|
|
451
|
+
this.handleError(err, 'verify');
|
|
452
|
+
},
|
|
453
|
+
});
|
|
454
|
+
}
|
|
455
|
+
// ==========================================
|
|
456
|
+
// MFA HANDLERS
|
|
457
|
+
// ==========================================
|
|
458
|
+
openMFAVerifyModal() {
|
|
459
|
+
const method = this.authService.mfaPending().method;
|
|
460
|
+
this.mfaVerifyFormProps.sections[0].name =
|
|
461
|
+
method === 'TOTP'
|
|
462
|
+
? 'Ingresa el código de tu app de autenticación'
|
|
463
|
+
: method === 'EMAIL'
|
|
464
|
+
? 'Ingresa el código enviado a tu correo'
|
|
465
|
+
: 'Ingresa el código enviado a tu teléfono';
|
|
466
|
+
this.mfaVerifyFormProps.state = ComponentStates.ENABLED;
|
|
467
|
+
this.isMFAVerifyModalOpen = true;
|
|
468
|
+
this.onMFARequired.emit({ method: method });
|
|
469
|
+
}
|
|
470
|
+
closeMFAVerifyModal() {
|
|
471
|
+
this.isMFAVerifyModalOpen = false;
|
|
472
|
+
}
|
|
473
|
+
verifyMFAHandler(event) {
|
|
474
|
+
const code = event.fields['code'];
|
|
475
|
+
this.mfaVerifyFormProps.state = ComponentStates.WORKING;
|
|
476
|
+
this.authService.verifyMFA(code).subscribe({
|
|
477
|
+
next: () => {
|
|
478
|
+
this.mfaVerifyFormProps.state = ComponentStates.ENABLED;
|
|
479
|
+
this.closeMFAVerifyModal();
|
|
480
|
+
this.handleLoginSuccess(undefined, true);
|
|
481
|
+
},
|
|
482
|
+
error: (err) => {
|
|
483
|
+
this.mfaVerifyFormProps.state = ComponentStates.ENABLED;
|
|
484
|
+
this.handleError(err, 'mfa');
|
|
485
|
+
},
|
|
486
|
+
});
|
|
487
|
+
}
|
|
488
|
+
// ==========================================
|
|
489
|
+
// FORGOT PASSWORD HANDLERS
|
|
490
|
+
// ==========================================
|
|
491
|
+
openForgotPasswordModal() {
|
|
492
|
+
this.isForgotPasswordModalOpen = true;
|
|
493
|
+
}
|
|
494
|
+
closeForgotPasswordModal() {
|
|
495
|
+
this.isForgotPasswordModalOpen = false;
|
|
496
|
+
}
|
|
497
|
+
forgotPasswordHandler(event) {
|
|
498
|
+
const email = event.fields['email'];
|
|
499
|
+
if (!email) {
|
|
500
|
+
this.showToast('Ingresa tu correo electrónico.');
|
|
501
|
+
return;
|
|
502
|
+
}
|
|
503
|
+
this.forgotPasswordFormProps.state = ComponentStates.WORKING;
|
|
504
|
+
this.authService.forgotPassword({ email }).subscribe({
|
|
505
|
+
next: () => {
|
|
506
|
+
this.forgotPasswordFormProps.state = ComponentStates.ENABLED;
|
|
507
|
+
this.closeForgotPasswordModal();
|
|
508
|
+
this.openResetPasswordModal(email);
|
|
509
|
+
this.showToast('Código enviado. Revisa tu email.');
|
|
510
|
+
},
|
|
511
|
+
error: (err) => {
|
|
512
|
+
this.forgotPasswordFormProps.state = ComponentStates.ENABLED;
|
|
513
|
+
this.handleError(err, 'forgot');
|
|
514
|
+
},
|
|
515
|
+
});
|
|
516
|
+
}
|
|
517
|
+
// ==========================================
|
|
518
|
+
// RESET PASSWORD HANDLERS
|
|
519
|
+
// ==========================================
|
|
520
|
+
openResetPasswordModal(email) {
|
|
521
|
+
this.pendingResetEmail = email;
|
|
522
|
+
this.resetPasswordFormProps.sections[0].name = `Ingresa el código enviado a ${email}`;
|
|
523
|
+
this.resetPasswordFormProps.state = ComponentStates.ENABLED;
|
|
524
|
+
this.isResetPasswordModalOpen = true;
|
|
525
|
+
this.startResetResendCooldown();
|
|
526
|
+
}
|
|
527
|
+
closeResetPasswordModal() {
|
|
528
|
+
this.isResetPasswordModalOpen = false;
|
|
529
|
+
this.pendingResetEmail = '';
|
|
530
|
+
this.stopResetResendCooldown();
|
|
531
|
+
}
|
|
532
|
+
resetPasswordHandler(event) {
|
|
533
|
+
const code = event.fields['code'];
|
|
534
|
+
const newPassword = event.fields['newPassword'];
|
|
535
|
+
this.resetPasswordFormProps.state = ComponentStates.WORKING;
|
|
536
|
+
this.authService
|
|
537
|
+
.resetPassword({
|
|
538
|
+
email: this.pendingResetEmail,
|
|
539
|
+
code,
|
|
540
|
+
newPassword,
|
|
541
|
+
})
|
|
542
|
+
.subscribe({
|
|
543
|
+
next: () => {
|
|
544
|
+
this.resetPasswordFormProps.state = ComponentStates.ENABLED;
|
|
545
|
+
this.showToast('¡Contraseña actualizada! Ya puedes iniciar sesión.');
|
|
546
|
+
this.closeResetPasswordModal();
|
|
547
|
+
},
|
|
548
|
+
error: (err) => {
|
|
549
|
+
this.resetPasswordFormProps.state = ComponentStates.ENABLED;
|
|
550
|
+
this.handleError(err, 'reset');
|
|
551
|
+
},
|
|
552
|
+
});
|
|
553
|
+
}
|
|
554
|
+
resendResetCode() {
|
|
555
|
+
if (this.resetResendCooldown > 0)
|
|
556
|
+
return;
|
|
557
|
+
this.authService
|
|
558
|
+
.resendCode({
|
|
559
|
+
email: this.pendingResetEmail,
|
|
560
|
+
type: 'PASSWORD_RESET',
|
|
561
|
+
})
|
|
562
|
+
.subscribe({
|
|
563
|
+
next: () => {
|
|
564
|
+
this.showToast('Código reenviado. Revisa tu email.');
|
|
565
|
+
this.startResetResendCooldown();
|
|
566
|
+
},
|
|
567
|
+
error: (err) => {
|
|
568
|
+
this.handleError(err, 'reset');
|
|
569
|
+
},
|
|
570
|
+
});
|
|
571
|
+
}
|
|
572
|
+
// ==========================================
|
|
573
|
+
// HELPERS
|
|
574
|
+
// ==========================================
|
|
575
|
+
handleLoginSuccess(oauthProvider, mfaCompleted) {
|
|
576
|
+
this.showToast('¡Bienvenido!');
|
|
577
|
+
this.onSuccess.emit({
|
|
578
|
+
user: this.authService.user(),
|
|
579
|
+
mfaCompleted,
|
|
580
|
+
oauthProvider,
|
|
581
|
+
});
|
|
582
|
+
if (this.props.redirectOnSuccess) {
|
|
583
|
+
this.router.navigate([this.props.redirectOnSuccess]);
|
|
584
|
+
}
|
|
585
|
+
}
|
|
586
|
+
handleError(err, operation) {
|
|
587
|
+
const message = this.getErrorMessage(err);
|
|
588
|
+
this.showToast(message);
|
|
589
|
+
this.onError.emit({
|
|
590
|
+
code: err?.code,
|
|
591
|
+
message,
|
|
592
|
+
operation,
|
|
593
|
+
});
|
|
594
|
+
}
|
|
595
|
+
showToast(message) {
|
|
596
|
+
this.toastService.show({
|
|
597
|
+
message,
|
|
598
|
+
duration: 3500,
|
|
599
|
+
position: 'top',
|
|
600
|
+
color: 'dark',
|
|
601
|
+
});
|
|
602
|
+
}
|
|
603
|
+
getErrorMessage(err) {
|
|
604
|
+
const error = err;
|
|
605
|
+
const errorMessages = {
|
|
606
|
+
// Validation
|
|
607
|
+
VALIDATION_MISSING_REQUIRED_FIELDS: 'Faltan campos requeridos',
|
|
608
|
+
VALIDATION_INVALID_REQUEST: 'Solicitud inválida',
|
|
609
|
+
// Signin
|
|
610
|
+
AUTHV2_INVALID_CREDENTIALS: 'Correo o contraseña incorrectos',
|
|
611
|
+
AUTHV2_EMAIL_NOT_VERIFIED: 'Debes verificar tu correo electrónico',
|
|
612
|
+
AUTHV2_ACCOUNT_SUSPENDED: 'Tu cuenta ha sido suspendida',
|
|
613
|
+
AUTHV2_SIGNIN_FAILED: 'Error al iniciar sesión',
|
|
614
|
+
// Signup
|
|
615
|
+
AUTHV2_EMAIL_EXISTS: 'Este correo ya está registrado',
|
|
616
|
+
AUTHV2_PHONE_EXISTS: 'Este teléfono ya está registrado',
|
|
617
|
+
AUTHV2_WEAK_PASSWORD: 'La contraseña es muy débil',
|
|
618
|
+
AUTHV2_SIGNUP_FAILED: 'Error al crear la cuenta',
|
|
619
|
+
// Verification codes
|
|
620
|
+
AUTHV2_INVALID_CODE: 'Código incorrecto',
|
|
621
|
+
AUTHV2_EXPIRED_CODE: 'El código ha expirado. Solicita uno nuevo.',
|
|
622
|
+
AUTHV2_CODE_EXPIRED: 'El código ha expirado. Solicita uno nuevo.',
|
|
623
|
+
AUTHV2_CODE_ALREADY_USED: 'Este código ya fue utilizado',
|
|
624
|
+
AUTHV2_TOO_MANY_ATTEMPTS: 'Demasiados intentos, intenta más tarde',
|
|
625
|
+
AUTHV2_SEND_CODE_FAILED: 'Error al enviar el código',
|
|
626
|
+
AUTHV2_ALREADY_VERIFIED: 'Este email ya está verificado',
|
|
627
|
+
// MFA
|
|
628
|
+
AUTHV2_MFA_INVALID_CODE: 'Código de verificación incorrecto',
|
|
629
|
+
// Password reset
|
|
630
|
+
AUTHV2_PASSWORD_RESET_FAILED: 'Error al restablecer la contraseña',
|
|
631
|
+
AUTHV2_INVALID_CURRENT_PASSWORD: 'Contraseña actual incorrecta',
|
|
632
|
+
AUTHV2_SAME_PASSWORD: 'La nueva contraseña debe ser diferente',
|
|
633
|
+
// Session/Tokens
|
|
634
|
+
AUTHV2_INVALID_TOKEN: 'Sesión inválida',
|
|
635
|
+
AUTHV2_EXPIRED_TOKEN: 'Tu sesión ha expirado',
|
|
636
|
+
AUTHV2_SESSION_EXPIRED: 'Tu sesión ha expirado',
|
|
637
|
+
// User
|
|
638
|
+
AUTHV2_USER_NOT_FOUND: 'Usuario no encontrado',
|
|
639
|
+
// OAuth
|
|
640
|
+
POPUP_BLOCKED: 'Por favor, permite ventanas emergentes para este sitio',
|
|
641
|
+
POPUP_CLOSED: 'Se canceló la autenticación',
|
|
642
|
+
INVALID_RESPONSE: 'Error en la respuesta del servidor',
|
|
643
|
+
OAUTH_FAILED: 'Error de autenticación',
|
|
644
|
+
OAUTH_EMAIL_EXISTS: 'Este correo ya está registrado con otro método',
|
|
645
|
+
};
|
|
646
|
+
const errorCode = error?.error?.code || error?.code;
|
|
647
|
+
if (errorCode && errorMessages[errorCode]) {
|
|
648
|
+
return errorMessages[errorCode];
|
|
649
|
+
}
|
|
650
|
+
if (error?.error?.message) {
|
|
651
|
+
return error.error.message;
|
|
652
|
+
}
|
|
653
|
+
return 'Ha ocurrido un error. Intenta de nuevo.';
|
|
654
|
+
}
|
|
655
|
+
// ==========================================
|
|
656
|
+
// COOLDOWN TIMERS
|
|
657
|
+
// ==========================================
|
|
658
|
+
startResendCooldown() {
|
|
659
|
+
this.resendCooldown = 30;
|
|
660
|
+
this.resendTimer = setInterval(() => {
|
|
661
|
+
this.resendCooldown--;
|
|
662
|
+
if (this.resendCooldown <= 0) {
|
|
663
|
+
this.stopResendCooldown();
|
|
664
|
+
}
|
|
665
|
+
}, 1000);
|
|
666
|
+
}
|
|
667
|
+
stopResendCooldown() {
|
|
668
|
+
if (this.resendTimer) {
|
|
669
|
+
clearInterval(this.resendTimer);
|
|
670
|
+
this.resendTimer = null;
|
|
671
|
+
}
|
|
672
|
+
this.resendCooldown = 0;
|
|
673
|
+
}
|
|
674
|
+
startResetResendCooldown() {
|
|
675
|
+
this.resetResendCooldown = 30;
|
|
676
|
+
this.resetResendTimer = setInterval(() => {
|
|
677
|
+
this.resetResendCooldown--;
|
|
678
|
+
if (this.resetResendCooldown <= 0) {
|
|
679
|
+
this.stopResetResendCooldown();
|
|
680
|
+
}
|
|
681
|
+
}, 1000);
|
|
682
|
+
}
|
|
683
|
+
stopResetResendCooldown() {
|
|
684
|
+
if (this.resetResendTimer) {
|
|
685
|
+
clearInterval(this.resetResendTimer);
|
|
686
|
+
this.resetResendTimer = null;
|
|
687
|
+
}
|
|
688
|
+
this.resetResendCooldown = 0;
|
|
689
|
+
}
|
|
690
|
+
ngOnDestroy() {
|
|
691
|
+
this.stopResendCooldown();
|
|
692
|
+
this.stopResetResendCooldown();
|
|
693
|
+
}
|
|
694
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: LoginComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
695
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: LoginComponent, isStandalone: true, selector: "val-login", inputs: { props: "props" }, outputs: { onSuccess: "onSuccess", onError: "onError", onMFARequired: "onMFARequired" }, ngImport: i0, template: "<div class=\"val-login\">\n <!-- Logo -->\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n\n <!-- Login Form -->\n <val-form [props]=\"loginFormProps\" (onSubmit)=\"loginHandler($event)\" />\n\n <!-- OAuth Section -->\n @if (config.showOAuth && config.oauthProviders.length > 0) {\n <div class=\"oauth-separator\">\n <span>o contin\u00FAa con</span>\n </div>\n\n <div class=\"oauth-buttons\">\n @for (provider of config.oauthProviders; track provider) {\n <ion-button\n expand=\"block\"\n fill=\"outline\"\n color=\"dark\"\n (click)=\"loginWithOAuth(provider)\"\n [disabled]=\"isOAuthLoading\"\n >\n @switch (provider) {\n @case ('google') {\n <ion-icon slot=\"start\" name=\"logo-google\"></ion-icon>\n {{ isOAuthLoading ? 'Conectando...' : 'Continuar con Google' }}\n }\n @case ('apple') {\n <ion-icon slot=\"start\" name=\"logo-apple\"></ion-icon>\n {{ isOAuthLoading ? 'Conectando...' : 'Continuar con Apple' }}\n }\n @case ('microsoft') {\n <ion-icon slot=\"start\" name=\"logo-microsoft\"></ion-icon>\n {{ isOAuthLoading ? 'Conectando...' : 'Continuar con Microsoft' }}\n }\n }\n </ion-button>\n }\n </div>\n }\n\n <!-- Register Link -->\n @if (config.showRegister) {\n <div class=\"auth-link\">\n <ion-text color=\"dark\">\n \u00BFNo tienes cuenta?\n <a (click)=\"openRegisterModal()\">Registrarse</a>\n </ion-text>\n </div>\n }\n\n <!-- Forgot Password Link -->\n @if (config.showForgotPassword) {\n <div class=\"auth-link forgot-password\">\n <ion-text color=\"dark\">\n \u00BFOlvidaste tu contrase\u00F1a?\n <a (click)=\"openForgotPasswordModal()\">Recuperar contrase\u00F1a</a>\n </ion-text>\n </div>\n }\n\n <!-- Legal Notice -->\n @if (props.legal) {\n <div class=\"legal-notice\">\n <ion-text color=\"medium\">\n <p>\n Utilizamos los servicios de\n @if (props.legal.companyLink) {\n <a [href]=\"props.legal.companyLink\"><strong>{{ props.legal.companyName }}</strong></a>\n } @else {\n <strong>{{ props.legal.companyName }}</strong>\n }\n para ofrecerte una experiencia segura. Al iniciar sesi\u00F3n, aceptas\n nuestros\n @if (props.legal.termsLink) {\n <a [href]=\"props.legal.termsLink\">T\u00E9rminos y Condiciones</a>\n } @else {\n <span>T\u00E9rminos y Condiciones</span>\n }\n y\n @if (props.legal.privacyLink) {\n <a [href]=\"props.legal.privacyLink\">Pol\u00EDtica de Privacidad</a>\n } @else {\n <span>Pol\u00EDtica de Privacidad</span>\n }.\n </p>\n </ion-text>\n </div>\n }\n</div>\n\n<!-- Register Modal -->\n<ion-modal [isOpen]=\"isRegisterModalOpen\" (didDismiss)=\"closeRegisterModal()\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeRegisterModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"registerFormProps\" (onSubmit)=\"registerHandler($event)\" />\n <div class=\"auth-link\">\n <ion-text color=\"dark\">\n \u00BFYa tienes cuenta?\n <a (click)=\"closeRegisterModal()\">Iniciar sesi\u00F3n</a>\n </ion-text>\n </div>\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n\n<!-- Verify Email Modal -->\n<ion-modal [isOpen]=\"isVerifyModalOpen\" [backdropDismiss]=\"false\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeVerifyModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"verifyFormProps\" (onSubmit)=\"verifyHandler($event)\" />\n <div class=\"resend-link\">\n <ion-text color=\"dark\">\n \u00BFNo has recibido tu c\u00F3digo?\n @if (resendCooldown > 0) {\n <span class=\"cooldown\">Reenviar en {{ resendCooldown }}s</span>\n } @else {\n <a (click)=\"resendCode()\">Reenviar</a>\n }\n </ion-text>\n </div>\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n\n<!-- Forgot Password Modal -->\n<ion-modal [isOpen]=\"isForgotPasswordModalOpen\" (didDismiss)=\"closeForgotPasswordModal()\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeForgotPasswordModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"forgotPasswordFormProps\" (onSubmit)=\"forgotPasswordHandler($event)\" />\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n\n<!-- Reset Password Modal -->\n<ion-modal [isOpen]=\"isResetPasswordModalOpen\" [backdropDismiss]=\"false\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeResetPasswordModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"resetPasswordFormProps\" (onSubmit)=\"resetPasswordHandler($event)\" />\n <div class=\"resend-link\">\n <ion-text color=\"dark\">\n \u00BFNo has recibido tu c\u00F3digo?\n @if (resetResendCooldown > 0) {\n <span class=\"cooldown\">Reenviar en {{ resetResendCooldown }}s</span>\n } @else {\n <a (click)=\"resendResetCode()\">Reenviar</a>\n }\n </ion-text>\n </div>\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n\n<!-- MFA Verify Modal -->\n<ion-modal [isOpen]=\"isMFAVerifyModalOpen\" [backdropDismiss]=\"false\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeMFAVerifyModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"mfaVerifyFormProps\" (onSubmit)=\"verifyMFAHandler($event)\" />\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n", styles: [".val-login{width:100%}.logo-container{max-width:130px;margin-bottom:1.5rem}.auth-link{text-align:center;margin-top:1rem;font-size:.9rem}.auth-link a{color:var(--ion-color-primary);text-decoration:none;font-weight:500;cursor:pointer}.auth-link a:hover{text-decoration:underline}.oauth-separator{display:flex;align-items:center;margin:1.5rem 0}.oauth-separator:before,.oauth-separator:after{content:\"\";flex:1;height:1px;background:var(--ion-color-medium-tint)}.oauth-separator span{padding:0 1rem;color:var(--ion-color-dark);font-size:.85rem}.oauth-buttons{margin-bottom:1rem}.oauth-buttons ion-button{--border-radius: 8px;--border-width: 1px}.oauth-buttons ion-icon{font-size:1.2rem;margin-right:.5rem}.legal-notice{margin-top:1.5rem;padding-top:1rem;border-top:1px solid var(--ion-color-medium-tint)}.legal-notice p{font-size:.75rem;line-height:1.5;text-align:center;margin:0}.legal-notice a{color:var(--ion-color-primary);text-decoration:none}.legal-notice a:hover{text-decoration:underline}.modal-form-section{padding:1rem}.resend-link{text-align:center;margin-top:1rem;font-size:.9rem}.resend-link a{color:var(--ion-color-primary);cursor:pointer;font-weight:500}.resend-link a:hover{text-decoration:underline}.resend-link .cooldown{color:var(--ion-color-medium)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: IonButtons, selector: "ion-buttons", inputs: ["collapse"] }, { kind: "component", type: IonContent, selector: "ion-content", inputs: ["color", "fixedSlotPlacement", "forceOverscroll", "fullscreen", "scrollEvents", "scrollX", "scrollY"] }, { kind: "component", type: IonHeader, selector: "ion-header", inputs: ["collapse", "mode", "translucent"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: IonModal, selector: "ion-modal" }, { kind: "component", type: IonText, selector: "ion-text", inputs: ["color", "mode"] }, { kind: "component", type: IonToolbar, selector: "ion-toolbar", inputs: ["color", "mode"] }, { kind: "component", type: FormComponent, selector: "val-form", inputs: ["props"], outputs: ["onSubmit", "onInvalid", "onSelectChange"] }, { kind: "component", type: ImageComponent, selector: "val-image", inputs: ["props"] }] }); }
|
|
696
|
+
}
|
|
697
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: LoginComponent, decorators: [{
|
|
698
|
+
type: Component,
|
|
699
|
+
args: [{ selector: 'val-login', standalone: true, imports: [
|
|
700
|
+
CommonModule,
|
|
701
|
+
IonButton,
|
|
702
|
+
IonButtons,
|
|
703
|
+
IonContent,
|
|
704
|
+
IonHeader,
|
|
705
|
+
IonIcon,
|
|
706
|
+
IonModal,
|
|
707
|
+
IonText,
|
|
708
|
+
IonToolbar,
|
|
709
|
+
FormComponent,
|
|
710
|
+
ImageComponent,
|
|
711
|
+
], template: "<div class=\"val-login\">\n <!-- Logo -->\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n\n <!-- Login Form -->\n <val-form [props]=\"loginFormProps\" (onSubmit)=\"loginHandler($event)\" />\n\n <!-- OAuth Section -->\n @if (config.showOAuth && config.oauthProviders.length > 0) {\n <div class=\"oauth-separator\">\n <span>o contin\u00FAa con</span>\n </div>\n\n <div class=\"oauth-buttons\">\n @for (provider of config.oauthProviders; track provider) {\n <ion-button\n expand=\"block\"\n fill=\"outline\"\n color=\"dark\"\n (click)=\"loginWithOAuth(provider)\"\n [disabled]=\"isOAuthLoading\"\n >\n @switch (provider) {\n @case ('google') {\n <ion-icon slot=\"start\" name=\"logo-google\"></ion-icon>\n {{ isOAuthLoading ? 'Conectando...' : 'Continuar con Google' }}\n }\n @case ('apple') {\n <ion-icon slot=\"start\" name=\"logo-apple\"></ion-icon>\n {{ isOAuthLoading ? 'Conectando...' : 'Continuar con Apple' }}\n }\n @case ('microsoft') {\n <ion-icon slot=\"start\" name=\"logo-microsoft\"></ion-icon>\n {{ isOAuthLoading ? 'Conectando...' : 'Continuar con Microsoft' }}\n }\n }\n </ion-button>\n }\n </div>\n }\n\n <!-- Register Link -->\n @if (config.showRegister) {\n <div class=\"auth-link\">\n <ion-text color=\"dark\">\n \u00BFNo tienes cuenta?\n <a (click)=\"openRegisterModal()\">Registrarse</a>\n </ion-text>\n </div>\n }\n\n <!-- Forgot Password Link -->\n @if (config.showForgotPassword) {\n <div class=\"auth-link forgot-password\">\n <ion-text color=\"dark\">\n \u00BFOlvidaste tu contrase\u00F1a?\n <a (click)=\"openForgotPasswordModal()\">Recuperar contrase\u00F1a</a>\n </ion-text>\n </div>\n }\n\n <!-- Legal Notice -->\n @if (props.legal) {\n <div class=\"legal-notice\">\n <ion-text color=\"medium\">\n <p>\n Utilizamos los servicios de\n @if (props.legal.companyLink) {\n <a [href]=\"props.legal.companyLink\"><strong>{{ props.legal.companyName }}</strong></a>\n } @else {\n <strong>{{ props.legal.companyName }}</strong>\n }\n para ofrecerte una experiencia segura. Al iniciar sesi\u00F3n, aceptas\n nuestros\n @if (props.legal.termsLink) {\n <a [href]=\"props.legal.termsLink\">T\u00E9rminos y Condiciones</a>\n } @else {\n <span>T\u00E9rminos y Condiciones</span>\n }\n y\n @if (props.legal.privacyLink) {\n <a [href]=\"props.legal.privacyLink\">Pol\u00EDtica de Privacidad</a>\n } @else {\n <span>Pol\u00EDtica de Privacidad</span>\n }.\n </p>\n </ion-text>\n </div>\n }\n</div>\n\n<!-- Register Modal -->\n<ion-modal [isOpen]=\"isRegisterModalOpen\" (didDismiss)=\"closeRegisterModal()\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeRegisterModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"registerFormProps\" (onSubmit)=\"registerHandler($event)\" />\n <div class=\"auth-link\">\n <ion-text color=\"dark\">\n \u00BFYa tienes cuenta?\n <a (click)=\"closeRegisterModal()\">Iniciar sesi\u00F3n</a>\n </ion-text>\n </div>\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n\n<!-- Verify Email Modal -->\n<ion-modal [isOpen]=\"isVerifyModalOpen\" [backdropDismiss]=\"false\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeVerifyModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"verifyFormProps\" (onSubmit)=\"verifyHandler($event)\" />\n <div class=\"resend-link\">\n <ion-text color=\"dark\">\n \u00BFNo has recibido tu c\u00F3digo?\n @if (resendCooldown > 0) {\n <span class=\"cooldown\">Reenviar en {{ resendCooldown }}s</span>\n } @else {\n <a (click)=\"resendCode()\">Reenviar</a>\n }\n </ion-text>\n </div>\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n\n<!-- Forgot Password Modal -->\n<ion-modal [isOpen]=\"isForgotPasswordModalOpen\" (didDismiss)=\"closeForgotPasswordModal()\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeForgotPasswordModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"forgotPasswordFormProps\" (onSubmit)=\"forgotPasswordHandler($event)\" />\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n\n<!-- Reset Password Modal -->\n<ion-modal [isOpen]=\"isResetPasswordModalOpen\" [backdropDismiss]=\"false\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeResetPasswordModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"resetPasswordFormProps\" (onSubmit)=\"resetPasswordHandler($event)\" />\n <div class=\"resend-link\">\n <ion-text color=\"dark\">\n \u00BFNo has recibido tu c\u00F3digo?\n @if (resetResendCooldown > 0) {\n <span class=\"cooldown\">Reenviar en {{ resetResendCooldown }}s</span>\n } @else {\n <a (click)=\"resendResetCode()\">Reenviar</a>\n }\n </ion-text>\n </div>\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n\n<!-- MFA Verify Modal -->\n<ion-modal [isOpen]=\"isMFAVerifyModalOpen\" [backdropDismiss]=\"false\">\n <ng-template>\n <ion-header>\n <ion-toolbar>\n <ion-buttons slot=\"end\">\n <ion-button fill=\"clear\" (click)=\"closeMFAVerifyModal()\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </ion-buttons>\n </ion-toolbar>\n </ion-header>\n <ion-content class=\"ion-padding\">\n <section class=\"modal-form-section\">\n @if (props.logo) {\n <div class=\"logo-container\">\n <val-image [props]=\"props.logo\" />\n </div>\n }\n <val-form [props]=\"mfaVerifyFormProps\" (onSubmit)=\"verifyMFAHandler($event)\" />\n </section>\n </ion-content>\n </ng-template>\n</ion-modal>\n", styles: [".val-login{width:100%}.logo-container{max-width:130px;margin-bottom:1.5rem}.auth-link{text-align:center;margin-top:1rem;font-size:.9rem}.auth-link a{color:var(--ion-color-primary);text-decoration:none;font-weight:500;cursor:pointer}.auth-link a:hover{text-decoration:underline}.oauth-separator{display:flex;align-items:center;margin:1.5rem 0}.oauth-separator:before,.oauth-separator:after{content:\"\";flex:1;height:1px;background:var(--ion-color-medium-tint)}.oauth-separator span{padding:0 1rem;color:var(--ion-color-dark);font-size:.85rem}.oauth-buttons{margin-bottom:1rem}.oauth-buttons ion-button{--border-radius: 8px;--border-width: 1px}.oauth-buttons ion-icon{font-size:1.2rem;margin-right:.5rem}.legal-notice{margin-top:1.5rem;padding-top:1rem;border-top:1px solid var(--ion-color-medium-tint)}.legal-notice p{font-size:.75rem;line-height:1.5;text-align:center;margin:0}.legal-notice a{color:var(--ion-color-primary);text-decoration:none}.legal-notice a:hover{text-decoration:underline}.modal-form-section{padding:1rem}.resend-link{text-align:center;margin-top:1rem;font-size:.9rem}.resend-link a{color:var(--ion-color-primary);cursor:pointer;font-weight:500}.resend-link a:hover{text-decoration:underline}.resend-link .cooldown{color:var(--ion-color-medium)}\n"] }]
|
|
712
|
+
}], propDecorators: { props: [{
|
|
713
|
+
type: Input
|
|
714
|
+
}], onSuccess: [{
|
|
715
|
+
type: Output
|
|
716
|
+
}], onError: [{
|
|
717
|
+
type: Output
|
|
718
|
+
}], onMFARequired: [{
|
|
719
|
+
type: Output
|
|
720
|
+
}] } });
|
|
721
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"login.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/login/login.component.ts","../../../../../../../src/lib/components/organisms/login/login.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  inject,\n  Input,\n  OnDestroy,\n  Output,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { Validators } from '@angular/forms';\nimport { Router } from '@angular/router';\nimport {\n  IonButton,\n  IonButtons,\n  IonContent,\n  IonHeader,\n  IonIcon,\n  IonModal,\n  IonText,\n  IonToolbar,\n} from '@ionic/angular/standalone';\n\nimport { AuthService } from '../../../services/auth';\nimport { ToastService } from '../../../services/toast.service';\nimport { FormComponent } from '../form/form.component';\nimport { ImageComponent } from '../../atoms/image/image.component';\nimport {\n  FormMetadata,\n  FormSubmit,\n  InputMetadata,\n  InputType,\n  ComponentStates,\n} from '../../types';\nimport { SolidDefaultBlock } from '../../atoms/button/factory';\nimport {\n  LoginMetadata,\n  LoginSuccessEvent,\n  LoginErrorEvent,\n  MFARequiredEvent,\n  LOGIN_DEFAULTS,\n} from './types';\n\n@Component({\n  selector: 'val-login',\n  standalone: true,\n  imports: [\n    CommonModule,\n    IonButton,\n    IonButtons,\n    IonContent,\n    IonHeader,\n    IonIcon,\n    IonModal,\n    IonText,\n    IonToolbar,\n    FormComponent,\n    ImageComponent,\n  ],\n  templateUrl: './login.component.html',\n  styleUrls: ['./login.component.scss'],\n})\nexport class LoginComponent implements OnDestroy {\n  @Input() props: LoginMetadata = {};\n\n  @Output() onSuccess = new EventEmitter<LoginSuccessEvent>();\n  @Output() onError = new EventEmitter<LoginErrorEvent>();\n  @Output() onMFARequired = new EventEmitter<MFARequiredEvent>();\n\n  // Services\n  authService = inject(AuthService);\n  private toastService = inject(ToastService);\n  private router = inject(Router);\n\n  // Timers\n  private resendTimer: ReturnType<typeof setInterval> | null = null;\n  private resetResendTimer: ReturnType<typeof setInterval> | null = null;\n\n  // State\n  isOAuthLoading = false;\n  isRegisterModalOpen = false;\n  isVerifyModalOpen = false;\n  isForgotPasswordModalOpen = false;\n  isResetPasswordModalOpen = false;\n  isMFAVerifyModalOpen = false;\n\n  pendingVerificationEmail = '';\n  pendingResetEmail = '';\n  resendCooldown = 0;\n  resetResendCooldown = 0;\n\n  // Resolved props with defaults\n  get config(): Required<Omit<LoginMetadata, 'logo' | 'legal' | 'redirectOnSuccess'>> & LoginMetadata {\n    return { ...LOGIN_DEFAULTS, ...this.props };\n  }\n\n  // ==========================================\n  // LOGIN FORM\n  // ==========================================\n\n  emailInput: InputMetadata = {\n    type: InputType.EMAIL,\n    label: 'Correo electrónico',\n    name: 'email',\n    token: 'login-email',\n    hint: '',\n    placeholder: 'tu@email.com',\n    errors: {\n      required: 'El correo es requerido',\n      email: 'Ingresa un correo válido',\n    },\n    validators: [Validators.required, Validators.email],\n    order: 0,\n    state: ComponentStates.ENABLED,\n    control: undefined,\n  };\n\n  passwordInput: InputMetadata = {\n    type: InputType.PASSWORD,\n    label: 'Contraseña',\n    name: 'password',\n    token: 'login-password',\n    hint: '',\n    placeholder: '••••••••',\n    errors: {\n      required: 'La contraseña es requerida',\n    },\n    validators: [Validators.required],\n    order: 1,\n    state: ComponentStates.ENABLED,\n    control: undefined,\n  };\n\n  loginFormProps: FormMetadata = {\n    name: 'Iniciar sesión',\n    sections: [\n      {\n        name: '',\n        order: 0,\n        fields: [this.emailInput, this.passwordInput],\n      },\n    ],\n    actions: {\n      ...SolidDefaultBlock('Iniciar sesión', 'submit'),\n      token: 'login-submit',\n    },\n    state: ComponentStates.ENABLED,\n  };\n\n  // ==========================================\n  // REGISTER FORM\n  // ==========================================\n\n  nameInput: InputMetadata = {\n    type: InputType.TEXT,\n    label: 'Nombre completo',\n    name: 'name',\n    token: 'register-name',\n    hint: '',\n    placeholder: 'Tu nombre',\n    errors: {\n      required: 'El nombre es requerido',\n      minlength: 'Mínimo 2 caracteres',\n    },\n    validators: [Validators.required, Validators.minLength(2)],\n    order: 0,\n    state: ComponentStates.ENABLED,\n    control: undefined,\n  };\n\n  registerEmailInput: InputMetadata = {\n    type: InputType.EMAIL,\n    label: 'Correo electrónico',\n    name: 'email',\n    token: 'register-email',\n    hint: '',\n    placeholder: 'tu@email.com',\n    errors: {\n      required: 'El correo es requerido',\n      email: 'Ingresa un correo válido',\n    },\n    validators: [Validators.required, Validators.email],\n    order: 1,\n    state: ComponentStates.ENABLED,\n    control: undefined,\n  };\n\n  registerPasswordInput: InputMetadata = {\n    type: InputType.PASSWORD,\n    label: 'Contraseña',\n    name: 'password',\n    token: 'register-password',\n    hint: '',\n    placeholder: '••••••••',\n    errors: {\n      required: 'La contraseña es requerida',\n      minlength: 'Mínimo 8 caracteres',\n    },\n    validators: [Validators.required, Validators.minLength(8)],\n    order: 2,\n    state: ComponentStates.ENABLED,\n    control: undefined,\n  };\n\n  registerFormProps: FormMetadata = {\n    name: 'Crear cuenta',\n    sections: [\n      {\n        name: '',\n        order: 0,\n        fields: [this.nameInput, this.registerEmailInput, this.registerPasswordInput],\n      },\n    ],\n    actions: {\n      ...SolidDefaultBlock('Registrarse', 'submit'),\n      token: 'register-submit',\n    },\n    state: ComponentStates.ENABLED,\n  };\n\n  // ==========================================\n  // VERIFY EMAIL FORM\n  // ==========================================\n\n  verifyPinInput: InputMetadata = {\n    type: InputType.PIN_CODE,\n    label: '',\n    name: 'code',\n    token: 'verify-pin',\n    hint: '',\n    placeholder: '',\n    errors: {\n      required: 'El código es requerido',\n      minlength: 'Ingresa los 6 dígitos',\n    },\n    validators: [Validators.required, Validators.minLength(6)],\n    order: 0,\n    state: ComponentStates.ENABLED,\n    control: undefined,\n    length: 6,\n    allowNumbersOnly: true,\n    autoFocus: true,\n  };\n\n  verifyFormProps: FormMetadata = {\n    name: 'Verificar correo',\n    sections: [\n      {\n        name: 'Ingresa el código de verificación enviado a tu correo.',\n        order: 0,\n        fields: [this.verifyPinInput],\n      },\n    ],\n    actions: {\n      ...SolidDefaultBlock('Verificar', 'submit'),\n      token: 'verify-submit',\n    },\n    state: ComponentStates.ENABLED,\n  };\n\n  // ==========================================\n  // MFA VERIFY FORM\n  // ==========================================\n\n  mfaPinInput: InputMetadata = {\n    type: InputType.PIN_CODE,\n    label: '',\n    name: 'code',\n    token: 'mfa-pin',\n    hint: '',\n    placeholder: '',\n    errors: {\n      required: 'El código es requerido',\n      minlength: 'Ingresa los 6 dígitos',\n    },\n    validators: [Validators.required, Validators.minLength(6)],\n    order: 0,\n    state: ComponentStates.ENABLED,\n    control: undefined,\n    length: 6,\n    allowNumbersOnly: true,\n    autoFocus: true,\n  };\n\n  mfaVerifyFormProps: FormMetadata = {\n    name: 'Verificación MFA',\n    sections: [\n      {\n        name: '',\n        order: 0,\n        fields: [this.mfaPinInput],\n      },\n    ],\n    actions: {\n      ...SolidDefaultBlock('Verificar', 'submit'),\n      token: 'mfa-verify-submit',\n    },\n    state: ComponentStates.ENABLED,\n  };\n\n  // ==========================================\n  // FORGOT PASSWORD FORM\n  // ==========================================\n\n  forgotEmailInput: InputMetadata = {\n    type: InputType.EMAIL,\n    label: '',\n    name: 'email',\n    token: 'forgot-email',\n    hint: '',\n    placeholder: 'tu@email.com',\n    errors: {\n      required: 'El correo es requerido',\n      email: 'Ingresa un correo válido',\n    },\n    validators: [Validators.required, Validators.email],\n    order: 0,\n    state: ComponentStates.ENABLED,\n    control: undefined,\n  };\n\n  forgotPasswordFormProps: FormMetadata = {\n    name: 'Recuperar contraseña',\n    sections: [\n      {\n        name: 'Ingresa tu correo electrónico y te enviaremos un código para restablecer tu contraseña.',\n        order: 0,\n        fields: [this.forgotEmailInput],\n      },\n    ],\n    actions: {\n      ...SolidDefaultBlock('Enviar código', 'submit'),\n      token: 'forgot-submit',\n    },\n    state: ComponentStates.ENABLED,\n  };\n\n  // ==========================================\n  // RESET PASSWORD FORM\n  // ==========================================\n\n  resetPinInput: InputMetadata = {\n    type: InputType.PIN_CODE,\n    label: '',\n    name: 'code',\n    token: 'reset-pin',\n    hint: '',\n    placeholder: '',\n    errors: {\n      required: 'El código es requerido',\n      minlength: 'Ingresa los 6 dígitos',\n    },\n    validators: [Validators.required, Validators.minLength(6)],\n    order: 0,\n    state: ComponentStates.ENABLED,\n    control: undefined,\n    length: 6,\n    allowNumbersOnly: true,\n    autoFocus: true,\n  };\n\n  newPasswordInput: InputMetadata = {\n    type: InputType.PASSWORD,\n    label: 'Nueva contraseña',\n    name: 'newPassword',\n    token: 'reset-new-password',\n    hint: 'Mínimo 8 caracteres',\n    placeholder: '••••••••',\n    errors: {\n      required: 'La contraseña es requerida',\n      minlength: 'Mínimo 8 caracteres',\n    },\n    validators: [Validators.required, Validators.minLength(8)],\n    order: 1,\n    state: ComponentStates.ENABLED,\n    control: undefined,\n  };\n\n  resetPasswordFormProps: FormMetadata = {\n    name: 'Restablecer contraseña',\n    sections: [\n      {\n        name: 'Hemos enviado un código de verificación a tu correo.',\n        order: 0,\n        fields: [this.resetPinInput, this.newPasswordInput],\n      },\n    ],\n    actions: {\n      ...SolidDefaultBlock('Cambiar contraseña', 'submit'),\n      token: 'reset-submit',\n    },\n    state: ComponentStates.ENABLED,\n  };\n\n  // ==========================================\n  // HANDLERS\n  // ==========================================\n\n  loginHandler(event: FormSubmit): void {\n    const email = event.fields['email'];\n    const password = event.fields['password'];\n\n    if (!email || !password) {\n      this.showToast('Completa todos los campos.');\n      return;\n    }\n\n    this.loginFormProps.state = ComponentStates.WORKING;\n\n    this.authService.signin({ email, password }).subscribe({\n      next: () => {\n        this.loginFormProps.state = ComponentStates.ENABLED;\n\n        if (this.authService.mfaPending().required) {\n          this.openMFAVerifyModal();\n          return;\n        }\n\n        this.handleLoginSuccess();\n      },\n      error: (err) => {\n        this.loginFormProps.state = ComponentStates.ENABLED;\n\n        const errorCode = (err as { code?: string })?.code;\n        if (errorCode === 'AUTHV2_EMAIL_NOT_VERIFIED') {\n          this.openVerifyModal(email);\n          return;\n        }\n\n        this.handleError(err, 'signin');\n      },\n    });\n  }\n\n  loginWithOAuth(provider: 'google' | 'apple' | 'microsoft'): void {\n    this.isOAuthLoading = true;\n\n    this.authService.signinWithOAuth(provider).subscribe({\n      next: () => {\n        this.isOAuthLoading = false;\n\n        if (this.authService.mfaPending().required) {\n          this.openMFAVerifyModal();\n          return;\n        }\n\n        this.handleLoginSuccess(provider);\n      },\n      error: (err) => {\n        this.isOAuthLoading = false;\n\n        const errorCode = (err as { code?: string })?.code;\n        if (errorCode === 'POPUP_CLOSED') {\n          return;\n        }\n\n        this.handleError(err, 'oauth');\n      },\n    });\n  }\n\n  // ==========================================\n  // REGISTER HANDLERS\n  // ==========================================\n\n  openRegisterModal(): void {\n    this.isRegisterModalOpen = true;\n  }\n\n  closeRegisterModal(): void {\n    this.isRegisterModalOpen = false;\n  }\n\n  registerHandler(event: FormSubmit): void {\n    const name = event.fields['name'];\n    const email = event.fields['email'];\n    const password = event.fields['password'];\n\n    if (!name || !email || !password) {\n      this.showToast('Completa todos los campos.');\n      return;\n    }\n\n    this.registerFormProps.state = ComponentStates.WORKING;\n\n    this.authService.signup({ name, email, password }).subscribe({\n      next: () => {\n        this.registerFormProps.state = ComponentStates.ENABLED;\n        this.closeRegisterModal();\n        this.openVerifyModal(email);\n      },\n      error: (err) => {\n        this.registerFormProps.state = ComponentStates.ENABLED;\n        this.handleError(err, 'signup');\n      },\n    });\n  }\n\n  // ==========================================\n  // VERIFY EMAIL HANDLERS\n  // ==========================================\n\n  openVerifyModal(email: string): void {\n    this.pendingVerificationEmail = email;\n    this.verifyFormProps.sections[0].name = `Ingresa el código enviado a ${email}`;\n    this.verifyFormProps.state = ComponentStates.ENABLED;\n    this.isVerifyModalOpen = true;\n    this.startResendCooldown();\n  }\n\n  closeVerifyModal(): void {\n    this.isVerifyModalOpen = false;\n    this.pendingVerificationEmail = '';\n    this.stopResendCooldown();\n  }\n\n  verifyHandler(event: FormSubmit): void {\n    const code = event.fields['code'];\n\n    this.verifyFormProps.state = ComponentStates.WORKING;\n\n    this.authService\n      .verifyEmail({\n        email: this.pendingVerificationEmail,\n        code,\n      })\n      .subscribe({\n        next: () => {\n          this.verifyFormProps.state = ComponentStates.ENABLED;\n          this.showToast('¡Email verificado! Bienvenido.');\n          this.closeVerifyModal();\n          this.handleLoginSuccess();\n        },\n        error: (err) => {\n          this.verifyFormProps.state = ComponentStates.ENABLED;\n          this.handleError(err, 'verify');\n        },\n      });\n  }\n\n  resendCode(): void {\n    if (this.resendCooldown > 0) return;\n\n    this.authService\n      .resendCode({\n        email: this.pendingVerificationEmail,\n        type: 'EMAIL_VERIFY',\n      })\n      .subscribe({\n        next: () => {\n          this.showToast('Código reenviado. Revisa tu email.');\n          this.startResendCooldown();\n        },\n        error: (err) => {\n          this.handleError(err, 'verify');\n        },\n      });\n  }\n\n  // ==========================================\n  // MFA HANDLERS\n  // ==========================================\n\n  openMFAVerifyModal(): void {\n    const method = this.authService.mfaPending().method;\n    this.mfaVerifyFormProps.sections[0].name =\n      method === 'TOTP'\n        ? 'Ingresa el código de tu app de autenticación'\n        : method === 'EMAIL'\n          ? 'Ingresa el código enviado a tu correo'\n          : 'Ingresa el código enviado a tu teléfono';\n    this.mfaVerifyFormProps.state = ComponentStates.ENABLED;\n    this.isMFAVerifyModalOpen = true;\n\n    this.onMFARequired.emit({ method: method as 'TOTP' | 'EMAIL' | 'SMS' });\n  }\n\n  closeMFAVerifyModal(): void {\n    this.isMFAVerifyModalOpen = false;\n  }\n\n  verifyMFAHandler(event: FormSubmit): void {\n    const code = event.fields['code'];\n\n    this.mfaVerifyFormProps.state = ComponentStates.WORKING;\n\n    this.authService.verifyMFA(code).subscribe({\n      next: () => {\n        this.mfaVerifyFormProps.state = ComponentStates.ENABLED;\n        this.closeMFAVerifyModal();\n        this.handleLoginSuccess(undefined, true);\n      },\n      error: (err) => {\n        this.mfaVerifyFormProps.state = ComponentStates.ENABLED;\n        this.handleError(err, 'mfa');\n      },\n    });\n  }\n\n  // ==========================================\n  // FORGOT PASSWORD HANDLERS\n  // ==========================================\n\n  openForgotPasswordModal(): void {\n    this.isForgotPasswordModalOpen = true;\n  }\n\n  closeForgotPasswordModal(): void {\n    this.isForgotPasswordModalOpen = false;\n  }\n\n  forgotPasswordHandler(event: FormSubmit): void {\n    const email = event.fields['email'];\n\n    if (!email) {\n      this.showToast('Ingresa tu correo electrónico.');\n      return;\n    }\n\n    this.forgotPasswordFormProps.state = ComponentStates.WORKING;\n\n    this.authService.forgotPassword({ email }).subscribe({\n      next: () => {\n        this.forgotPasswordFormProps.state = ComponentStates.ENABLED;\n        this.closeForgotPasswordModal();\n        this.openResetPasswordModal(email);\n        this.showToast('Código enviado. Revisa tu email.');\n      },\n      error: (err) => {\n        this.forgotPasswordFormProps.state = ComponentStates.ENABLED;\n        this.handleError(err, 'forgot');\n      },\n    });\n  }\n\n  // ==========================================\n  // RESET PASSWORD HANDLERS\n  // ==========================================\n\n  openResetPasswordModal(email: string): void {\n    this.pendingResetEmail = email;\n    this.resetPasswordFormProps.sections[0].name = `Ingresa el código enviado a ${email}`;\n    this.resetPasswordFormProps.state = ComponentStates.ENABLED;\n    this.isResetPasswordModalOpen = true;\n    this.startResetResendCooldown();\n  }\n\n  closeResetPasswordModal(): void {\n    this.isResetPasswordModalOpen = false;\n    this.pendingResetEmail = '';\n    this.stopResetResendCooldown();\n  }\n\n  resetPasswordHandler(event: FormSubmit): void {\n    const code = event.fields['code'];\n    const newPassword = event.fields['newPassword'];\n\n    this.resetPasswordFormProps.state = ComponentStates.WORKING;\n\n    this.authService\n      .resetPassword({\n        email: this.pendingResetEmail,\n        code,\n        newPassword,\n      })\n      .subscribe({\n        next: () => {\n          this.resetPasswordFormProps.state = ComponentStates.ENABLED;\n          this.showToast('¡Contraseña actualizada! Ya puedes iniciar sesión.');\n          this.closeResetPasswordModal();\n        },\n        error: (err) => {\n          this.resetPasswordFormProps.state = ComponentStates.ENABLED;\n          this.handleError(err, 'reset');\n        },\n      });\n  }\n\n  resendResetCode(): void {\n    if (this.resetResendCooldown > 0) return;\n\n    this.authService\n      .resendCode({\n        email: this.pendingResetEmail,\n        type: 'PASSWORD_RESET',\n      })\n      .subscribe({\n        next: () => {\n          this.showToast('Código reenviado. Revisa tu email.');\n          this.startResetResendCooldown();\n        },\n        error: (err) => {\n          this.handleError(err, 'reset');\n        },\n      });\n  }\n\n  // ==========================================\n  // HELPERS\n  // ==========================================\n\n  private handleLoginSuccess(oauthProvider?: 'google' | 'apple' | 'microsoft', mfaCompleted?: boolean): void {\n    this.showToast('¡Bienvenido!');\n\n    this.onSuccess.emit({\n      user: this.authService.user(),\n      mfaCompleted,\n      oauthProvider,\n    });\n\n    if (this.props.redirectOnSuccess) {\n      this.router.navigate([this.props.redirectOnSuccess]);\n    }\n  }\n\n  private handleError(err: unknown, operation: LoginErrorEvent['operation']): void {\n    const message = this.getErrorMessage(err);\n    this.showToast(message);\n\n    this.onError.emit({\n      code: (err as { code?: string })?.code,\n      message,\n      operation,\n    });\n  }\n\n  private showToast(message: string): void {\n    this.toastService.show({\n      message,\n      duration: 3500,\n      position: 'top',\n      color: 'dark',\n    });\n  }\n\n  private getErrorMessage(err: unknown): string {\n    const error = err as {\n      error?: { message?: string; code?: string };\n      code?: string;\n    };\n\n    const errorMessages: Record<string, string> = {\n      // Validation\n      VALIDATION_MISSING_REQUIRED_FIELDS: 'Faltan campos requeridos',\n      VALIDATION_INVALID_REQUEST: 'Solicitud inválida',\n\n      // Signin\n      AUTHV2_INVALID_CREDENTIALS: 'Correo o contraseña incorrectos',\n      AUTHV2_EMAIL_NOT_VERIFIED: 'Debes verificar tu correo electrónico',\n      AUTHV2_ACCOUNT_SUSPENDED: 'Tu cuenta ha sido suspendida',\n      AUTHV2_SIGNIN_FAILED: 'Error al iniciar sesión',\n\n      // Signup\n      AUTHV2_EMAIL_EXISTS: 'Este correo ya está registrado',\n      AUTHV2_PHONE_EXISTS: 'Este teléfono ya está registrado',\n      AUTHV2_WEAK_PASSWORD: 'La contraseña es muy débil',\n      AUTHV2_SIGNUP_FAILED: 'Error al crear la cuenta',\n\n      // Verification codes\n      AUTHV2_INVALID_CODE: 'Código incorrecto',\n      AUTHV2_EXPIRED_CODE: 'El código ha expirado. Solicita uno nuevo.',\n      AUTHV2_CODE_EXPIRED: 'El código ha expirado. Solicita uno nuevo.',\n      AUTHV2_CODE_ALREADY_USED: 'Este código ya fue utilizado',\n      AUTHV2_TOO_MANY_ATTEMPTS: 'Demasiados intentos, intenta más tarde',\n      AUTHV2_SEND_CODE_FAILED: 'Error al enviar el código',\n      AUTHV2_ALREADY_VERIFIED: 'Este email ya está verificado',\n\n      // MFA\n      AUTHV2_MFA_INVALID_CODE: 'Código de verificación incorrecto',\n\n      // Password reset\n      AUTHV2_PASSWORD_RESET_FAILED: 'Error al restablecer la contraseña',\n      AUTHV2_INVALID_CURRENT_PASSWORD: 'Contraseña actual incorrecta',\n      AUTHV2_SAME_PASSWORD: 'La nueva contraseña debe ser diferente',\n\n      // Session/Tokens\n      AUTHV2_INVALID_TOKEN: 'Sesión inválida',\n      AUTHV2_EXPIRED_TOKEN: 'Tu sesión ha expirado',\n      AUTHV2_SESSION_EXPIRED: 'Tu sesión ha expirado',\n\n      // User\n      AUTHV2_USER_NOT_FOUND: 'Usuario no encontrado',\n\n      // OAuth\n      POPUP_BLOCKED: 'Por favor, permite ventanas emergentes para este sitio',\n      POPUP_CLOSED: 'Se canceló la autenticación',\n      INVALID_RESPONSE: 'Error en la respuesta del servidor',\n      OAUTH_FAILED: 'Error de autenticación',\n      OAUTH_EMAIL_EXISTS: 'Este correo ya está registrado con otro método',\n    };\n\n    const errorCode = error?.error?.code || error?.code;\n    if (errorCode && errorMessages[errorCode]) {\n      return errorMessages[errorCode];\n    }\n\n    if (error?.error?.message) {\n      return error.error.message;\n    }\n\n    return 'Ha ocurrido un error. Intenta de nuevo.';\n  }\n\n  // ==========================================\n  // COOLDOWN TIMERS\n  // ==========================================\n\n  private startResendCooldown(): void {\n    this.resendCooldown = 30;\n    this.resendTimer = setInterval(() => {\n      this.resendCooldown--;\n      if (this.resendCooldown <= 0) {\n        this.stopResendCooldown();\n      }\n    }, 1000);\n  }\n\n  private stopResendCooldown(): void {\n    if (this.resendTimer) {\n      clearInterval(this.resendTimer);\n      this.resendTimer = null;\n    }\n    this.resendCooldown = 0;\n  }\n\n  private startResetResendCooldown(): void {\n    this.resetResendCooldown = 30;\n    this.resetResendTimer = setInterval(() => {\n      this.resetResendCooldown--;\n      if (this.resetResendCooldown <= 0) {\n        this.stopResetResendCooldown();\n      }\n    }, 1000);\n  }\n\n  private stopResetResendCooldown(): void {\n    if (this.resetResendTimer) {\n      clearInterval(this.resetResendTimer);\n      this.resetResendTimer = null;\n    }\n    this.resetResendCooldown = 0;\n  }\n\n  ngOnDestroy(): void {\n    this.stopResendCooldown();\n    this.stopResetResendCooldown();\n  }\n}\n","<div class=\"val-login\">\n  <!-- Logo -->\n  @if (props.logo) {\n    <div class=\"logo-container\">\n      <val-image [props]=\"props.logo\" />\n    </div>\n  }\n\n  <!-- Login Form -->\n  <val-form [props]=\"loginFormProps\" (onSubmit)=\"loginHandler($event)\" />\n\n  <!-- OAuth Section -->\n  @if (config.showOAuth && config.oauthProviders.length > 0) {\n    <div class=\"oauth-separator\">\n      <span>o continúa con</span>\n    </div>\n\n    <div class=\"oauth-buttons\">\n      @for (provider of config.oauthProviders; track provider) {\n        <ion-button\n          expand=\"block\"\n          fill=\"outline\"\n          color=\"dark\"\n          (click)=\"loginWithOAuth(provider)\"\n          [disabled]=\"isOAuthLoading\"\n        >\n          @switch (provider) {\n            @case ('google') {\n              <ion-icon slot=\"start\" name=\"logo-google\"></ion-icon>\n              {{ isOAuthLoading ? 'Conectando...' : 'Continuar con Google' }}\n            }\n            @case ('apple') {\n              <ion-icon slot=\"start\" name=\"logo-apple\"></ion-icon>\n              {{ isOAuthLoading ? 'Conectando...' : 'Continuar con Apple' }}\n            }\n            @case ('microsoft') {\n              <ion-icon slot=\"start\" name=\"logo-microsoft\"></ion-icon>\n              {{ isOAuthLoading ? 'Conectando...' : 'Continuar con Microsoft' }}\n            }\n          }\n        </ion-button>\n      }\n    </div>\n  }\n\n  <!-- Register Link -->\n  @if (config.showRegister) {\n    <div class=\"auth-link\">\n      <ion-text color=\"dark\">\n        ¿No tienes cuenta?\n        <a (click)=\"openRegisterModal()\">Registrarse</a>\n      </ion-text>\n    </div>\n  }\n\n  <!-- Forgot Password Link -->\n  @if (config.showForgotPassword) {\n    <div class=\"auth-link forgot-password\">\n      <ion-text color=\"dark\">\n        ¿Olvidaste tu contraseña?\n        <a (click)=\"openForgotPasswordModal()\">Recuperar contraseña</a>\n      </ion-text>\n    </div>\n  }\n\n  <!-- Legal Notice -->\n  @if (props.legal) {\n    <div class=\"legal-notice\">\n      <ion-text color=\"medium\">\n        <p>\n          Utilizamos los servicios de\n          @if (props.legal.companyLink) {\n            <a [href]=\"props.legal.companyLink\"><strong>{{ props.legal.companyName }}</strong></a>\n          } @else {\n            <strong>{{ props.legal.companyName }}</strong>\n          }\n          para ofrecerte una experiencia segura. Al iniciar sesión, aceptas\n          nuestros\n          @if (props.legal.termsLink) {\n            <a [href]=\"props.legal.termsLink\">Términos y Condiciones</a>\n          } @else {\n            <span>Términos y Condiciones</span>\n          }\n          y\n          @if (props.legal.privacyLink) {\n            <a [href]=\"props.legal.privacyLink\">Política de Privacidad</a>\n          } @else {\n            <span>Política de Privacidad</span>\n          }.\n        </p>\n      </ion-text>\n    </div>\n  }\n</div>\n\n<!-- Register Modal -->\n<ion-modal [isOpen]=\"isRegisterModalOpen\" (didDismiss)=\"closeRegisterModal()\">\n  <ng-template>\n    <ion-header>\n      <ion-toolbar>\n        <ion-buttons slot=\"end\">\n          <ion-button fill=\"clear\" (click)=\"closeRegisterModal()\">\n            <ion-icon name=\"close-outline\"></ion-icon>\n          </ion-button>\n        </ion-buttons>\n      </ion-toolbar>\n    </ion-header>\n    <ion-content class=\"ion-padding\">\n      <section class=\"modal-form-section\">\n        @if (props.logo) {\n          <div class=\"logo-container\">\n            <val-image [props]=\"props.logo\" />\n          </div>\n        }\n        <val-form [props]=\"registerFormProps\" (onSubmit)=\"registerHandler($event)\" />\n        <div class=\"auth-link\">\n          <ion-text color=\"dark\">\n            ¿Ya tienes cuenta?\n            <a (click)=\"closeRegisterModal()\">Iniciar sesión</a>\n          </ion-text>\n        </div>\n      </section>\n    </ion-content>\n  </ng-template>\n</ion-modal>\n\n<!-- Verify Email Modal -->\n<ion-modal [isOpen]=\"isVerifyModalOpen\" [backdropDismiss]=\"false\">\n  <ng-template>\n    <ion-header>\n      <ion-toolbar>\n        <ion-buttons slot=\"end\">\n          <ion-button fill=\"clear\" (click)=\"closeVerifyModal()\">\n            <ion-icon name=\"close-outline\"></ion-icon>\n          </ion-button>\n        </ion-buttons>\n      </ion-toolbar>\n    </ion-header>\n    <ion-content class=\"ion-padding\">\n      <section class=\"modal-form-section\">\n        @if (props.logo) {\n          <div class=\"logo-container\">\n            <val-image [props]=\"props.logo\" />\n          </div>\n        }\n        <val-form [props]=\"verifyFormProps\" (onSubmit)=\"verifyHandler($event)\" />\n        <div class=\"resend-link\">\n          <ion-text color=\"dark\">\n            ¿No has recibido tu código?\n            @if (resendCooldown > 0) {\n              <span class=\"cooldown\">Reenviar en {{ resendCooldown }}s</span>\n            } @else {\n              <a (click)=\"resendCode()\">Reenviar</a>\n            }\n          </ion-text>\n        </div>\n      </section>\n    </ion-content>\n  </ng-template>\n</ion-modal>\n\n<!-- Forgot Password Modal -->\n<ion-modal [isOpen]=\"isForgotPasswordModalOpen\" (didDismiss)=\"closeForgotPasswordModal()\">\n  <ng-template>\n    <ion-header>\n      <ion-toolbar>\n        <ion-buttons slot=\"end\">\n          <ion-button fill=\"clear\" (click)=\"closeForgotPasswordModal()\">\n            <ion-icon name=\"close-outline\"></ion-icon>\n          </ion-button>\n        </ion-buttons>\n      </ion-toolbar>\n    </ion-header>\n    <ion-content class=\"ion-padding\">\n      <section class=\"modal-form-section\">\n        @if (props.logo) {\n          <div class=\"logo-container\">\n            <val-image [props]=\"props.logo\" />\n          </div>\n        }\n        <val-form [props]=\"forgotPasswordFormProps\" (onSubmit)=\"forgotPasswordHandler($event)\" />\n      </section>\n    </ion-content>\n  </ng-template>\n</ion-modal>\n\n<!-- Reset Password Modal -->\n<ion-modal [isOpen]=\"isResetPasswordModalOpen\" [backdropDismiss]=\"false\">\n  <ng-template>\n    <ion-header>\n      <ion-toolbar>\n        <ion-buttons slot=\"end\">\n          <ion-button fill=\"clear\" (click)=\"closeResetPasswordModal()\">\n            <ion-icon name=\"close-outline\"></ion-icon>\n          </ion-button>\n        </ion-buttons>\n      </ion-toolbar>\n    </ion-header>\n    <ion-content class=\"ion-padding\">\n      <section class=\"modal-form-section\">\n        @if (props.logo) {\n          <div class=\"logo-container\">\n            <val-image [props]=\"props.logo\" />\n          </div>\n        }\n        <val-form [props]=\"resetPasswordFormProps\" (onSubmit)=\"resetPasswordHandler($event)\" />\n        <div class=\"resend-link\">\n          <ion-text color=\"dark\">\n            ¿No has recibido tu código?\n            @if (resetResendCooldown > 0) {\n              <span class=\"cooldown\">Reenviar en {{ resetResendCooldown }}s</span>\n            } @else {\n              <a (click)=\"resendResetCode()\">Reenviar</a>\n            }\n          </ion-text>\n        </div>\n      </section>\n    </ion-content>\n  </ng-template>\n</ion-modal>\n\n<!-- MFA Verify Modal -->\n<ion-modal [isOpen]=\"isMFAVerifyModalOpen\" [backdropDismiss]=\"false\">\n  <ng-template>\n    <ion-header>\n      <ion-toolbar>\n        <ion-buttons slot=\"end\">\n          <ion-button fill=\"clear\" (click)=\"closeMFAVerifyModal()\">\n            <ion-icon name=\"close-outline\"></ion-icon>\n          </ion-button>\n        </ion-buttons>\n      </ion-toolbar>\n    </ion-header>\n    <ion-content class=\"ion-padding\">\n      <section class=\"modal-form-section\">\n        @if (props.logo) {\n          <div class=\"logo-container\">\n            <val-image [props]=\"props.logo\" />\n          </div>\n        }\n        <val-form [props]=\"mfaVerifyFormProps\" (onSubmit)=\"verifyMFAHandler($event)\" />\n      </section>\n    </ion-content>\n  </ng-template>\n</ion-modal>\n"]}
|