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.
- package/esm2022/lib/components/organisms/login/login.component.mjs +353 -306
- package/esm2022/lib/services/i18n/default-content.mjs +165 -1
- package/esm2022/lib/services/i18n/index.mjs +3 -1
- package/esm2022/lib/services/i18n/input-i18n.helper.mjs +93 -0
- package/fesm2022/valtech-components.mjs +606 -306
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/atoms/rights-footer/rights-footer.component.d.ts +1 -1
- package/lib/components/atoms/text/text.component.d.ts +1 -1
- package/lib/components/organisms/article/article.component.d.ts +3 -3
- package/lib/components/organisms/login/login.component.d.ts +32 -17
- package/lib/services/i18n/index.d.ts +1 -0
- package/lib/services/i18n/input-i18n.helper.d.ts +69 -0
- package/package.json +1 -1
|
@@ -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.
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
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: [
|
|
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('
|
|
122
|
+
...SolidDefaultBlock('', 'submit'),
|
|
83
123
|
token: 'login-submit',
|
|
124
|
+
textKey: 'loginSubmit',
|
|
84
125
|
},
|
|
85
|
-
state:
|
|
86
|
-
};
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
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: [
|
|
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('
|
|
190
|
+
...SolidDefaultBlock('', 'submit'),
|
|
149
191
|
token: 'register-submit',
|
|
192
|
+
textKey: 'registerSubmit',
|
|
150
193
|
},
|
|
151
|
-
state:
|
|
152
|
-
};
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
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:
|
|
206
|
+
name: this._verifyFormSectionName || this.t('verifyDescription'),
|
|
180
207
|
order: 0,
|
|
181
|
-
fields: [
|
|
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('
|
|
231
|
+
...SolidDefaultBlock('', 'submit'),
|
|
186
232
|
token: 'verify-submit',
|
|
233
|
+
textKey: 'verifySubmit',
|
|
187
234
|
},
|
|
188
|
-
state:
|
|
189
|
-
};
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
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: [
|
|
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('
|
|
280
|
+
...SolidDefaultBlock('', 'submit'),
|
|
223
281
|
token: 'mfa-verify-submit',
|
|
282
|
+
textKey: 'verifySubmit',
|
|
224
283
|
},
|
|
225
|
-
state:
|
|
226
|
-
};
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
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:
|
|
299
|
+
name: this.t('forgotDescription'),
|
|
251
300
|
order: 0,
|
|
252
|
-
fields: [
|
|
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('
|
|
321
|
+
...SolidDefaultBlock('', 'submit'),
|
|
257
322
|
token: 'forgot-submit',
|
|
323
|
+
textKey: 'forgotSubmit',
|
|
258
324
|
},
|
|
259
|
-
state:
|
|
260
|
-
};
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
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:
|
|
337
|
+
name: this._resetFormSectionName || this.t('resetDescription'),
|
|
304
338
|
order: 0,
|
|
305
|
-
fields: [
|
|
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('
|
|
377
|
+
...SolidDefaultBlock('', 'submit'),
|
|
310
378
|
token: 'reset-submit',
|
|
379
|
+
textKey: 'resetSubmit',
|
|
311
380
|
},
|
|
312
|
-
state:
|
|
313
|
-
};
|
|
381
|
+
state: this._resetPasswordFormState,
|
|
382
|
+
});
|
|
314
383
|
}
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
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(
|
|
397
|
+
this.showToast(this.t('completeAllFields'));
|
|
327
398
|
return;
|
|
328
399
|
}
|
|
329
|
-
this.
|
|
400
|
+
this._loginFormState = ComponentStates.WORKING;
|
|
330
401
|
this.authService.signin({ email, password }).subscribe({
|
|
331
402
|
next: () => {
|
|
332
|
-
this.
|
|
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.
|
|
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(
|
|
456
|
+
this.showToast(this.t('completeAllFields'));
|
|
386
457
|
return;
|
|
387
458
|
}
|
|
388
|
-
this.
|
|
459
|
+
this._registerFormState = ComponentStates.WORKING;
|
|
389
460
|
this.authService.signup({ name, email, password }).subscribe({
|
|
390
461
|
next: () => {
|
|
391
|
-
this.
|
|
462
|
+
this._registerFormState = ComponentStates.ENABLED;
|
|
392
463
|
this.closeRegisterModal();
|
|
393
464
|
this.openVerifyModal(email);
|
|
394
465
|
},
|
|
395
466
|
error: (err) => {
|
|
396
|
-
this.
|
|
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.
|
|
407
|
-
this.
|
|
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.
|
|
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.
|
|
427
|
-
this.showToast(
|
|
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.
|
|
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(
|
|
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.
|
|
461
|
-
|
|
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
|
|
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.
|
|
542
|
+
this._mfaVerifyFormState = ComponentStates.WORKING;
|
|
476
543
|
this.authService.verifyMFA(code).subscribe({
|
|
477
544
|
next: () => {
|
|
478
|
-
this.
|
|
545
|
+
this._mfaVerifyFormState = ComponentStates.ENABLED;
|
|
479
546
|
this.closeMFAVerifyModal();
|
|
480
547
|
this.handleLoginSuccess(undefined, true);
|
|
481
548
|
},
|
|
482
549
|
error: (err) => {
|
|
483
|
-
this.
|
|
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(
|
|
567
|
+
this.showToast(this.t('enterEmail'));
|
|
501
568
|
return;
|
|
502
569
|
}
|
|
503
|
-
this.
|
|
570
|
+
this._forgotPasswordFormState = ComponentStates.WORKING;
|
|
504
571
|
this.authService.forgotPassword({ email }).subscribe({
|
|
505
572
|
next: () => {
|
|
506
|
-
this.
|
|
573
|
+
this._forgotPasswordFormState = ComponentStates.ENABLED;
|
|
507
574
|
this.closeForgotPasswordModal();
|
|
508
575
|
this.openResetPasswordModal(email);
|
|
509
|
-
this.showToast(
|
|
576
|
+
this.showToast(this.t('codeSent'));
|
|
510
577
|
},
|
|
511
578
|
error: (err) => {
|
|
512
|
-
this.
|
|
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.
|
|
523
|
-
this.
|
|
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.
|
|
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.
|
|
545
|
-
this.showToast(
|
|
612
|
+
this._resetPasswordFormState = ComponentStates.ENABLED;
|
|
613
|
+
this.showToast(this.t('passwordUpdated'));
|
|
546
614
|
this.closeResetPasswordModal();
|
|
547
615
|
},
|
|
548
616
|
error: (err) => {
|
|
549
|
-
this.
|
|
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(
|
|
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('
|
|
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
|
-
|
|
606
|
-
|
|
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: '
|
|
611
|
-
AUTHV2_EMAIL_NOT_VERIFIED: '
|
|
612
|
-
AUTHV2_ACCOUNT_SUSPENDED: '
|
|
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: '
|
|
616
|
-
|
|
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: '
|
|
621
|
-
AUTHV2_EXPIRED_CODE: '
|
|
622
|
-
AUTHV2_CODE_EXPIRED: '
|
|
623
|
-
|
|
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: '
|
|
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: '
|
|
641
|
-
|
|
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 &&
|
|
648
|
-
return
|
|
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
|
|
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>
|
|
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"]}
|