ng-easycommerce-v18 0.3.18-beta.1 → 0.3.18-beta.2

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/README.md CHANGED
@@ -1,3 +1,5 @@
1
+ # Version 0.3.18-beta.2
2
+ - Se optimizar y comenta codigo para que el runtime no sea un bucle y se haga antes de la primera carga
1
3
  # Version 0.3.18-beta.1
2
4
  - Se modifican servicios y componentes para que funcione el deploy SSR.
3
5
  - Se elimina uso de DOM ya que en SSR tira error.
@@ -1,7 +1,7 @@
1
1
  import { Injectable, Inject, PLATFORM_ID } from '@angular/core';
2
2
  import { isPlatformBrowser } from '@angular/common';
3
3
  import { of, BehaviorSubject } from 'rxjs';
4
- import { catchError, tap } from 'rxjs/operators';
4
+ import { catchError, tap, shareReplay } from 'rxjs/operators';
5
5
  import * as i0 from "@angular/core";
6
6
  import * as i1 from "@angular/common/http";
7
7
  export class RuntimeConfigService {
@@ -10,34 +10,71 @@ export class RuntimeConfigService {
10
10
  configSubject = new BehaviorSubject(null);
11
11
  config$ = this.configSubject.asObservable();
12
12
  _config = null;
13
+ _loadPromise = null;
14
+ _isLoaded = false;
13
15
  constructor(platformId, http) {
14
16
  this.platformId = platformId;
15
17
  this.http = http;
18
+ // En SSR o browser, intentar cargar inmediatamente desde window si está disponible
19
+ if (typeof window !== 'undefined' && window.__RUNTIME_CONFIG__ && !this._isLoaded) {
20
+ this._config = window.__RUNTIME_CONFIG__;
21
+ this._isLoaded = true;
22
+ this.configSubject.next(this._config);
23
+ }
24
+ }
25
+ /**
26
+ * Inicialización única - llamada por APP_INITIALIZER
27
+ * Garantiza que la configuración se carga solo una vez al inicio
28
+ */
29
+ initialize() {
30
+ if (this._isLoaded && this._config) {
31
+ return Promise.resolve(this._config);
32
+ }
33
+ return new Promise((resolve, reject) => {
34
+ this.loadConfig().subscribe({
35
+ next: (config) => {
36
+ this._isLoaded = true;
37
+ resolve(config);
38
+ },
39
+ error: (error) => {
40
+ reject(error);
41
+ }
42
+ });
43
+ });
16
44
  }
17
45
  /**
18
- * Carga la configuración en runtime
46
+ * Carga la configuración en runtime (optimizada para una sola carga)
19
47
  * - En SSR: Lee desde el objeto window inyectado por el servidor
20
- * - En Browser: Hace petición HTTP a /runtime-config.json
48
+ * - En Browser: Hace petición HTTP a /runtime-config.json (solo si no está en window)
21
49
  */
22
50
  loadConfig() {
23
- // Si ya tenemos la config cargada, la devolvemos
24
- if (this._config) {
51
+ // Si ya tenemos la config cargada, la devolvemos inmediatamente
52
+ if (this._isLoaded && this._config) {
25
53
  return of(this._config);
26
54
  }
55
+ // Si ya hay una carga en progreso, reutilizamos esa petición
56
+ if (this._loadPromise) {
57
+ return this._loadPromise;
58
+ }
27
59
  if (isPlatformBrowser(this.platformId)) {
28
60
  // En el navegador, primero intentamos obtener la config desde window
29
- if (window.__RUNTIME_CONFIG__) {
61
+ if (typeof window !== 'undefined' && window.__RUNTIME_CONFIG__ && !this._isLoaded) {
30
62
  this._config = window.__RUNTIME_CONFIG__;
63
+ this._isLoaded = true;
31
64
  this.configSubject.next(this._config);
32
65
  return of(this._config);
33
66
  }
34
- // Si no está en window, hacemos petición HTTP
35
- return this.http.get('/runtime-config.json').pipe(tap(config => {
67
+ // Si no está en window, hacemos petición HTTP (solo una vez)
68
+ this._loadPromise = this.http.get('/runtime-config.json').pipe(tap(config => {
36
69
  this._config = config;
70
+ this._isLoaded = true;
37
71
  this.configSubject.next(config);
38
72
  // Guardamos en window para siguientes accesos
39
- window.__RUNTIME_CONFIG__ = config;
40
- }), catchError(error => {
73
+ if (typeof window !== 'undefined') {
74
+ window.__RUNTIME_CONFIG__ = config;
75
+ }
76
+ }), shareReplay(1), // Cache la respuesta para múltiples suscripciones
77
+ catchError(error => {
41
78
  console.error('Error loading runtime config:', error);
42
79
  // Fallback config si falla la carga
43
80
  const fallbackConfig = {
@@ -48,14 +85,17 @@ export class RuntimeConfigService {
48
85
  channel: 'minorista'
49
86
  };
50
87
  this._config = fallbackConfig;
88
+ this._isLoaded = true;
51
89
  this.configSubject.next(fallbackConfig);
52
90
  return of(fallbackConfig);
53
91
  }));
92
+ return this._loadPromise;
54
93
  }
55
94
  else {
56
95
  // En SSR, leemos desde window que fue inyectado por el servidor
57
- if (typeof window !== 'undefined' && window.__RUNTIME_CONFIG__) {
96
+ if (typeof window !== 'undefined' && window.__RUNTIME_CONFIG__ && !this._isLoaded) {
58
97
  this._config = window.__RUNTIME_CONFIG__;
98
+ this._isLoaded = true;
59
99
  this.configSubject.next(this._config);
60
100
  return of(this._config);
61
101
  }
@@ -69,6 +109,7 @@ export class RuntimeConfigService {
69
109
  channel: 'minorista'
70
110
  };
71
111
  this._config = fallbackConfig;
112
+ this._isLoaded = true;
72
113
  this.configSubject.next(fallbackConfig);
73
114
  return of(fallbackConfig);
74
115
  }
@@ -146,4 +187,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
146
187
  type: Inject,
147
188
  args: [PLATFORM_ID]
148
189
  }] }, { type: i1.HttpClient }] });
149
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"runtime-config.service.js","sourceRoot":"","sources":["../../../../../projects/ng-easycommerce-v18/src/lib/ec-services/runtime-config.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Inject, PLATFORM_ID } from '@angular/core';\r\nimport { isPlatformBrowser } from '@angular/common';\r\nimport { HttpClient } from '@angular/common/http';\r\nimport { Observable, of, BehaviorSubject } from 'rxjs';\r\nimport { catchError, tap } from 'rxjs/operators';\r\nimport { RuntimeConfig } from '../interfaces/runtime-config';\r\n\r\ndeclare const window: any;\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class RuntimeConfigService {\r\n  private configSubject = new BehaviorSubject<RuntimeConfig | null>(null);\r\n  public config$ = this.configSubject.asObservable();\r\n  \r\n  private _config: RuntimeConfig | null = null;\r\n\r\n  constructor(\r\n    @Inject(PLATFORM_ID) private platformId: Object,\r\n    private http: HttpClient\r\n  ) {}\r\n\r\n  /**\r\n   * Carga la configuración en runtime\r\n   * - En SSR: Lee desde el objeto window inyectado por el servidor\r\n   * - En Browser: Hace petición HTTP a /runtime-config.json\r\n   */\r\n  loadConfig(): Observable<RuntimeConfig> {\r\n    // Si ya tenemos la config cargada, la devolvemos\r\n    if (this._config) {\r\n      return of(this._config as RuntimeConfig);\r\n    }\r\n\r\n    if (isPlatformBrowser(this.platformId)) {\r\n      // En el navegador, primero intentamos obtener la config desde window\r\n      if (window.__RUNTIME_CONFIG__) {\r\n        this._config = window.__RUNTIME_CONFIG__;\r\n        this.configSubject.next(this._config);\r\n        return of(this._config as RuntimeConfig);\r\n      }\r\n\r\n      // Si no está en window, hacemos petición HTTP\r\n      return this.http.get<RuntimeConfig>('/runtime-config.json').pipe(\r\n        tap(config => {\r\n          this._config = config;\r\n          this.configSubject.next(config);\r\n          // Guardamos en window para siguientes accesos\r\n          window.__RUNTIME_CONFIG__ = config;\r\n        }),\r\n        catchError(error => {\r\n          console.error('Error loading runtime config:', error);\r\n          // Fallback config si falla la carga\r\n          const fallbackConfig: RuntimeConfig = {\r\n            apiUrl: 'https://elden-preprod.backend.easycommerce.com.ar/',\r\n            appEnv: 'preprod',\r\n            production: false,\r\n            locale: 'es_AR',\r\n            channel: 'minorista'\r\n          };\r\n          this._config = fallbackConfig;\r\n          this.configSubject.next(fallbackConfig);\r\n          return of(fallbackConfig);\r\n        })\r\n      );\r\n    } else {\r\n      // En SSR, leemos desde window que fue inyectado por el servidor\r\n      if (typeof window !== 'undefined' && window.__RUNTIME_CONFIG__) {\r\n        this._config = window.__RUNTIME_CONFIG__;\r\n        this.configSubject.next(this._config);\r\n        return of(this._config as RuntimeConfig);\r\n      }\r\n\r\n      // Fallback para SSR si no hay config inyectada\r\n      // En el servidor, las variables de entorno se configuran desde el server.ts\r\n      const fallbackConfig: RuntimeConfig = {\r\n        apiUrl: 'https://elden-preprod.backend.easycommerce.com.ar/',\r\n        appEnv: 'preprod',\r\n        production: false,\r\n        locale: 'es_AR',\r\n        channel: 'minorista'\r\n      };\r\n      \r\n      this._config = fallbackConfig;\r\n      this.configSubject.next(fallbackConfig);\r\n      return of(fallbackConfig);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Obtiene la configuración actual (síncrono)\r\n   */\r\n  getConfig(): RuntimeConfig | null {\r\n    return this._config;\r\n  }\r\n\r\n  /**\r\n   * Obtiene la API URL actual\r\n   */\r\n  getApiUrl(): string {\r\n    return this._config?.apiUrl || 'https://elden-preprod.backend.easycommerce.com.ar/';\r\n  }\r\n\r\n  /**\r\n   * Obtiene la Frontend URL actual\r\n   */\r\n  getFrontendUrl(): string | undefined {\r\n    return this._config?.frontendUrl;\r\n  }\r\n\r\n  /**\r\n   * Obtiene el entorno actual\r\n   */\r\n  getAppEnv(): string {\r\n    return this._config?.appEnv || 'development';\r\n  }\r\n\r\n  /**\r\n   * Verifica si estamos en producción\r\n   */\r\n  isProduction(): boolean {\r\n    return this._config?.production || false;\r\n  }\r\n\r\n  /**\r\n   * Obtiene el locale actual\r\n   */\r\n  getLocale(): string {\r\n    return this._config?.locale || 'es_AR';\r\n  }\r\n\r\n  /**\r\n   * Obtiene el canal actual\r\n   */\r\n  getChannel(): string {\r\n    return this._config?.channel || 'minorista';\r\n  }\r\n\r\n  /**\r\n   * Método helper para construir URLs de API\r\n   */\r\n  buildApiUrl(endpoint: string): string {\r\n    const apiUrl = this.getApiUrl();\r\n    const cleanEndpoint = endpoint.startsWith('/') ? endpoint.slice(1) : endpoint;\r\n    return `${apiUrl}${cleanEndpoint}`;\r\n  }\r\n\r\n  /**\r\n   * Método helper para construir URLs del frontend\r\n   */\r\n  buildFrontendUrl(path: string): string {\r\n    const frontendUrl = this.getFrontendUrl();\r\n    if (!frontendUrl) {\r\n      return path;\r\n    }\r\n    const cleanPath = path.startsWith('/') ? path.slice(1) : path;\r\n    return `${frontendUrl}${cleanPath}`;\r\n  }\r\n}"]}
190
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"runtime-config.service.js","sourceRoot":"","sources":["../../../../../projects/ng-easycommerce-v18/src/lib/ec-services/runtime-config.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Inject, PLATFORM_ID } from '@angular/core';\r\nimport { isPlatformBrowser } from '@angular/common';\r\nimport { HttpClient } from '@angular/common/http';\r\nimport { Observable, of, BehaviorSubject } from 'rxjs';\r\nimport { catchError, tap, shareReplay } from 'rxjs/operators';\r\nimport { RuntimeConfig } from '../interfaces/runtime-config';\r\n\r\ndeclare const window: any;\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class RuntimeConfigService {\r\n  private configSubject = new BehaviorSubject<RuntimeConfig | null>(null);\r\n  public config$ = this.configSubject.asObservable();\r\n  \r\n  private _config: RuntimeConfig | null = null;\r\n  private _loadPromise: Observable<RuntimeConfig> | null = null;\r\n  private _isLoaded = false;\r\n\r\n  constructor(\r\n    @Inject(PLATFORM_ID) private platformId: Object,\r\n    private http: HttpClient\r\n  ) {\r\n    // En SSR o browser, intentar cargar inmediatamente desde window si está disponible\r\n    if (typeof window !== 'undefined' && window.__RUNTIME_CONFIG__ && !this._isLoaded) {\r\n      this._config = window.__RUNTIME_CONFIG__;\r\n      this._isLoaded = true;\r\n      this.configSubject.next(this._config);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Inicialización única - llamada por APP_INITIALIZER\r\n   * Garantiza que la configuración se carga solo una vez al inicio\r\n   */\r\n  initialize(): Promise<RuntimeConfig> {\r\n    if (this._isLoaded && this._config) {\r\n      return Promise.resolve(this._config);\r\n    }\r\n\r\n    return new Promise((resolve, reject) => {\r\n      this.loadConfig().subscribe({\r\n        next: (config: RuntimeConfig) => {\r\n          this._isLoaded = true;\r\n          resolve(config);\r\n        },\r\n        error: (error) => {\r\n          reject(error);\r\n        }\r\n      });\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Carga la configuración en runtime (optimizada para una sola carga)\r\n   * - En SSR: Lee desde el objeto window inyectado por el servidor\r\n   * - En Browser: Hace petición HTTP a /runtime-config.json (solo si no está en window)\r\n   */\r\n  loadConfig(): Observable<RuntimeConfig> {\r\n    // Si ya tenemos la config cargada, la devolvemos inmediatamente\r\n    if (this._isLoaded && this._config) {\r\n      return of(this._config as RuntimeConfig);\r\n    }\r\n\r\n    // Si ya hay una carga en progreso, reutilizamos esa petición\r\n    if (this._loadPromise) {\r\n      return this._loadPromise;\r\n    }\r\n\r\n    if (isPlatformBrowser(this.platformId)) {\r\n      // En el navegador, primero intentamos obtener la config desde window\r\n      if (typeof window !== 'undefined' && window.__RUNTIME_CONFIG__ && !this._isLoaded) {\r\n        this._config = window.__RUNTIME_CONFIG__;\r\n        this._isLoaded = true;\r\n        this.configSubject.next(this._config);\r\n        return of(this._config as RuntimeConfig);\r\n      }\r\n\r\n      // Si no está en window, hacemos petición HTTP (solo una vez)\r\n      this._loadPromise = this.http.get<RuntimeConfig>('/runtime-config.json').pipe(\r\n        tap(config => {\r\n          this._config = config;\r\n          this._isLoaded = true;\r\n          this.configSubject.next(config);\r\n          // Guardamos en window para siguientes accesos\r\n          if (typeof window !== 'undefined') {\r\n            window.__RUNTIME_CONFIG__ = config;\r\n          }\r\n        }),\r\n        shareReplay(1), // Cache la respuesta para múltiples suscripciones\r\n        catchError(error => {\r\n          console.error('Error loading runtime config:', error);\r\n          // Fallback config si falla la carga\r\n          const fallbackConfig: RuntimeConfig = {\r\n            apiUrl: 'https://elden-preprod.backend.easycommerce.com.ar/',\r\n            appEnv: 'preprod',\r\n            production: false,\r\n            locale: 'es_AR',\r\n            channel: 'minorista'\r\n          };\r\n          this._config = fallbackConfig;\r\n          this._isLoaded = true;\r\n          this.configSubject.next(fallbackConfig);\r\n          return of(fallbackConfig);\r\n        })\r\n      );\r\n      \r\n      return this._loadPromise;\r\n    } else {\r\n      // En SSR, leemos desde window que fue inyectado por el servidor\r\n      if (typeof window !== 'undefined' && window.__RUNTIME_CONFIG__ && !this._isLoaded) {\r\n        this._config = window.__RUNTIME_CONFIG__;\r\n        this._isLoaded = true;\r\n        this.configSubject.next(this._config);\r\n        return of(this._config as RuntimeConfig);\r\n      }\r\n\r\n      // Fallback para SSR si no hay config inyectada\r\n      // En el servidor, las variables de entorno se configuran desde el server.ts\r\n      const fallbackConfig: RuntimeConfig = {\r\n        apiUrl: 'https://elden-preprod.backend.easycommerce.com.ar/',\r\n        appEnv: 'preprod',\r\n        production: false,\r\n        locale: 'es_AR',\r\n        channel: 'minorista'\r\n      };\r\n      \r\n      this._config = fallbackConfig;\r\n      this._isLoaded = true;\r\n      this.configSubject.next(fallbackConfig);\r\n      return of(fallbackConfig);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Obtiene la configuración actual (síncrono)\r\n   */\r\n  getConfig(): RuntimeConfig | null {\r\n    return this._config;\r\n  }\r\n\r\n  /**\r\n   * Obtiene la API URL actual\r\n   */\r\n  getApiUrl(): string {\r\n    return this._config?.apiUrl || 'https://elden-preprod.backend.easycommerce.com.ar/';\r\n  }\r\n\r\n  /**\r\n   * Obtiene la Frontend URL actual\r\n   */\r\n  getFrontendUrl(): string | undefined {\r\n    return this._config?.frontendUrl;\r\n  }\r\n\r\n  /**\r\n   * Obtiene el entorno actual\r\n   */\r\n  getAppEnv(): string {\r\n    return this._config?.appEnv || 'development';\r\n  }\r\n\r\n  /**\r\n   * Verifica si estamos en producción\r\n   */\r\n  isProduction(): boolean {\r\n    return this._config?.production || false;\r\n  }\r\n\r\n  /**\r\n   * Obtiene el locale actual\r\n   */\r\n  getLocale(): string {\r\n    return this._config?.locale || 'es_AR';\r\n  }\r\n\r\n  /**\r\n   * Obtiene el canal actual\r\n   */\r\n  getChannel(): string {\r\n    return this._config?.channel || 'minorista';\r\n  }\r\n\r\n  /**\r\n   * Método helper para construir URLs de API\r\n   */\r\n  buildApiUrl(endpoint: string): string {\r\n    const apiUrl = this.getApiUrl();\r\n    const cleanEndpoint = endpoint.startsWith('/') ? endpoint.slice(1) : endpoint;\r\n    return `${apiUrl}${cleanEndpoint}`;\r\n  }\r\n\r\n  /**\r\n   * Método helper para construir URLs del frontend\r\n   */\r\n  buildFrontendUrl(path: string): string {\r\n    const frontendUrl = this.getFrontendUrl();\r\n    if (!frontendUrl) {\r\n      return path;\r\n    }\r\n    const cleanPath = path.startsWith('/') ? path.slice(1) : path;\r\n    return `${frontendUrl}${cleanPath}`;\r\n  }\r\n}"]}
@@ -1,2 +1,3 @@
1
1
  export * from './auth.interceptor';
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1lYXN5Y29tbWVyY2UtdjE4L3NyYy9saWIvaW50ZXJjZXB0b3JzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsb0JBQW9CLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2F1dGguaW50ZXJjZXB0b3InOyJdfQ==
2
+ export * from './runtime-config.interceptor';
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1lYXN5Y29tbWVyY2UtdjE4L3NyYy9saWIvaW50ZXJjZXB0b3JzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsb0JBQW9CLENBQUM7QUFDbkMsY0FBYyw4QkFBOEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vYXV0aC5pbnRlcmNlcHRvcic7XHJcbmV4cG9ydCAqIGZyb20gJy4vcnVudGltZS1jb25maWcuaW50ZXJjZXB0b3InOyJdfQ==
@@ -0,0 +1,41 @@
1
+ import { inject } from '@angular/core';
2
+ import { RuntimeConfigService } from '../ec-services/runtime-config.service';
3
+ import { switchMap, take, filter } from 'rxjs/operators';
4
+ /**
5
+ * Interceptor que garantiza que las peticiones HTTP no se hagan hasta que
6
+ * el runtime config esté completamente cargado
7
+ * Evita que se hagan peticiones con URLs incorrectas
8
+ */
9
+ export const runtimeConfigInterceptor = (req, next) => {
10
+ const runtimeConfigService = inject(RuntimeConfigService);
11
+ // Si la petición es para obtener el runtime config, la dejamos pasar inmediatamente
12
+ if (req.url.includes('/runtime-config.json')) {
13
+ return next(req);
14
+ }
15
+ // Si la petición es para assets estáticos, la dejamos pasar
16
+ if (req.url.includes('/assets/') || req.url.includes('.json') || req.url.includes('.css') || req.url.includes('.js')) {
17
+ return next(req);
18
+ }
19
+ // Para las demás peticiones, esperamos a que el config esté listo
20
+ return runtimeConfigService.config$.pipe(filter(config => config !== null), // Esperamos hasta que haya config
21
+ take(1), // Solo tomamos el primer valor válido
22
+ switchMap(config => {
23
+ // Si la URL ya es completa (empieza con http), la dejamos como está
24
+ if (req.url.startsWith('http')) {
25
+ return next(req);
26
+ }
27
+ // Si es una URL relativa que debería ir al API, la completamos
28
+ if (req.url.startsWith('/api/') || req.url.startsWith('api/')) {
29
+ const apiUrl = config.apiUrl;
30
+ const cleanUrl = req.url.startsWith('/') ? req.url.substring(1) : req.url;
31
+ const fullUrl = `${apiUrl}${cleanUrl}`;
32
+ const modifiedReq = req.clone({
33
+ url: fullUrl
34
+ });
35
+ return next(modifiedReq);
36
+ }
37
+ // Para otras URLs, las dejamos pasar sin modificar
38
+ return next(req);
39
+ }));
40
+ };
41
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicnVudGltZS1jb25maWcuaW50ZXJjZXB0b3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1lYXN5Y29tbWVyY2UtdjE4L3NyYy9saWIvaW50ZXJjZXB0b3JzL3J1bnRpbWUtY29uZmlnLmludGVyY2VwdG9yLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUNBLE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDdkMsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sdUNBQXVDLENBQUM7QUFDN0UsT0FBTyxFQUFFLFNBQVMsRUFBRSxJQUFJLEVBQUUsTUFBTSxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFFekQ7Ozs7R0FJRztBQUNILE1BQU0sQ0FBQyxNQUFNLHdCQUF3QixHQUFzQixDQUFDLEdBQXlCLEVBQUUsSUFBbUIsRUFBRSxFQUFFO0lBQzVHLE1BQU0sb0JBQW9CLEdBQUcsTUFBTSxDQUFDLG9CQUFvQixDQUFDLENBQUM7SUFFMUQsb0ZBQW9GO0lBQ3BGLElBQUksR0FBRyxDQUFDLEdBQUcsQ0FBQyxRQUFRLENBQUMsc0JBQXNCLENBQUMsRUFBRSxDQUFDO1FBQzdDLE9BQU8sSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQ25CLENBQUM7SUFFRCw0REFBNEQ7SUFDNUQsSUFBSSxHQUFHLENBQUMsR0FBRyxDQUFDLFFBQVEsQ0FBQyxVQUFVLENBQUMsSUFBSSxHQUFHLENBQUMsR0FBRyxDQUFDLFFBQVEsQ0FBQyxPQUFPLENBQUMsSUFBSSxHQUFHLENBQUMsR0FBRyxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsSUFBSSxHQUFHLENBQUMsR0FBRyxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDO1FBQ3JILE9BQU8sSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQ25CLENBQUM7SUFFRCxrRUFBa0U7SUFDbEUsT0FBTyxvQkFBb0IsQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUN0QyxNQUFNLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQyxNQUFNLEtBQUssSUFBSSxDQUFDLEVBQUUsa0NBQWtDO0lBQ3JFLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxzQ0FBc0M7SUFDL0MsU0FBUyxDQUFDLE1BQU0sQ0FBQyxFQUFFO1FBQ2pCLG9FQUFvRTtRQUNwRSxJQUFJLEdBQUcsQ0FBQyxHQUFHLENBQUMsVUFBVSxDQUFDLE1BQU0sQ0FBQyxFQUFFLENBQUM7WUFDL0IsT0FBTyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7UUFDbkIsQ0FBQztRQUVELCtEQUErRDtRQUMvRCxJQUFJLEdBQUcsQ0FBQyxHQUFHLENBQUMsVUFBVSxDQUFDLE9BQU8sQ0FBQyxJQUFJLEdBQUcsQ0FBQyxHQUFHLENBQUMsVUFBVSxDQUFDLE1BQU0sQ0FBQyxFQUFFLENBQUM7WUFDOUQsTUFBTSxNQUFNLEdBQUcsTUFBTyxDQUFDLE1BQU0sQ0FBQztZQUM5QixNQUFNLFFBQVEsR0FBRyxHQUFHLENBQUMsR0FBRyxDQUFDLFVBQVUsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLEdBQUcsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxHQUFHLENBQUM7WUFDMUUsTUFBTSxPQUFPLEdBQUcsR0FBRyxNQUFNLEdBQUcsUUFBUSxFQUFFLENBQUM7WUFFdkMsTUFBTSxXQUFXLEdBQUcsR0FBRyxDQUFDLEtBQUssQ0FBQztnQkFDNUIsR0FBRyxFQUFFLE9BQU87YUFDYixDQUFDLENBQUM7WUFFSCxPQUFPLElBQUksQ0FBQyxXQUFXLENBQUMsQ0FBQztRQUMzQixDQUFDO1FBRUQsbURBQW1EO1FBQ25ELE9BQU8sSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQ25CLENBQUMsQ0FBQyxDQUNILENBQUM7QUFDSixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBIdHRwSW50ZXJjZXB0b3JGbiwgSHR0cFJlcXVlc3QsIEh0dHBIYW5kbGVyRm4gfSBmcm9tICdAYW5ndWxhci9jb21tb24vaHR0cCc7XHJcbmltcG9ydCB7IGluamVjdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyBSdW50aW1lQ29uZmlnU2VydmljZSB9IGZyb20gJy4uL2VjLXNlcnZpY2VzL3J1bnRpbWUtY29uZmlnLnNlcnZpY2UnO1xyXG5pbXBvcnQgeyBzd2l0Y2hNYXAsIHRha2UsIGZpbHRlciB9IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcclxuXHJcbi8qKlxyXG4gKiBJbnRlcmNlcHRvciBxdWUgZ2FyYW50aXphIHF1ZSBsYXMgcGV0aWNpb25lcyBIVFRQIG5vIHNlIGhhZ2FuIGhhc3RhIHF1ZSBcclxuICogZWwgcnVudGltZSBjb25maWcgZXN0w6kgY29tcGxldGFtZW50ZSBjYXJnYWRvXHJcbiAqIEV2aXRhIHF1ZSBzZSBoYWdhbiBwZXRpY2lvbmVzIGNvbiBVUkxzIGluY29ycmVjdGFzXHJcbiAqL1xyXG5leHBvcnQgY29uc3QgcnVudGltZUNvbmZpZ0ludGVyY2VwdG9yOiBIdHRwSW50ZXJjZXB0b3JGbiA9IChyZXE6IEh0dHBSZXF1ZXN0PHVua25vd24+LCBuZXh0OiBIdHRwSGFuZGxlckZuKSA9PiB7XHJcbiAgY29uc3QgcnVudGltZUNvbmZpZ1NlcnZpY2UgPSBpbmplY3QoUnVudGltZUNvbmZpZ1NlcnZpY2UpO1xyXG4gIFxyXG4gIC8vIFNpIGxhIHBldGljacOzbiBlcyBwYXJhIG9idGVuZXIgZWwgcnVudGltZSBjb25maWcsIGxhIGRlamFtb3MgcGFzYXIgaW5tZWRpYXRhbWVudGVcclxuICBpZiAocmVxLnVybC5pbmNsdWRlcygnL3J1bnRpbWUtY29uZmlnLmpzb24nKSkge1xyXG4gICAgcmV0dXJuIG5leHQocmVxKTtcclxuICB9XHJcbiAgXHJcbiAgLy8gU2kgbGEgcGV0aWNpw7NuIGVzIHBhcmEgYXNzZXRzIGVzdMOhdGljb3MsIGxhIGRlamFtb3MgcGFzYXJcclxuICBpZiAocmVxLnVybC5pbmNsdWRlcygnL2Fzc2V0cy8nKSB8fCByZXEudXJsLmluY2x1ZGVzKCcuanNvbicpIHx8IHJlcS51cmwuaW5jbHVkZXMoJy5jc3MnKSB8fCByZXEudXJsLmluY2x1ZGVzKCcuanMnKSkge1xyXG4gICAgcmV0dXJuIG5leHQocmVxKTtcclxuICB9XHJcblxyXG4gIC8vIFBhcmEgbGFzIGRlbcOhcyBwZXRpY2lvbmVzLCBlc3BlcmFtb3MgYSBxdWUgZWwgY29uZmlnIGVzdMOpIGxpc3RvXHJcbiAgcmV0dXJuIHJ1bnRpbWVDb25maWdTZXJ2aWNlLmNvbmZpZyQucGlwZShcclxuICAgIGZpbHRlcihjb25maWcgPT4gY29uZmlnICE9PSBudWxsKSwgLy8gRXNwZXJhbW9zIGhhc3RhIHF1ZSBoYXlhIGNvbmZpZ1xyXG4gICAgdGFrZSgxKSwgLy8gU29sbyB0b21hbW9zIGVsIHByaW1lciB2YWxvciB2w6FsaWRvXHJcbiAgICBzd2l0Y2hNYXAoY29uZmlnID0+IHtcclxuICAgICAgLy8gU2kgbGEgVVJMIHlhIGVzIGNvbXBsZXRhIChlbXBpZXphIGNvbiBodHRwKSwgbGEgZGVqYW1vcyBjb21vIGVzdMOhXHJcbiAgICAgIGlmIChyZXEudXJsLnN0YXJ0c1dpdGgoJ2h0dHAnKSkge1xyXG4gICAgICAgIHJldHVybiBuZXh0KHJlcSk7XHJcbiAgICAgIH1cclxuICAgICAgXHJcbiAgICAgIC8vIFNpIGVzIHVuYSBVUkwgcmVsYXRpdmEgcXVlIGRlYmVyw61hIGlyIGFsIEFQSSwgbGEgY29tcGxldGFtb3NcclxuICAgICAgaWYgKHJlcS51cmwuc3RhcnRzV2l0aCgnL2FwaS8nKSB8fCByZXEudXJsLnN0YXJ0c1dpdGgoJ2FwaS8nKSkge1xyXG4gICAgICAgIGNvbnN0IGFwaVVybCA9IGNvbmZpZyEuYXBpVXJsO1xyXG4gICAgICAgIGNvbnN0IGNsZWFuVXJsID0gcmVxLnVybC5zdGFydHNXaXRoKCcvJykgPyByZXEudXJsLnN1YnN0cmluZygxKSA6IHJlcS51cmw7XHJcbiAgICAgICAgY29uc3QgZnVsbFVybCA9IGAke2FwaVVybH0ke2NsZWFuVXJsfWA7XHJcbiAgICAgICAgXHJcbiAgICAgICAgY29uc3QgbW9kaWZpZWRSZXEgPSByZXEuY2xvbmUoe1xyXG4gICAgICAgICAgdXJsOiBmdWxsVXJsXHJcbiAgICAgICAgfSk7XHJcbiAgICAgICAgXHJcbiAgICAgICAgcmV0dXJuIG5leHQobW9kaWZpZWRSZXEpO1xyXG4gICAgICB9XHJcbiAgICAgIFxyXG4gICAgICAvLyBQYXJhIG90cmFzIFVSTHMsIGxhcyBkZWphbW9zIHBhc2FyIHNpbiBtb2RpZmljYXJcclxuICAgICAgcmV0dXJuIG5leHQocmVxKTtcclxuICAgIH0pXHJcbiAgKTtcclxufTsiXX0=
@@ -1,23 +1,26 @@
1
1
  import { APP_INITIALIZER, makeEnvironmentProviders } from '@angular/core';
2
2
  import { RuntimeConfigService } from '../ec-services/runtime-config.service';
3
3
  /**
4
- * Factory function para APP_INITIALIZER
5
- * Carga la configuración runtime antes de que Angular bootstrap la aplicación
4
+ * Factory function para APP_INITIALIZER (OPTIMIZADA)
5
+ * Carga la configuración runtime UNA SOLA VEZ antes de que Angular bootstrap la aplicación
6
+ * Evita múltiples cargas y mejora el rendimiento
6
7
  */
7
8
  export function initializeRuntimeConfig(runtimeConfigService) {
8
9
  return () => {
9
- return new Promise((resolve, reject) => {
10
- runtimeConfigService.loadConfig().subscribe({
11
- next: (config) => {
12
- console.log('🔧 Runtime config loaded:', config);
13
- resolve(config);
14
- },
15
- error: (error) => {
16
- console.error('❌ Failed to load runtime config:', error);
17
- // Aún en caso de error, resolvemos para no bloquear la app
18
- resolve(null);
19
- }
10
+ console.log('🚀 Inicializando Runtime Config...');
11
+ return runtimeConfigService.initialize()
12
+ .then((config) => {
13
+ console.log(' Runtime config inicializado correctamente:', {
14
+ apiUrl: config.apiUrl,
15
+ appEnv: config.appEnv,
16
+ production: config.production
20
17
  });
18
+ return config;
19
+ })
20
+ .catch((error) => {
21
+ console.error('❌ Error al inicializar runtime config:', error);
22
+ // Aún en caso de error, continuamos para no bloquear la app
23
+ return null;
21
24
  });
22
25
  };
23
26
  }
@@ -36,4 +39,4 @@ export function provideRuntimeConfig() {
36
39
  }
37
40
  ]);
38
41
  }
39
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvdmlkZVJ1bnRpbWVDb25maWcuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1lYXN5Y29tbWVyY2UtdjE4L3NyYy9saWIvcHJvdmlkZXJzL3Byb3ZpZGVSdW50aW1lQ29uZmlnLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxlQUFlLEVBQXdCLHdCQUF3QixFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ2hHLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLHVDQUF1QyxDQUFDO0FBRTdFOzs7R0FHRztBQUNILE1BQU0sVUFBVSx1QkFBdUIsQ0FBQyxvQkFBMEM7SUFDaEYsT0FBTyxHQUFpQixFQUFFO1FBQ3hCLE9BQU8sSUFBSSxPQUFPLENBQUMsQ0FBQyxPQUFPLEVBQUUsTUFBTSxFQUFFLEVBQUU7WUFDckMsb0JBQW9CLENBQUMsVUFBVSxFQUFFLENBQUMsU0FBUyxDQUFDO2dCQUMxQyxJQUFJLEVBQUUsQ0FBQyxNQUFNLEVBQUUsRUFBRTtvQkFDZixPQUFPLENBQUMsR0FBRyxDQUFDLDJCQUEyQixFQUFFLE1BQU0sQ0FBQyxDQUFDO29CQUNqRCxPQUFPLENBQUMsTUFBTSxDQUFDLENBQUM7Z0JBQ2xCLENBQUM7Z0JBQ0QsS0FBSyxFQUFFLENBQUMsS0FBSyxFQUFFLEVBQUU7b0JBQ2YsT0FBTyxDQUFDLEtBQUssQ0FBQyxrQ0FBa0MsRUFBRSxLQUFLLENBQUMsQ0FBQztvQkFDekQsMkRBQTJEO29CQUMzRCxPQUFPLENBQUMsSUFBSSxDQUFDLENBQUM7Z0JBQ2hCLENBQUM7YUFDRixDQUFDLENBQUM7UUFDTCxDQUFDLENBQUMsQ0FBQztJQUNMLENBQUMsQ0FBQztBQUNKLENBQUM7QUFFRDs7O0dBR0c7QUFDSCxNQUFNLFVBQVUsb0JBQW9CO0lBQ2xDLE9BQU8sd0JBQXdCLENBQUM7UUFDOUIsb0JBQW9CO1FBQ3BCO1lBQ0UsT0FBTyxFQUFFLGVBQWU7WUFDeEIsVUFBVSxFQUFFLHVCQUF1QjtZQUNuQyxJQUFJLEVBQUUsQ0FBQyxvQkFBb0IsQ0FBQztZQUM1QixLQUFLLEVBQUUsSUFBSTtTQUNaO0tBQ0YsQ0FBQyxDQUFDO0FBQ0wsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEFQUF9JTklUSUFMSVpFUiwgRW52aXJvbm1lbnRQcm92aWRlcnMsIG1ha2VFbnZpcm9ubWVudFByb3ZpZGVycyB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyBSdW50aW1lQ29uZmlnU2VydmljZSB9IGZyb20gJy4uL2VjLXNlcnZpY2VzL3J1bnRpbWUtY29uZmlnLnNlcnZpY2UnO1xyXG5cclxuLyoqXHJcbiAqIEZhY3RvcnkgZnVuY3Rpb24gcGFyYSBBUFBfSU5JVElBTElaRVJcclxuICogQ2FyZ2EgbGEgY29uZmlndXJhY2nDs24gcnVudGltZSBhbnRlcyBkZSBxdWUgQW5ndWxhciBib290c3RyYXAgbGEgYXBsaWNhY2nDs25cclxuICovXHJcbmV4cG9ydCBmdW5jdGlvbiBpbml0aWFsaXplUnVudGltZUNvbmZpZyhydW50aW1lQ29uZmlnU2VydmljZTogUnVudGltZUNvbmZpZ1NlcnZpY2UpOiAoKSA9PiBQcm9taXNlPGFueT4ge1xyXG4gIHJldHVybiAoKTogUHJvbWlzZTxhbnk+ID0+IHtcclxuICAgIHJldHVybiBuZXcgUHJvbWlzZSgocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XHJcbiAgICAgIHJ1bnRpbWVDb25maWdTZXJ2aWNlLmxvYWRDb25maWcoKS5zdWJzY3JpYmUoe1xyXG4gICAgICAgIG5leHQ6IChjb25maWcpID0+IHtcclxuICAgICAgICAgIGNvbnNvbGUubG9nKCfwn5SnIFJ1bnRpbWUgY29uZmlnIGxvYWRlZDonLCBjb25maWcpO1xyXG4gICAgICAgICAgcmVzb2x2ZShjb25maWcpO1xyXG4gICAgICAgIH0sXHJcbiAgICAgICAgZXJyb3I6IChlcnJvcikgPT4ge1xyXG4gICAgICAgICAgY29uc29sZS5lcnJvcign4p2MIEZhaWxlZCB0byBsb2FkIHJ1bnRpbWUgY29uZmlnOicsIGVycm9yKTtcclxuICAgICAgICAgIC8vIEHDum4gZW4gY2FzbyBkZSBlcnJvciwgcmVzb2x2ZW1vcyBwYXJhIG5vIGJsb3F1ZWFyIGxhIGFwcFxyXG4gICAgICAgICAgcmVzb2x2ZShudWxsKTtcclxuICAgICAgICB9XHJcbiAgICAgIH0pO1xyXG4gICAgfSk7XHJcbiAgfTtcclxufVxyXG5cclxuLyoqXHJcbiAqIEZ1bmNpw7NuIHF1ZSBjb25maWd1cmEgZWwgUnVudGltZUNvbmZpZ1NlcnZpY2UgeSBzdSBpbmljaWFsaXphY2nDs25cclxuICogRGViZSBzZXIgdXNhZG8gZW4gYXBwLmNvbmZpZy50cyBqdW50byBjb24gcHJvdmlkZUVudmlyb25tZW50XHJcbiAqL1xyXG5leHBvcnQgZnVuY3Rpb24gcHJvdmlkZVJ1bnRpbWVDb25maWcoKTogRW52aXJvbm1lbnRQcm92aWRlcnMge1xyXG4gIHJldHVybiBtYWtlRW52aXJvbm1lbnRQcm92aWRlcnMoW1xyXG4gICAgUnVudGltZUNvbmZpZ1NlcnZpY2UsXHJcbiAgICB7XHJcbiAgICAgIHByb3ZpZGU6IEFQUF9JTklUSUFMSVpFUixcclxuICAgICAgdXNlRmFjdG9yeTogaW5pdGlhbGl6ZVJ1bnRpbWVDb25maWcsXHJcbiAgICAgIGRlcHM6IFtSdW50aW1lQ29uZmlnU2VydmljZV0sXHJcbiAgICAgIG11bHRpOiB0cnVlXHJcbiAgICB9XHJcbiAgXSk7XHJcbn0iXX0=
42
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvdmlkZVJ1bnRpbWVDb25maWcuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1lYXN5Y29tbWVyY2UtdjE4L3NyYy9saWIvcHJvdmlkZXJzL3Byb3ZpZGVSdW50aW1lQ29uZmlnLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxlQUFlLEVBQXdCLHdCQUF3QixFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ2hHLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLHVDQUF1QyxDQUFDO0FBRTdFOzs7O0dBSUc7QUFDSCxNQUFNLFVBQVUsdUJBQXVCLENBQUMsb0JBQTBDO0lBQ2hGLE9BQU8sR0FBaUIsRUFBRTtRQUN4QixPQUFPLENBQUMsR0FBRyxDQUFDLG9DQUFvQyxDQUFDLENBQUM7UUFDbEQsT0FBTyxvQkFBb0IsQ0FBQyxVQUFVLEVBQUU7YUFDckMsSUFBSSxDQUFDLENBQUMsTUFBTSxFQUFFLEVBQUU7WUFDZixPQUFPLENBQUMsR0FBRyxDQUFDLDhDQUE4QyxFQUFFO2dCQUMxRCxNQUFNLEVBQUUsTUFBTSxDQUFDLE1BQU07Z0JBQ3JCLE1BQU0sRUFBRSxNQUFNLENBQUMsTUFBTTtnQkFDckIsVUFBVSxFQUFFLE1BQU0sQ0FBQyxVQUFVO2FBQzlCLENBQUMsQ0FBQztZQUNILE9BQU8sTUFBTSxDQUFDO1FBQ2hCLENBQUMsQ0FBQzthQUNELEtBQUssQ0FBQyxDQUFDLEtBQUssRUFBRSxFQUFFO1lBQ2YsT0FBTyxDQUFDLEtBQUssQ0FBQyx3Q0FBd0MsRUFBRSxLQUFLLENBQUMsQ0FBQztZQUMvRCw0REFBNEQ7WUFDNUQsT0FBTyxJQUFJLENBQUM7UUFDZCxDQUFDLENBQUMsQ0FBQztJQUNQLENBQUMsQ0FBQztBQUNKLENBQUM7QUFFRDs7O0dBR0c7QUFDSCxNQUFNLFVBQVUsb0JBQW9CO0lBQ2xDLE9BQU8sd0JBQXdCLENBQUM7UUFDOUIsb0JBQW9CO1FBQ3BCO1lBQ0UsT0FBTyxFQUFFLGVBQWU7WUFDeEIsVUFBVSxFQUFFLHVCQUF1QjtZQUNuQyxJQUFJLEVBQUUsQ0FBQyxvQkFBb0IsQ0FBQztZQUM1QixLQUFLLEVBQUUsSUFBSTtTQUNaO0tBQ0YsQ0FBQyxDQUFDO0FBQ0wsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEFQUF9JTklUSUFMSVpFUiwgRW52aXJvbm1lbnRQcm92aWRlcnMsIG1ha2VFbnZpcm9ubWVudFByb3ZpZGVycyB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyBSdW50aW1lQ29uZmlnU2VydmljZSB9IGZyb20gJy4uL2VjLXNlcnZpY2VzL3J1bnRpbWUtY29uZmlnLnNlcnZpY2UnO1xyXG5cclxuLyoqXHJcbiAqIEZhY3RvcnkgZnVuY3Rpb24gcGFyYSBBUFBfSU5JVElBTElaRVIgKE9QVElNSVpBREEpXHJcbiAqIENhcmdhIGxhIGNvbmZpZ3VyYWNpw7NuIHJ1bnRpbWUgVU5BIFNPTEEgVkVaIGFudGVzIGRlIHF1ZSBBbmd1bGFyIGJvb3RzdHJhcCBsYSBhcGxpY2FjacOzblxyXG4gKiBFdml0YSBtw7psdGlwbGVzIGNhcmdhcyB5IG1lam9yYSBlbCByZW5kaW1pZW50b1xyXG4gKi9cclxuZXhwb3J0IGZ1bmN0aW9uIGluaXRpYWxpemVSdW50aW1lQ29uZmlnKHJ1bnRpbWVDb25maWdTZXJ2aWNlOiBSdW50aW1lQ29uZmlnU2VydmljZSk6ICgpID0+IFByb21pc2U8YW55PiB7XHJcbiAgcmV0dXJuICgpOiBQcm9taXNlPGFueT4gPT4ge1xyXG4gICAgY29uc29sZS5sb2coJ/CfmoAgSW5pY2lhbGl6YW5kbyBSdW50aW1lIENvbmZpZy4uLicpO1xyXG4gICAgcmV0dXJuIHJ1bnRpbWVDb25maWdTZXJ2aWNlLmluaXRpYWxpemUoKVxyXG4gICAgICAudGhlbigoY29uZmlnKSA9PiB7XHJcbiAgICAgICAgY29uc29sZS5sb2coJ+KchSBSdW50aW1lIGNvbmZpZyBpbmljaWFsaXphZG8gY29ycmVjdGFtZW50ZTonLCB7XHJcbiAgICAgICAgICBhcGlVcmw6IGNvbmZpZy5hcGlVcmwsXHJcbiAgICAgICAgICBhcHBFbnY6IGNvbmZpZy5hcHBFbnYsXHJcbiAgICAgICAgICBwcm9kdWN0aW9uOiBjb25maWcucHJvZHVjdGlvblxyXG4gICAgICAgIH0pO1xyXG4gICAgICAgIHJldHVybiBjb25maWc7XHJcbiAgICAgIH0pXHJcbiAgICAgIC5jYXRjaCgoZXJyb3IpID0+IHtcclxuICAgICAgICBjb25zb2xlLmVycm9yKCfinYwgRXJyb3IgYWwgaW5pY2lhbGl6YXIgcnVudGltZSBjb25maWc6JywgZXJyb3IpO1xyXG4gICAgICAgIC8vIEHDum4gZW4gY2FzbyBkZSBlcnJvciwgY29udGludWFtb3MgcGFyYSBubyBibG9xdWVhciBsYSBhcHBcclxuICAgICAgICByZXR1cm4gbnVsbDtcclxuICAgICAgfSk7XHJcbiAgfTtcclxufVxyXG5cclxuLyoqXHJcbiAqIEZ1bmNpw7NuIHF1ZSBjb25maWd1cmEgZWwgUnVudGltZUNvbmZpZ1NlcnZpY2UgeSBzdSBpbmljaWFsaXphY2nDs25cclxuICogRGViZSBzZXIgdXNhZG8gZW4gYXBwLmNvbmZpZy50cyBqdW50byBjb24gcHJvdmlkZUVudmlyb25tZW50XHJcbiAqL1xyXG5leHBvcnQgZnVuY3Rpb24gcHJvdmlkZVJ1bnRpbWVDb25maWcoKTogRW52aXJvbm1lbnRQcm92aWRlcnMge1xyXG4gIHJldHVybiBtYWtlRW52aXJvbm1lbnRQcm92aWRlcnMoW1xyXG4gICAgUnVudGltZUNvbmZpZ1NlcnZpY2UsXHJcbiAgICB7XHJcbiAgICAgIHByb3ZpZGU6IEFQUF9JTklUSUFMSVpFUixcclxuICAgICAgdXNlRmFjdG9yeTogaW5pdGlhbGl6ZVJ1bnRpbWVDb25maWcsXHJcbiAgICAgIGRlcHM6IFtSdW50aW1lQ29uZmlnU2VydmljZV0sXHJcbiAgICAgIG11bHRpOiB0cnVlXHJcbiAgICB9XHJcbiAgXSk7XHJcbn0iXX0=
@@ -2,10 +2,10 @@ import * as i0 from '@angular/core';
2
2
  import { InjectionToken, makeEnvironmentProviders, PLATFORM_ID, Injectable, Inject, APP_INITIALIZER, inject, RendererFactory2, afterNextRender, signal, EnvironmentInjector, runInInjectionContext, Component, ChangeDetectorRef, HostListener, CUSTOM_ELEMENTS_SCHEMA, Input, Pipe, Injector, EventEmitter, Output, forwardRef, afterRender, ViewChild, computed, Renderer2, ChangeDetectionStrategy, Directive } from '@angular/core';
3
3
  import * as i1$1 from '@angular/common';
4
4
  import { isPlatformBrowser, DOCUMENT, AsyncPipe, CommonModule, TitleCasePipe, JsonPipe, UpperCasePipe, Location } from '@angular/common';
5
- import { BehaviorSubject, of, take, shareReplay, map, catchError as catchError$1, filter, ReplaySubject, firstValueFrom, concatMap, throwError, switchMap, combineLatest } from 'rxjs';
5
+ import { BehaviorSubject, of, take, shareReplay as shareReplay$1, map, catchError as catchError$1, filter, ReplaySubject, firstValueFrom, concatMap, throwError, switchMap, combineLatest } from 'rxjs';
6
6
  import * as i1 from '@angular/common/http';
7
7
  import { HttpClient, HttpHeaders } from '@angular/common/http';
8
- import { tap, catchError, filter as filter$1, skipWhile } from 'rxjs/operators';
8
+ import { tap, shareReplay, catchError, filter as filter$1, skipWhile, take as take$1, switchMap as switchMap$1 } from 'rxjs/operators';
9
9
  import * as i1$2 from '@ngx-translate/core';
10
10
  import { TranslateService, TranslateModule } from '@ngx-translate/core';
11
11
  import { CookieService } from 'ngx-cookie-service';
@@ -50,34 +50,71 @@ class RuntimeConfigService {
50
50
  configSubject = new BehaviorSubject(null);
51
51
  config$ = this.configSubject.asObservable();
52
52
  _config = null;
53
+ _loadPromise = null;
54
+ _isLoaded = false;
53
55
  constructor(platformId, http) {
54
56
  this.platformId = platformId;
55
57
  this.http = http;
58
+ // En SSR o browser, intentar cargar inmediatamente desde window si está disponible
59
+ if (typeof window !== 'undefined' && window.__RUNTIME_CONFIG__ && !this._isLoaded) {
60
+ this._config = window.__RUNTIME_CONFIG__;
61
+ this._isLoaded = true;
62
+ this.configSubject.next(this._config);
63
+ }
56
64
  }
57
65
  /**
58
- * Carga la configuración en runtime
66
+ * Inicialización única - llamada por APP_INITIALIZER
67
+ * Garantiza que la configuración se carga solo una vez al inicio
68
+ */
69
+ initialize() {
70
+ if (this._isLoaded && this._config) {
71
+ return Promise.resolve(this._config);
72
+ }
73
+ return new Promise((resolve, reject) => {
74
+ this.loadConfig().subscribe({
75
+ next: (config) => {
76
+ this._isLoaded = true;
77
+ resolve(config);
78
+ },
79
+ error: (error) => {
80
+ reject(error);
81
+ }
82
+ });
83
+ });
84
+ }
85
+ /**
86
+ * Carga la configuración en runtime (optimizada para una sola carga)
59
87
  * - En SSR: Lee desde el objeto window inyectado por el servidor
60
- * - En Browser: Hace petición HTTP a /runtime-config.json
88
+ * - En Browser: Hace petición HTTP a /runtime-config.json (solo si no está en window)
61
89
  */
62
90
  loadConfig() {
63
- // Si ya tenemos la config cargada, la devolvemos
64
- if (this._config) {
91
+ // Si ya tenemos la config cargada, la devolvemos inmediatamente
92
+ if (this._isLoaded && this._config) {
65
93
  return of(this._config);
66
94
  }
95
+ // Si ya hay una carga en progreso, reutilizamos esa petición
96
+ if (this._loadPromise) {
97
+ return this._loadPromise;
98
+ }
67
99
  if (isPlatformBrowser(this.platformId)) {
68
100
  // En el navegador, primero intentamos obtener la config desde window
69
- if (window.__RUNTIME_CONFIG__) {
101
+ if (typeof window !== 'undefined' && window.__RUNTIME_CONFIG__ && !this._isLoaded) {
70
102
  this._config = window.__RUNTIME_CONFIG__;
103
+ this._isLoaded = true;
71
104
  this.configSubject.next(this._config);
72
105
  return of(this._config);
73
106
  }
74
- // Si no está en window, hacemos petición HTTP
75
- return this.http.get('/runtime-config.json').pipe(tap(config => {
107
+ // Si no está en window, hacemos petición HTTP (solo una vez)
108
+ this._loadPromise = this.http.get('/runtime-config.json').pipe(tap(config => {
76
109
  this._config = config;
110
+ this._isLoaded = true;
77
111
  this.configSubject.next(config);
78
112
  // Guardamos en window para siguientes accesos
79
- window.__RUNTIME_CONFIG__ = config;
80
- }), catchError(error => {
113
+ if (typeof window !== 'undefined') {
114
+ window.__RUNTIME_CONFIG__ = config;
115
+ }
116
+ }), shareReplay(1), // Cache la respuesta para múltiples suscripciones
117
+ catchError(error => {
81
118
  console.error('Error loading runtime config:', error);
82
119
  // Fallback config si falla la carga
83
120
  const fallbackConfig = {
@@ -88,14 +125,17 @@ class RuntimeConfigService {
88
125
  channel: 'minorista'
89
126
  };
90
127
  this._config = fallbackConfig;
128
+ this._isLoaded = true;
91
129
  this.configSubject.next(fallbackConfig);
92
130
  return of(fallbackConfig);
93
131
  }));
132
+ return this._loadPromise;
94
133
  }
95
134
  else {
96
135
  // En SSR, leemos desde window que fue inyectado por el servidor
97
- if (typeof window !== 'undefined' && window.__RUNTIME_CONFIG__) {
136
+ if (typeof window !== 'undefined' && window.__RUNTIME_CONFIG__ && !this._isLoaded) {
98
137
  this._config = window.__RUNTIME_CONFIG__;
138
+ this._isLoaded = true;
99
139
  this.configSubject.next(this._config);
100
140
  return of(this._config);
101
141
  }
@@ -109,6 +149,7 @@ class RuntimeConfigService {
109
149
  channel: 'minorista'
110
150
  };
111
151
  this._config = fallbackConfig;
152
+ this._isLoaded = true;
112
153
  this.configSubject.next(fallbackConfig);
113
154
  return of(fallbackConfig);
114
155
  }
@@ -188,23 +229,26 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
188
229
  }] }, { type: i1.HttpClient }] });
189
230
 
190
231
  /**
191
- * Factory function para APP_INITIALIZER
192
- * Carga la configuración runtime antes de que Angular bootstrap la aplicación
232
+ * Factory function para APP_INITIALIZER (OPTIMIZADA)
233
+ * Carga la configuración runtime UNA SOLA VEZ antes de que Angular bootstrap la aplicación
234
+ * Evita múltiples cargas y mejora el rendimiento
193
235
  */
194
236
  function initializeRuntimeConfig(runtimeConfigService) {
195
237
  return () => {
196
- return new Promise((resolve, reject) => {
197
- runtimeConfigService.loadConfig().subscribe({
198
- next: (config) => {
199
- console.log('🔧 Runtime config loaded:', config);
200
- resolve(config);
201
- },
202
- error: (error) => {
203
- console.error('❌ Failed to load runtime config:', error);
204
- // Aún en caso de error, resolvemos para no bloquear la app
205
- resolve(null);
206
- }
238
+ console.log('🚀 Inicializando Runtime Config...');
239
+ return runtimeConfigService.initialize()
240
+ .then((config) => {
241
+ console.log(' Runtime config inicializado correctamente:', {
242
+ apiUrl: config.apiUrl,
243
+ appEnv: config.appEnv,
244
+ production: config.production
207
245
  });
246
+ return config;
247
+ })
248
+ .catch((error) => {
249
+ console.error('❌ Error al inicializar runtime config:', error);
250
+ // Aún en caso de error, continuamos para no bloquear la app
251
+ return null;
208
252
  });
209
253
  };
210
254
  }
@@ -773,7 +817,7 @@ class OptionsService {
773
817
  */
774
818
  getCategories() {
775
819
  if (this.categoriesSubject.getValue().length === 0) {
776
- this.connection.get(this.taxonsApi()).pipe(shareReplay(1), map((response) => {
820
+ this.connection.get(this.taxonsApi()).pipe(shareReplay$1(1), map((response) => {
777
821
  const categories = this.appendPaths(true, response);
778
822
  this.categoriesSubject.next(categories);
779
823
  return categories;
@@ -791,7 +835,7 @@ class OptionsService {
791
835
  */
792
836
  getSections() {
793
837
  if (this.sectionsSubject.getValue().length === 0) {
794
- this.connection.get(this.sectionsApi()).pipe(shareReplay(1), map((response) => {
838
+ this.connection.get(this.sectionsApi()).pipe(shareReplay$1(1), map((response) => {
795
839
  let sections = response.filter((section) => section.code != 'home');
796
840
  sections = sections.map((section) => this.appendSection(section));
797
841
  this.sectionsSubject.next(sections);
@@ -807,7 +851,7 @@ class OptionsService {
807
851
  */
808
852
  getAttributes(params) {
809
853
  if (this.attributesSubject.getValue().length === 0) {
810
- this.connection.get(this.attributesApi(params)).pipe(shareReplay(1), map((response) => {
854
+ this.connection.get(this.attributesApi(params)).pipe(shareReplay$1(1), map((response) => {
811
855
  const attributes = this.appendPaths(false, (response.attributes[0] || response));
812
856
  this.attributesSubject.next(attributes);
813
857
  return attributes;
@@ -1039,7 +1083,7 @@ class ParametersService {
1039
1083
  */
1040
1084
  getParameters() {
1041
1085
  if (this.parametersSubject.getValue() === null) {
1042
- this.connection.get(this.getAllParametersAPI()).pipe(shareReplay(1), map((response) => {
1086
+ this.connection.get(this.getAllParametersAPI()).pipe(shareReplay$1(1), map((response) => {
1043
1087
  this.parametersSubject.next(response);
1044
1088
  return response;
1045
1089
  }), catchError$1((error) => {
@@ -1103,7 +1147,7 @@ class BlocksRepositoryService {
1103
1147
  */
1104
1148
  getBlocks(section) {
1105
1149
  //if(this.blocksSubject.getValue() != null){
1106
- this.connection.get(this.blocksAPI(section)).pipe(shareReplay(1), map((response) => {
1150
+ this.connection.get(this.blocksAPI(section)).pipe(shareReplay$1(1), map((response) => {
1107
1151
  //console.log(response)
1108
1152
  this.blocksSubject.next(response.items);
1109
1153
  return response;
@@ -2285,7 +2329,7 @@ class ChannelService {
2285
2329
  * @returns Observable con el canal.
2286
2330
  */
2287
2331
  getChannel = () => {
2288
- this.connection.get(this.channelApi()).pipe(shareReplay(1), map((response) => {
2332
+ this.connection.get(this.channelApi()).pipe(shareReplay$1(1), map((response) => {
2289
2333
  this.setChannelInfo(response);
2290
2334
  this.channelSubject.next(response);
2291
2335
  this.channelType = response.type;
@@ -2725,7 +2769,7 @@ class FormService {
2725
2769
  * @returns
2726
2770
  */
2727
2771
  getCountriesData() {
2728
- this.connection.get(this.countriesApi()).pipe(shareReplay(1), map((response) => {
2772
+ this.connection.get(this.countriesApi()).pipe(shareReplay$1(1), map((response) => {
2729
2773
  //console.log(response);
2730
2774
  const data = response._embedded.items;
2731
2775
  this.countriesSubject.next(data);
@@ -4011,7 +4055,7 @@ class PaginationService {
4011
4055
  _connectionService = inject(ConnectionService);
4012
4056
  _filtersService = inject(FiltersService);
4013
4057
  _constants = inject(CoreConstantsService);
4014
- paginationData$ = this._filtersService.filters$.pipe(shareReplay(1), filter(filters => filters && filters.length > 0), switchMap((filters) => {
4058
+ paginationData$ = this._filtersService.filters$.pipe(shareReplay$1(1), filter(filters => filters && filters.length > 0), switchMap((filters) => {
4015
4059
  const url = this.buildUrl(filters);
4016
4060
  return this.getData(url);
4017
4061
  }));
@@ -12246,6 +12290,44 @@ const authInterceptor = (req, next) => {
12246
12290
  return next(req);
12247
12291
  };
12248
12292
 
12293
+ /**
12294
+ * Interceptor que garantiza que las peticiones HTTP no se hagan hasta que
12295
+ * el runtime config esté completamente cargado
12296
+ * Evita que se hagan peticiones con URLs incorrectas
12297
+ */
12298
+ const runtimeConfigInterceptor = (req, next) => {
12299
+ const runtimeConfigService = inject(RuntimeConfigService);
12300
+ // Si la petición es para obtener el runtime config, la dejamos pasar inmediatamente
12301
+ if (req.url.includes('/runtime-config.json')) {
12302
+ return next(req);
12303
+ }
12304
+ // Si la petición es para assets estáticos, la dejamos pasar
12305
+ if (req.url.includes('/assets/') || req.url.includes('.json') || req.url.includes('.css') || req.url.includes('.js')) {
12306
+ return next(req);
12307
+ }
12308
+ // Para las demás peticiones, esperamos a que el config esté listo
12309
+ return runtimeConfigService.config$.pipe(filter$1(config => config !== null), // Esperamos hasta que haya config
12310
+ take$1(1), // Solo tomamos el primer valor válido
12311
+ switchMap$1(config => {
12312
+ // Si la URL ya es completa (empieza con http), la dejamos como está
12313
+ if (req.url.startsWith('http')) {
12314
+ return next(req);
12315
+ }
12316
+ // Si es una URL relativa que debería ir al API, la completamos
12317
+ if (req.url.startsWith('/api/') || req.url.startsWith('api/')) {
12318
+ const apiUrl = config.apiUrl;
12319
+ const cleanUrl = req.url.startsWith('/') ? req.url.substring(1) : req.url;
12320
+ const fullUrl = `${apiUrl}${cleanUrl}`;
12321
+ const modifiedReq = req.clone({
12322
+ url: fullUrl
12323
+ });
12324
+ return next(modifiedReq);
12325
+ }
12326
+ // Para otras URLs, las dejamos pasar sin modificar
12327
+ return next(req);
12328
+ }));
12329
+ };
12330
+
12249
12331
  class ProductStockDirective {
12250
12332
  templateRef;
12251
12333
  viewContainer;
@@ -12447,5 +12529,5 @@ const directives = [
12447
12529
  * Generated bundle index. Do not edit.
12448
12530
  */
12449
12531
 
12450
- export { AccountEcComponent, AddressingService, AnalyticsService, AuthEcComponent, AuthService, AuthStorageService, BaseApiService, BlockBannerBoxEcComponent, BlockBannerFullEcComponent, BlockFormContactEcComponent, BlockHtmlEcComponent, BlockNewsletterEcComponent, BlockProductsEcComponent, BlocksEcComponent, BlocksRepositoryService, BlocksService, BreadcrumbEcComponent, CartEcComponent, CartItemEcComponent, CartService, ChannelService, CheckoutEcComponent, CheckoutService, CollectionEcComponent, ConfirmAccountEcComponent, ContactEcComponent, CoreConstantsService, CouponEcComponent, CurrencyService, DopplerService, ENVIRONMENT_TOKEN, EcCurrencySymbolPipe, EcSafeHtmlPipe, FacebookPixelService, FaqsEcComponent, FiltersEcComponent, FiltersService, FiltersSortEcComponent, FooterEcComponent, ForgotPasswordEcComponent, FormService, GTMService, GoogleAnalyticsService, HeaderEcComponent, HomeEcComponent, LoadingFullEcComponent, LoadingInlineEcComponent, LoadingSectionEcComponent, LocalStorageService, LoginFormEcComponent, MagnizoomEcComponent, MetricoolPixelService, NgxLocalStorageService, OptionsService, OrderEcComponent, OrderUtilityService, OrdersListEcComponent, OrdersService, PaginationService, ParametersService, ParamsContext, PasswordResetEcComponent, PaymentService, PriceEcComponent, PriceRangeFilterComponent, ProductDetailEcComponent, ProductDetailService, ProductEcComponent, ProductOffDirective, ProductStockDirective, ProductsService, ReCaptchaEcComponent, ReCaptchaService, RedsysCatchEcComponent, RegisterFormEcComponent, RegisterWholesalerFormEcComponent, RelatedProductsEcComponent, ReviewsEcComponent, ReviewsFormEcComponent, RuntimeConfigService, SectionContainerEcComponent, ShareEcComponent, ShipmentService, SidebarEcComponent, StoresEcComponent, SuccessEcComponent, TestService, ToastService, VariantsEcComponent, authGuard, authInterceptor, directives, initializeRuntimeConfig, provideEnvironment, provideRuntimeConfig };
12532
+ export { AccountEcComponent, AddressingService, AnalyticsService, AuthEcComponent, AuthService, AuthStorageService, BaseApiService, BlockBannerBoxEcComponent, BlockBannerFullEcComponent, BlockFormContactEcComponent, BlockHtmlEcComponent, BlockNewsletterEcComponent, BlockProductsEcComponent, BlocksEcComponent, BlocksRepositoryService, BlocksService, BreadcrumbEcComponent, CartEcComponent, CartItemEcComponent, CartService, ChannelService, CheckoutEcComponent, CheckoutService, CollectionEcComponent, ConfirmAccountEcComponent, ContactEcComponent, CoreConstantsService, CouponEcComponent, CurrencyService, DopplerService, ENVIRONMENT_TOKEN, EcCurrencySymbolPipe, EcSafeHtmlPipe, FacebookPixelService, FaqsEcComponent, FiltersEcComponent, FiltersService, FiltersSortEcComponent, FooterEcComponent, ForgotPasswordEcComponent, FormService, GTMService, GoogleAnalyticsService, HeaderEcComponent, HomeEcComponent, LoadingFullEcComponent, LoadingInlineEcComponent, LoadingSectionEcComponent, LocalStorageService, LoginFormEcComponent, MagnizoomEcComponent, MetricoolPixelService, NgxLocalStorageService, OptionsService, OrderEcComponent, OrderUtilityService, OrdersListEcComponent, OrdersService, PaginationService, ParametersService, ParamsContext, PasswordResetEcComponent, PaymentService, PriceEcComponent, PriceRangeFilterComponent, ProductDetailEcComponent, ProductDetailService, ProductEcComponent, ProductOffDirective, ProductStockDirective, ProductsService, ReCaptchaEcComponent, ReCaptchaService, RedsysCatchEcComponent, RegisterFormEcComponent, RegisterWholesalerFormEcComponent, RelatedProductsEcComponent, ReviewsEcComponent, ReviewsFormEcComponent, RuntimeConfigService, SectionContainerEcComponent, ShareEcComponent, ShipmentService, SidebarEcComponent, StoresEcComponent, SuccessEcComponent, TestService, ToastService, VariantsEcComponent, authGuard, authInterceptor, directives, initializeRuntimeConfig, provideEnvironment, provideRuntimeConfig, runtimeConfigInterceptor };
12451
12533
  //# sourceMappingURL=ng-easycommerce-v18.mjs.map