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,152 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Ads Consent Service
|
|
3
|
+
*
|
|
4
|
+
* Integra el servicio de Ads con el Consent Mode v2 existente en AnalyticsService.
|
|
5
|
+
* Proporciona estado de consent especifico para ads (advertising).
|
|
6
|
+
*/
|
|
7
|
+
import { Injectable, Injector, PLATFORM_ID, inject, signal, computed } from '@angular/core';
|
|
8
|
+
import { isPlatformBrowser } from '@angular/common';
|
|
9
|
+
import * as i0 from "@angular/core";
|
|
10
|
+
/**
|
|
11
|
+
* Servicio que maneja el consent para ads.
|
|
12
|
+
*
|
|
13
|
+
* Se integra con AnalyticsService para reutilizar el estado de consent GDPR.
|
|
14
|
+
* Si AnalyticsService no esta disponible, usa valores por defecto conservadores.
|
|
15
|
+
*/
|
|
16
|
+
export class AdsConsentService {
|
|
17
|
+
constructor() {
|
|
18
|
+
this.platformId = inject(PLATFORM_ID);
|
|
19
|
+
this.injector = inject(Injector);
|
|
20
|
+
// Cache del estado de consent
|
|
21
|
+
this._consentState = signal({
|
|
22
|
+
adStorage: false,
|
|
23
|
+
adPersonalization: false,
|
|
24
|
+
adUserData: false,
|
|
25
|
+
});
|
|
26
|
+
/** Estado de consent para ads (readonly) */
|
|
27
|
+
this.adsConsentState = this._consentState.asReadonly();
|
|
28
|
+
/** Indica si se pueden mostrar ads (al menos consent basico) */
|
|
29
|
+
this.canShowAds = computed(() => {
|
|
30
|
+
// Siempre permitimos mostrar ads (pueden ser no personalizados)
|
|
31
|
+
// El control real esta en canPersonalize
|
|
32
|
+
return true;
|
|
33
|
+
});
|
|
34
|
+
/** Indica si se pueden mostrar ads personalizados */
|
|
35
|
+
this.canPersonalize = computed(() => {
|
|
36
|
+
const state = this._consentState();
|
|
37
|
+
return state.adStorage && state.adPersonalization && state.adUserData;
|
|
38
|
+
});
|
|
39
|
+
this.analyticsService = null;
|
|
40
|
+
this.consentSyncInterval = null;
|
|
41
|
+
if (isPlatformBrowser(this.platformId)) {
|
|
42
|
+
this.initializeConsentSync();
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
/**
|
|
46
|
+
* Inicializa la sincronizacion con AnalyticsService.
|
|
47
|
+
*/
|
|
48
|
+
initializeConsentSync() {
|
|
49
|
+
// Intentar obtener AnalyticsService de forma lazy
|
|
50
|
+
// Usamos setTimeout para dar tiempo a que se inicialice
|
|
51
|
+
setTimeout(() => {
|
|
52
|
+
this.syncWithAnalyticsConsent();
|
|
53
|
+
}, 100);
|
|
54
|
+
// Re-sincronizar periodicamente por si el usuario cambia consent
|
|
55
|
+
this.consentSyncInterval = setInterval(() => {
|
|
56
|
+
this.syncWithAnalyticsConsent();
|
|
57
|
+
}, 5000);
|
|
58
|
+
}
|
|
59
|
+
/**
|
|
60
|
+
* Sincroniza el estado de consent desde AnalyticsService.
|
|
61
|
+
*/
|
|
62
|
+
async syncWithAnalyticsConsent() {
|
|
63
|
+
try {
|
|
64
|
+
// Importar AnalyticsService dinamicamente
|
|
65
|
+
const { AnalyticsService } = await import('../firebase/analytics.service');
|
|
66
|
+
if (!this.analyticsService) {
|
|
67
|
+
this.analyticsService = this.injector.get(AnalyticsService, null);
|
|
68
|
+
}
|
|
69
|
+
if (this.analyticsService) {
|
|
70
|
+
// AnalyticsService expone consentState como signal
|
|
71
|
+
const consentState = this.analyticsService.consentState();
|
|
72
|
+
this.updateFromAnalyticsConsent(consentState.settings);
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
catch {
|
|
76
|
+
// AnalyticsService no disponible - usar defaults
|
|
77
|
+
this.setDefaultConsent();
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
/**
|
|
81
|
+
* Actualiza el estado de consent desde AnalyticsService.
|
|
82
|
+
*/
|
|
83
|
+
updateFromAnalyticsConsent(settings) {
|
|
84
|
+
const hasAdvertisingConsent = settings['advertising'] === true;
|
|
85
|
+
this._consentState.set({
|
|
86
|
+
adStorage: hasAdvertisingConsent,
|
|
87
|
+
adPersonalization: hasAdvertisingConsent,
|
|
88
|
+
adUserData: hasAdvertisingConsent,
|
|
89
|
+
});
|
|
90
|
+
}
|
|
91
|
+
/**
|
|
92
|
+
* Establece consent por defecto (sin datos personales).
|
|
93
|
+
*/
|
|
94
|
+
setDefaultConsent() {
|
|
95
|
+
this._consentState.set({
|
|
96
|
+
adStorage: false,
|
|
97
|
+
adPersonalization: false,
|
|
98
|
+
adUserData: false,
|
|
99
|
+
});
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* Actualiza manualmente el consent para ads.
|
|
103
|
+
* Usar cuando el usuario actualiza sus preferencias directamente.
|
|
104
|
+
*
|
|
105
|
+
* @param consent - Nuevo estado de consent parcial
|
|
106
|
+
*/
|
|
107
|
+
updateConsent(consent) {
|
|
108
|
+
this._consentState.update((current) => ({
|
|
109
|
+
...current,
|
|
110
|
+
...consent,
|
|
111
|
+
}));
|
|
112
|
+
// Notificar a GPT si ya esta cargado
|
|
113
|
+
this.applyConsentToGPT();
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* Aplica el estado de consent actual a GPT.
|
|
117
|
+
*/
|
|
118
|
+
applyConsentToGPT() {
|
|
119
|
+
if (!isPlatformBrowser(this.platformId))
|
|
120
|
+
return;
|
|
121
|
+
const googletag = window.googletag;
|
|
122
|
+
if (!googletag)
|
|
123
|
+
return;
|
|
124
|
+
googletag.cmd.push(() => {
|
|
125
|
+
const canPersonalize = this.canPersonalize();
|
|
126
|
+
// 0 = personalized, 1 = non-personalized
|
|
127
|
+
googletag.pubads().setRequestNonPersonalizedAds(canPersonalize ? 0 : 1);
|
|
128
|
+
});
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Obtiene el estado actual de consent.
|
|
132
|
+
*/
|
|
133
|
+
getConsentState() {
|
|
134
|
+
return this._consentState();
|
|
135
|
+
}
|
|
136
|
+
/**
|
|
137
|
+
* Destruye el servicio y limpia recursos.
|
|
138
|
+
*/
|
|
139
|
+
destroy() {
|
|
140
|
+
if (this.consentSyncInterval) {
|
|
141
|
+
clearInterval(this.consentSyncInterval);
|
|
142
|
+
this.consentSyncInterval = null;
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AdsConsentService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
146
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AdsConsentService, providedIn: 'root' }); }
|
|
147
|
+
}
|
|
148
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AdsConsentService, decorators: [{
|
|
149
|
+
type: Injectable,
|
|
150
|
+
args: [{ providedIn: 'root' }]
|
|
151
|
+
}], ctorParameters: () => [] });
|
|
152
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ads-consent.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/ads/ads-consent.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Ads Consent Service\n *\n * Integra el servicio de Ads con el Consent Mode v2 existente en AnalyticsService.\n * Proporciona estado de consent especifico para ads (advertising).\n */\n\nimport { Injectable, Injector, PLATFORM_ID, inject, signal, computed, effect } from '@angular/core';\nimport { isPlatformBrowser } from '@angular/common';\nimport { AdsConsentState } from './types';\n\n/**\n * Servicio que maneja el consent para ads.\n *\n * Se integra con AnalyticsService para reutilizar el estado de consent GDPR.\n * Si AnalyticsService no esta disponible, usa valores por defecto conservadores.\n */\n@Injectable({ providedIn: 'root' })\nexport class AdsConsentService {\n  private readonly platformId = inject(PLATFORM_ID);\n  private readonly injector = inject(Injector);\n\n  // Cache del estado de consent\n  private readonly _consentState = signal<AdsConsentState>({\n    adStorage: false,\n    adPersonalization: false,\n    adUserData: false,\n  });\n\n  /** Estado de consent para ads (readonly) */\n  readonly adsConsentState = this._consentState.asReadonly();\n\n  /** Indica si se pueden mostrar ads (al menos consent basico) */\n  readonly canShowAds = computed(() => {\n    // Siempre permitimos mostrar ads (pueden ser no personalizados)\n    // El control real esta en canPersonalize\n    return true;\n  });\n\n  /** Indica si se pueden mostrar ads personalizados */\n  readonly canPersonalize = computed(() => {\n    const state = this._consentState();\n    return state.adStorage && state.adPersonalization && state.adUserData;\n  });\n\n  private analyticsService: unknown = null;\n  private consentSyncInterval: ReturnType<typeof setInterval> | null = null;\n\n  constructor() {\n    if (isPlatformBrowser(this.platformId)) {\n      this.initializeConsentSync();\n    }\n  }\n\n  /**\n   * Inicializa la sincronizacion con AnalyticsService.\n   */\n  private initializeConsentSync(): void {\n    // Intentar obtener AnalyticsService de forma lazy\n    // Usamos setTimeout para dar tiempo a que se inicialice\n    setTimeout(() => {\n      this.syncWithAnalyticsConsent();\n    }, 100);\n\n    // Re-sincronizar periodicamente por si el usuario cambia consent\n    this.consentSyncInterval = setInterval(() => {\n      this.syncWithAnalyticsConsent();\n    }, 5000);\n  }\n\n  /**\n   * Sincroniza el estado de consent desde AnalyticsService.\n   */\n  private async syncWithAnalyticsConsent(): Promise<void> {\n    try {\n      // Importar AnalyticsService dinamicamente\n      const { AnalyticsService } = await import('../firebase/analytics.service');\n\n      if (!this.analyticsService) {\n        this.analyticsService = this.injector.get(AnalyticsService, null);\n      }\n\n      if (this.analyticsService) {\n        // AnalyticsService expone consentState como signal\n        const consentState = (this.analyticsService as { consentState: () => { settings: Record<string, boolean | undefined> } }).consentState();\n        this.updateFromAnalyticsConsent(consentState.settings);\n      }\n    } catch {\n      // AnalyticsService no disponible - usar defaults\n      this.setDefaultConsent();\n    }\n  }\n\n  /**\n   * Actualiza el estado de consent desde AnalyticsService.\n   */\n  private updateFromAnalyticsConsent(settings: Record<string, boolean | undefined>): void {\n    const hasAdvertisingConsent = settings['advertising'] === true;\n\n    this._consentState.set({\n      adStorage: hasAdvertisingConsent,\n      adPersonalization: hasAdvertisingConsent,\n      adUserData: hasAdvertisingConsent,\n    });\n  }\n\n  /**\n   * Establece consent por defecto (sin datos personales).\n   */\n  private setDefaultConsent(): void {\n    this._consentState.set({\n      adStorage: false,\n      adPersonalization: false,\n      adUserData: false,\n    });\n  }\n\n  /**\n   * Actualiza manualmente el consent para ads.\n   * Usar cuando el usuario actualiza sus preferencias directamente.\n   *\n   * @param consent - Nuevo estado de consent parcial\n   */\n  updateConsent(consent: Partial<AdsConsentState>): void {\n    this._consentState.update((current) => ({\n      ...current,\n      ...consent,\n    }));\n\n    // Notificar a GPT si ya esta cargado\n    this.applyConsentToGPT();\n  }\n\n  /**\n   * Aplica el estado de consent actual a GPT.\n   */\n  applyConsentToGPT(): void {\n    if (!isPlatformBrowser(this.platformId)) return;\n\n    const googletag = window.googletag;\n    if (!googletag) return;\n\n    googletag.cmd.push(() => {\n      const canPersonalize = this.canPersonalize();\n      // 0 = personalized, 1 = non-personalized\n      googletag.pubads().setRequestNonPersonalizedAds(canPersonalize ? 0 : 1);\n    });\n  }\n\n  /**\n   * Obtiene el estado actual de consent.\n   */\n  getConsentState(): AdsConsentState {\n    return this._consentState();\n  }\n\n  /**\n   * Destruye el servicio y limpia recursos.\n   */\n  destroy(): void {\n    if (this.consentSyncInterval) {\n      clearInterval(this.consentSyncInterval);\n      this.consentSyncInterval = null;\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,160 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Ads Loader Service
|
|
3
|
+
*
|
|
4
|
+
* Maneja la carga lazy del script GPT (Google Publisher Tag).
|
|
5
|
+
* Solo carga el script cuando se necesita el primer ad.
|
|
6
|
+
*/
|
|
7
|
+
import { Injectable, Inject, PLATFORM_ID, signal } from '@angular/core';
|
|
8
|
+
import { isPlatformBrowser } from '@angular/common';
|
|
9
|
+
import { VALTECH_ADS_CONFIG } from './config';
|
|
10
|
+
import * as i0 from "@angular/core";
|
|
11
|
+
import * as i1 from "./ads-consent.service";
|
|
12
|
+
/** URL del script GPT */
|
|
13
|
+
const GPT_SCRIPT_URL = 'https://securepubads.g.doubleclick.net/tag/js/gpt.js';
|
|
14
|
+
/**
|
|
15
|
+
* Servicio para cargar el script de Google Publisher Tags.
|
|
16
|
+
*
|
|
17
|
+
* Implementa lazy loading: el script solo se carga cuando
|
|
18
|
+
* se solicita renderizar el primer ad slot.
|
|
19
|
+
*/
|
|
20
|
+
export class AdsLoaderService {
|
|
21
|
+
constructor(config, platformId, consentService) {
|
|
22
|
+
this.config = config;
|
|
23
|
+
this.platformId = platformId;
|
|
24
|
+
this.consentService = consentService;
|
|
25
|
+
this._isLoading = signal(false);
|
|
26
|
+
this._isLoaded = signal(false);
|
|
27
|
+
this._error = signal(null);
|
|
28
|
+
/** Indica si el script esta cargando */
|
|
29
|
+
this.isLoading = this._isLoading.asReadonly();
|
|
30
|
+
/** Indica si el script esta cargado */
|
|
31
|
+
this.isLoaded = this._isLoaded.asReadonly();
|
|
32
|
+
/** Error de carga (si existe) */
|
|
33
|
+
this.error = this._error.asReadonly();
|
|
34
|
+
this.loadPromise = null;
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* Carga el script GPT de forma lazy.
|
|
38
|
+
* Retorna la instancia de googletag o null si falla.
|
|
39
|
+
*
|
|
40
|
+
* @returns Promise con googletag o null
|
|
41
|
+
*/
|
|
42
|
+
loadGPT() {
|
|
43
|
+
// SSR check
|
|
44
|
+
if (!isPlatformBrowser(this.platformId)) {
|
|
45
|
+
return Promise.resolve(null);
|
|
46
|
+
}
|
|
47
|
+
// Ya cargado
|
|
48
|
+
if (this._isLoaded() && window.googletag) {
|
|
49
|
+
return Promise.resolve(window.googletag);
|
|
50
|
+
}
|
|
51
|
+
// Ya hay una carga en progreso
|
|
52
|
+
if (this.loadPromise) {
|
|
53
|
+
return this.loadPromise;
|
|
54
|
+
}
|
|
55
|
+
this._isLoading.set(true);
|
|
56
|
+
this._error.set(null);
|
|
57
|
+
this.loadPromise = new Promise((resolve) => {
|
|
58
|
+
// Inicializar cmd queue
|
|
59
|
+
window.googletag = window.googletag || { cmd: [] };
|
|
60
|
+
const googletag = window.googletag;
|
|
61
|
+
// Crear script
|
|
62
|
+
const script = document.createElement('script');
|
|
63
|
+
script.async = true;
|
|
64
|
+
script.src = GPT_SCRIPT_URL;
|
|
65
|
+
script.onload = () => {
|
|
66
|
+
googletag.cmd.push(() => {
|
|
67
|
+
this.configureGPT(googletag);
|
|
68
|
+
this._isLoaded.set(true);
|
|
69
|
+
this._isLoading.set(false);
|
|
70
|
+
if (this.config.debugMode) {
|
|
71
|
+
console.log('[ValtechAds] Script GPT cargado y configurado');
|
|
72
|
+
}
|
|
73
|
+
resolve(googletag);
|
|
74
|
+
});
|
|
75
|
+
};
|
|
76
|
+
script.onerror = (error) => {
|
|
77
|
+
console.error('[ValtechAds] Error cargando GPT:', error);
|
|
78
|
+
this._error.set(new Error('Error cargando Google Publisher Tag'));
|
|
79
|
+
this._isLoading.set(false);
|
|
80
|
+
this.loadPromise = null;
|
|
81
|
+
resolve(null);
|
|
82
|
+
};
|
|
83
|
+
// Insertar script
|
|
84
|
+
const firstScript = document.getElementsByTagName('script')[0];
|
|
85
|
+
firstScript.parentNode?.insertBefore(script, firstScript);
|
|
86
|
+
});
|
|
87
|
+
return this.loadPromise;
|
|
88
|
+
}
|
|
89
|
+
/**
|
|
90
|
+
* Configura GPT con las opciones de la aplicacion.
|
|
91
|
+
*/
|
|
92
|
+
configureGPT(googletag) {
|
|
93
|
+
const pubads = googletag.pubads();
|
|
94
|
+
// Single Request Architecture para mejor performance
|
|
95
|
+
pubads.enableSingleRequest();
|
|
96
|
+
// Colapsar divs vacios antes de fetch
|
|
97
|
+
pubads.collapseEmptyDivs(true);
|
|
98
|
+
// Lazy loading nativo de GPT
|
|
99
|
+
if (this.config.lazyLoad && this.config.lazyLoadConfig) {
|
|
100
|
+
pubads.enableLazyLoad({
|
|
101
|
+
fetchMarginPercent: this.config.lazyLoadConfig.fetchMarginPercent,
|
|
102
|
+
renderMarginPercent: this.config.lazyLoadConfig.renderMarginPercent,
|
|
103
|
+
mobileScaling: 2.0,
|
|
104
|
+
});
|
|
105
|
+
}
|
|
106
|
+
// Targeting global
|
|
107
|
+
if (this.config.globalTargeting) {
|
|
108
|
+
for (const [key, value] of Object.entries(this.config.globalTargeting)) {
|
|
109
|
+
pubads.setTargeting(key, value);
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
// Non-personalized ads segun consent
|
|
113
|
+
if (!this.consentService.canPersonalize()) {
|
|
114
|
+
pubads.setRequestNonPersonalizedAds(1);
|
|
115
|
+
}
|
|
116
|
+
// Event listeners
|
|
117
|
+
pubads.addEventListener('slotRenderEnded', (event) => {
|
|
118
|
+
const slotId = event.slot.getSlotElementId();
|
|
119
|
+
if (this.config.debugMode) {
|
|
120
|
+
console.log(`[ValtechAds] Slot ${slotId} rendered, empty: ${event.isEmpty}`);
|
|
121
|
+
}
|
|
122
|
+
});
|
|
123
|
+
// Debug mode
|
|
124
|
+
if (this.config.debugMode) {
|
|
125
|
+
console.log('[ValtechAds] GPT configurado:', {
|
|
126
|
+
lazyLoad: this.config.lazyLoad,
|
|
127
|
+
personalized: this.consentService.canPersonalize(),
|
|
128
|
+
targeting: this.config.globalTargeting,
|
|
129
|
+
});
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
/**
|
|
133
|
+
* Verifica si el script GPT esta disponible.
|
|
134
|
+
*/
|
|
135
|
+
isGPTAvailable() {
|
|
136
|
+
return isPlatformBrowser(this.platformId) && !!window.googletag && this._isLoaded();
|
|
137
|
+
}
|
|
138
|
+
/**
|
|
139
|
+
* Obtiene la instancia de googletag si esta cargada.
|
|
140
|
+
*/
|
|
141
|
+
getGPT() {
|
|
142
|
+
if (this.isGPTAvailable()) {
|
|
143
|
+
return window.googletag;
|
|
144
|
+
}
|
|
145
|
+
return null;
|
|
146
|
+
}
|
|
147
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AdsLoaderService, deps: [{ token: VALTECH_ADS_CONFIG }, { token: PLATFORM_ID }, { token: i1.AdsConsentService }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
148
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AdsLoaderService, providedIn: 'root' }); }
|
|
149
|
+
}
|
|
150
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AdsLoaderService, decorators: [{
|
|
151
|
+
type: Injectable,
|
|
152
|
+
args: [{ providedIn: 'root' }]
|
|
153
|
+
}], ctorParameters: () => [{ type: undefined, decorators: [{
|
|
154
|
+
type: Inject,
|
|
155
|
+
args: [VALTECH_ADS_CONFIG]
|
|
156
|
+
}] }, { type: Object, decorators: [{
|
|
157
|
+
type: Inject,
|
|
158
|
+
args: [PLATFORM_ID]
|
|
159
|
+
}] }, { type: i1.AdsConsentService }] });
|
|
160
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ads-loader.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/ads/ads-loader.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Ads Loader Service\n *\n * Maneja la carga lazy del script GPT (Google Publisher Tag).\n * Solo carga el script cuando se necesita el primer ad.\n */\n\nimport { Injectable, Inject, PLATFORM_ID, signal } from '@angular/core';\nimport { isPlatformBrowser } from '@angular/common';\n\nimport { VALTECH_ADS_CONFIG } from './config';\nimport { ValtechAdsConfig, GPTNamespace } from './types';\nimport { AdsConsentService } from './ads-consent.service';\n\n/** URL del script GPT */\nconst GPT_SCRIPT_URL = 'https://securepubads.g.doubleclick.net/tag/js/gpt.js';\n\n/**\n * Servicio para cargar el script de Google Publisher Tags.\n *\n * Implementa lazy loading: el script solo se carga cuando\n * se solicita renderizar el primer ad slot.\n */\n@Injectable({ providedIn: 'root' })\nexport class AdsLoaderService {\n  private readonly _isLoading = signal<boolean>(false);\n  private readonly _isLoaded = signal<boolean>(false);\n  private readonly _error = signal<Error | null>(null);\n\n  /** Indica si el script esta cargando */\n  readonly isLoading = this._isLoading.asReadonly();\n\n  /** Indica si el script esta cargado */\n  readonly isLoaded = this._isLoaded.asReadonly();\n\n  /** Error de carga (si existe) */\n  readonly error = this._error.asReadonly();\n\n  private loadPromise: Promise<GPTNamespace | null> | null = null;\n\n  constructor(\n    @Inject(VALTECH_ADS_CONFIG) private config: ValtechAdsConfig,\n    @Inject(PLATFORM_ID) private platformId: Object,\n    private consentService: AdsConsentService\n  ) {}\n\n  /**\n   * Carga el script GPT de forma lazy.\n   * Retorna la instancia de googletag o null si falla.\n   *\n   * @returns Promise con googletag o null\n   */\n  loadGPT(): Promise<GPTNamespace | null> {\n    // SSR check\n    if (!isPlatformBrowser(this.platformId)) {\n      return Promise.resolve(null);\n    }\n\n    // Ya cargado\n    if (this._isLoaded() && window.googletag) {\n      return Promise.resolve(window.googletag);\n    }\n\n    // Ya hay una carga en progreso\n    if (this.loadPromise) {\n      return this.loadPromise;\n    }\n\n    this._isLoading.set(true);\n    this._error.set(null);\n\n    this.loadPromise = new Promise<GPTNamespace | null>((resolve) => {\n      // Inicializar cmd queue\n      window.googletag = window.googletag || ({ cmd: [] } as GPTNamespace);\n      const googletag = window.googletag;\n\n      // Crear script\n      const script = document.createElement('script');\n      script.async = true;\n      script.src = GPT_SCRIPT_URL;\n\n      script.onload = () => {\n        googletag.cmd.push(() => {\n          this.configureGPT(googletag);\n          this._isLoaded.set(true);\n          this._isLoading.set(false);\n\n          if (this.config.debugMode) {\n            console.log('[ValtechAds] Script GPT cargado y configurado');\n          }\n\n          resolve(googletag);\n        });\n      };\n\n      script.onerror = (error) => {\n        console.error('[ValtechAds] Error cargando GPT:', error);\n        this._error.set(new Error('Error cargando Google Publisher Tag'));\n        this._isLoading.set(false);\n        this.loadPromise = null;\n        resolve(null);\n      };\n\n      // Insertar script\n      const firstScript = document.getElementsByTagName('script')[0];\n      firstScript.parentNode?.insertBefore(script, firstScript);\n    });\n\n    return this.loadPromise;\n  }\n\n  /**\n   * Configura GPT con las opciones de la aplicacion.\n   */\n  private configureGPT(googletag: GPTNamespace): void {\n    const pubads = googletag.pubads();\n\n    // Single Request Architecture para mejor performance\n    pubads.enableSingleRequest();\n\n    // Colapsar divs vacios antes de fetch\n    pubads.collapseEmptyDivs(true);\n\n    // Lazy loading nativo de GPT\n    if (this.config.lazyLoad && this.config.lazyLoadConfig) {\n      pubads.enableLazyLoad({\n        fetchMarginPercent: this.config.lazyLoadConfig.fetchMarginPercent,\n        renderMarginPercent: this.config.lazyLoadConfig.renderMarginPercent,\n        mobileScaling: 2.0,\n      });\n    }\n\n    // Targeting global\n    if (this.config.globalTargeting) {\n      for (const [key, value] of Object.entries(this.config.globalTargeting)) {\n        pubads.setTargeting(key, value);\n      }\n    }\n\n    // Non-personalized ads segun consent\n    if (!this.consentService.canPersonalize()) {\n      pubads.setRequestNonPersonalizedAds(1);\n    }\n\n    // Event listeners\n    pubads.addEventListener('slotRenderEnded', (event) => {\n      const slotId = event.slot.getSlotElementId();\n      if (this.config.debugMode) {\n        console.log(`[ValtechAds] Slot ${slotId} rendered, empty: ${event.isEmpty}`);\n      }\n    });\n\n    // Debug mode\n    if (this.config.debugMode) {\n      console.log('[ValtechAds] GPT configurado:', {\n        lazyLoad: this.config.lazyLoad,\n        personalized: this.consentService.canPersonalize(),\n        targeting: this.config.globalTargeting,\n      });\n    }\n  }\n\n  /**\n   * Verifica si el script GPT esta disponible.\n   */\n  isGPTAvailable(): boolean {\n    return isPlatformBrowser(this.platformId) && !!window.googletag && this._isLoaded();\n  }\n\n  /**\n   * Obtiene la instancia de googletag si esta cargada.\n   */\n  getGPT(): GPTNamespace | null {\n    if (this.isGPTAvailable()) {\n      return window.googletag!;\n    }\n    return null;\n  }\n}\n"]}
|