valtech-components 2.0.558 → 2.0.560

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.
@@ -5,6 +5,7 @@ import { Router } from '@angular/router';
5
5
  import { IonButton, IonButtons, IonContent, IonHeader, IonIcon, IonModal, IonText, IonToolbar, } from '@ionic/angular/standalone';
6
6
  import { AuthService } from '../../../services/auth';
7
7
  import { ToastService } from '../../../services/toast.service';
8
+ import { I18nService, InputI18nHelper } from '../../../services/i18n';
8
9
  import { FormComponent } from '../form/form.component';
9
10
  import { ImageComponent } from '../../atoms/image/image.component';
10
11
  import { InputType, ComponentStates, } from '../../types';
@@ -21,6 +22,8 @@ export class LoginComponent {
21
22
  this.authService = inject(AuthService);
22
23
  this.toastService = inject(ToastService);
23
24
  this.router = inject(Router);
25
+ this.i18n = inject(I18nService);
26
+ this.i18nHelper = inject(InputI18nHelper);
24
27
  // Timers
25
28
  this.resendTimer = null;
26
29
  this.resetResendTimer = null;
@@ -38,283 +41,351 @@ export class LoginComponent {
38
41
  // ==========================================
39
42
  // LOGIN FORM
40
43
  // ==========================================
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',
44
+ this._loginFormState = ComponentStates.ENABLED;
45
+ // ==========================================
46
+ // REGISTER FORM
47
+ // ==========================================
48
+ this._registerFormState = ComponentStates.ENABLED;
49
+ // ==========================================
50
+ // VERIFY EMAIL FORM
51
+ // ==========================================
52
+ this._verifyFormState = ComponentStates.ENABLED;
53
+ this._verifyFormSectionName = '';
54
+ // ==========================================
55
+ // MFA VERIFY FORM
56
+ // ==========================================
57
+ this._mfaVerifyFormState = ComponentStates.ENABLED;
58
+ this._mfaMethod = 'TOTP';
59
+ // ==========================================
60
+ // FORGOT PASSWORD FORM
61
+ // ==========================================
62
+ this._forgotPasswordFormState = ComponentStates.ENABLED;
63
+ // ==========================================
64
+ // RESET PASSWORD FORM
65
+ // ==========================================
66
+ this._resetPasswordFormState = ComponentStates.ENABLED;
67
+ this._resetFormSectionName = '';
68
+ }
69
+ /**
70
+ * Helper method for translating auth-related strings.
71
+ * Exposed to template for dynamic text.
72
+ */
73
+ t(key, data) {
74
+ return this.i18n.t(key, '_auth', data);
75
+ }
76
+ // Resolved props with defaults
77
+ get config() {
78
+ return { ...LOGIN_DEFAULTS, ...this.props };
79
+ }
80
+ get loginFormProps() {
81
+ return this.i18nHelper.resolveForm({
82
+ nameKey: 'loginTitle',
83
+ i18nNamespace: '_auth',
74
84
  sections: [
75
85
  {
76
86
  name: '',
77
87
  order: 0,
78
- fields: [this.emailInput, this.passwordInput],
88
+ fields: [
89
+ {
90
+ type: InputType.EMAIL,
91
+ name: 'email',
92
+ token: 'login-email',
93
+ hint: '',
94
+ labelKey: 'email',
95
+ placeholderKey: 'emailPlaceholder',
96
+ errorKeys: {
97
+ required: 'emailRequired',
98
+ email: 'emailInvalid',
99
+ },
100
+ validators: [Validators.required, Validators.email],
101
+ order: 0,
102
+ state: ComponentStates.ENABLED,
103
+ },
104
+ {
105
+ type: InputType.PASSWORD,
106
+ name: 'password',
107
+ token: 'login-password',
108
+ hint: '',
109
+ labelKey: 'password',
110
+ placeholderKey: 'passwordPlaceholder',
111
+ errorKeys: {
112
+ required: 'passwordRequired',
113
+ },
114
+ validators: [Validators.required],
115
+ order: 1,
116
+ state: ComponentStates.ENABLED,
117
+ },
118
+ ],
79
119
  },
80
120
  ],
81
121
  actions: {
82
- ...SolidDefaultBlock('Iniciar sesión', 'submit'),
122
+ ...SolidDefaultBlock('', 'submit'),
83
123
  token: 'login-submit',
124
+ textKey: 'loginSubmit',
84
125
  },
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',
126
+ state: this._loginFormState,
127
+ });
128
+ }
129
+ set loginFormState(value) {
130
+ this._loginFormState = value;
131
+ }
132
+ get registerFormProps() {
133
+ return this.i18nHelper.resolveForm({
134
+ nameKey: 'registerTitle',
135
+ i18nNamespace: '_auth',
140
136
  sections: [
141
137
  {
142
138
  name: '',
143
139
  order: 0,
144
- fields: [this.nameInput, this.registerEmailInput, this.registerPasswordInput],
140
+ fields: [
141
+ {
142
+ type: InputType.TEXT,
143
+ name: 'name',
144
+ token: 'register-name',
145
+ hint: '',
146
+ labelKey: 'fullName',
147
+ placeholderKey: 'namePlaceholder',
148
+ errorKeys: {
149
+ required: 'nameRequired',
150
+ minlength: 'nameMinLength',
151
+ },
152
+ validators: [Validators.required, Validators.minLength(2)],
153
+ order: 0,
154
+ state: ComponentStates.ENABLED,
155
+ },
156
+ {
157
+ type: InputType.EMAIL,
158
+ name: 'email',
159
+ token: 'register-email',
160
+ hint: '',
161
+ labelKey: 'email',
162
+ placeholderKey: 'emailPlaceholder',
163
+ errorKeys: {
164
+ required: 'emailRequired',
165
+ email: 'emailInvalid',
166
+ },
167
+ validators: [Validators.required, Validators.email],
168
+ order: 1,
169
+ state: ComponentStates.ENABLED,
170
+ },
171
+ {
172
+ type: InputType.PASSWORD,
173
+ name: 'password',
174
+ token: 'register-password',
175
+ hint: '',
176
+ labelKey: 'password',
177
+ placeholderKey: 'passwordPlaceholder',
178
+ errorKeys: {
179
+ required: 'passwordRequired',
180
+ minlength: 'passwordMinLength',
181
+ },
182
+ validators: [Validators.required, Validators.minLength(8)],
183
+ order: 2,
184
+ state: ComponentStates.ENABLED,
185
+ },
186
+ ],
145
187
  },
146
188
  ],
147
189
  actions: {
148
- ...SolidDefaultBlock('Registrarse', 'submit'),
190
+ ...SolidDefaultBlock('', 'submit'),
149
191
  token: 'register-submit',
192
+ textKey: 'registerSubmit',
150
193
  },
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',
194
+ state: this._registerFormState,
195
+ });
196
+ }
197
+ set registerFormState(value) {
198
+ this._registerFormState = value;
199
+ }
200
+ get verifyFormProps() {
201
+ return this.i18nHelper.resolveForm({
202
+ nameKey: 'verifyTitle',
203
+ i18nNamespace: '_auth',
177
204
  sections: [
178
205
  {
179
- name: 'Ingresa el código de verificación enviado a tu correo.',
206
+ name: this._verifyFormSectionName || this.t('verifyDescription'),
180
207
  order: 0,
181
- fields: [this.verifyPinInput],
208
+ fields: [
209
+ {
210
+ type: InputType.PIN_CODE,
211
+ label: '',
212
+ name: 'code',
213
+ token: 'verify-pin',
214
+ hint: '',
215
+ placeholder: '',
216
+ errorKeys: {
217
+ required: 'codeRequired',
218
+ minlength: 'codeMinLength',
219
+ },
220
+ validators: [Validators.required, Validators.minLength(6)],
221
+ order: 0,
222
+ state: ComponentStates.ENABLED,
223
+ length: 6,
224
+ allowNumbersOnly: true,
225
+ autoFocus: true,
226
+ },
227
+ ],
182
228
  },
183
229
  ],
184
230
  actions: {
185
- ...SolidDefaultBlock('Verificar', 'submit'),
231
+ ...SolidDefaultBlock('', 'submit'),
186
232
  token: 'verify-submit',
233
+ textKey: 'verifySubmit',
187
234
  },
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',
235
+ state: this._verifyFormState,
236
+ });
237
+ }
238
+ set verifyFormState(value) {
239
+ this._verifyFormState = value;
240
+ }
241
+ set verifyFormSectionName(value) {
242
+ this._verifyFormSectionName = value;
243
+ }
244
+ get mfaVerifyFormProps() {
245
+ const sectionName = this._mfaMethod === 'TOTP'
246
+ ? this.t('mfaTOTP')
247
+ : this._mfaMethod === 'EMAIL'
248
+ ? this.t('mfaEmail')
249
+ : this.t('mfaSMS');
250
+ return this.i18nHelper.resolveForm({
251
+ nameKey: 'mfaTitle',
252
+ i18nNamespace: '_auth',
214
253
  sections: [
215
254
  {
216
- name: '',
255
+ name: sectionName,
217
256
  order: 0,
218
- fields: [this.mfaPinInput],
257
+ fields: [
258
+ {
259
+ type: InputType.PIN_CODE,
260
+ label: '',
261
+ name: 'code',
262
+ token: 'mfa-pin',
263
+ hint: '',
264
+ placeholder: '',
265
+ errorKeys: {
266
+ required: 'codeRequired',
267
+ minlength: 'codeMinLength',
268
+ },
269
+ validators: [Validators.required, Validators.minLength(6)],
270
+ order: 0,
271
+ state: ComponentStates.ENABLED,
272
+ length: 6,
273
+ allowNumbersOnly: true,
274
+ autoFocus: true,
275
+ },
276
+ ],
219
277
  },
220
278
  ],
221
279
  actions: {
222
- ...SolidDefaultBlock('Verificar', 'submit'),
280
+ ...SolidDefaultBlock('', 'submit'),
223
281
  token: 'mfa-verify-submit',
282
+ textKey: 'verifySubmit',
224
283
  },
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',
284
+ state: this._mfaVerifyFormState,
285
+ });
286
+ }
287
+ set mfaVerifyFormState(value) {
288
+ this._mfaVerifyFormState = value;
289
+ }
290
+ set mfaMethod(value) {
291
+ this._mfaMethod = value;
292
+ }
293
+ get forgotPasswordFormProps() {
294
+ return this.i18nHelper.resolveForm({
295
+ nameKey: 'forgotTitle',
296
+ i18nNamespace: '_auth',
248
297
  sections: [
249
298
  {
250
- name: 'Ingresa tu correo electrónico y te enviaremos un código para restablecer tu contraseña.',
299
+ name: this.t('forgotDescription'),
251
300
  order: 0,
252
- fields: [this.forgotEmailInput],
301
+ fields: [
302
+ {
303
+ type: InputType.EMAIL,
304
+ label: '',
305
+ name: 'email',
306
+ token: 'forgot-email',
307
+ hint: '',
308
+ placeholderKey: 'emailPlaceholder',
309
+ errorKeys: {
310
+ required: 'emailRequired',
311
+ email: 'emailInvalid',
312
+ },
313
+ validators: [Validators.required, Validators.email],
314
+ order: 0,
315
+ state: ComponentStates.ENABLED,
316
+ },
317
+ ],
253
318
  },
254
319
  ],
255
320
  actions: {
256
- ...SolidDefaultBlock('Enviar código', 'submit'),
321
+ ...SolidDefaultBlock('', 'submit'),
257
322
  token: 'forgot-submit',
323
+ textKey: 'forgotSubmit',
258
324
  },
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',
325
+ state: this._forgotPasswordFormState,
326
+ });
327
+ }
328
+ set forgotPasswordFormState(value) {
329
+ this._forgotPasswordFormState = value;
330
+ }
331
+ get resetPasswordFormProps() {
332
+ return this.i18nHelper.resolveForm({
333
+ nameKey: 'resetTitle',
334
+ i18nNamespace: '_auth',
301
335
  sections: [
302
336
  {
303
- name: 'Hemos enviado un código de verificación a tu correo.',
337
+ name: this._resetFormSectionName || this.t('resetDescription'),
304
338
  order: 0,
305
- fields: [this.resetPinInput, this.newPasswordInput],
339
+ fields: [
340
+ {
341
+ type: InputType.PIN_CODE,
342
+ label: '',
343
+ name: 'code',
344
+ token: 'reset-pin',
345
+ hint: '',
346
+ placeholder: '',
347
+ errorKeys: {
348
+ required: 'codeRequired',
349
+ minlength: 'codeMinLength',
350
+ },
351
+ validators: [Validators.required, Validators.minLength(6)],
352
+ order: 0,
353
+ state: ComponentStates.ENABLED,
354
+ length: 6,
355
+ allowNumbersOnly: true,
356
+ autoFocus: true,
357
+ },
358
+ {
359
+ type: InputType.PASSWORD,
360
+ name: 'newPassword',
361
+ token: 'reset-new-password',
362
+ labelKey: 'newPassword',
363
+ hintKey: 'newPasswordHint',
364
+ placeholderKey: 'passwordPlaceholder',
365
+ errorKeys: {
366
+ required: 'passwordRequired',
367
+ minlength: 'passwordMinLength',
368
+ },
369
+ validators: [Validators.required, Validators.minLength(8)],
370
+ order: 1,
371
+ state: ComponentStates.ENABLED,
372
+ },
373
+ ],
306
374
  },
307
375
  ],
308
376
  actions: {
309
- ...SolidDefaultBlock('Cambiar contraseña', 'submit'),
377
+ ...SolidDefaultBlock('', 'submit'),
310
378
  token: 'reset-submit',
379
+ textKey: 'resetSubmit',
311
380
  },
312
- state: ComponentStates.ENABLED,
313
- };
381
+ state: this._resetPasswordFormState,
382
+ });
314
383
  }
