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.
@@ -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"]}