valtech-components 2.0.498 → 2.0.501

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.
Files changed (49) hide show
  1. package/esm2022/lib/components/atoms/button/button.component.mjs +87 -48
  2. package/esm2022/lib/components/molecules/action-header/action-header.component.mjs +1 -1
  3. package/esm2022/lib/components/molecules/ad-slot/ad-slot.component.mjs +249 -0
  4. package/esm2022/lib/components/molecules/button-group/button-group.component.mjs +1 -1
  5. package/esm2022/lib/components/molecules/card/card.component.mjs +2 -2
  6. package/esm2022/lib/components/molecules/file-input/file-input.component.mjs +1 -1
  7. package/esm2022/lib/components/molecules/raffle-status-card/raffle-status-card.component.mjs +2 -2
  8. package/esm2022/lib/components/organisms/article/article.component.mjs +2 -2
  9. package/esm2022/lib/components/organisms/menu/menu.component.mjs +1 -1
  10. package/esm2022/lib/components/templates/page-template/page-template.component.mjs +1 -1
  11. package/esm2022/lib/services/ads/ads-consent.service.mjs +152 -0
  12. package/esm2022/lib/services/ads/ads-loader.service.mjs +160 -0
  13. package/esm2022/lib/services/ads/ads.service.mjs +449 -0
  14. package/esm2022/lib/services/ads/config.mjs +118 -0
  15. package/esm2022/lib/services/ads/index.mjs +14 -0
  16. package/esm2022/lib/services/ads/types.mjs +23 -0
  17. package/esm2022/lib/services/auth/auth.service.mjs +103 -6
  18. package/esm2022/lib/services/auth/index.mjs +4 -1
  19. package/esm2022/lib/services/auth/oauth-callback.component.mjs +141 -0
  20. package/esm2022/lib/services/auth/oauth.service.mjs +250 -0
  21. package/esm2022/lib/services/auth/types.mjs +1 -1
  22. package/esm2022/lib/services/firebase/analytics-error-handler.mjs +141 -0
  23. package/esm2022/lib/services/firebase/analytics-router-tracker.mjs +99 -0
  24. package/esm2022/lib/services/firebase/analytics.service.mjs +597 -0
  25. package/esm2022/lib/services/firebase/config.mjs +21 -2
  26. package/esm2022/lib/services/firebase/index.mjs +6 -1
  27. package/esm2022/public-api.mjs +6 -1
  28. package/fesm2022/valtech-components.mjs +2739 -239
  29. package/fesm2022/valtech-components.mjs.map +1 -1
  30. package/lib/components/atoms/button/button.component.d.ts +30 -6
  31. package/lib/components/molecules/ad-slot/ad-slot.component.d.ts +78 -0
  32. package/lib/components/organisms/article/article.component.d.ts +3 -3
  33. package/lib/services/ads/ads-consent.service.d.ts +59 -0
  34. package/lib/services/ads/ads-loader.service.d.ts +46 -0
  35. package/lib/services/ads/ads.service.d.ts +123 -0
  36. package/lib/services/ads/config.d.ts +69 -0
  37. package/lib/services/ads/index.d.ts +10 -0
  38. package/lib/services/ads/types.d.ts +163 -0
  39. package/lib/services/auth/auth.service.d.ts +56 -3
  40. package/lib/services/auth/index.d.ts +2 -0
  41. package/lib/services/auth/oauth-callback.component.d.ts +34 -0
  42. package/lib/services/auth/oauth.service.d.ts +90 -0
  43. package/lib/services/auth/types.d.ts +69 -0
  44. package/lib/services/firebase/analytics-error-handler.d.ts +54 -0
  45. package/lib/services/firebase/analytics-router-tracker.d.ts +51 -0
  46. package/lib/services/firebase/analytics.service.d.ts +256 -0
  47. package/lib/services/firebase/index.d.ts +4 -0
  48. package/package.json +1 -1
  49. package/public-api.d.ts +2 -0
