valtech-components 2.0.500 → 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.
- package/esm2022/lib/components/atoms/button/button.component.mjs +87 -48
- package/esm2022/lib/components/molecules/action-header/action-header.component.mjs +1 -1
- package/esm2022/lib/components/molecules/ad-slot/ad-slot.component.mjs +249 -0
- package/esm2022/lib/components/molecules/button-group/button-group.component.mjs +1 -1
- package/esm2022/lib/components/molecules/card/card.component.mjs +2 -2
- package/esm2022/lib/components/molecules/file-input/file-input.component.mjs +1 -1
- package/esm2022/lib/components/molecules/raffle-status-card/raffle-status-card.component.mjs +2 -2
- package/esm2022/lib/components/organisms/article/article.component.mjs +2 -2
- package/esm2022/lib/components/organisms/menu/menu.component.mjs +1 -1
- package/esm2022/lib/components/templates/page-template/page-template.component.mjs +1 -1
- package/esm2022/lib/services/ads/ads-consent.service.mjs +152 -0
- package/esm2022/lib/services/ads/ads-loader.service.mjs +160 -0
- package/esm2022/lib/services/ads/ads.service.mjs +449 -0
- package/esm2022/lib/services/ads/config.mjs +118 -0
- package/esm2022/lib/services/ads/index.mjs +14 -0
- package/esm2022/lib/services/ads/types.mjs +23 -0
- package/esm2022/public-api.mjs +6 -1
- package/fesm2022/valtech-components.mjs +1330 -154
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/atoms/button/button.component.d.ts +30 -6
- package/lib/components/molecules/ad-slot/ad-slot.component.d.ts +78 -0
- package/lib/components/organisms/article/article.component.d.ts +3 -3
- package/lib/services/ads/ads-consent.service.d.ts +59 -0
- package/lib/services/ads/ads-loader.service.d.ts +46 -0
- package/lib/services/ads/ads.service.d.ts +123 -0
- package/lib/services/ads/config.d.ts +69 -0
- package/lib/services/ads/index.d.ts +10 -0
- package/lib/services/ads/types.d.ts +163 -0
- package/package.json +1 -1
- 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==
|