ngx-dsxlibrary 1.0.0 → 1.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,553 @@
1
+ import * as i0 from '@angular/core';
2
+ import { InjectionToken, signal, Injectable, inject, Component, isDevMode, Input, input, Pipe } from '@angular/core';
3
+ import { HttpClient, HttpHeaders, HttpParams, HttpStatusCode, HttpErrorResponse } from '@angular/common/http';
4
+ import { throwError, catchError, switchMap, finalize, Observable } from 'rxjs';
5
+ import { JwtHelperService } from '@auth0/angular-jwt';
6
+ import { CookieService } from 'ngx-cookie-service';
7
+ import { ToastrService } from 'ngx-toastr';
8
+ import Swal from 'sweetalert2';
9
+ import * as i1 from 'primeng/tag';
10
+ import { TagModule } from 'primeng/tag';
11
+ import * as i1$1 from 'ngx-countup';
12
+ import { CountUpModule } from 'ngx-countup';
13
+
14
+ const ENVIRONMENT = new InjectionToken('EnvironmentConfig');
15
+
16
+ class SpinnerLoadingService {
17
+ // Creamos una señal para el estado de visibilidad
18
+ spinnerVisible = signal(false);
19
+ constructor() { }
20
+ // Método para mostrar el spinner
21
+ show() {
22
+ this.spinnerVisible.set(true);
23
+ }
24
+ // Método para ocultar el spinner
25
+ hide() {
26
+ this.spinnerVisible.set(false);
27
+ }
28
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: SpinnerLoadingService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
29
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: SpinnerLoadingService, providedIn: 'root' });
30
+ }
31
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: SpinnerLoadingService, decorators: [{
32
+ type: Injectable,
33
+ args: [{
34
+ providedIn: 'root',
35
+ }]
36
+ }], ctorParameters: () => [] });
37
+
38
+ class LoadingComponent {
39
+ // Accedemos directamente a la señal del servicio
40
+ _spinnerService = inject(SpinnerLoadingService);
41
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: LoadingComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
42
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "19.2.0", type: LoadingComponent, isStandalone: true, selector: "app-loading", ngImport: i0, template: "<!-- Actualizaci\u00F3n 2025-31-01 12:00 -->\r\n@if(_spinnerService.spinnerVisible()){\r\n<div class=\"spinner-overlay\">\r\n <div class=\"loader\">\r\n <div class=\"external-shadow\">\r\n <div class=\"central\"></div>\r\n </div>\r\n <img src=\"assets/icon/secure.png\" class=\"spinner-image\" />\r\n </div>\r\n <p class=\"loading-text\">Desarollo Software Xela</p>\r\n</div>\r\n}\r\n", styles: ["@charset \"UTF-8\";.spinner-overlay{position:fixed;top:0;left:0;width:100vw;height:100vh;background-color:#000c;display:flex;flex-direction:column;justify-content:center;align-items:center;z-index:9999}.loader{display:flex;justify-content:center;align-items:center;position:relative;cursor:not-allowed;scale:.7}.central{display:flex;justify-content:center;align-items:center;position:relative;width:10em;height:10em;border-radius:50%;box-shadow:.5em 1em 1em #8a2be2,-.5em .5em 1em #00f,.5em -.5em 1em purple,-.5em -.5em 1em #0ff;background-color:#0000004d}.external-shadow{width:10em;height:10em;border-radius:50%;display:flex;justify-content:center;align-items:center;position:relative;box-shadow:.5em .5em 3em #8a2be2,-.5em .5em 3em #00f,.5em -.5em 3em purple,-.5em -.5em 3em #0ff;z-index:999;animation:rotate 3s linear infinite;background-color:#21212180}@keyframes rotate{0%{transform:rotate(0)}50%{transform:rotate(180deg)}to{transform:rotate(360deg)}}.spinner-image{width:70%;height:70%;border-radius:50%;object-fit:cover;position:absolute;z-index:1000}.loading-text{font-family:Montserrat,sans-serif;text-transform:uppercase;letter-spacing:3px;background:linear-gradient(45deg,#0ff,#f0f);-webkit-background-clip:text;background-clip:text;color:transparent}@keyframes pulse{0%,to{opacity:.7}50%{opacity:1}}\n"] });
43
+ }
44
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: LoadingComponent, decorators: [{
45
+ type: Component,
46
+ args: [{ selector: 'app-loading', imports: [], template: "<!-- Actualizaci\u00F3n 2025-31-01 12:00 -->\r\n@if(_spinnerService.spinnerVisible()){\r\n<div class=\"spinner-overlay\">\r\n <div class=\"loader\">\r\n <div class=\"external-shadow\">\r\n <div class=\"central\"></div>\r\n </div>\r\n <img src=\"assets/icon/secure.png\" class=\"spinner-image\" />\r\n </div>\r\n <p class=\"loading-text\">Desarollo Software Xela</p>\r\n</div>\r\n}\r\n", styles: ["@charset \"UTF-8\";.spinner-overlay{position:fixed;top:0;left:0;width:100vw;height:100vh;background-color:#000c;display:flex;flex-direction:column;justify-content:center;align-items:center;z-index:9999}.loader{display:flex;justify-content:center;align-items:center;position:relative;cursor:not-allowed;scale:.7}.central{display:flex;justify-content:center;align-items:center;position:relative;width:10em;height:10em;border-radius:50%;box-shadow:.5em 1em 1em #8a2be2,-.5em .5em 1em #00f,.5em -.5em 1em purple,-.5em -.5em 1em #0ff;background-color:#0000004d}.external-shadow{width:10em;height:10em;border-radius:50%;display:flex;justify-content:center;align-items:center;position:relative;box-shadow:.5em .5em 3em #8a2be2,-.5em .5em 3em #00f,.5em -.5em 3em purple,-.5em -.5em 3em #0ff;z-index:999;animation:rotate 3s linear infinite;background-color:#21212180}@keyframes rotate{0%{transform:rotate(0)}50%{transform:rotate(180deg)}to{transform:rotate(360deg)}}.spinner-image{width:70%;height:70%;border-radius:50%;object-fit:cover;position:absolute;z-index:1000}.loading-text{font-family:Montserrat,sans-serif;text-transform:uppercase;letter-spacing:3px;background:linear-gradient(45deg,#0ff,#f0f);-webkit-background-clip:text;background-clip:text;color:transparent}@keyframes pulse{0%,to{opacity:.7}50%{opacity:1}}\n"] }]
47
+ }] });
48
+
49
+ class AuthorizeService {
50
+ environment = inject(ENVIRONMENT);
51
+ _cookieService = inject(CookieService);
52
+ helperJwt = new JwtHelperService();
53
+ _isRefreshing = false;
54
+ // Función para obtener opciones de cookies estándar
55
+ getCookieOptions(expiryDate) {
56
+ const isProduction = this.environment.production;
57
+ const cookieOptions = {
58
+ path: '/',
59
+ expires: expiryDate,
60
+ secure: isProduction,
61
+ sameSite: isProduction ? 'None' : 'Lax', // sameSite: 'None',
62
+ };
63
+ if (isProduction) {
64
+ cookieOptions.domain = '.itgtxela.com';
65
+ }
66
+ return cookieOptions;
67
+ }
68
+ // Función para establecer una cookie
69
+ setCookie(name, value, expiryDate) {
70
+ const options = this.getCookieOptions(expiryDate);
71
+ this._cookieService.set(name, value, options);
72
+ }
73
+ get isRefreshing() {
74
+ return this._isRefreshing;
75
+ }
76
+ set isRefreshing(value) {
77
+ this._isRefreshing = value;
78
+ }
79
+ getToken() {
80
+ return this._cookieService.get(this.environment.tokenName);
81
+ //return localStorage.getItem(environment.tokenName);
82
+ }
83
+ getTokenRefresh() {
84
+ //USO EN MODO DESARROLLO
85
+ if (isDevMode()) {
86
+ const tokenRefresh = this._cookieService.get(this.environment.tokenNameRF);
87
+ const tokens = {
88
+ token: '',
89
+ tokenRefresh: '8508408a-6cbc-4ebb-b105-d851655a3b0c',
90
+ refreshTokenExpiry: new Date(),
91
+ };
92
+ !!tokenRefresh ? null : this.tokenReload(tokens);
93
+ }
94
+ return this._cookieService.get(this.environment.tokenNameRF);
95
+ //return localStorage.getItem(environment.tokenNameRF);
96
+ }
97
+ // Actualizar valores de status
98
+ setLastActivity() {
99
+ const tokenExpiryDate = new Date();
100
+ // Expira en 30 minutos
101
+ tokenExpiryDate.setMinutes(tokenExpiryDate.getMinutes() + 30);
102
+ this.setCookie(this.environment.sessionStatus, Date.now().toString(), tokenExpiryDate);
103
+ }
104
+ // Almacenar los tokens en cookies
105
+ tokenReload(tokens) {
106
+ const tokenExpiryDate = new Date();
107
+ tokenExpiryDate.setMinutes(tokenExpiryDate.getMinutes() + 30); // Access token expiry: 30 mins
108
+ const refreshTokenExpiryDate = new Date();
109
+ refreshTokenExpiryDate.setDate(refreshTokenExpiryDate.getDate() + 7); // Refresh token expiry: 7 days
110
+ // Guardar el access token y refresh token
111
+ this.setCookie(this.environment.tokenName, tokens.token, tokenExpiryDate);
112
+ this.setCookie(this.environment.tokenNameRF, tokens.tokenRefresh, refreshTokenExpiryDate);
113
+ // Guardar el estado de sesión
114
+ this.setLastActivity();
115
+ // Almacenar la fecha de expiración del refresh token en localStorage
116
+ const expiryDate = new Date(tokens.refreshTokenExpiry).toISOString();
117
+ localStorage.setItem(this.environment.refreshTokenExpiry, expiryDate);
118
+ }
119
+ getTokenValid(token) {
120
+ // Devuelve true si el token NO ha expirado
121
+ return !this.helperJwt.isTokenExpired(token);
122
+ }
123
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: AuthorizeService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
124
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: AuthorizeService, providedIn: 'root' });
125
+ }
126
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: AuthorizeService, decorators: [{
127
+ type: Injectable,
128
+ args: [{
129
+ providedIn: 'root',
130
+ }]
131
+ }] });
132
+
133
+ class SecurityService {
134
+ http = inject(HttpClient);
135
+ environment = inject(ENVIRONMENT);
136
+ urlApi = `${this.environment.myAppUrl}api/authorize`;
137
+ tokenRefresh(refreshToken) {
138
+ const body = { refreshToken }; // Enviar el token en formato JSON
139
+ return this.http.post(`${this.urlApi}/TokenRefresh/`, body, {
140
+ headers: new HttpHeaders({
141
+ 'Content-Type': 'application/json', // Asegúrate de que se está enviando como JSON
142
+ }),
143
+ });
144
+ }
145
+ getParameterSecurity(invalidCacheParam = false) {
146
+ const params = new HttpParams().set('invalidCacheParam', invalidCacheParam);
147
+ return this.http.get(`${this.urlApi}/securityParameter/`, { params });
148
+ }
149
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: SecurityService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
150
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: SecurityService, providedIn: 'root' });
151
+ }
152
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: SecurityService, decorators: [{
153
+ type: Injectable,
154
+ args: [{
155
+ providedIn: 'root',
156
+ }]
157
+ }] });
158
+
159
+ class AlertaService {
160
+ //Update: 2024-10-28 14:58 Arrendamientos
161
+ toastrService = inject(ToastrService);
162
+ /**
163
+ * @param {number} toastrAlign - Alineación de la alerta, por defecto 1. OPCIONES: 0. top-left 1. top-center 2. top-right 3. bottom-left 5. bottom-center 6. bottom-right
164
+ * @param {number} toastrType - 1. Success 2. Info 3. Warning 4. Error
165
+ * @param {number} toastrTimer - Tiempo de la alerta default 3000
166
+ * @param {string} toastrTitle - Titulo de la alerta
167
+ * @param {string} toastrMessage - Mensaje de la alerta
168
+ * @returns - Retonar una alerta toastr
169
+ * */
170
+ toastrAlerts(toastrType, toastrTitle, toastrMessage, toastrAlign = 1, toastrTimer = 3000) {
171
+ const alignMessage = [
172
+ { id: 0, align: 'toast-top-left' },
173
+ { id: 1, align: 'toast-top-center' },
174
+ { id: 2, align: 'toast-top-right' },
175
+ { id: 3, align: 'toast-bottom-left' },
176
+ { id: 4, align: 'toast-bottom-center' },
177
+ { id: 5, align: 'toast-bottom-right' },
178
+ ];
179
+ const valueAlign = alignMessage.find((c) => c.id == toastrAlign)?.align;
180
+ const toastrMethods = {
181
+ 1: this.toastrService.success.bind(this.toastrService),
182
+ 2: this.toastrService.info.bind(this.toastrService),
183
+ 3: this.toastrService.warning.bind(this.toastrService),
184
+ 4: this.toastrService.error.bind(this.toastrService),
185
+ };
186
+ const showToast = toastrMethods[toastrType];
187
+ if (showToast) {
188
+ showToast(toastrMessage, toastrTitle, {
189
+ enableHtml: true,
190
+ closeButton: true,
191
+ progressBar: true,
192
+ positionClass: valueAlign,
193
+ timeOut: toastrTimer,
194
+ });
195
+ }
196
+ }
197
+ alertaHtml(titleAlert, message) {
198
+ Swal.fire({
199
+ title: titleAlert,
200
+ imageUrl: 'assets/image/notFound01.png',
201
+ imageWidth: 145,
202
+ imageHeight: 125,
203
+ imageAlt: 'error 404',
204
+ html: `
205
+ <strong>ALERTA: </strong><i class='text-red-300'>${message}</i> <span class='text-blue-600'> comuniquese con el administrador. </span>
206
+ `,
207
+ showConfirmButton: false,
208
+ timerProgressBar: true,
209
+ timer: 2500,
210
+ });
211
+ }
212
+ alertaHtmlSuccess(titleAlert, message) {
213
+ Swal.fire({
214
+ title: titleAlert,
215
+ imageUrl: 'https://firebasestorage.googleapis.com/v0/b/seguridadintegrada-36204.appspot.com/o/Logos%2FSuccess.png?alt=media&token=275135d6-277e-4619-baa5-f76da95ca0eb',
216
+ imageWidth: 150,
217
+ imageHeight: 120,
218
+ imageAlt: 'success 200',
219
+ html: `
220
+ <strong>Alerta: </strong><b class='text-success'>${message}</b>
221
+ `,
222
+ showConfirmButton: false,
223
+ timerProgressBar: true,
224
+ timer: 2000,
225
+ });
226
+ }
227
+ preloadImage(icono) {
228
+ return new Promise((resolve, reject) => {
229
+ const img = new Image();
230
+ img.src = `assets/icon/${icono}`;
231
+ img.onload = () => resolve();
232
+ img.onerror = (err) => reject(err);
233
+ });
234
+ }
235
+ alertConfirm(title, text, icono) {
236
+ return this.preloadImage(icono).then(() => Swal.fire({
237
+ title: title,
238
+ html: text,
239
+ footer: '<strong>DevSoftXela</strong> 2025',
240
+ imageUrl: `assets/icon/${icono}`,
241
+ imageWidth: 150,
242
+ imageHeight: 150,
243
+ imageAlt: 'icon',
244
+ showCloseButton: true,
245
+ showCancelButton: true,
246
+ confirmButtonColor: '#3085d6',
247
+ cancelButtonColor: '#d33',
248
+ confirmButtonText: `<i class="fa fa-thumbs-up"></i> Aceptar!`,
249
+ cancelButtonText: `<i class="fa fa-thumbs-down"></i> Cancelar`,
250
+ }).then((result) => result.isConfirmed));
251
+ }
252
+ //notifyAlertSuccess(message: string) {
253
+ // this._notyf.success({
254
+ // message: message,
255
+ // duration: 1500,
256
+ // position: {
257
+ // x: 'center',
258
+ // y: 'top'
259
+ // }
260
+ // });
261
+ //}
262
+ toastrHttpResponse(response) {
263
+ const time = 3000;
264
+ const align = 'toast-top-center';
265
+ if (response.isSuccess) {
266
+ this.toastrService.success(response.statusMessage, response.title, {
267
+ enableHtml: true,
268
+ closeButton: true,
269
+ progressBar: true,
270
+ positionClass: align,
271
+ timeOut: time,
272
+ });
273
+ }
274
+ else {
275
+ this.toastrService.error(response.statusMessage, response.title, {
276
+ enableHtml: true,
277
+ closeButton: true,
278
+ progressBar: true,
279
+ positionClass: align,
280
+ timeOut: time,
281
+ });
282
+ }
283
+ }
284
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: AlertaService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
285
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: AlertaService, providedIn: 'root' });
286
+ }
287
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: AlertaService, decorators: [{
288
+ type: Injectable,
289
+ args: [{
290
+ providedIn: 'root',
291
+ }]
292
+ }] });
293
+
294
+ class ErrorHandlerService {
295
+ _serviceAlerta = inject(AlertaService);
296
+ handleErrorResponse(error) {
297
+ const err = error;
298
+ const technicalMessage = `Error status: ${error.status}, Message: ${error.message}, URL: ${error.url}`;
299
+ let userMessage = 'Ha ocurrido un error inesperado.';
300
+ switch (error.status) {
301
+ case HttpStatusCode.BadRequest:
302
+ userMessage =
303
+ 'Solicitud incorrecta (400). Verifica los datos ingresados o validaciones existentes.';
304
+ break;
305
+ case HttpStatusCode.Unauthorized:
306
+ userMessage = 'Acceso no autorizado (401). Por favor inicia sesión.';
307
+ break;
308
+ case HttpStatusCode.Forbidden:
309
+ userMessage = 'No tienes permisos para realizar esta acción (403).';
310
+ break;
311
+ case HttpStatusCode.NotFound:
312
+ userMessage = 'No se encontró el recurso solicitado (404).';
313
+ break;
314
+ case HttpStatusCode.Conflict: // Capturar error de concurrencia
315
+ userMessage =
316
+ 'El registro fue modificado por otro usuario. Por favor, actualiza los datos antes de continuar.';
317
+ break;
318
+ case HttpStatusCode.InternalServerError:
319
+ userMessage =
320
+ 'Ocurrió un error en el servidor (500). Inténtalo más tarde.';
321
+ break;
322
+ default:
323
+ userMessage = 'Un error inesperado ha ocurrido.';
324
+ break;
325
+ }
326
+ // Mostrar mensaje para el usuario
327
+ this._serviceAlerta.alertaHtml('Servicios de Errores', '<i>Código:</i> ' +
328
+ err.status +
329
+ ' <i>Message:</i> ' +
330
+ userMessage +
331
+ `. (${error.error})`);
332
+ // Imprimir el mensaje técnico
333
+ if (isDevMode()) {
334
+ console.error(err);
335
+ }
336
+ // Retornar el error para continuar con el flujo de manejo de errores
337
+ return throwError(() => new Error(technicalMessage));
338
+ }
339
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: ErrorHandlerService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
340
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: ErrorHandlerService, providedIn: 'root' });
341
+ }
342
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: ErrorHandlerService, decorators: [{
343
+ type: Injectable,
344
+ args: [{
345
+ providedIn: 'root',
346
+ }]
347
+ }] });
348
+
349
+ //Update: 2024-01-29 12:00
350
+ let _activeRequest = 0;
351
+ //let isRefreshing = false; // Estado de si se está refrescando el token
352
+ let pendingRequests = []; // Cola de solicitudes pendientes
353
+ const httpAuthorizeInterceptor = (req, next) => {
354
+ //console.log('------ INGRESO A HTTPINTERCEPTOR -------');
355
+ const _authorizeService = inject(AuthorizeService);
356
+ const _securityService = inject(SecurityService);
357
+ const _spinnerService = inject(SpinnerLoadingService);
358
+ const _handleErrorService = inject(ErrorHandlerService);
359
+ //Variables
360
+ const _token = _authorizeService.getToken();
361
+ const _tokenRefresh = _authorizeService.getTokenRefresh();
362
+ //Generar la petición con el token
363
+ const authReq = req.clone({
364
+ setHeaders: {
365
+ Authorization: `Bearer ${_token}`,
366
+ },
367
+ });
368
+ if (_activeRequest == 0) {
369
+ _spinnerService.show();
370
+ }
371
+ _activeRequest++;
372
+ return next(authReq).pipe(catchError((error) => {
373
+ // Si el error es 401
374
+ if (error.status === HttpStatusCode.Unauthorized) {
375
+ // Si no hay refreshToken, finalizar la sesión
376
+ if (!_tokenRefresh) {
377
+ _spinnerService.hide();
378
+ return _handleErrorService.handleErrorResponse(new HttpErrorResponse({
379
+ status: 401,
380
+ statusText: 'No autorizado. El token de refresco no está disponible.',
381
+ }));
382
+ }
383
+ // Intentar refrescar el token si no se está haciendo ya
384
+ if (!_authorizeService.isRefreshing) {
385
+ // Marcamos que se está refrescando el token
386
+ _authorizeService.isRefreshing = true;
387
+ // Intentar refrescar el token
388
+ return _securityService.tokenRefresh(_tokenRefresh).pipe(switchMap((response) => {
389
+ if (!response.token || !response.tokenRefresh) {
390
+ _spinnerService.hide();
391
+ return _handleErrorService.handleErrorResponse(new HttpErrorResponse({
392
+ status: 401,
393
+ statusText: 'No autorizado. El token de refresco ha expirado.',
394
+ }));
395
+ }
396
+ // Recargar el token en el servicio
397
+ _authorizeService.tokenReload(response);
398
+ // Generar una nueva solicitud con el token actualizado
399
+ const newReq = req.clone({
400
+ setHeaders: {
401
+ Authorization: `Bearer ${response.token}`,
402
+ },
403
+ });
404
+ // Refresco completado
405
+ _authorizeService.isRefreshing = false;
406
+ pendingRequests.forEach((pendingReq) => {
407
+ const clonedReq = pendingReq.authReq.clone({
408
+ setHeaders: {
409
+ Authorization: `Bearer ${_authorizeService.getToken()}`,
410
+ },
411
+ });
412
+ // Volver a ejecutar las solicitudes pendientes con el nuevo token
413
+ next(clonedReq).subscribe({
414
+ next: pendingReq.next,
415
+ error: pendingReq.error,
416
+ complete: pendingReq.complete,
417
+ });
418
+ });
419
+ pendingRequests = []; // Limpiar la cola
420
+ return next(newReq); // Procesar la solicitud original con el nuevo token
421
+ }), catchError((refreshError) => {
422
+ // Finalizamos el refresco si hay error
423
+ _authorizeService.isRefreshing = false;
424
+ _spinnerService.hide();
425
+ return _handleErrorService.handleErrorResponse(new HttpErrorResponse({
426
+ status: 401,
427
+ statusText: 'Error al refrescar el token. Redirigiendo al login.',
428
+ }));
429
+ }), finalize(() => {
430
+ _authorizeService.isRefreshing = false;
431
+ }));
432
+ }
433
+ else {
434
+ // Si ya se está refrescando, añadir la solicitud a la cola
435
+ return new Observable((observer) => {
436
+ pendingRequests.push({
437
+ authReq,
438
+ next: observer.next.bind(observer),
439
+ error: observer.error.bind(observer),
440
+ complete: observer.complete.bind(observer),
441
+ });
442
+ });
443
+ }
444
+ }
445
+ // Si no es un error 401, manejar otros errores
446
+ return _handleErrorService.handleErrorResponse(error);
447
+ }), finalize(() => {
448
+ _activeRequest--;
449
+ if (_activeRequest === 0) {
450
+ _spinnerService.hide();
451
+ }
452
+ }));
453
+ };
454
+
455
+ class AppMessageErrorComponent {
456
+ // Control de formulario que se pasa como input
457
+ control;
458
+ form = null;
459
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: AppMessageErrorComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
460
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "19.2.0", type: AppMessageErrorComponent, isStandalone: true, selector: "app-message-error", inputs: { control: "control", form: "form" }, ngImport: i0, template: "@if(control?.touched && control?.invalid){\r\n<div class=\"absolute dsx-error-message\">\r\n @if(control?.errors?.['required']){ El campo <strong>es requerido.</strong>\r\n } @else if(control?.errors?.['invalidDateRange']){\r\n <strong>{{ control?.errors?.['invalidDateRange']?.message }}</strong\r\n >. } @else if(control?.errors?.['dateNotRange']){\r\n <strong>{{ control?.errors?.['dateNotRange']?.message }}</strong\r\n >. } @else if(control?.errors?.['minlength']){ Debe tener al menos\r\n <strong>{{ control?.errors?.['minlength']?.requiredLength }}</strong>\r\n caracteres. } @else if(control?.errors?.['maxlength']){ Debe tener como m\u00E1ximo\r\n <strong>{{ control?.errors?.['maxlength']?.requiredLength }}</strong>\r\n caracteres. } @else if(control?.errors?.['min']){ El valor m\u00EDnimo permitido es\r\n <strong>{{ control?.errors?.['min']?.min }}</strong\r\n >. } @else if(control?.errors?.['max']){ El valor m\u00E1ximo permitido es\r\n <strong>{{ control?.errors?.['max']?.max }}</strong\r\n >. } @else if(control?.errors?.['email']){ Debe ser una direcci\u00F3n de correo\r\n v\u00E1lida. } @else if(control?.errors?.['pattern']){ El campo no tiene el formato\r\n requerido. } @else{ Existe un error a\u00FAn no identificado. }\r\n</div>\r\n}\r\n<!-- mensaje para formulario en general -->\r\n@if(form?.invalid && form?.touched){\r\n<div class=\"mt-2 mb-2\">\r\n @if(this.form?.errors?.['atLeastOneRequired']){\r\n <p-tag severity=\"danger\" [rounded]=\"true\">\r\n {{ form?.getError(\"atLeastOneRequired\")?.message }}</p-tag\r\n >\r\n }\r\n</div>\r\n}\r\n", styles: [""], dependencies: [{ kind: "ngmodule", type: TagModule }, { kind: "component", type: i1.Tag, selector: "p-tag", inputs: ["style", "styleClass", "severity", "value", "icon", "rounded"] }] });
461
+ }
462
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: AppMessageErrorComponent, decorators: [{
463
+ type: Component,
464
+ args: [{ selector: 'app-message-error', imports: [TagModule], template: "@if(control?.touched && control?.invalid){\r\n<div class=\"absolute dsx-error-message\">\r\n @if(control?.errors?.['required']){ El campo <strong>es requerido.</strong>\r\n } @else if(control?.errors?.['invalidDateRange']){\r\n <strong>{{ control?.errors?.['invalidDateRange']?.message }}</strong\r\n >. } @else if(control?.errors?.['dateNotRange']){\r\n <strong>{{ control?.errors?.['dateNotRange']?.message }}</strong\r\n >. } @else if(control?.errors?.['minlength']){ Debe tener al menos\r\n <strong>{{ control?.errors?.['minlength']?.requiredLength }}</strong>\r\n caracteres. } @else if(control?.errors?.['maxlength']){ Debe tener como m\u00E1ximo\r\n <strong>{{ control?.errors?.['maxlength']?.requiredLength }}</strong>\r\n caracteres. } @else if(control?.errors?.['min']){ El valor m\u00EDnimo permitido es\r\n <strong>{{ control?.errors?.['min']?.min }}</strong\r\n >. } @else if(control?.errors?.['max']){ El valor m\u00E1ximo permitido es\r\n <strong>{{ control?.errors?.['max']?.max }}</strong\r\n >. } @else if(control?.errors?.['email']){ Debe ser una direcci\u00F3n de correo\r\n v\u00E1lida. } @else if(control?.errors?.['pattern']){ El campo no tiene el formato\r\n requerido. } @else{ Existe un error a\u00FAn no identificado. }\r\n</div>\r\n}\r\n<!-- mensaje para formulario en general -->\r\n@if(form?.invalid && form?.touched){\r\n<div class=\"mt-2 mb-2\">\r\n @if(this.form?.errors?.['atLeastOneRequired']){\r\n <p-tag severity=\"danger\" [rounded]=\"true\">\r\n {{ form?.getError(\"atLeastOneRequired\")?.message }}</p-tag\r\n >\r\n }\r\n</div>\r\n}\r\n" }]
465
+ }], propDecorators: { control: [{
466
+ type: Input
467
+ }], form: [{
468
+ type: Input
469
+ }] } });
470
+
471
+ class KpicardComponent {
472
+ //type = input.required<string>();
473
+ option = input.required();
474
+ label = input.required();
475
+ iconType = input('fa-regular fa-bell');
476
+ color = input();
477
+ valor = input(0);
478
+ theme = input('light');
479
+ options = {
480
+ currency: {
481
+ startVal: 0,
482
+ duration: 0.5,
483
+ prefix: 'Q',
484
+ decimalPlaces: 2, // Decimales
485
+ useGrouping: true, // Separador de miles
486
+ separator: ',', // Carácter separador
487
+ decimal: '.', // Carácter decimal
488
+ suffix: ' GTQ', // Sufijo
489
+ scrollSpyDelay: 200, // Retardo antes de la animación
490
+ enableScrollSpy: true,
491
+ scrollSpyOnce: true,
492
+ },
493
+ integer: {
494
+ startVal: 0,
495
+ duration: 0.5,
496
+ prefix: 'Cant. ',
497
+ decimalPlaces: 2, // Decimales
498
+ useGrouping: true, // Separador de miles
499
+ separator: ',', // Carácter separador
500
+ decimal: '.', // Carácter decimal
501
+ scrollSpyDelay: 200, // Retardo antes de la animación
502
+ enableScrollSpy: true,
503
+ scrollSpyOnce: true,
504
+ },
505
+ };
506
+ // ✅ Método auxiliar para obtener la opción con seguridad
507
+ getSelectedOption() {
508
+ return this.options[this.option()] ?? this.options.currency; // Si no existe, usa `currency`
509
+ }
510
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: KpicardComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
511
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "19.2.0", type: KpicardComponent, isStandalone: true, selector: "app-kpicard", inputs: { option: { classPropertyName: "option", publicName: "option", isSignal: true, isRequired: true, transformFunction: null }, label: { classPropertyName: "label", publicName: "label", isSignal: true, isRequired: true, transformFunction: null }, iconType: { classPropertyName: "iconType", publicName: "iconType", isSignal: true, isRequired: false, transformFunction: null }, color: { classPropertyName: "color", publicName: "color", isSignal: true, isRequired: false, transformFunction: null }, valor: { classPropertyName: "valor", publicName: "valor", isSignal: true, isRequired: false, transformFunction: null }, theme: { classPropertyName: "theme", publicName: "theme", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: "<div class=\"kpi-card {{ color() }}\">\r\n <span\r\n [countUp]=\"valor()\"\r\n [options]=\"getSelectedOption()\"\r\n [class]=\"theme() === 'light' ? 'card-value-light' : 'card-value-dark'\"\r\n >0</span\r\n >\r\n <span [class]=\"theme() === 'light' ? 'card-text-light' : 'card-text-dark'\">{{\r\n label()\r\n }}</span>\r\n <i\r\n [class]=\"\r\n theme() === 'light'\r\n ? iconType() + ' icon icon-light'\r\n : iconType() + ' icon icon-dark'\r\n \"\r\n ></i>\r\n</div>\r\n", styles: [".icon{float:right;font-size:500%;position:absolute;top:0rem;right:-.3rem;opacity:.2}.icon-light{color:#000}.icon-dark{color:#fff}#container{width:1200px;display:flex}.grey-dark{background:#495057;color:#efefef}.red-gradient{background:linear-gradient(180deg,#cf5252,#790909 80%);color:#fff}.red{background:#a83b3b;color:#fff}.purple{background:#886ab5;color:#fff}.orange{background:#ffc241;color:#fff}.kpi-card{overflow:hidden;position:relative;box-shadow:1px 1px 3px #000000bf;display:inline-block;float:left;padding:1em;border-radius:.3em;font-family:sans-serif;width:240px;min-width:180px;margin-left:.5em;margin-top:.5em}.card-value-light{display:block;font-size:200%;font-weight:bolder;background:linear-gradient(45deg,#00fffc,#a200ff,#0094ff);-webkit-background-clip:text;background-clip:text;color:transparent;text-shadow:0 0 5px rgba(0,255,252,.3),0 0 10px rgba(162,0,255,.2),0 0 15px rgba(0,148,255,.1)}.card-value-dark{display:block;font-size:200%;font-weight:bolder;background:linear-gradient(45deg,#ff7e5f,#feb47b,#ff6a00);-webkit-background-clip:text;background-clip:text;color:transparent;text-shadow:0 0 5px rgba(255,126,95,.5),0 0 10px rgba(254,180,123,.4),0 0 15px rgba(255,106,0,.3)}.card-text-light{display:block;font-family:Orbitron,sans-serif;font-size:90%;padding-left:.2em;color:#495057}.card-text-dark{display:block;font-family:Orbitron,sans-serif;font-size:90%;padding-left:.2em;color:#d5d7d8}\n"], dependencies: [{ kind: "ngmodule", type: CountUpModule }, { kind: "directive", type: i1$1.CountUpDirective, selector: "[countUp]", inputs: ["countUp", "options", "reanimateOnClick"], outputs: ["complete"] }] });
512
+ }
513
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: KpicardComponent, decorators: [{
514
+ type: Component,
515
+ args: [{ selector: 'app-kpicard', imports: [CountUpModule], template: "<div class=\"kpi-card {{ color() }}\">\r\n <span\r\n [countUp]=\"valor()\"\r\n [options]=\"getSelectedOption()\"\r\n [class]=\"theme() === 'light' ? 'card-value-light' : 'card-value-dark'\"\r\n >0</span\r\n >\r\n <span [class]=\"theme() === 'light' ? 'card-text-light' : 'card-text-dark'\">{{\r\n label()\r\n }}</span>\r\n <i\r\n [class]=\"\r\n theme() === 'light'\r\n ? iconType() + ' icon icon-light'\r\n : iconType() + ' icon icon-dark'\r\n \"\r\n ></i>\r\n</div>\r\n", styles: [".icon{float:right;font-size:500%;position:absolute;top:0rem;right:-.3rem;opacity:.2}.icon-light{color:#000}.icon-dark{color:#fff}#container{width:1200px;display:flex}.grey-dark{background:#495057;color:#efefef}.red-gradient{background:linear-gradient(180deg,#cf5252,#790909 80%);color:#fff}.red{background:#a83b3b;color:#fff}.purple{background:#886ab5;color:#fff}.orange{background:#ffc241;color:#fff}.kpi-card{overflow:hidden;position:relative;box-shadow:1px 1px 3px #000000bf;display:inline-block;float:left;padding:1em;border-radius:.3em;font-family:sans-serif;width:240px;min-width:180px;margin-left:.5em;margin-top:.5em}.card-value-light{display:block;font-size:200%;font-weight:bolder;background:linear-gradient(45deg,#00fffc,#a200ff,#0094ff);-webkit-background-clip:text;background-clip:text;color:transparent;text-shadow:0 0 5px rgba(0,255,252,.3),0 0 10px rgba(162,0,255,.2),0 0 15px rgba(0,148,255,.1)}.card-value-dark{display:block;font-size:200%;font-weight:bolder;background:linear-gradient(45deg,#ff7e5f,#feb47b,#ff6a00);-webkit-background-clip:text;background-clip:text;color:transparent;text-shadow:0 0 5px rgba(255,126,95,.5),0 0 10px rgba(254,180,123,.4),0 0 15px rgba(255,106,0,.3)}.card-text-light{display:block;font-family:Orbitron,sans-serif;font-size:90%;padding-left:.2em;color:#495057}.card-text-dark{display:block;font-family:Orbitron,sans-serif;font-size:90%;padding-left:.2em;color:#d5d7d8}\n"] }]
516
+ }] });
517
+
518
+ class JsonHighlightPipe {
519
+ transform(value) {
520
+ if (!value) {
521
+ return '';
522
+ }
523
+ const json = JSON.stringify(value, null, 2); // Formatea el JSON con sangría
524
+ // Aplica estilo y colores básicos para diferenciar claves y valores
525
+ return json
526
+ .replace(/&/g, '&amp;')
527
+ .replace(/</g, '&lt;')
528
+ .replace(/>/g, '&gt;')
529
+ .replace(/(".*?"):/g, '<span class="json-key">$1</span>:')
530
+ .replace(/:\s(".*?")/g, ': <span class="json-string">$1</span>')
531
+ .replace(/\b(true|false|null)\b/g, '<span class="json-boolean">$1</span>')
532
+ .replace(/:\s(\d+)/g, ': <span class="json-number">$1</span>');
533
+ }
534
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: JsonHighlightPipe, deps: [], target: i0.ɵɵFactoryTarget.Pipe });
535
+ static ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "19.2.0", ngImport: i0, type: JsonHighlightPipe, isStandalone: true, name: "jsonHighlight" });
536
+ }
537
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.0", ngImport: i0, type: JsonHighlightPipe, decorators: [{
538
+ type: Pipe,
539
+ args: [{
540
+ name: 'jsonHighlight',
541
+ }]
542
+ }] });
543
+
544
+ /*
545
+ * Public API Surface of ngx-dsx
546
+ */
547
+
548
+ /**
549
+ * Generated bundle index. Do not edit.
550
+ */
551
+
552
+ export { AlertaService, AppMessageErrorComponent, AuthorizeService, ENVIRONMENT, JsonHighlightPipe, KpicardComponent, LoadingComponent, SecurityService, httpAuthorizeInterceptor };
553
+ //# sourceMappingURL=ngx-dsx.mjs.map