@@ -0,0 +1,449 @@
1
+ /**
2
+ * Ads Service
3
+ *
4
+ * Servicio principal para Google Ad Manager (GPT).
5
+ * Integra con el sistema de consent existente y respeta usuarios premium.
6
+ */
7
+ import { Inject, Injectable, PLATFORM_ID, signal, computed } from '@angular/core';
8
+ import { isPlatformBrowser } from '@angular/common';
9
+ import { NavigationEnd } from '@angular/router';
10
+ import { filter } from 'rxjs/operators';
11
+ import { VALTECH_ADS_CONFIG } from './config';
12
+ import { AD_SIZE_MAP, } from './types';
13
+ import * as i0 from "@angular/core";
14
+ import * as i1 from "@angular/router";
15
+ import * as i2 from "./ads-loader.service";
16
+ import * as i3 from "./ads-consent.service";
17
+ /**
18
+ * Servicio principal de Google Ad Manager.
19
+ *
20
+ * Maneja la creacion, destruccion y refresh de ad slots.
21
+ * Integra automaticamente con consent mode y detecta usuarios premium.
22
+ *
23
+ * @example
24
+ * ```typescript
25
+ * @Component({...})
26
+ * export class MyComponent {
27
+ * private ads = inject(AdsService);
28
+ *
29
+ * // Verificar si mostrar ads
30
+ * showAds = this.ads.isEnabled;
31
+ *
32
+ * // Verificar si usuario es premium
33
+ * isPremium = this.ads.isPremiumUser;
34
+ * }
35
+ * ```
36
+ */
37
+ export class AdsService {
38
+ constructor(injector, config, platformId, router, loaderService, consentService) {
39
+ this.injector = injector;
40
+ this.config = config;
41
+ this.platformId = platformId;
42
+ this.router = router;
43
+ this.loaderService = loaderService;
44
+ this.consentService = consentService;
45
+ // ===========================================================================
46
+ // ESTADO (Signals)
47
+ // ===========================================================================
48
+ this._isInitialized = signal(false);
49
+ this._isEnabled = signal(false);
50
+ this._isPremiumUser = signal(false);
51
+ this._isDebugMode = signal(false);
52
+ this._slots = signal(new Map());
53
+ this._slotStates = signal(new Map());
54
+ this._events = signal([]);
55
+ /** Indica si el servicio esta inicializado */
56
+ this.isInitialized = this._isInitialized.asReadonly();
57
+ /** Indica si los ads estan habilitados (consent + no premium) */
58
+ this.isEnabled = computed(() => {
59
+ return (this._isInitialized() &&
60
+ this._isEnabled() &&
61
+ !this._isPremiumUser() &&
62
+ this.consentService.canShowAds());
63
+ });
64
+ /** Indica si el usuario es premium (no ve ads) */
65
+ this.isPremiumUser = this._isPremiumUser.asReadonly();
66
+ /** Indica si esta en modo debug */
67
+ this.isDebugMode = this._isDebugMode.asReadonly();
68
+ /** Estado de consent para ads */
69
+ this.consentState = this.consentService.adsConsentState;
70
+ /** Eventos de ads (historial) */
71
+ this.events = this._events.asReadonly();
72
+ // ===========================================================================
73
+ // INTERNOS
74
+ // ===========================================================================
75
+ this.activeSlots = new Map();
76
+ this.routerSubscription = null;
77
+ this.refreshTimerId = null;
78
+ this._isDebugMode.set(config.debugMode ?? false);
79
+ }
80
+ // ===========================================================================
81
+ // INICIALIZACION
82
+ // ===========================================================================
83
+ /**
84
+ * Inicializa el servicio de ads.
85
+ * Llamado automaticamente por APP_INITIALIZER.
86
+ * NO carga el script GPT hasta que se necesite el primer ad.
87
+ */
88
+ async initialize() {
89
+ if (!isPlatformBrowser(this.platformId)) {
90
+ return;
91
+ }
92
+ // Verificar si usuario es premium
93
+ if (this.config.isPremiumUser) {
94
+ try {
95
+ this._isPremiumUser.set(this.config.isPremiumUser());
96
+ }
97
+ catch {
98
+ this._isPremiumUser.set(false);
99
+ }
100
+ }
101
+ // Si es premium, no inicializar ads
102
+ if (this._isPremiumUser()) {
103
+ if (this._isDebugMode()) {
104
+ console.log('[ValtechAds] Usuario premium detectado - ads deshabilitados');
105
+ }
106
+ this._isInitialized.set(true);
107
+ return;
108
+ }
109
+ // Verificar consent basico
110
+ if (!this.consentService.canShowAds() && !this.config.showNonPersonalizedAds) {
111
+ if (this._isDebugMode()) {
112
+ console.log('[ValtechAds] Sin consent para ads');
113
+ }
114
+ this._isInitialized.set(true);
115
+ return;
116
+ }
117
+ // Suscribirse a cambios de ruta para verificar exclusiones
118
+ this.setupRouteListener();
119
+ // Configurar auto-refresh si esta habilitado
120
+ if (this.config.autoRefreshInterval && this.config.autoRefreshInterval > 0) {
121
+ this.setupAutoRefresh();
122
+ }
123
+ this._isEnabled.set(true);
124
+ this._isInitialized.set(true);
125
+ if (this._isDebugMode()) {
126
+ console.log('[ValtechAds] Inicializado en modo debug', {
127
+ networkId: this.config.networkId,
128
+ lazyLoad: this.config.lazyLoad,
129
+ excludeRoutes: this.config.excludeRoutes,
130
+ });
131
+ }
132
+ }
133
+ /**
134
+ * Carga el script GPT de forma lazy.
135
+ * Llamado automaticamente cuando se renderiza el primer ad slot.
136
+ */
137
+ async ensureGPTLoaded() {
138
+ if (!this.isEnabled()) {
139
+ return null;
140
+ }
141
+ return this.loaderService.loadGPT();
142
+ }
143
+ // ===========================================================================
144
+ // GESTION DE SLOTS
145
+ // ===========================================================================
146
+ /**
147
+ * Define y muestra un ad slot.
148
+ * Llamado internamente por el componente AdSlot.
149
+ *
150
+ * @param config - Configuracion del slot
151
+ * @returns ID del slot o null si no se puede mostrar
152
+ */
153
+ async defineSlot(config) {
154
+ if (!this.isEnabled()) {
155
+ this.updateSlotState(config.slotId, 'hidden');
156
+ return null;
157
+ }
158
+ // Verificar si la ruta actual esta excluida
159
+ if (this.isRouteExcluded()) {
160
+ this.updateSlotState(config.slotId, 'hidden');
161
+ return null;
162
+ }
163
+ this.updateSlotState(config.slotId, 'loading');
164
+ const googletag = await this.ensureGPTLoaded();
165
+ if (!googletag) {
166
+ this.updateSlotState(config.slotId, 'error');
167
+ return null;
168
+ }
169
+ return new Promise((resolve) => {
170
+ googletag.cmd.push(() => {
171
+ try {
172
+ const adUnitPath = `${this.config.networkId}${config.adUnitPath}`;
173
+ const sizes = this.resolveSizes(config.size);
174
+ const slot = googletag.defineSlot(adUnitPath, sizes, config.slotId);
175
+ if (!slot) {
176
+ this.updateSlotState(config.slotId, 'error');
177
+ this.emitEvent({ type: 'error', slotId: config.slotId, error: new Error('No se pudo crear el slot') });
178
+ resolve(null);
179
+ return;
180
+ }
181
+ // Agregar al servicio pubads
182
+ slot.addService(googletag.pubads());
183
+ // Configurar size mapping responsivo
184
+ if (config.sizeMapping && config.sizeMapping.length > 0) {
185
+ const mapping = this.buildSizeMapping(googletag, config.sizeMapping);
186
+ slot.defineSizeMapping(mapping);
187
+ }
188
+ // Configurar targeting
189
+ this.applyTargeting(slot, config.targeting);
190
+ // Colapsar si esta vacio
191
+ if (config.collapseEmpty ?? this.config.defaultSlotConfig?.collapseEmpty) {
192
+ slot.setCollapseEmptyDiv(true, true);
193
+ }
194
+ // Guardar referencia
195
+ this.activeSlots.set(config.slotId, slot);
196
+ this._slots.update((slots) => new Map(slots).set(config.slotId, slot));
197
+ // Mostrar el ad
198
+ googletag.enableServices();
199
+ googletag.display(config.slotId);
200
+ this.updateSlotState(config.slotId, 'rendered');
201
+ this.emitEvent({ type: 'loaded', slotId: config.slotId, isEmpty: false });
202
+ if (this._isDebugMode()) {
203
+ console.log(`[ValtechAds] Slot definido: ${config.slotId}`, {
204
+ adUnitPath,
205
+ sizes: config.size,
206
+ });
207
+ }
208
+ resolve(config.slotId);
209
+ }
210
+ catch (error) {
211
+ console.error('[ValtechAds] Error definiendo slot:', error);
212
+ this.updateSlotState(config.slotId, 'error');
213
+ this.emitEvent({ type: 'error', slotId: config.slotId, error: error });
214
+ resolve(null);
215
+ }
216
+ });
217
+ });
218
+ }
219
+ /**
220
+ * Destruye un slot y libera recursos.
221
+ *
222
+ * @param slotId - ID del slot a destruir
223
+ */
224
+ destroySlot(slotId) {
225
+ const slot = this.activeSlots.get(slotId);
226
+ if (!slot)
227
+ return;
228
+ const googletag = window.googletag;
229
+ if (googletag) {
230
+ googletag.cmd.push(() => {
231
+ googletag.destroySlots([slot]);
232
+ });
233
+ }
234
+ this.activeSlots.delete(slotId);
235
+ this._slots.update((slots) => {
236
+ const newSlots = new Map(slots);
237
+ newSlots.delete(slotId);
238
+ return newSlots;
239
+ });
240
+ this._slotStates.update((states) => {
241
+ const newStates = new Map(states);
242
+ newStates.delete(slotId);
243
+ return newStates;
244
+ });
245
+ if (this._isDebugMode()) {
246
+ console.log(`[ValtechAds] Slot destruido: ${slotId}`);
247
+ }
248
+ }
249
+ /**
250
+ * Refresca un slot especifico o todos los slots activos.
251
+ *
252
+ * @param slotIds - IDs de slots a refrescar (undefined = todos)
253
+ */
254
+ refreshSlots(slotIds) {
255
+ if (!this.isEnabled())
256
+ return;
257
+ const googletag = window.googletag;
258
+ if (!googletag)
259
+ return;
260
+ googletag.cmd.push(() => {
261
+ if (slotIds && slotIds.length > 0) {
262
+ const slots = slotIds.map((id) => this.activeSlots.get(id)).filter((s) => !!s);
263
+ if (slots.length > 0) {
264
+ googletag.pubads().refresh(slots);
265
+ if (this._isDebugMode()) {
266
+ console.log(`[ValtechAds] Slots refrescados: ${slotIds.join(', ')}`);
267
+ }
268
+ }
269
+ }
270
+ else {
271
+ googletag.pubads().refresh();
272
+ if (this._isDebugMode()) {
273
+ console.log('[ValtechAds] Todos los slots refrescados');
274
+ }
275
+ }
276
+ });
277
+ }
278
+ /**
279
+ * Obtiene el estado de un slot.
280
+ *
281
+ * @param slotId - ID del slot
282
+ * @returns Estado actual del slot
283
+ */
284
+ getSlotState(slotId) {
285
+ return this._slotStates().get(slotId) ?? 'idle';
286
+ }
287
+ // ===========================================================================
288
+ // PREMIUM USER
289
+ // ===========================================================================
290
+ /**
291
+ * Actualiza el estado premium del usuario.
292
+ * Llamar cuando cambie el estado de suscripcion.
293
+ *
294
+ * @param isPremium - Nuevo estado premium
295
+ */
296
+ updatePremiumStatus(isPremium) {
297
+ const wasEnabled = this.isEnabled();
298
+ this._isPremiumUser.set(isPremium);
299
+ if (isPremium && wasEnabled) {
300
+ // Destruir todos los slots activos
301
+ this.destroyAllSlots();
302
+ if (this._isDebugMode()) {
303
+ console.log('[ValtechAds] Usuario ahora es premium - ads deshabilitados');
304
+ }
305
+ }
306
+ else if (!isPremium && !wasEnabled && this._isInitialized()) {
307
+ if (this._isDebugMode()) {
308
+ console.log('[ValtechAds] Usuario ya no es premium - ads habilitados');
309
+ }
310
+ }
311
+ }
312
+ /**
313
+ * Destruye todos los slots activos.
314
+ */
315
+ destroyAllSlots() {
316
+ const googletag = window.googletag;
317
+ if (googletag) {
318
+ googletag.cmd.push(() => {
319
+ googletag.destroySlots();
320
+ });
321
+ }
322
+ this.activeSlots.clear();
323
+ this._slots.set(new Map());
324
+ this._slotStates.set(new Map());
325
+ if (this._isDebugMode()) {
326
+ console.log('[ValtechAds] Todos los slots destruidos');
327
+ }
328
+ }
329
+ // ===========================================================================
330
+ // PRIVATE METHODS
331
+ // ===========================================================================
332
+ resolveSizes(size) {
333
+ if (typeof size === 'string') {
334
+ // Single preset size
335
+ const mapped = AD_SIZE_MAP[size];
336
+ return mapped === 'fluid' ? 'fluid' : [mapped];
337
+ }
338
+ if (Array.isArray(size)) {
339
+ if (typeof size[0] === 'string') {
340
+ // Array of preset sizes
341
+ return size.map((s) => {
342
+ const mapped = AD_SIZE_MAP[s];
343
+ return mapped === 'fluid' ? 'fluid' : mapped;
344
+ });
345
+ }
346
+ // Direct size array
347
+ return size;
348
+ }
349
+ return 'fluid';
350
+ }
351
+ buildSizeMapping(googletag, mapping) {
352
+ const builder = googletag.sizeMapping();
353
+ // Ordenar por viewport descendente
354
+ const sorted = [...mapping].sort((a, b) => b.viewportWidth - a.viewportWidth);
355
+ for (const m of sorted) {
356
+ builder.addSize([m.viewportWidth, 0], m.sizes);
357
+ }
358
+ return builder.build();
359
+ }
360
+ applyTargeting(slot, targeting) {
361
+ // Aplicar targeting global
362
+ if (this.config.globalTargeting) {
363
+ for (const [key, value] of Object.entries(this.config.globalTargeting)) {
364
+ slot.setTargeting(key, value);
365
+ }
366
+ }
367
+ // Aplicar targeting del slot
368
+ if (targeting) {
369
+ for (const [key, value] of Object.entries(targeting)) {
370
+ slot.setTargeting(key, value);
371
+ }
372
+ }
373
+ // Agregar targeting de debug si esta habilitado
374
+ if (this._isDebugMode()) {
375
+ slot.setTargeting('test', 'true');
376
+ }
377
+ }
378
+ updateSlotState(slotId, state) {
379
+ this._slotStates.update((states) => new Map(states).set(slotId, state));
380
+ }
381
+ emitEvent(event) {
382
+ const fullEvent = { ...event, timestamp: new Date() };
383
+ this._events.update((events) => [...events.slice(-99), fullEvent]);
384
+ // Callbacks de configuracion
385
+ if (event.type === 'loaded' || event.type === 'empty') {
386
+ this.config.onAdLoaded?.(event.slotId, event.isEmpty ?? false);
387
+ }
388
+ else if (event.type === 'error' && event.error) {
389
+ this.config.onAdError?.(event.slotId, event.error);
390
+ }
391
+ }
392
+ setupRouteListener() {
393
+ if (!this.config.excludeRoutes || this.config.excludeRoutes.length === 0) {
394
+ return;
395
+ }
396
+ this.routerSubscription = this.router.events
397
+ .pipe(filter((event) => event instanceof NavigationEnd))
398
+ .subscribe((event) => {
399
+ const shouldHide = this.isRouteExcluded(event.urlAfterRedirects);
400
+ if (shouldHide) {
401
+ this.destroyAllSlots();
402
+ }
403
+ });
404
+ }
405
+ setupAutoRefresh() {
406
+ const intervalMs = (this.config.autoRefreshInterval ?? 0) * 1000;
407
+ if (intervalMs <= 0)
408
+ return;
409
+ this.refreshTimerId = setInterval(() => {
410
+ if (this.isEnabled() && this.activeSlots.size > 0) {
411
+ this.refreshSlots();
412
+ }
413
+ }, intervalMs);
414
+ }
415
+ isRouteExcluded(url) {
416
+ if (!this.config.excludeRoutes || this.config.excludeRoutes.length === 0) {
417
+ return false;
418
+ }
419
+ const currentUrl = url ?? this.router.url;
420
+ return this.config.excludeRoutes.some((pattern) => {
421
+ const regex = new RegExp(pattern);
422
+ return regex.test(currentUrl);
423
+ });
424
+ }
425
+ // ===========================================================================
426
+ // LIFECYCLE
427
+ // ===========================================================================
428
+ ngOnDestroy() {
429
+ this.routerSubscription?.unsubscribe();
430
+ if (this.refreshTimerId) {
431
+ clearInterval(this.refreshTimerId);
432
+ }
433
+ this.destroyAllSlots();
434
+ this.consentService.destroy();
435
+ }
436
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AdsService, deps: [{ token: i0.Injector }, { token: VALTECH_ADS_CONFIG }, { token: PLATFORM_ID }, { token: i1.Router }, { token: i2.AdsLoaderService }, { token: i3.AdsConsentService }], target: i0.ɵɵFactoryTarget.Injectable }); }
437
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AdsService, providedIn: 'root' }); }
438
+ }
439
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AdsService, decorators: [{
440
+ type: Injectable,
441
+ args: [{ providedIn: 'root' }]
442
+ }], ctorParameters: () => [{ type: i0.Injector }, { type: undefined, decorators: [{
443
+ type: Inject,
444
+ args: [VALTECH_ADS_CONFIG]
445
+ }] }, { type: Object, decorators: [{
446
+ type: Inject,
447
+ args: [PLATFORM_ID]
448
+ }] }, { type: i1.Router }, { type: i2.AdsLoaderService }, { type: i3.AdsConsentService }] });
449
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ads.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/ads/ads.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Ads Service\n *\n * Servicio principal para Google Ad Manager (GPT).\n * Integra con el sistema de consent existente y respeta usuarios premium.\n */\n\nimport { Inject, Injectable, Injector, PLATFORM_ID, signal, computed, OnDestroy } from '@angular/core';\nimport { isPlatformBrowser } from '@angular/common';\nimport { Router, NavigationEnd } from '@angular/router';\nimport { filter } from 'rxjs/operators';\nimport { Subscription } from 'rxjs';\n\nimport { VALTECH_ADS_CONFIG } from './config';\nimport { AdsLoaderService } from './ads-loader.service';\nimport { AdsConsentService } from './ads-consent.service';\nimport {\n  ValtechAdsConfig,\n  AdSlotConfig,\n  AdSlotState,\n  AdEvent,\n  GPTSlot,\n  GPTNamespace,\n  SizeMapping,\n  AD_SIZE_MAP,\n  AdSlotSize,\n} from './types';\n\n/**\n * Servicio principal de Google Ad Manager.\n *\n * Maneja la creacion, destruccion y refresh de ad slots.\n * Integra automaticamente con consent mode y detecta usuarios premium.\n *\n * @example\n * ```typescript\n * @Component({...})\n * export class MyComponent {\n *   private ads = inject(AdsService);\n *\n *   // Verificar si mostrar ads\n *   showAds = this.ads.isEnabled;\n *\n *   // Verificar si usuario es premium\n *   isPremium = this.ads.isPremiumUser;\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class AdsService implements OnDestroy {\n  // ===========================================================================\n  // ESTADO (Signals)\n  // ===========================================================================\n\n  private readonly _isInitialized = signal<boolean>(false);\n  private readonly _isEnabled = signal<boolean>(false);\n  private readonly _isPremiumUser = signal<boolean>(false);\n  private readonly _isDebugMode = signal<boolean>(false);\n  private readonly _slots = signal<Map<string, GPTSlot>>(new Map());\n  private readonly _slotStates = signal<Map<string, AdSlotState>>(new Map());\n  private readonly _events = signal<AdEvent[]>([]);\n\n  /** Indica si el servicio esta inicializado */\n  readonly isInitialized = this._isInitialized.asReadonly();\n\n  /** Indica si los ads estan habilitados (consent + no premium) */\n  readonly isEnabled = computed(() => {\n    return (\n      this._isInitialized() &&\n      this._isEnabled() &&\n      !this._isPremiumUser() &&\n      this.consentService.canShowAds()\n    );\n  });\n\n  /** Indica si el usuario es premium (no ve ads) */\n  readonly isPremiumUser = this._isPremiumUser.asReadonly();\n\n  /** Indica si esta en modo debug */\n  readonly isDebugMode = this._isDebugMode.asReadonly();\n\n  /** Estado de consent para ads */\n  readonly consentState = this.consentService.adsConsentState;\n\n  /** Eventos de ads (historial) */\n  readonly events = this._events.asReadonly();\n\n  // ===========================================================================\n  // INTERNOS\n  // ===========================================================================\n\n  private activeSlots: Map<string, GPTSlot> = new Map();\n  private routerSubscription: Subscription | null = null;\n  private refreshTimerId: ReturnType<typeof setInterval> | null = null;\n\n  constructor(\n    private injector: Injector,\n    @Inject(VALTECH_ADS_CONFIG) private config: ValtechAdsConfig,\n    @Inject(PLATFORM_ID) private platformId: Object,\n    private router: Router,\n    private loaderService: AdsLoaderService,\n    private consentService: AdsConsentService\n  ) {\n    this._isDebugMode.set(config.debugMode ?? false);\n  }\n\n  // ===========================================================================\n  // INICIALIZACION\n  // ===========================================================================\n\n  /**\n   * Inicializa el servicio de ads.\n   * Llamado automaticamente por APP_INITIALIZER.\n   * NO carga el script GPT hasta que se necesite el primer ad.\n   */\n  async initialize(): Promise<void> {\n    if (!isPlatformBrowser(this.platformId)) {\n      return;\n    }\n\n    // Verificar si usuario es premium\n    if (this.config.isPremiumUser) {\n      try {\n        this._isPremiumUser.set(this.config.isPremiumUser());\n      } catch {\n        this._isPremiumUser.set(false);\n      }\n    }\n\n    // Si es premium, no inicializar ads\n    if (this._isPremiumUser()) {\n      if (this._isDebugMode()) {\n        console.log('[ValtechAds] Usuario premium detectado - ads deshabilitados');\n      }\n      this._isInitialized.set(true);\n      return;\n    }\n\n    // Verificar consent basico\n    if (!this.consentService.canShowAds() && !this.config.showNonPersonalizedAds) {\n      if (this._isDebugMode()) {\n        console.log('[ValtechAds] Sin consent para ads');\n      }\n      this._isInitialized.set(true);\n      return;\n    }\n\n    // Suscribirse a cambios de ruta para verificar exclusiones\n    this.setupRouteListener();\n\n    // Configurar auto-refresh si esta habilitado\n    if (this.config.autoRefreshInterval && this.config.autoRefreshInterval > 0) {\n      this.setupAutoRefresh();\n    }\n\n    this._isEnabled.set(true);\n    this._isInitialized.set(true);\n\n    if (this._isDebugMode()) {\n      console.log('[ValtechAds] Inicializado en modo debug', {\n        networkId: this.config.networkId,\n        lazyLoad: this.config.lazyLoad,\n        excludeRoutes: this.config.excludeRoutes,\n      });\n    }\n  }\n\n  /**\n   * Carga el script GPT de forma lazy.\n   * Llamado automaticamente cuando se renderiza el primer ad slot.\n   */\n  async ensureGPTLoaded(): Promise<GPTNamespace | null> {\n    if (!this.isEnabled()) {\n      return null;\n    }\n\n    return this.loaderService.loadGPT();\n  }\n\n  // ===========================================================================\n  // GESTION DE SLOTS\n  // ===========================================================================\n\n  /**\n   * Define y muestra un ad slot.\n   * Llamado internamente por el componente AdSlot.\n   *\n   * @param config - Configuracion del slot\n   * @returns ID del slot o null si no se puede mostrar\n   */\n  async defineSlot(config: AdSlotConfig): Promise<string | null> {\n    if (!this.isEnabled()) {\n      this.updateSlotState(config.slotId, 'hidden');\n      return null;\n    }\n\n    // Verificar si la ruta actual esta excluida\n    if (this.isRouteExcluded()) {\n      this.updateSlotState(config.slotId, 'hidden');\n      return null;\n    }\n\n    this.updateSlotState(config.slotId, 'loading');\n\n    const googletag = await this.ensureGPTLoaded();\n    if (!googletag) {\n      this.updateSlotState(config.slotId, 'error');\n      return null;\n    }\n\n    return new Promise((resolve) => {\n      googletag.cmd.push(() => {\n        try {\n          const adUnitPath = `${this.config.networkId}${config.adUnitPath}`;\n          const sizes = this.resolveSizes(config.size);\n\n          const slot = googletag.defineSlot(adUnitPath, sizes, config.slotId);\n          if (!slot) {\n            this.updateSlotState(config.slotId, 'error');\n            this.emitEvent({ type: 'error', slotId: config.slotId, error: new Error('No se pudo crear el slot') });\n            resolve(null);\n            return;\n          }\n\n          // Agregar al servicio pubads\n          slot.addService(googletag.pubads());\n\n          // Configurar size mapping responsivo\n          if (config.sizeMapping && config.sizeMapping.length > 0) {\n            const mapping = this.buildSizeMapping(googletag, config.sizeMapping);\n            slot.defineSizeMapping(mapping);\n          }\n\n          // Configurar targeting\n          this.applyTargeting(slot, config.targeting);\n\n          // Colapsar si esta vacio\n          if (config.collapseEmpty ?? this.config.defaultSlotConfig?.collapseEmpty) {\n            slot.setCollapseEmptyDiv(true, true);\n          }\n\n          // Guardar referencia\n          this.activeSlots.set(config.slotId, slot);\n          this._slots.update((slots) => new Map(slots).set(config.slotId, slot));\n\n          // Mostrar el ad\n          googletag.enableServices();\n          googletag.display(config.slotId);\n\n          this.updateSlotState(config.slotId, 'rendered');\n          this.emitEvent({ type: 'loaded', slotId: config.slotId, isEmpty: false });\n\n          if (this._isDebugMode()) {\n            console.log(`[ValtechAds] Slot definido: ${config.slotId}`, {\n              adUnitPath,\n              sizes: config.size,\n            });\n          }\n\n          resolve(config.slotId);\n        } catch (error) {\n          console.error('[ValtechAds] Error definiendo slot:', error);\n          this.updateSlotState(config.slotId, 'error');\n          this.emitEvent({ type: 'error', slotId: config.slotId, error: error as Error });\n          resolve(null);\n        }\n      });\n    });\n  }\n\n  /**\n   * Destruye un slot y libera recursos.\n   *\n   * @param slotId - ID del slot a destruir\n   */\n  destroySlot(slotId: string): void {\n    const slot = this.activeSlots.get(slotId);\n    if (!slot) return;\n\n    const googletag = window.googletag;\n    if (googletag) {\n      googletag.cmd.push(() => {\n        googletag.destroySlots([slot]);\n      });\n    }\n\n    this.activeSlots.delete(slotId);\n    this._slots.update((slots) => {\n      const newSlots = new Map(slots);\n      newSlots.delete(slotId);\n      return newSlots;\n    });\n    this._slotStates.update((states) => {\n      const newStates = new Map(states);\n      newStates.delete(slotId);\n      return newStates;\n    });\n\n    if (this._isDebugMode()) {\n      console.log(`[ValtechAds] Slot destruido: ${slotId}`);\n    }\n  }\n\n  /**\n   * Refresca un slot especifico o todos los slots activos.\n   *\n   * @param slotIds - IDs de slots a refrescar (undefined = todos)\n   */\n  refreshSlots(slotIds?: string[]): void {\n    if (!this.isEnabled()) return;\n\n    const googletag = window.googletag;\n    if (!googletag) return;\n\n    googletag.cmd.push(() => {\n      if (slotIds && slotIds.length > 0) {\n        const slots = slotIds.map((id) => this.activeSlots.get(id)).filter((s): s is GPTSlot => !!s);\n        if (slots.length > 0) {\n          googletag.pubads().refresh(slots);\n          if (this._isDebugMode()) {\n            console.log(`[ValtechAds] Slots refrescados: ${slotIds.join(', ')}`);\n          }\n        }\n      } else {\n        googletag.pubads().refresh();\n        if (this._isDebugMode()) {\n          console.log('[ValtechAds] Todos los slots refrescados');\n        }\n      }\n    });\n  }\n\n  /**\n   * Obtiene el estado de un slot.\n   *\n   * @param slotId - ID del slot\n   * @returns Estado actual del slot\n   */\n  getSlotState(slotId: string): AdSlotState {\n    return this._slotStates().get(slotId) ?? 'idle';\n  }\n\n  // ===========================================================================\n  // PREMIUM USER\n  // ===========================================================================\n\n  /**\n   * Actualiza el estado premium del usuario.\n   * Llamar cuando cambie el estado de suscripcion.\n   *\n   * @param isPremium - Nuevo estado premium\n   */\n  updatePremiumStatus(isPremium: boolean): void {\n    const wasEnabled = this.isEnabled();\n    this._isPremiumUser.set(isPremium);\n\n    if (isPremium && wasEnabled) {\n      // Destruir todos los slots activos\n      this.destroyAllSlots();\n      if (this._isDebugMode()) {\n        console.log('[ValtechAds] Usuario ahora es premium - ads deshabilitados');\n      }\n    } else if (!isPremium && !wasEnabled && this._isInitialized()) {\n      if (this._isDebugMode()) {\n        console.log('[ValtechAds] Usuario ya no es premium - ads habilitados');\n      }\n    }\n  }\n\n  /**\n   * Destruye todos los slots activos.\n   */\n  destroyAllSlots(): void {\n    const googletag = window.googletag;\n    if (googletag) {\n      googletag.cmd.push(() => {\n        googletag.destroySlots();\n      });\n    }\n    this.activeSlots.clear();\n    this._slots.set(new Map());\n    this._slotStates.set(new Map());\n\n    if (this._isDebugMode()) {\n      console.log('[ValtechAds] Todos los slots destruidos');\n    }\n  }\n\n  // ===========================================================================\n  // PRIVATE METHODS\n  // ===========================================================================\n\n  private resolveSizes(size: AdSlotConfig['size']): unknown {\n    if (typeof size === 'string') {\n      // Single preset size\n      const mapped = AD_SIZE_MAP[size as AdSlotSize];\n      return mapped === 'fluid' ? 'fluid' : [mapped];\n    }\n\n    if (Array.isArray(size)) {\n      if (typeof size[0] === 'string') {\n        // Array of preset sizes\n        return (size as AdSlotSize[]).map((s) => {\n          const mapped = AD_SIZE_MAP[s];\n          return mapped === 'fluid' ? 'fluid' : mapped;\n        });\n      }\n      // Direct size array\n      return size;\n    }\n\n    return 'fluid';\n  }\n\n  private buildSizeMapping(googletag: GPTNamespace, mapping: SizeMapping[]): unknown {\n    const builder = googletag.sizeMapping();\n\n    // Ordenar por viewport descendente\n    const sorted = [...mapping].sort((a, b) => b.viewportWidth - a.viewportWidth);\n\n    for (const m of sorted) {\n      builder.addSize([m.viewportWidth, 0], m.sizes);\n    }\n\n    return builder.build();\n  }\n\n  private applyTargeting(slot: GPTSlot, targeting?: Record<string, string | string[]>): void {\n    // Aplicar targeting global\n    if (this.config.globalTargeting) {\n      for (const [key, value] of Object.entries(this.config.globalTargeting)) {\n        slot.setTargeting(key, value);\n      }\n    }\n\n    // Aplicar targeting del slot\n    if (targeting) {\n      for (const [key, value] of Object.entries(targeting)) {\n        slot.setTargeting(key, value);\n      }\n    }\n\n    // Agregar targeting de debug si esta habilitado\n    if (this._isDebugMode()) {\n      slot.setTargeting('test', 'true');\n    }\n  }\n\n  private updateSlotState(slotId: string, state: AdSlotState): void {\n    this._slotStates.update((states) => new Map(states).set(slotId, state));\n  }\n\n  private emitEvent(event: Omit<AdEvent, 'timestamp'>): void {\n    const fullEvent: AdEvent = { ...event, timestamp: new Date() };\n    this._events.update((events) => [...events.slice(-99), fullEvent]);\n\n    // Callbacks de configuracion\n    if (event.type === 'loaded' || event.type === 'empty') {\n      this.config.onAdLoaded?.(event.slotId, event.isEmpty ?? false);\n    } else if (event.type === 'error' && event.error) {\n      this.config.onAdError?.(event.slotId, event.error);\n    }\n  }\n\n  private setupRouteListener(): void {\n    if (!this.config.excludeRoutes || this.config.excludeRoutes.length === 0) {\n      return;\n    }\n\n    this.routerSubscription = this.router.events\n      .pipe(filter((event): event is NavigationEnd => event instanceof NavigationEnd))\n      .subscribe((event: NavigationEnd) => {\n        const shouldHide = this.isRouteExcluded(event.urlAfterRedirects);\n\n        if (shouldHide) {\n          this.destroyAllSlots();\n        }\n      });\n  }\n\n  private setupAutoRefresh(): void {\n    const intervalMs = (this.config.autoRefreshInterval ?? 0) * 1000;\n    if (intervalMs <= 0) return;\n\n    this.refreshTimerId = setInterval(() => {\n      if (this.isEnabled() && this.activeSlots.size > 0) {\n        this.refreshSlots();\n      }\n    }, intervalMs);\n  }\n\n  private isRouteExcluded(url?: string): boolean {\n    if (!this.config.excludeRoutes || this.config.excludeRoutes.length === 0) {\n      return false;\n    }\n\n    const currentUrl = url ?? this.router.url;\n    return this.config.excludeRoutes.some((pattern) => {\n      const regex = new RegExp(pattern);\n      return regex.test(currentUrl);\n    });\n  }\n\n  // ===========================================================================\n  // LIFECYCLE\n  // ===========================================================================\n\n  ngOnDestroy(): void {\n    this.routerSubscription?.unsubscribe();\n\n    if (this.refreshTimerId) {\n      clearInterval(this.refreshTimerId);\n    }\n\n    this.destroyAllSlots();\n    this.consentService.destroy();\n  }\n}\n"]}
@@ -0,0 +1,118 @@
1
+ /**
2
+ * Ads Configuration
3
+ *
4
+ * Configuracion e inicializacion de Google Ad Manager para Angular.
5
+ * Usa provideValtechAds() en el bootstrap de tu aplicacion.
6
+ */
7
+ import { InjectionToken, makeEnvironmentProviders, APP_INITIALIZER, } from '@angular/core';
8
+ import { AdsService } from './ads.service';
9
+ /**
10
+ * Token de inyeccion para la configuracion de Ads.
11
+ * Usado internamente por los servicios de ads.
12
+ */
13
+ export const VALTECH_ADS_CONFIG = new InjectionToken('ValtechAdsConfig');
14
+ /**
15
+ * Configuracion por defecto de lazy loading.
16
+ */
17
+ export const DEFAULT_LAZY_LOAD_CONFIG = {
18
+ rootMargin: '200px',
19
+ threshold: 0,
20
+ fetchMarginPercent: 200,
21
+ renderMarginPercent: 100,
22
+ };
23
+ /**
24
+ * Configuracion por defecto del servicio de ads.
25
+ */
26
+ export const DEFAULT_ADS_CONFIG = {
27
+ debugMode: false,
28
+ lazyLoad: true,
29
+ lazyLoadConfig: DEFAULT_LAZY_LOAD_CONFIG,
30
+ enablePersonalization: true,
31
+ showNonPersonalizedAds: true,
32
+ autoRefreshInterval: 0, // Deshabilitado por defecto
33
+ defaultSlotConfig: {
34
+ collapseEmpty: true,
35
+ showSkeleton: true,
36
+ minHeight: '90px',
37
+ },
38
+ };
39
+ /**
40
+ * Factory para inicializar AdsService.
41
+ * Se ejecuta durante el bootstrap de la aplicacion.
42
+ */
43
+ function initializeAds(adsService) {
44
+ return () => adsService.initialize();
45
+ }
46
+ /**
47
+ * Provee el servicio de Ads a la aplicacion Angular.
48
+ *
49
+ * @param config - Configuracion de Google Ad Manager
50
+ * @returns EnvironmentProviders para usar en bootstrapApplication
51
+ *
52
+ * @example
53
+ * ```typescript
54
+ * // main.ts
55
+ * import { bootstrapApplication } from '@angular/platform-browser';
56
+ * import { provideValtechAds, provideValtechFirebase } from 'valtech-components';
57
+ * import { environment } from './environments/environment';
58
+ *
59
+ * bootstrapApplication(AppComponent, {
60
+ * providers: [
61
+ * // Firebase primero (para consent mode)
62
+ * provideValtechFirebase({
63
+ * firebase: environment.firebase,
64
+ * enableAnalytics: true,
65
+ * }),
66
+ *
67
+ * // Luego Ads
68
+ * provideValtechAds({
69
+ * networkId: '/12345678',
70
+ * debugMode: !environment.production,
71
+ * isPremiumUser: () => inject(AuthService).hasRole('premium'),
72
+ * }),
73
+ * ],
74
+ * });
75
+ * ```
76
+ *
77
+ * @example Con targeting global y rutas excluidas
78
+ * ```typescript
79
+ * provideValtechAds({
80
+ * networkId: '/12345678',
81
+ * globalTargeting: {
82
+ * site: 'myvaltech',
83
+ * section: 'app',
84
+ * },
85
+ * excludeRoutes: [
86
+ * '^/checkout',
87
+ * '^/payment',
88
+ * '^/premium',
89
+ * ],
90
+ * })
91
+ * ```
92
+ */
93
+ export function provideValtechAds(config) {
94
+ // Merge config con defaults
95
+ const mergedConfig = {
96
+ ...DEFAULT_ADS_CONFIG,
97
+ ...config,
98
+ lazyLoadConfig: {
99
+ ...DEFAULT_LAZY_LOAD_CONFIG,
100
+ ...config.lazyLoadConfig,
101
+ },
102
+ defaultSlotConfig: {
103
+ ...DEFAULT_ADS_CONFIG.defaultSlotConfig,
104
+ ...config.defaultSlotConfig,
105
+ },
106
+ };
107
+ return makeEnvironmentProviders([
108
+ { provide: VALTECH_ADS_CONFIG, useValue: mergedConfig },
109
+ // Inicializar AdsService al arrancar (pero NO cargar script GPT aun)
110
+ {
111
+ provide: APP_INITIALIZER,
112
+ useFactory: initializeAds,
113
+ deps: [AdsService],
114
+ multi: true,
115
+ },
116
+ ]);
117
+ }
118
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9zZXJ2aWNlcy9hZHMvY29uZmlnLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7OztHQUtHO0FBRUgsT0FBTyxFQUVMLGNBQWMsRUFDZCx3QkFBd0IsRUFDeEIsZUFBZSxHQUNoQixNQUFNLGVBQWUsQ0FBQztBQUV2QixPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRTNDOzs7R0FHRztBQUNILE1BQU0sQ0FBQyxNQUFNLGtCQUFrQixHQUFHLElBQUksY0FBYyxDQUFtQixrQkFBa0IsQ0FBQyxDQUFDO0FBRTNGOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sd0JBQXdCLEdBQW1CO0lBQ3RELFVBQVUsRUFBRSxPQUFPO0lBQ25CLFNBQVMsRUFBRSxDQUFDO0lBQ1osa0JBQWtCLEVBQUUsR0FBRztJQUN2QixtQkFBbUIsRUFBRSxHQUFHO0NBQ3pCLENBQUM7QUFFRjs7R0FFRztBQUNILE1BQU0sQ0FBQyxNQUFNLGtCQUFrQixHQUE4QjtJQUMzRCxTQUFTLEVBQUUsS0FBSztJQUNoQixRQUFRLEVBQUUsSUFBSTtJQUNkLGNBQWMsRUFBRSx3QkFBd0I7SUFDeEMscUJBQXFCLEVBQUUsSUFBSTtJQUMzQixzQkFBc0IsRUFBRSxJQUFJO0lBQzVCLG1CQUFtQixFQUFFLENBQUMsRUFBRSw0QkFBNEI7SUFDcEQsaUJBQWlCLEVBQUU7UUFDakIsYUFBYSxFQUFFLElBQUk7UUFDbkIsWUFBWSxFQUFFLElBQUk7UUFDbEIsU0FBUyxFQUFFLE1BQU07S0FDbEI7Q0FDRixDQUFDO0FBRUY7OztHQUdHO0FBQ0gsU0FBUyxhQUFhLENBQUMsVUFBc0I7SUFDM0MsT0FBTyxHQUFHLEVBQUUsQ0FBQyxVQUFVLENBQUMsVUFBVSxFQUFFLENBQUM7QUFDdkMsQ0FBQztBQUVEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBOENHO0FBQ0gsTUFBTSxVQUFVLGlCQUFpQixDQUFDLE1BQXdCO0lBQ3hELDRCQUE0QjtJQUM1QixNQUFNLFlBQVksR0FBcUI7UUFDckMsR0FBRyxrQkFBa0I7UUFDckIsR0FBRyxNQUFNO1FBQ1QsY0FBYyxFQUFFO1lBQ2QsR0FBRyx3QkFBd0I7WUFDM0IsR0FBRyxNQUFNLENBQUMsY0FBYztTQUN6QjtRQUNELGlCQUFpQixFQUFFO1lBQ2pCLEdBQUcsa0JBQWtCLENBQUMsaUJBQWlCO1lBQ3ZDLEdBQUcsTUFBTSxDQUFDLGlCQUFpQjtTQUM1QjtLQUNGLENBQUM7SUFFRixPQUFPLHdCQUF3QixDQUFDO1FBQzlCLEVBQUUsT0FBTyxFQUFFLGtCQUFrQixFQUFFLFFBQVEsRUFBRSxZQUFZLEVBQUU7UUFDdkQscUVBQXFFO1FBQ3JFO1lBQ0UsT0FBTyxFQUFFLGVBQWU7WUFDeEIsVUFBVSxFQUFFLGFBQWE7WUFDekIsSUFBSSxFQUFFLENBQUMsVUFBVSxDQUFDO1lBQ2xCLEtBQUssRUFBRSxJQUFJO1NBQ1o7S0FDRixDQUFDLENBQUM7QUFDTCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBBZHMgQ29uZmlndXJhdGlvblxuICpcbiAqIENvbmZpZ3VyYWNpb24gZSBpbmljaWFsaXphY2lvbiBkZSBHb29nbGUgQWQgTWFuYWdlciBwYXJhIEFuZ3VsYXIuXG4gKiBVc2EgcHJvdmlkZVZhbHRlY2hBZHMoKSBlbiBlbCBib290c3RyYXAgZGUgdHUgYXBsaWNhY2lvbi5cbiAqL1xuXG5pbXBvcnQge1xuICBFbnZpcm9ubWVudFByb3ZpZGVycyxcbiAgSW5qZWN0aW9uVG9rZW4sXG4gIG1ha2VFbnZpcm9ubWVudFByb3ZpZGVycyxcbiAgQVBQX0lOSVRJQUxJWkVSLFxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFZhbHRlY2hBZHNDb25maWcsIExhenlMb2FkQ29uZmlnIH0gZnJvbSAnLi90eXBlcyc7XG5pbXBvcnQgeyBBZHNTZXJ2aWNlIH0gZnJvbSAnLi9hZHMuc2VydmljZSc7XG5cbi8qKlxuICogVG9rZW4gZGUgaW55ZWNjaW9uIHBhcmEgbGEgY29uZmlndXJhY2lvbiBkZSBBZHMuXG4gKiBVc2FkbyBpbnRlcm5hbWVudGUgcG9yIGxvcyBzZXJ2aWNpb3MgZGUgYWRzLlxuICovXG5leHBvcnQgY29uc3QgVkFMVEVDSF9BRFNfQ09ORklHID0gbmV3IEluamVjdGlvblRva2VuPFZhbHRlY2hBZHNDb25maWc+KCdWYWx0ZWNoQWRzQ29uZmlnJyk7XG5cbi8qKlxuICogQ29uZmlndXJhY2lvbiBwb3IgZGVmZWN0byBkZSBsYXp5IGxvYWRpbmcuXG4gKi9cbmV4cG9ydCBjb25zdCBERUZBVUxUX0xBWllfTE9BRF9DT05GSUc6IExhenlMb2FkQ29uZmlnID0ge1xuICByb290TWFyZ2luOiAnMjAwcHgnLFxuICB0aHJlc2hvbGQ6IDAsXG4gIGZldGNoTWFyZ2luUGVyY2VudDogMjAwLFxuICByZW5kZXJNYXJnaW5QZXJjZW50OiAxMDAsXG59O1xuXG4vKipcbiAqIENvbmZpZ3VyYWNpb24gcG9yIGRlZmVjdG8gZGVsIHNlcnZpY2lvIGRlIGFkcy5cbiAqL1xuZXhwb3J0IGNvbnN0IERFRkFVTFRfQURTX0NPTkZJRzogUGFydGlhbDxWYWx0ZWNoQWRzQ29uZmlnPiA9IHtcbiAgZGVidWdNb2RlOiBmYWxzZSxcbiAgbGF6eUxvYWQ6IHRydWUsXG4gIGxhenlMb2FkQ29uZmlnOiBERUZBVUxUX0xBWllfTE9BRF9DT05GSUcsXG4gIGVuYWJsZVBlcnNvbmFsaXphdGlvbjogdHJ1ZSxcbiAgc2hvd05vblBlcnNvbmFsaXplZEFkczogdHJ1ZSxcbiAgYXV0b1JlZnJlc2hJbnRlcnZhbDogMCwgLy8gRGVzaGFiaWxpdGFkbyBwb3IgZGVmZWN0b1xuICBkZWZhdWx0U2xvdENvbmZpZzoge1xuICAgIGNvbGxhcHNlRW1wdHk6IHRydWUsXG4gICAgc2hvd1NrZWxldG9uOiB0cnVlLFxuICAgIG1pbkhlaWdodDogJzkwcHgnLFxuICB9LFxufTtcblxuLyoqXG4gKiBGYWN0b3J5IHBhcmEgaW5pY2lhbGl6YXIgQWRzU2VydmljZS5cbiAqIFNlIGVqZWN1dGEgZHVyYW50ZSBlbCBib290c3RyYXAgZGUgbGEgYXBsaWNhY2lvbi5cbiAqL1xuZnVuY3Rpb24gaW5pdGlhbGl6ZUFkcyhhZHNTZXJ2aWNlOiBBZHNTZXJ2aWNlKTogKCkgPT4gUHJvbWlzZTx2b2lkPiB7XG4gIHJldHVybiAoKSA9PiBhZHNTZXJ2aWNlLmluaXRpYWxpemUoKTtcbn1cblxuLyoqXG4gKiBQcm92ZWUgZWwgc2VydmljaW8gZGUgQWRzIGEgbGEgYXBsaWNhY2lvbiBBbmd1bGFyLlxuICpcbiAqIEBwYXJhbSBjb25maWcgLSBDb25maWd1cmFjaW9uIGRlIEdvb2dsZSBBZCBNYW5hZ2VyXG4gKiBAcmV0dXJucyBFbnZpcm9ubWVudFByb3ZpZGVycyBwYXJhIHVzYXIgZW4gYm9vdHN0cmFwQXBwbGljYXRpb25cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogLy8gbWFpbi50c1xuICogaW1wb3J0IHsgYm9vdHN0cmFwQXBwbGljYXRpb24gfSBmcm9tICdAYW5ndWxhci9wbGF0Zm9ybS1icm93c2VyJztcbiAqIGltcG9ydCB7IHByb3ZpZGVWYWx0ZWNoQWRzLCBwcm92aWRlVmFsdGVjaEZpcmViYXNlIH0gZnJvbSAndmFsdGVjaC1jb21wb25lbnRzJztcbiAqIGltcG9ydCB7IGVudmlyb25tZW50IH0gZnJvbSAnLi9lbnZpcm9ubWVudHMvZW52aXJvbm1lbnQnO1xuICpcbiAqIGJvb3RzdHJhcEFwcGxpY2F0aW9uKEFwcENvbXBvbmVudCwge1xuICogICBwcm92aWRlcnM6IFtcbiAqICAgICAvLyBGaXJlYmFzZSBwcmltZXJvIChwYXJhIGNvbnNlbnQgbW9kZSlcbiAqICAgICBwcm92aWRlVmFsdGVjaEZpcmViYXNlKHtcbiAqICAgICAgIGZpcmViYXNlOiBlbnZpcm9ubWVudC5maXJlYmFzZSxcbiAqICAgICAgIGVuYWJsZUFuYWx5dGljczogdHJ1ZSxcbiAqICAgICB9KSxcbiAqXG4gKiAgICAgLy8gTHVlZ28gQWRzXG4gKiAgICAgcHJvdmlkZVZhbHRlY2hBZHMoe1xuICogICAgICAgbmV0d29ya0lkOiAnLzEyMzQ1Njc4JyxcbiAqICAgICAgIGRlYnVnTW9kZTogIWVudmlyb25tZW50LnByb2R1Y3Rpb24sXG4gKiAgICAgICBpc1ByZW1pdW1Vc2VyOiAoKSA9PiBpbmplY3QoQXV0aFNlcnZpY2UpLmhhc1JvbGUoJ3ByZW1pdW0nKSxcbiAqICAgICB9KSxcbiAqICAgXSxcbiAqIH0pO1xuICogYGBgXG4gKlxuICogQGV4YW1wbGUgQ29uIHRhcmdldGluZyBnbG9iYWwgeSBydXRhcyBleGNsdWlkYXNcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIHByb3ZpZGVWYWx0ZWNoQWRzKHtcbiAqICAgbmV0d29ya0lkOiAnLzEyMzQ1Njc4JyxcbiAqICAgZ2xvYmFsVGFyZ2V0aW5nOiB7XG4gKiAgICAgc2l0ZTogJ215dmFsdGVjaCcsXG4gKiAgICAgc2VjdGlvbjogJ2FwcCcsXG4gKiAgIH0sXG4gKiAgIGV4Y2x1ZGVSb3V0ZXM6IFtcbiAqICAgICAnXi9jaGVja291dCcsXG4gKiAgICAgJ14vcGF5bWVudCcsXG4gKiAgICAgJ14vcHJlbWl1bScsXG4gKiAgIF0sXG4gKiB9KVxuICogYGBgXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBwcm92aWRlVmFsdGVjaEFkcyhjb25maWc6IFZhbHRlY2hBZHNDb25maWcpOiBFbnZpcm9ubWVudFByb3ZpZGVycyB7XG4gIC8vIE1lcmdlIGNvbmZpZyBjb24gZGVmYXVsdHNcbiAgY29uc3QgbWVyZ2VkQ29uZmlnOiBWYWx0ZWNoQWRzQ29uZmlnID0ge1xuICAgIC4uLkRFRkFVTFRfQURTX0NPTkZJRyxcbiAgICAuLi5jb25maWcsXG4gICAgbGF6eUxvYWRDb25maWc6IHtcbiAgICAgIC4uLkRFRkFVTFRfTEFaWV9MT0FEX0NPTkZJRyxcbiAgICAgIC4uLmNvbmZpZy5sYXp5TG9hZENvbmZpZyxcbiAgICB9LFxuICAgIGRlZmF1bHRTbG90Q29uZmlnOiB7XG4gICAgICAuLi5ERUZBVUxUX0FEU19DT05GSUcuZGVmYXVsdFNsb3RDb25maWcsXG4gICAgICAuLi5jb25maWcuZGVmYXVsdFNsb3RDb25maWcsXG4gICAgfSxcbiAgfTtcblxuICByZXR1cm4gbWFrZUVudmlyb25tZW50UHJvdmlkZXJzKFtcbiAgICB7IHByb3ZpZGU6IFZBTFRFQ0hfQURTX0NPTkZJRywgdXNlVmFsdWU6IG1lcmdlZENvbmZpZyB9LFxuICAgIC8vIEluaWNpYWxpemFyIEFkc1NlcnZpY2UgYWwgYXJyYW5jYXIgKHBlcm8gTk8gY2FyZ2FyIHNjcmlwdCBHUFQgYXVuKVxuICAgIHtcbiAgICAgIHByb3ZpZGU6IEFQUF9JTklUSUFMSVpFUixcbiAgICAgIHVzZUZhY3Rvcnk6IGluaXRpYWxpemVBZHMsXG4gICAgICBkZXBzOiBbQWRzU2VydmljZV0sXG4gICAgICBtdWx0aTogdHJ1ZSxcbiAgICB9LFxuICBdKTtcbn1cbiJdfQ==