315
- // Resolved props with defaults
316
- get config() {
317
- return { ...LOGIN_DEFAULTS, ...this.props };
384
+ set resetPasswordFormState(value) {
385
+ this._resetPasswordFormState = value;
386
+ }
387
+ set resetFormSectionName(value) {
388
+ this._resetFormSectionName = value;
318
389
  }
319
390
  // ==========================================
320
391
  // HANDLERS
@@ -323,13 +394,13 @@ export class LoginComponent {
323
394
  const email = event.fields['email'];
324
395
  const password = event.fields['password'];
325
396
  if (!email || !password) {
326
- this.showToast('Completa todos los campos.');
397
+ this.showToast(this.t('completeAllFields'));
327
398
  return;
328
399
  }
329
- this.loginFormProps.state = ComponentStates.WORKING;
400
+ this._loginFormState = ComponentStates.WORKING;
330
401
  this.authService.signin({ email, password }).subscribe({
331
402
  next: () => {
332
- this.loginFormProps.state = ComponentStates.ENABLED;
403
+ this._loginFormState = ComponentStates.ENABLED;
333
404
  if (this.authService.mfaPending().required) {
334
405
  this.openMFAVerifyModal();
335
406
  return;
@@ -337,7 +408,7 @@ export class LoginComponent {
337
408
  this.handleLoginSuccess();
338
409
  },
339
410
  error: (err) => {
340
- this.loginFormProps.state = ComponentStates.ENABLED;
411
+ this._loginFormState = ComponentStates.ENABLED;
341
412
  const errorCode = err?.code;
342
413
  if (errorCode === 'AUTHV2_EMAIL_NOT_VERIFIED') {
343
414
  this.openVerifyModal(email);
@@ -382,18 +453,18 @@ export class LoginComponent {
382
453
  const email = event.fields['email'];
383
454
  const password = event.fields['password'];
384
455
  if (!name || !email || !password) {
385
- this.showToast('Completa todos los campos.');
456
+ this.showToast(this.t('completeAllFields'));
386
457
  return;
387
458
  }
388
- this.registerFormProps.state = ComponentStates.WORKING;
459
+ this._registerFormState = ComponentStates.WORKING;
389
460
  this.authService.signup({ name, email, password }).subscribe({
390
461
  next: () => {
391
- this.registerFormProps.state = ComponentStates.ENABLED;
462
+ this._registerFormState = ComponentStates.ENABLED;
392
463
  this.closeRegisterModal();
393
464
  this.openVerifyModal(email);
394
465
  },
395
466
  error: (err) => {
396
- this.registerFormProps.state = ComponentStates.ENABLED;
467
+ this._registerFormState = ComponentStates.ENABLED;
397
468
  this.handleError(err, 'signup');
398
469
  },
399
470
  });
@@ -403,19 +474,20 @@ export class LoginComponent {
403
474
  // ==========================================
404
475
  openVerifyModal(email) {
405
476
  this.pendingVerificationEmail = email;
406
- this.verifyFormProps.sections[0].name = `Ingresa el código enviado a ${email}`;
407
- this.verifyFormProps.state = ComponentStates.ENABLED;
477
+ this._verifyFormSectionName = `${this.t('verifyDescription').replace('.', '')} (${email})`;
478
+ this._verifyFormState = ComponentStates.ENABLED;
408
479
  this.isVerifyModalOpen = true;
409
480
  this.startResendCooldown();
410
481
  }
411
482
  closeVerifyModal() {
412
483
  this.isVerifyModalOpen = false;
413
484
  this.pendingVerificationEmail = '';
485
+ this._verifyFormSectionName = '';
414
486
  this.stopResendCooldown();
415
487
  }
416
488
  verifyHandler(event) {
417
489
  const code = event.fields['code'];
418
- this.verifyFormProps.state = ComponentStates.WORKING;
490
+ this._verifyFormState = ComponentStates.WORKING;
419
491
  this.authService
420
492
  .verifyEmail({
421
493
  email: this.pendingVerificationEmail,
@@ -423,13 +495,13 @@ export class LoginComponent {
423
495
  })
424
496
  .subscribe({
425
497
  next: () => {
426
- this.verifyFormProps.state = ComponentStates.ENABLED;
427
- this.showToast('¡Email verificado! Bienvenido.');
498
+ this._verifyFormState = ComponentStates.ENABLED;
499
+ this.showToast(this.t('emailVerified'));
428
500
  this.closeVerifyModal();
429
501
  this.handleLoginSuccess();
430
502
  },
431
503
  error: (err) => {
432
- this.verifyFormProps.state = ComponentStates.ENABLED;
504
+ this._verifyFormState = ComponentStates.ENABLED;
433
505
  this.handleError(err, 'verify');
434
506
  },
435
507
  });
@@ -444,7 +516,7 @@ export class LoginComponent {
444
516
  })
445
517
  .subscribe({
446
518
  next: () => {
447
- this.showToast('Código reenviado. Revisa tu email.');
519
+ this.showToast(this.t('codeSent'));
448
520
  this.startResendCooldown();
449
521
  },
450
522
  error: (err) => {
@@ -457,30 +529,25 @@ export class LoginComponent {
457
529
  // ==========================================
458
530
  openMFAVerifyModal() {
459
531
  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;
532
+ this._mfaMethod = method;
533
+ this._mfaVerifyFormState = ComponentStates.ENABLED;
467
534
  this.isMFAVerifyModalOpen = true;
468
- this.onMFARequired.emit({ method: method });
535
+ this.onMFARequired.emit({ method });
469
536
  }
470
537
  closeMFAVerifyModal() {
471
538
  this.isMFAVerifyModalOpen = false;
472
539
  }
473
540
  verifyMFAHandler(event) {
474
541
  const code = event.fields['code'];
475
- this.mfaVerifyFormProps.state = ComponentStates.WORKING;
542
+ this._mfaVerifyFormState = ComponentStates.WORKING;
476
543
  this.authService.verifyMFA(code).subscribe({
477
544
  next: () => {
478
- this.mfaVerifyFormProps.state = ComponentStates.ENABLED;
545
+ this._mfaVerifyFormState = ComponentStates.ENABLED;
479
546
  this.closeMFAVerifyModal();
480
547
  this.handleLoginSuccess(undefined, true);
481
548
  },
482
549
  error: (err) => {
483
- this.mfaVerifyFormProps.state = ComponentStates.ENABLED;
550
+ this._mfaVerifyFormState = ComponentStates.ENABLED;
484
551
  this.handleError(err, 'mfa');
485
552
  },
486
553
  });
@@ -497,19 +564,19 @@ export class LoginComponent {
497
564
  forgotPasswordHandler(event) {
498
565
  const email = event.fields['email'];
499
566
  if (!email) {
500
- this.showToast('Ingresa tu correo electrónico.');
567
+ this.showToast(this.t('enterEmail'));
501
568
  return;
502
569
  }
503
- this.forgotPasswordFormProps.state = ComponentStates.WORKING;
570
+ this._forgotPasswordFormState = ComponentStates.WORKING;
504
571
  this.authService.forgotPassword({ email }).subscribe({
505
572
  next: () => {
506
- this.forgotPasswordFormProps.state = ComponentStates.ENABLED;
573
+ this._forgotPasswordFormState = ComponentStates.ENABLED;
507
574
  this.closeForgotPasswordModal();
508
575
  this.openResetPasswordModal(email);
509
- this.showToast('Código enviado. Revisa tu email.');
576
+ this.showToast(this.t('codeSent'));
510
577
  },
511
578
  error: (err) => {
512
- this.forgotPasswordFormProps.state = ComponentStates.ENABLED;
579
+ this._forgotPasswordFormState = ComponentStates.ENABLED;
513
580
  this.handleError(err, 'forgot');
514
581
  },
515
582
  });
@@ -519,20 +586,21 @@ export class LoginComponent {
519
586
  // ==========================================
520
587
  openResetPasswordModal(email) {
521
588
  this.pendingResetEmail = email;
522
- this.resetPasswordFormProps.sections[0].name = `Ingresa el código enviado a ${email}`;
523
- this.resetPasswordFormProps.state = ComponentStates.ENABLED;
589
+ this._resetFormSectionName = `${this.t('resetDescription').replace('.', '')} (${email})`;
590
+ this._resetPasswordFormState = ComponentStates.ENABLED;
524
591
  this.isResetPasswordModalOpen = true;
525
592
  this.startResetResendCooldown();
526
593
  }
527
594
  closeResetPasswordModal() {
528
595
  this.isResetPasswordModalOpen = false;
529
596
  this.pendingResetEmail = '';
597
+ this._resetFormSectionName = '';
530
598
  this.stopResetResendCooldown();
531
599
  }
532
600
  resetPasswordHandler(event) {
533
601
  const code = event.fields['code'];
534
602
  const newPassword = event.fields['newPassword'];
535
- this.resetPasswordFormProps.state = ComponentStates.WORKING;
603
+ this._resetPasswordFormState = ComponentStates.WORKING;
536
604
  this.authService
537
605
  .resetPassword({
538
606
  email: this.pendingResetEmail,
@@ -541,12 +609,12 @@ export class LoginComponent {
541
609
  })
542
610
  .subscribe({
543
611
  next: () => {
544
- this.resetPasswordFormProps.state = ComponentStates.ENABLED;
545
- this.showToast('¡Contraseña actualizada! Ya puedes iniciar sesión.');
612
+ this._resetPasswordFormState = ComponentStates.ENABLED;
613
+ this.showToast(this.t('passwordUpdated'));
546
614
  this.closeResetPasswordModal();
547
615
  },
548
616
  error: (err) => {
549
- this.resetPasswordFormProps.state = ComponentStates.ENABLED;
617
+ this._resetPasswordFormState = ComponentStates.ENABLED;
550
618
  this.handleError(err, 'reset');
551
619
  },
552
620
  });
@@ -561,7 +629,7 @@ export class LoginComponent {
561
629
  })
562
630
  .subscribe({
563
631
  next: () => {
564
- this.showToast('Código reenviado. Revisa tu email.');
632
+ this.showToast(this.t('codeSent'));
565
633
  this.startResetResendCooldown();
566
634
  },
567
635
  error: (err) => {
@@ -573,7 +641,7 @@ export class LoginComponent {
573
641
  // HELPERS
574
642
  // ==========================================
575
643
  handleLoginSuccess(oauthProvider, mfaCompleted) {
576
- this.showToast('¡Bienvenido!');
644
+ this.showToast(this.t('welcome'));
577
645
  this.onSuccess.emit({
578
646
  user: this.authService.user(),
579
647
  mfaCompleted,
@@ -602,55 +670,34 @@ export class LoginComponent {
602
670
  }
603
671
  getErrorMessage(err) {
604
672
  const error = err;
605
- const errorMessages = {
606
- // Validation
607
- VALIDATION_MISSING_REQUIRED_FIELDS: 'Faltan campos requeridos',
608
- VALIDATION_INVALID_REQUEST: 'Solicitud inválida',
673
+ // Map error codes to i18n keys
674
+ const errorKeyMap = {
609
675
  // 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',
676
+ AUTHV2_INVALID_CREDENTIALS: 'errorInvalidCredentials',
677
+ AUTHV2_EMAIL_NOT_VERIFIED: 'errorEmailNotVerified',
678
+ AUTHV2_ACCOUNT_SUSPENDED: 'errorAccountSuspended',
614
679
  // 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',
680
+ AUTHV2_EMAIL_EXISTS: 'errorEmailExists',
681
+ AUTHV2_WEAK_PASSWORD: 'errorWeakPassword',
619
682
  // 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',
683
+ AUTHV2_INVALID_CODE: 'errorInvalidCode',
684
+ AUTHV2_EXPIRED_CODE: 'errorExpiredCode',
685
+ AUTHV2_CODE_EXPIRED: 'errorExpiredCode',
686
+ AUTHV2_TOO_MANY_ATTEMPTS: 'errorTooManyAttempts',
627
687
  // 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',
688
+ AUTHV2_MFA_INVALID_CODE: 'errorMFAInvalidCode',
639
689
  // 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',
690
+ POPUP_BLOCKED: 'errorPopupBlocked',
691
+ OAUTH_FAILED: 'errorOAuthFailed',
645
692
  };
646
693
  const errorCode = error?.error?.code || error?.code;
647
- if (errorCode && errorMessages[errorCode]) {
648
- return errorMessages[errorCode];
694
+ if (errorCode && errorKeyMap[errorCode]) {
695
+ return this.t(errorKeyMap[errorCode]);
649
696
  }
650
697
  if (error?.error?.message) {
651
698
  return error.error.message;
652
699
  }
653
- return 'Ha ocurrido un error. Intenta de nuevo.';
700
+ return this.t('errorGeneric');
654
701
  }
655
702
  // ==========================================
656
703
  // COOLDOWN TIMERS
@@ -692,7 +739,7 @@ export class LoginComponent {
692
739
  this.stopResetResendCooldown();
693
740
  }
694
741
  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\" [class.val-login--card]=\"config.showCard\">\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%}.val-login--card{padding:1rem;background:var(--ion-color-light);border-radius:16px;box-shadow:0 2px 8px #00000014}.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"] }] }); }
742
+ 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\" [class.val-login--card]=\"config.showCard\">\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>{{ t('orContinueWith') }}</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 ? t('connecting') : t('continueWithGoogle') }}\n }\n @case ('apple') {\n <ion-icon slot=\"start\" name=\"logo-apple\"></ion-icon>\n {{ isOAuthLoading ? t('connecting') : t('continueWithApple') }}\n }\n @case ('microsoft') {\n <ion-icon slot=\"start\" name=\"logo-microsoft\"></ion-icon>\n {{ isOAuthLoading ? t('connecting') : t('continueWithMicrosoft') }}\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 {{ t('noAccount') }}\n <a (click)=\"openRegisterModal()\">{{ t('register') }}</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 {{ t('forgotLink') }}\n <a (click)=\"openForgotPasswordModal()\">{{ t('recoverPassword') }}</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 {{ t('legalPrefix') }}\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 {{ t('legalSuffix') }}\n @if (props.legal.termsLink) {\n <a [href]=\"props.legal.termsLink\">{{ t('termsAndConditions') }}</a>\n } @else {\n <span>{{ t('termsAndConditions') }}</span>\n }\n {{ t('and') }}\n @if (props.legal.privacyLink) {\n <a [href]=\"props.legal.privacyLink\">{{ t('privacyPolicy') }}</a>\n } @else {\n <span>{{ t('privacyPolicy') }}</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 {{ t('hasAccount') }}\n <a (click)=\"closeRegisterModal()\">{{ t('signIn') }}</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 {{ t('noCodeReceived') }}\n @if (resendCooldown > 0) {\n <span class=\"cooldown\">{{ t('resendIn', { seconds: resendCooldown.toString() }) }}</span>\n } @else {\n <a (click)=\"resendCode()\">{{ t('resend') }}</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 {{ t('noCodeReceived') }}\n @if (resetResendCooldown > 0) {\n <span class=\"cooldown\">{{ t('resendIn', { seconds: resetResendCooldown.toString() }) }}</span>\n } @else {\n <a (click)=\"resendResetCode()\">{{ t('resend') }}</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%}.val-login--card{padding:1rem;background:var(--ion-color-light);border-radius:16px;box-shadow:0 2px 8px #00000014}.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
743
  }
697
744
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: LoginComponent, decorators: [{
698
745
  type: Component,
@@ -708,7 +755,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
708
755
  IonToolbar,
709
756
  FormComponent,
710
757
  ImageComponent,
711
- ], template: "<div class=\"val-login\" [class.val-login--card]=\"config.showCard\">\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%}.val-login--card{padding:1rem;background:var(--ion-color-light);border-radius:16px;box-shadow:0 2px 8px #00000014}.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"] }]
758
+ ], template: "<div class=\"val-login\" [class.val-login--card]=\"config.showCard\">\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>{{ t('orContinueWith') }}</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 ? t('connecting') : t('continueWithGoogle') }}\n }\n @case ('apple') {\n <ion-icon slot=\"start\" name=\"logo-apple\"></ion-icon>\n {{ isOAuthLoading ? t('connecting') : t('continueWithApple') }}\n }\n @case ('microsoft') {\n <ion-icon slot=\"start\" name=\"logo-microsoft\"></ion-icon>\n {{ isOAuthLoading ? t('connecting') : t('continueWithMicrosoft') }}\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 {{ t('noAccount') }}\n <a (click)=\"openRegisterModal()\">{{ t('register') }}</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 {{ t('forgotLink') }}\n <a (click)=\"openForgotPasswordModal()\">{{ t('recoverPassword') }}</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 {{ t('legalPrefix') }}\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 {{ t('legalSuffix') }}\n @if (props.legal.termsLink) {\n <a [href]=\"props.legal.termsLink\">{{ t('termsAndConditions') }}</a>\n } @else {\n <span>{{ t('termsAndConditions') }}</span>\n }\n {{ t('and') }}\n @if (props.legal.privacyLink) {\n <a [href]=\"props.legal.privacyLink\">{{ t('privacyPolicy') }}</a>\n } @else {\n <span>{{ t('privacyPolicy') }}</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 {{ t('hasAccount') }}\n <a (click)=\"closeRegisterModal()\">{{ t('signIn') }}</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 {{ t('noCodeReceived') }}\n @if (resendCooldown > 0) {\n <span class=\"cooldown\">{{ t('resendIn', { seconds: resendCooldown.toString() }) }}</span>\n } @else {\n <a (click)=\"resendCode()\">{{ t('resend') }}</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 {{ t('noCodeReceived') }}\n @if (resetResendCooldown > 0) {\n <span class=\"cooldown\">{{ t('resendIn', { seconds: resetResendCooldown.toString() }) }}</span>\n } @else {\n <a (click)=\"resendResetCode()\">{{ t('resend') }}</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%}.val-login--card{padding:1rem;background:var(--ion-color-light);border-radius:16px;box-shadow:0 2px 8px #00000014}.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
759
  }], propDecorators: { props: [{
713
760
  type: Input
714
761
  }], onSuccess: [{
@@ -718,4 +765,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
718
765
  }], onMFARequired: [{
719
766
  type: Output
720
767
  }] } });
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\" [class.val-login--card]=\"config.showCard\">\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"]}
768
+ //# 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 { I18nService, InputI18nHelper } from '../../../services/i18n';\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  ComponentState,\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  private i18n = inject(I18nService);\n  private i18nHelper = inject(InputI18nHelper);\n\n  /**\n   * Helper method for translating auth-related strings.\n   * Exposed to template for dynamic text.\n   */\n  t(key: string, data?: Record<string, string>): string {\n    return this.i18n.t(key, '_auth', data);\n  }\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  private _loginFormState: ComponentState = ComponentStates.ENABLED;\n\n  get loginFormProps(): FormMetadata {\n    return this.i18nHelper.resolveForm({\n      nameKey: 'loginTitle',\n      i18nNamespace: '_auth',\n      sections: [\n        {\n          name: '',\n          order: 0,\n          fields: [\n            {\n              type: InputType.EMAIL,\n              name: 'email',\n              token: 'login-email',\n              hint: '',\n              labelKey: 'email',\n              placeholderKey: 'emailPlaceholder',\n              errorKeys: {\n                required: 'emailRequired',\n                email: 'emailInvalid',\n              },\n              validators: [Validators.required, Validators.email],\n              order: 0,\n              state: ComponentStates.ENABLED,\n            },\n            {\n              type: InputType.PASSWORD,\n              name: 'password',\n              token: 'login-password',\n              hint: '',\n              labelKey: 'password',\n              placeholderKey: 'passwordPlaceholder',\n              errorKeys: {\n                required: 'passwordRequired',\n              },\n              validators: [Validators.required],\n              order: 1,\n              state: ComponentStates.ENABLED,\n            },\n          ],\n        },\n      ],\n      actions: {\n        ...SolidDefaultBlock('', 'submit'),\n        token: 'login-submit',\n        textKey: 'loginSubmit',\n      },\n      state: this._loginFormState,\n    });\n  }\n\n  set loginFormState(value: ComponentState) {\n    this._loginFormState = value;\n  }\n\n  // ==========================================\n  // REGISTER FORM\n  // ==========================================\n\n  private _registerFormState: ComponentState = ComponentStates.ENABLED;\n\n  get registerFormProps(): FormMetadata {\n    return this.i18nHelper.resolveForm({\n      nameKey: 'registerTitle',\n      i18nNamespace: '_auth',\n      sections: [\n        {\n          name: '',\n          order: 0,\n          fields: [\n            {\n              type: InputType.TEXT,\n              name: 'name',\n              token: 'register-name',\n              hint: '',\n              labelKey: 'fullName',\n              placeholderKey: 'namePlaceholder',\n              errorKeys: {\n                required: 'nameRequired',\n                minlength: 'nameMinLength',\n              },\n              validators: [Validators.required, Validators.minLength(2)],\n              order: 0,\n              state: ComponentStates.ENABLED,\n            },\n            {\n              type: InputType.EMAIL,\n              name: 'email',\n              token: 'register-email',\n              hint: '',\n              labelKey: 'email',\n              placeholderKey: 'emailPlaceholder',\n              errorKeys: {\n                required: 'emailRequired',\n                email: 'emailInvalid',\n              },\n              validators: [Validators.required, Validators.email],\n              order: 1,\n              state: ComponentStates.ENABLED,\n            },\n            {\n              type: InputType.PASSWORD,\n              name: 'password',\n              token: 'register-password',\n              hint: '',\n              labelKey: 'password',\n              placeholderKey: 'passwordPlaceholder',\n              errorKeys: {\n                required: 'passwordRequired',\n                minlength: 'passwordMinLength',\n              },\n              validators: [Validators.required, Validators.minLength(8)],\n              order: 2,\n              state: ComponentStates.ENABLED,\n            },\n          ],\n        },\n      ],\n      actions: {\n        ...SolidDefaultBlock('', 'submit'),\n        token: 'register-submit',\n        textKey: 'registerSubmit',\n      },\n      state: this._registerFormState,\n    });\n  }\n\n  set registerFormState(value: ComponentState) {\n    this._registerFormState = value;\n  }\n\n  // ==========================================\n  // VERIFY EMAIL FORM\n  // ==========================================\n\n  private _verifyFormState: ComponentState = ComponentStates.ENABLED;\n  private _verifyFormSectionName = '';\n\n  get verifyFormProps(): FormMetadata {\n    return this.i18nHelper.resolveForm({\n      nameKey: 'verifyTitle',\n      i18nNamespace: '_auth',\n      sections: [\n        {\n          name: this._verifyFormSectionName || this.t('verifyDescription'),\n          order: 0,\n          fields: [\n            {\n              type: InputType.PIN_CODE,\n              label: '',\n              name: 'code',\n              token: 'verify-pin',\n              hint: '',\n              placeholder: '',\n              errorKeys: {\n                required: 'codeRequired',\n                minlength: 'codeMinLength',\n              },\n              validators: [Validators.required, Validators.minLength(6)],\n              order: 0,\n              state: ComponentStates.ENABLED,\n              length: 6,\n              allowNumbersOnly: true,\n              autoFocus: true,\n            },\n          ],\n        },\n      ],\n      actions: {\n        ...SolidDefaultBlock('', 'submit'),\n        token: 'verify-submit',\n        textKey: 'verifySubmit',\n      },\n      state: this._verifyFormState,\n    });\n  }\n\n  set verifyFormState(value: ComponentState) {\n    this._verifyFormState = value;\n  }\n\n  set verifyFormSectionName(value: string) {\n    this._verifyFormSectionName = value;\n  }\n\n  // ==========================================\n  // MFA VERIFY FORM\n  // ==========================================\n\n  private _mfaVerifyFormState: ComponentState = ComponentStates.ENABLED;\n  private _mfaMethod: 'TOTP' | 'EMAIL' | 'SMS' = 'TOTP';\n\n  get mfaVerifyFormProps(): FormMetadata {\n    const sectionName =\n      this._mfaMethod === 'TOTP'\n        ? this.t('mfaTOTP')\n        : this._mfaMethod === 'EMAIL'\n          ? this.t('mfaEmail')\n          : this.t('mfaSMS');\n\n    return this.i18nHelper.resolveForm({\n      nameKey: 'mfaTitle',\n      i18nNamespace: '_auth',\n      sections: [\n        {\n          name: sectionName,\n          order: 0,\n          fields: [\n            {\n              type: InputType.PIN_CODE,\n              label: '',\n              name: 'code',\n              token: 'mfa-pin',\n              hint: '',\n              placeholder: '',\n              errorKeys: {\n                required: 'codeRequired',\n                minlength: 'codeMinLength',\n              },\n              validators: [Validators.required, Validators.minLength(6)],\n              order: 0,\n              state: ComponentStates.ENABLED,\n              length: 6,\n              allowNumbersOnly: true,\n              autoFocus: true,\n            },\n          ],\n        },\n      ],\n      actions: {\n        ...SolidDefaultBlock('', 'submit'),\n        token: 'mfa-verify-submit',\n        textKey: 'verifySubmit',\n      },\n      state: this._mfaVerifyFormState,\n    });\n  }\n\n  set mfaVerifyFormState(value: ComponentState) {\n    this._mfaVerifyFormState = value;\n  }\n\n  set mfaMethod(value: 'TOTP' | 'EMAIL' | 'SMS') {\n    this._mfaMethod = value;\n  }\n\n  // ==========================================\n  // FORGOT PASSWORD FORM\n  // ==========================================\n\n  private _forgotPasswordFormState: ComponentState = ComponentStates.ENABLED;\n\n  get forgotPasswordFormProps(): FormMetadata {\n    return this.i18nHelper.resolveForm({\n      nameKey: 'forgotTitle',\n      i18nNamespace: '_auth',\n      sections: [\n        {\n          name: this.t('forgotDescription'),\n          order: 0,\n          fields: [\n            {\n              type: InputType.EMAIL,\n              label: '',\n              name: 'email',\n              token: 'forgot-email',\n              hint: '',\n              placeholderKey: 'emailPlaceholder',\n              errorKeys: {\n                required: 'emailRequired',\n                email: 'emailInvalid',\n              },\n              validators: [Validators.required, Validators.email],\n              order: 0,\n              state: ComponentStates.ENABLED,\n            },\n          ],\n        },\n      ],\n      actions: {\n        ...SolidDefaultBlock('', 'submit'),\n        token: 'forgot-submit',\n        textKey: 'forgotSubmit',\n      },\n      state: this._forgotPasswordFormState,\n    });\n  }\n\n  set forgotPasswordFormState(value: ComponentState) {\n    this._forgotPasswordFormState = value;\n  }\n\n  // ==========================================\n  // RESET PASSWORD FORM\n  // ==========================================\n\n  private _resetPasswordFormState: ComponentState = ComponentStates.ENABLED;\n  private _resetFormSectionName = '';\n\n  get resetPasswordFormProps(): FormMetadata {\n    return this.i18nHelper.resolveForm({\n      nameKey: 'resetTitle',\n      i18nNamespace: '_auth',\n      sections: [\n        {\n          name: this._resetFormSectionName || this.t('resetDescription'),\n          order: 0,\n          fields: [\n            {\n              type: InputType.PIN_CODE,\n              label: '',\n              name: 'code',\n              token: 'reset-pin',\n              hint: '',\n              placeholder: '',\n              errorKeys: {\n                required: 'codeRequired',\n                minlength: 'codeMinLength',\n              },\n              validators: [Validators.required, Validators.minLength(6)],\n              order: 0,\n              state: ComponentStates.ENABLED,\n              length: 6,\n              allowNumbersOnly: true,\n              autoFocus: true,\n            },\n            {\n              type: InputType.PASSWORD,\n              name: 'newPassword',\n              token: 'reset-new-password',\n              labelKey: 'newPassword',\n              hintKey: 'newPasswordHint',\n              placeholderKey: 'passwordPlaceholder',\n              errorKeys: {\n                required: 'passwordRequired',\n                minlength: 'passwordMinLength',\n              },\n              validators: [Validators.required, Validators.minLength(8)],\n              order: 1,\n              state: ComponentStates.ENABLED,\n            },\n          ],\n        },\n      ],\n      actions: {\n        ...SolidDefaultBlock('', 'submit'),\n        token: 'reset-submit',\n        textKey: 'resetSubmit',\n      },\n      state: this._resetPasswordFormState,\n    });\n  }\n\n  set resetPasswordFormState(value: ComponentState) {\n    this._resetPasswordFormState = value;\n  }\n\n  set resetFormSectionName(value: string) {\n    this._resetFormSectionName = value;\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(this.t('completeAllFields'));\n      return;\n    }\n\n    this._loginFormState = ComponentStates.WORKING;\n\n    this.authService.signin({ email, password }).subscribe({\n      next: () => {\n        this._loginFormState = 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._loginFormState = 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(this.t('completeAllFields'));\n      return;\n    }\n\n    this._registerFormState = ComponentStates.WORKING;\n\n    this.authService.signup({ name, email, password }).subscribe({\n      next: () => {\n        this._registerFormState = ComponentStates.ENABLED;\n        this.closeRegisterModal();\n        this.openVerifyModal(email);\n      },\n      error: (err) => {\n        this._registerFormState = 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._verifyFormSectionName = `${this.t('verifyDescription').replace('.', '')} (${email})`;\n    this._verifyFormState = ComponentStates.ENABLED;\n    this.isVerifyModalOpen = true;\n    this.startResendCooldown();\n  }\n\n  closeVerifyModal(): void {\n    this.isVerifyModalOpen = false;\n    this.pendingVerificationEmail = '';\n    this._verifyFormSectionName = '';\n    this.stopResendCooldown();\n  }\n\n  verifyHandler(event: FormSubmit): void {\n    const code = event.fields['code'];\n\n    this._verifyFormState = ComponentStates.WORKING;\n\n    this.authService\n      .verifyEmail({\n        email: this.pendingVerificationEmail,\n        code,\n      })\n      .subscribe({\n        next: () => {\n          this._verifyFormState = ComponentStates.ENABLED;\n          this.showToast(this.t('emailVerified'));\n          this.closeVerifyModal();\n          this.handleLoginSuccess();\n        },\n        error: (err) => {\n          this._verifyFormState = 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(this.t('codeSent'));\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 as 'TOTP' | 'EMAIL' | 'SMS';\n    this._mfaMethod = method;\n    this._mfaVerifyFormState = ComponentStates.ENABLED;\n    this.isMFAVerifyModalOpen = true;\n\n    this.onMFARequired.emit({ method });\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._mfaVerifyFormState = ComponentStates.WORKING;\n\n    this.authService.verifyMFA(code).subscribe({\n      next: () => {\n        this._mfaVerifyFormState = ComponentStates.ENABLED;\n        this.closeMFAVerifyModal();\n        this.handleLoginSuccess(undefined, true);\n      },\n      error: (err) => {\n        this._mfaVerifyFormState = 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(this.t('enterEmail'));\n      return;\n    }\n\n    this._forgotPasswordFormState = ComponentStates.WORKING;\n\n    this.authService.forgotPassword({ email }).subscribe({\n      next: () => {\n        this._forgotPasswordFormState = ComponentStates.ENABLED;\n        this.closeForgotPasswordModal();\n        this.openResetPasswordModal(email);\n        this.showToast(this.t('codeSent'));\n      },\n      error: (err) => {\n        this._forgotPasswordFormState = 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._resetFormSectionName = `${this.t('resetDescription').replace('.', '')} (${email})`;\n    this._resetPasswordFormState = ComponentStates.ENABLED;\n    this.isResetPasswordModalOpen = true;\n    this.startResetResendCooldown();\n  }\n\n  closeResetPasswordModal(): void {\n    this.isResetPasswordModalOpen = false;\n    this.pendingResetEmail = '';\n    this._resetFormSectionName = '';\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._resetPasswordFormState = ComponentStates.WORKING;\n\n    this.authService\n      .resetPassword({\n        email: this.pendingResetEmail,\n        code,\n        newPassword,\n      })\n      .subscribe({\n        next: () => {\n          this._resetPasswordFormState = ComponentStates.ENABLED;\n          this.showToast(this.t('passwordUpdated'));\n          this.closeResetPasswordModal();\n        },\n        error: (err) => {\n          this._resetPasswordFormState = 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(this.t('codeSent'));\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(this.t('welcome'));\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    // Map error codes to i18n keys\n    const errorKeyMap: Record<string, string> = {\n      // Signin\n      AUTHV2_INVALID_CREDENTIALS: 'errorInvalidCredentials',\n      AUTHV2_EMAIL_NOT_VERIFIED: 'errorEmailNotVerified',\n      AUTHV2_ACCOUNT_SUSPENDED: 'errorAccountSuspended',\n\n      // Signup\n      AUTHV2_EMAIL_EXISTS: 'errorEmailExists',\n      AUTHV2_WEAK_PASSWORD: 'errorWeakPassword',\n\n      // Verification codes\n      AUTHV2_INVALID_CODE: 'errorInvalidCode',\n      AUTHV2_EXPIRED_CODE: 'errorExpiredCode',\n      AUTHV2_CODE_EXPIRED: 'errorExpiredCode',\n      AUTHV2_TOO_MANY_ATTEMPTS: 'errorTooManyAttempts',\n\n      // MFA\n      AUTHV2_MFA_INVALID_CODE: 'errorMFAInvalidCode',\n\n      // OAuth\n      POPUP_BLOCKED: 'errorPopupBlocked',\n      OAUTH_FAILED: 'errorOAuthFailed',\n    };\n\n    const errorCode = error?.error?.code || error?.code;\n    if (errorCode && errorKeyMap[errorCode]) {\n      return this.t(errorKeyMap[errorCode]);\n    }\n\n    if (error?.error?.message) {\n      return error.error.message;\n    }\n\n    return this.t('errorGeneric');\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\" [class.val-login--card]=\"config.showCard\">\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>{{ t('orContinueWith') }}</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 ? t('connecting') : t('continueWithGoogle') }}\n            }\n            @case ('apple') {\n              <ion-icon slot=\"start\" name=\"logo-apple\"></ion-icon>\n              {{ isOAuthLoading ? t('connecting') : t('continueWithApple') }}\n            }\n            @case ('microsoft') {\n              <ion-icon slot=\"start\" name=\"logo-microsoft\"></ion-icon>\n              {{ isOAuthLoading ? t('connecting') : t('continueWithMicrosoft') }}\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        {{ t('noAccount') }}\n        <a (click)=\"openRegisterModal()\">{{ t('register') }}</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        {{ t('forgotLink') }}\n        <a (click)=\"openForgotPasswordModal()\">{{ t('recoverPassword') }}</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          {{ t('legalPrefix') }}\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          {{ t('legalSuffix') }}\n          @if (props.legal.termsLink) {\n            <a [href]=\"props.legal.termsLink\">{{ t('termsAndConditions') }}</a>\n          } @else {\n            <span>{{ t('termsAndConditions') }}</span>\n          }\n          {{ t('and') }}\n          @if (props.legal.privacyLink) {\n            <a [href]=\"props.legal.privacyLink\">{{ t('privacyPolicy') }}</a>\n          } @else {\n            <span>{{ t('privacyPolicy') }}</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            {{ t('hasAccount') }}\n            <a (click)=\"closeRegisterModal()\">{{ t('signIn') }}</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            {{ t('noCodeReceived') }}\n            @if (resendCooldown > 0) {\n              <span class=\"cooldown\">{{ t('resendIn', { seconds: resendCooldown.toString() }) }}</span>\n            } @else {\n              <a (click)=\"resendCode()\">{{ t('resend') }}</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            {{ t('noCodeReceived') }}\n            @if (resetResendCooldown > 0) {\n              <span class=\"cooldown\">{{ t('resendIn', { seconds: resetResendCooldown.toString() }) }}</span>\n            } @else {\n              <a (click)=\"resendResetCode()\">{{ t('resend') }}</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"]}