valtech-components 2.0.504 → 2.0.506
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/molecules/ad-slot/ad-slot.component.mjs +75 -107
- package/esm2022/lib/services/ads/ads-loader.service.mjs +47 -87
- package/esm2022/lib/services/ads/ads.service.mjs +62 -246
- package/esm2022/lib/services/ads/config.mjs +12 -49
- package/esm2022/lib/services/ads/index.mjs +3 -4
- package/esm2022/lib/services/ads/types.mjs +4 -4
- package/esm2022/lib/services/firebase/analytics-types.mjs +1 -1
- package/esm2022/lib/services/firebase/analytics.service.mjs +33 -10
- package/fesm2022/valtech-components.mjs +247 -667
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/molecules/ad-slot/ad-slot.component.d.ts +25 -39
- package/lib/services/ads/ads-loader.service.d.ts +12 -17
- package/lib/services/ads/ads.service.d.ts +26 -44
- package/lib/services/ads/config.d.ts +9 -25
- package/lib/services/ads/index.d.ts +3 -4
- package/lib/services/ads/types.d.ts +34 -99
- package/lib/services/firebase/analytics-types.d.ts +3 -1
- package/lib/services/firebase/analytics.service.d.ts +1 -1
- package/package.json +1 -1
- package/esm2022/lib/services/ads/ads-consent.service.mjs +0 -152
- package/lib/services/ads/ads-consent.service.d.ts +0 -59
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Ad Slot Component
|
|
3
3
|
*
|
|
4
|
-
* Componente standalone para mostrar ads de Google
|
|
4
|
+
* Componente standalone para mostrar ads de Google AdSense.
|
|
5
5
|
* Se integra automaticamente con el servicio de Ads y respeta consent + premium.
|
|
6
6
|
*
|
|
7
7
|
* @example
|
|
@@ -9,46 +9,38 @@
|
|
|
9
9
|
* <!-- Banner basico -->
|
|
10
10
|
* <val-ad-slot
|
|
11
11
|
* slotId="homepage-top"
|
|
12
|
-
*
|
|
13
|
-
*
|
|
12
|
+
* format="horizontal"
|
|
13
|
+
* [fullWidth]="true"
|
|
14
14
|
* />
|
|
15
15
|
*
|
|
16
|
-
* <!--
|
|
16
|
+
* <!-- Rectangle ad -->
|
|
17
17
|
* <val-ad-slot
|
|
18
18
|
* slotId="sidebar-ad"
|
|
19
|
-
*
|
|
20
|
-
*
|
|
21
|
-
* [sizeMapping]="[
|
|
22
|
-
* { viewportWidth: 1024, sizes: [[300, 600]] },
|
|
23
|
-
* { viewportWidth: 768, sizes: [[300, 250]] },
|
|
24
|
-
* { viewportWidth: 0, sizes: [[320, 50]] }
|
|
25
|
-
* ]"
|
|
19
|
+
* adSlot="1234567890"
|
|
20
|
+
* format="rectangle"
|
|
26
21
|
* />
|
|
27
22
|
*
|
|
28
|
-
* <!--
|
|
23
|
+
* <!-- Auto format (Google decides) -->
|
|
29
24
|
* <val-ad-slot
|
|
30
|
-
* slotId="article-
|
|
31
|
-
*
|
|
32
|
-
*
|
|
33
|
-
* [isNative]="true"
|
|
25
|
+
* slotId="article-ad"
|
|
26
|
+
* format="auto"
|
|
27
|
+
* [fullWidth]="true"
|
|
34
28
|
* />
|
|
35
29
|
* ```
|
|
36
30
|
*/
|
|
37
|
-
import { Component, Input, inject, signal, computed, PLATFORM_ID, ChangeDetectionStrategy, } from '@angular/core';
|
|
31
|
+
import { Component, Input, inject, signal, computed, PLATFORM_ID, ChangeDetectionStrategy, ElementRef, } from '@angular/core';
|
|
38
32
|
import { CommonModule, isPlatformBrowser } from '@angular/common';
|
|
39
33
|
import { AdsService } from '../../../services/ads/ads.service';
|
|
40
|
-
import { AD_SIZE_MAP } from '../../../services/ads/types';
|
|
41
34
|
import * as i0 from "@angular/core";
|
|
42
35
|
export class AdSlotComponent {
|
|
43
36
|
constructor() {
|
|
44
37
|
this.adsService = inject(AdsService);
|
|
45
38
|
this.platformId = inject(PLATFORM_ID);
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
this.isNative = false;
|
|
39
|
+
this.elementRef = inject(ElementRef);
|
|
40
|
+
/** Formato del ad */
|
|
41
|
+
this.format = 'auto';
|
|
42
|
+
/** Full width responsive */
|
|
43
|
+
this.fullWidth = true;
|
|
52
44
|
/** CSS class adicional */
|
|
53
45
|
this.cssClass = '';
|
|
54
46
|
/** Altura minima */
|
|
@@ -60,6 +52,7 @@ export class AdSlotComponent {
|
|
|
60
52
|
// ===========================================================================
|
|
61
53
|
this._state = signal('idle');
|
|
62
54
|
this.state = this._state.asReadonly();
|
|
55
|
+
this.adInitialized = false;
|
|
63
56
|
/** Indica si el componente debe renderizarse */
|
|
64
57
|
this.shouldRender = computed(() => {
|
|
65
58
|
// No renderizar en SSR
|
|
@@ -76,16 +69,6 @@ export class AdSlotComponent {
|
|
|
76
69
|
}
|
|
77
70
|
return true;
|
|
78
71
|
});
|
|
79
|
-
/** Ancho del skeleton */
|
|
80
|
-
this.skeletonWidth = computed(() => {
|
|
81
|
-
const firstSize = this.getFirstSize();
|
|
82
|
-
return firstSize ? `${firstSize[0]}px` : '100%';
|
|
83
|
-
});
|
|
84
|
-
/** Altura del skeleton */
|
|
85
|
-
this.skeletonHeight = computed(() => {
|
|
86
|
-
const firstSize = this.getFirstSize();
|
|
87
|
-
return firstSize ? `${firstSize[1]}px` : this.minHeight;
|
|
88
|
-
});
|
|
89
72
|
}
|
|
90
73
|
// ===========================================================================
|
|
91
74
|
// LIFECYCLE
|
|
@@ -94,59 +77,47 @@ export class AdSlotComponent {
|
|
|
94
77
|
if (!this.shouldRender()) {
|
|
95
78
|
return;
|
|
96
79
|
}
|
|
97
|
-
|
|
80
|
+
this._state.set('loading');
|
|
98
81
|
}
|
|
99
|
-
|
|
100
|
-
if (
|
|
101
|
-
|
|
82
|
+
async ngAfterViewInit() {
|
|
83
|
+
if (!this.shouldRender() || this.adInitialized) {
|
|
84
|
+
return;
|
|
102
85
|
}
|
|
86
|
+
await this.initializeAd();
|
|
87
|
+
}
|
|
88
|
+
ngOnDestroy() {
|
|
89
|
+
// AdSense no requiere cleanup manual como GPT
|
|
90
|
+
this.adInitialized = false;
|
|
103
91
|
}
|
|
104
92
|
// ===========================================================================
|
|
105
93
|
// PRIVATE
|
|
106
94
|
// ===========================================================================
|
|
107
|
-
async
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
const result = await this.adsService.defineSlot(config);
|
|
122
|
-
if (result) {
|
|
123
|
-
this._state.set('rendered');
|
|
124
|
-
}
|
|
125
|
-
else {
|
|
126
|
-
this._state.set(this.adsService.getSlotState(this.slotId));
|
|
127
|
-
}
|
|
128
|
-
}
|
|
129
|
-
getFirstSize() {
|
|
130
|
-
if (typeof this.size === 'string') {
|
|
131
|
-
const mapped = AD_SIZE_MAP[this.size];
|
|
132
|
-
return mapped === 'fluid' ? null : mapped;
|
|
133
|
-
}
|
|
134
|
-
if (Array.isArray(this.size)) {
|
|
135
|
-
if (typeof this.size[0] === 'number') {
|
|
136
|
-
return this.size;
|
|
95
|
+
async initializeAd() {
|
|
96
|
+
try {
|
|
97
|
+
// Registrar slot y cargar AdSense si no esta cargado
|
|
98
|
+
const success = await this.adsService.registerSlot(this.slotId);
|
|
99
|
+
if (success) {
|
|
100
|
+
this._state.set('rendered');
|
|
101
|
+
this.adInitialized = true;
|
|
102
|
+
if (this.adsService.isDebugMode()) {
|
|
103
|
+
console.log(`[ValtechAds] Ad slot '${this.slotId}' rendered`, {
|
|
104
|
+
format: this.format,
|
|
105
|
+
adSlot: this.adSlot,
|
|
106
|
+
fullWidth: this.fullWidth,
|
|
107
|
+
});
|
|
108
|
+
}
|
|
137
109
|
}
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
return first === 'fluid' ? null : first;
|
|
141
|
-
}
|
|
142
|
-
if (Array.isArray(this.size[0])) {
|
|
143
|
-
return this.size[0];
|
|
110
|
+
else {
|
|
111
|
+
this._state.set('empty');
|
|
144
112
|
}
|
|
145
113
|
}
|
|
146
|
-
|
|
114
|
+
catch (error) {
|
|
115
|
+
console.error(`[ValtechAds] Error initializing ad slot '${this.slotId}':`, error);
|
|
116
|
+
this._state.set('error');
|
|
117
|
+
}
|
|
147
118
|
}
|
|
148
119
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AdSlotComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
149
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: AdSlotComponent, isStandalone: true, selector: "val-ad-slot", inputs: { slotId: "slotId",
|
|
120
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: AdSlotComponent, isStandalone: true, selector: "val-ad-slot", inputs: { slotId: "slotId", adSlot: "adSlot", format: "format", fullWidth: "fullWidth", cssClass: "cssClass", minHeight: "minHeight", showSkeleton: "showSkeleton" }, ngImport: i0, template: `
|
|
150
121
|
@if (shouldRender()) {
|
|
151
122
|
<div
|
|
152
123
|
class="val-ad-slot"
|
|
@@ -154,7 +125,6 @@ export class AdSlotComponent {
|
|
|
154
125
|
[class.val-ad-slot--rendered]="state() === 'rendered'"
|
|
155
126
|
[class.val-ad-slot--empty]="state() === 'empty'"
|
|
156
127
|
[class.val-ad-slot--hidden]="state() === 'hidden'"
|
|
157
|
-
[class.val-ad-slot--native]="isNative"
|
|
158
128
|
[class]="cssClass"
|
|
159
129
|
[style.min-height]="minHeight"
|
|
160
130
|
>
|
|
@@ -162,27 +132,30 @@ export class AdSlotComponent {
|
|
|
162
132
|
@if (showSkeleton && state() === 'loading') {
|
|
163
133
|
<div
|
|
164
134
|
class="val-ad-slot__skeleton"
|
|
165
|
-
[style.
|
|
166
|
-
[style.height]="skeletonHeight()"
|
|
135
|
+
[style.height]="minHeight"
|
|
167
136
|
></div>
|
|
168
137
|
}
|
|
169
138
|
|
|
170
|
-
<!--
|
|
171
|
-
<
|
|
172
|
-
|
|
173
|
-
class="val-ad-slot__container"
|
|
139
|
+
<!-- AdSense ins element -->
|
|
140
|
+
<ins
|
|
141
|
+
class="adsbygoogle val-ad-slot__container"
|
|
174
142
|
[class.val-ad-slot__container--hidden]="state() === 'loading' && showSkeleton"
|
|
175
|
-
|
|
143
|
+
[style.display]="'block'"
|
|
144
|
+
[attr.data-ad-client]="adsService.adClient()"
|
|
145
|
+
[attr.data-ad-slot]="adSlot || null"
|
|
146
|
+
[attr.data-ad-format]="format"
|
|
147
|
+
[attr.data-full-width-responsive]="fullWidth ? 'true' : null"
|
|
148
|
+
></ins>
|
|
176
149
|
|
|
177
150
|
<!-- Debug info -->
|
|
178
151
|
@if (adsService.isDebugMode()) {
|
|
179
152
|
<div class="val-ad-slot__debug">
|
|
180
|
-
<small>{{ slotId }} | {{
|
|
153
|
+
<small>{{ slotId }} | {{ format }} | {{ state() }}</small>
|
|
181
154
|
</div>
|
|
182
155
|
}
|
|
183
156
|
</div>
|
|
184
157
|
}
|
|
185
|
-
`, isInline: true, styles: [".val-ad-slot{position:relative;display:flex;justify-content:center;align-items:center;overflow:hidden}.val-ad-slot--loading{background-color:var(--ion-color-light, #f4f4f4)}.val-ad-slot--empty,.val-ad-slot--hidden{display:none!important}.val-ad-
|
|
158
|
+
`, isInline: true, styles: [".val-ad-slot{position:relative;display:flex;justify-content:center;align-items:center;overflow:hidden;width:100%}.val-ad-slot--loading{background-color:var(--ion-color-light, #f4f4f4)}.val-ad-slot--empty,.val-ad-slot--hidden{display:none!important}.val-ad-slot__skeleton{width:100%;background:linear-gradient(90deg,var(--ion-color-light, #f4f4f4) 25%,var(--ion-color-light-shade, #e0e0e0) 50%,var(--ion-color-light, #f4f4f4) 75%);background-size:200% 100%;animation:skeleton-loading 1.5s infinite;border-radius:4px}.val-ad-slot__container{width:100%}.val-ad-slot__container--hidden{visibility:hidden;position:absolute}.val-ad-slot__debug{position:absolute;bottom:0;left:0;background:#000000b3;color:#fff;padding:2px 6px;font-size:10px;z-index:1000;font-family:monospace}@keyframes skeleton-loading{0%{background-position:200% 0}to{background-position:-200% 0}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
186
159
|
}
|
|
187
160
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AdSlotComponent, decorators: [{
|
|
188
161
|
type: Component,
|
|
@@ -194,7 +167,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
194
167
|
[class.val-ad-slot--rendered]="state() === 'rendered'"
|
|
195
168
|
[class.val-ad-slot--empty]="state() === 'empty'"
|
|
196
169
|
[class.val-ad-slot--hidden]="state() === 'hidden'"
|
|
197
|
-
[class.val-ad-slot--native]="isNative"
|
|
198
170
|
[class]="cssClass"
|
|
199
171
|
[style.min-height]="minHeight"
|
|
200
172
|
>
|
|
@@ -202,42 +174,38 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
202
174
|
@if (showSkeleton && state() === 'loading') {
|
|
203
175
|
<div
|
|
204
176
|
class="val-ad-slot__skeleton"
|
|
205
|
-
[style.
|
|
206
|
-
[style.height]="skeletonHeight()"
|
|
177
|
+
[style.height]="minHeight"
|
|
207
178
|
></div>
|
|
208
179
|
}
|
|
209
180
|
|
|
210
|
-
<!--
|
|
211
|
-
<
|
|
212
|
-
|
|
213
|
-
class="val-ad-slot__container"
|
|
181
|
+
<!-- AdSense ins element -->
|
|
182
|
+
<ins
|
|
183
|
+
class="adsbygoogle val-ad-slot__container"
|
|
214
184
|
[class.val-ad-slot__container--hidden]="state() === 'loading' && showSkeleton"
|
|
215
|
-
|
|
185
|
+
[style.display]="'block'"
|
|
186
|
+
[attr.data-ad-client]="adsService.adClient()"
|
|
187
|
+
[attr.data-ad-slot]="adSlot || null"
|
|
188
|
+
[attr.data-ad-format]="format"
|
|
189
|
+
[attr.data-full-width-responsive]="fullWidth ? 'true' : null"
|
|
190
|
+
></ins>
|
|
216
191
|
|
|
217
192
|
<!-- Debug info -->
|
|
218
193
|
@if (adsService.isDebugMode()) {
|
|
219
194
|
<div class="val-ad-slot__debug">
|
|
220
|
-
<small>{{ slotId }} | {{
|
|
195
|
+
<small>{{ slotId }} | {{ format }} | {{ state() }}</small>
|
|
221
196
|
</div>
|
|
222
197
|
}
|
|
223
198
|
</div>
|
|
224
199
|
}
|
|
225
|
-
`, styles: [".val-ad-slot{position:relative;display:flex;justify-content:center;align-items:center;overflow:hidden}.val-ad-slot--loading{background-color:var(--ion-color-light, #f4f4f4)}.val-ad-slot--empty,.val-ad-slot--hidden{display:none!important}.val-ad-
|
|
200
|
+
`, styles: [".val-ad-slot{position:relative;display:flex;justify-content:center;align-items:center;overflow:hidden;width:100%}.val-ad-slot--loading{background-color:var(--ion-color-light, #f4f4f4)}.val-ad-slot--empty,.val-ad-slot--hidden{display:none!important}.val-ad-slot__skeleton{width:100%;background:linear-gradient(90deg,var(--ion-color-light, #f4f4f4) 25%,var(--ion-color-light-shade, #e0e0e0) 50%,var(--ion-color-light, #f4f4f4) 75%);background-size:200% 100%;animation:skeleton-loading 1.5s infinite;border-radius:4px}.val-ad-slot__container{width:100%}.val-ad-slot__container--hidden{visibility:hidden;position:absolute}.val-ad-slot__debug{position:absolute;bottom:0;left:0;background:#000000b3;color:#fff;padding:2px 6px;font-size:10px;z-index:1000;font-family:monospace}@keyframes skeleton-loading{0%{background-position:200% 0}to{background-position:-200% 0}}\n"] }]
|
|
226
201
|
}], propDecorators: { slotId: [{
|
|
227
202
|
type: Input,
|
|
228
203
|
args: [{ required: true }]
|
|
229
|
-
}],
|
|
230
|
-
type: Input,
|
|
231
|
-
args: [{ required: true }]
|
|
232
|
-
}], size: [{
|
|
233
|
-
type: Input
|
|
234
|
-
}], sizeMapping: [{
|
|
235
|
-
type: Input
|
|
236
|
-
}], targeting: [{
|
|
204
|
+
}], adSlot: [{
|
|
237
205
|
type: Input
|
|
238
|
-
}],
|
|
206
|
+
}], format: [{
|
|
239
207
|
type: Input
|
|
240
|
-
}],
|
|
208
|
+
}], fullWidth: [{
|
|
241
209
|
type: Input
|
|
242
210
|
}], cssClass: [{
|
|
243
211
|
type: Input
|
|
@@ -246,4 +214,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
246
214
|
}], showSkeleton: [{
|
|
247
215
|
type: Input
|
|
248
216
|
}] } });
|
|
249
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ad-slot.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/ad-slot/ad-slot.component.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAmCG;AAEH,OAAO,EACL,SAAS,EACT,KAAK,EAGL,MAAM,EACN,MAAM,EACN,QAAQ,EACR,WAAW,EACX,uBAAuB,GACxB,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,YAAY,EAAE,iBAAiB,EAAE,MAAM,iBAAiB,CAAC;AAClE,OAAO,EAAE,UAAU,EAAE,MAAM,mCAAmC,CAAC;AAC/D,OAAO,EAAsD,WAAW,EAAE,MAAM,6BAA6B,CAAC;;AA2G9G,MAAM,OAAO,eAAe;IAzG5B;QA0GW,eAAU,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;QACxB,eAAU,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;QAYlD,oBAAoB;QACX,SAAI,GAAsE,kBAAkB,CAAC;QAQtG,wBAAwB;QACf,kBAAa,GAAG,IAAI,CAAC;QAE9B,mBAAmB;QACV,aAAQ,GAAG,KAAK,CAAC;QAE1B,0BAA0B;QACjB,aAAQ,GAAG,EAAE,CAAC;QAEvB,oBAAoB;QACX,cAAS,GAAG,MAAM,CAAC;QAE5B,uBAAuB;QACd,iBAAY,GAAG,IAAI,CAAC;QAE7B,8EAA8E;QAC9E,SAAS;QACT,8EAA8E;QAE7D,WAAM,GAAG,MAAM,CAAc,MAAM,CAAC,CAAC;QAC7C,UAAK,GAAG,IAAI,CAAC,MAAM,CAAC,UAAU,EAAE,CAAC;QAE1C,gDAAgD;QACvC,iBAAY,GAAG,QAAQ,CAAC,GAAG,EAAE;YACpC,uBAAuB;YACvB,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;gBACxC,OAAO,KAAK,CAAC;YACf,CAAC;YAED,sCAAsC;YACtC,IAAI,IAAI,CAAC,UAAU,CAAC,aAAa,EAAE,EAAE,CAAC;gBACpC,OAAO,KAAK,CAAC;YACf,CAAC;YAED,kEAAkE;YAClE,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,aAAa,EAAE,EAAE,CAAC;gBACrC,OAAO,KAAK,CAAC;YACf,CAAC;YAED,OAAO,IAAI,CAAC;QACd,CAAC,CAAC,CAAC;QAEH,yBAAyB;QAChB,kBAAa,GAAG,QAAQ,CAAC,GAAG,EAAE;YACrC,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;YACtC,OAAO,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC;QAClD,CAAC,CAAC,CAAC;QAEH,0BAA0B;QACjB,mBAAc,GAAG,QAAQ,CAAC,GAAG,EAAE;YACtC,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;YACtC,OAAO,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC;QAC1D,CAAC,CAAC,CAAC;KAsEJ;IApEC,8EAA8E;IAC9E,YAAY;IACZ,8EAA8E;IAE9E,KAAK,CAAC,QAAQ;QACZ,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC;YACzB,OAAO;QACT,CAAC;QAED,MAAM,IAAI,CAAC,cAAc,EAAE,CAAC;IAC9B,CAAC;IAED,WAAW;QACT,IAAI,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;YACvC,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAC3C,CAAC;IACH,CAAC;IAED,8EAA8E;IAC9E,UAAU;IACV,8EAA8E;IAEtE,KAAK,CAAC,cAAc;QAC1B,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAE3B,MAAM,MAAM,GAAiB;YAC3B,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,UAAU,EAAE,IAAI,CAAC,UAAU;YAC3B,IAAI,EAAE,IAAI,CAAC,IAAI;YACf,WAAW,EAAE,IAAI,CAAC,WAAW;YAC7B,SAAS,EAAE,IAAI,CAAC,SAAS;YACzB,aAAa,EAAE,IAAI,CAAC,aAAa;YACjC,QAAQ,EAAE,IAAI,CAAC,QAAQ;YACvB,QAAQ,EAAE,IAAI,CAAC,QAAQ;YACvB,SAAS,EAAE,IAAI,CAAC,SAAS;YACzB,YAAY,EAAE,IAAI,CAAC,YAAY;SAChC,CAAC;QAEF,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;QAExD,IAAI,MAAM,EAAE,CAAC;YACX,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;QAC9B,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,UAAU,CAAC,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC;QAC7D,CAAC;IACH,CAAC;IAEO,YAAY;QAClB,IAAI,OAAO,IAAI,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;YAClC,MAAM,MAAM,GAAG,WAAW,CAAC,IAAI,CAAC,IAAkB,CAAC,CAAC;YACpD,OAAO,MAAM,KAAK,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC;QAC5C,CAAC;QAED,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC;YAC7B,IAAI,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,QAAQ,EAAE,CAAC;gBACrC,OAAO,IAAI,CAAC,IAAwB,CAAC;YACvC,CAAC;YACD,IAAI,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,QAAQ,EAAE,CAAC;gBACrC,MAAM,KAAK,GAAG,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAe,CAAC,CAAC;gBACtD,OAAO,KAAK,KAAK,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC;YAC1C,CAAC;YACD,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;gBAChC,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAqB,CAAC;YAC1C,CAAC;QACH,CAAC;QAED,OAAO,IAAI,CAAC;IACd,CAAC;+GAhJU,eAAe;mGAAf,eAAe,mUApGhB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAoCT,84BAtCS,YAAY;;4FAsGX,eAAe;kBAzG3B,SAAS;+BACE,aAAa,cACX,IAAI,WACP,CAAC,YAAY,CAAC,mBACN,uBAAuB,CAAC,MAAM,YACrC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAoCT;8BAyE0B,MAAM;sBAAhC,KAAK;uBAAC,EAAE,QAAQ,EAAE,IAAI,EAAE;gBAGE,UAAU;sBAApC,KAAK;uBAAC,EAAE,QAAQ,EAAE,IAAI,EAAE;gBAGhB,IAAI;sBAAZ,KAAK;gBAGG,WAAW;sBAAnB,KAAK;gBAGG,SAAS;sBAAjB,KAAK;gBAGG,aAAa;sBAArB,KAAK;gBAGG,QAAQ;sBAAhB,KAAK;gBAGG,QAAQ;sBAAhB,KAAK;gBAGG,SAAS;sBAAjB,KAAK;gBAGG,YAAY;sBAApB,KAAK","sourcesContent":["/**\n * Ad Slot Component\n *\n * Componente standalone para mostrar ads de Google Ad Manager.\n * Se integra automaticamente con el servicio de Ads y respeta consent + premium.\n *\n * @example\n * ```html\n * <!-- Banner basico -->\n * <val-ad-slot\n *   slotId=\"homepage-top\"\n *   adUnitPath=\"/homepage/top\"\n *   size=\"leaderboard\"\n * />\n *\n * <!-- Con responsivo -->\n * <val-ad-slot\n *   slotId=\"sidebar-ad\"\n *   adUnitPath=\"/sidebar\"\n *   [size]=\"['medium-rectangle', 'half-page']\"\n *   [sizeMapping]=\"[\n *     { viewportWidth: 1024, sizes: [[300, 600]] },\n *     { viewportWidth: 768, sizes: [[300, 250]] },\n *     { viewportWidth: 0, sizes: [[320, 50]] }\n *   ]\"\n * />\n *\n * <!-- Native ad -->\n * <val-ad-slot\n *   slotId=\"article-native\"\n *   adUnitPath=\"/article/native\"\n *   size=\"native\"\n *   [isNative]=\"true\"\n * />\n * ```\n */\n\nimport {\n  Component,\n  Input,\n  OnInit,\n  OnDestroy,\n  inject,\n  signal,\n  computed,\n  PLATFORM_ID,\n  ChangeDetectionStrategy,\n} from '@angular/core';\nimport { CommonModule, isPlatformBrowser } from '@angular/common';\nimport { AdsService } from '../../../services/ads/ads.service';\nimport { AdSlotConfig, AdSlotSize, AdSlotState, SizeMapping, AD_SIZE_MAP } from '../../../services/ads/types';\n\n@Component({\n  selector: 'val-ad-slot',\n  standalone: true,\n  imports: [CommonModule],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  template: `\n    @if (shouldRender()) {\n      <div\n        class=\"val-ad-slot\"\n        [class.val-ad-slot--loading]=\"state() === 'loading'\"\n        [class.val-ad-slot--rendered]=\"state() === 'rendered'\"\n        [class.val-ad-slot--empty]=\"state() === 'empty'\"\n        [class.val-ad-slot--hidden]=\"state() === 'hidden'\"\n        [class.val-ad-slot--native]=\"isNative\"\n        [class]=\"cssClass\"\n        [style.min-height]=\"minHeight\"\n      >\n        <!-- Skeleton mientras carga -->\n        @if (showSkeleton && state() === 'loading') {\n          <div\n            class=\"val-ad-slot__skeleton\"\n            [style.width]=\"skeletonWidth()\"\n            [style.height]=\"skeletonHeight()\"\n          ></div>\n        }\n\n        <!-- Container del ad -->\n        <div\n          [id]=\"slotId\"\n          class=\"val-ad-slot__container\"\n          [class.val-ad-slot__container--hidden]=\"state() === 'loading' && showSkeleton\"\n        ></div>\n\n        <!-- Debug info -->\n        @if (adsService.isDebugMode()) {\n          <div class=\"val-ad-slot__debug\">\n            <small>{{ slotId }} | {{ adUnitPath }} | {{ state() }}</small>\n          </div>\n        }\n      </div>\n    }\n  `,\n  styles: [\n    `\n      .val-ad-slot {\n        position: relative;\n        display: flex;\n        justify-content: center;\n        align-items: center;\n        overflow: hidden;\n      }\n\n      .val-ad-slot--loading {\n        background-color: var(--ion-color-light, #f4f4f4);\n      }\n\n      .val-ad-slot--empty,\n      .val-ad-slot--hidden {\n        display: none !important;\n      }\n\n      .val-ad-slot--native {\n        width: 100%;\n      }\n\n      .val-ad-slot__skeleton {\n        background: linear-gradient(\n          90deg,\n          var(--ion-color-light, #f4f4f4) 25%,\n          var(--ion-color-light-shade, #e0e0e0) 50%,\n          var(--ion-color-light, #f4f4f4) 75%\n        );\n        background-size: 200% 100%;\n        animation: skeleton-loading 1.5s infinite;\n        border-radius: 4px;\n      }\n\n      .val-ad-slot__container--hidden {\n        visibility: hidden;\n        position: absolute;\n      }\n\n      .val-ad-slot__debug {\n        position: absolute;\n        bottom: 0;\n        left: 0;\n        background: rgba(0, 0, 0, 0.7);\n        color: white;\n        padding: 2px 6px;\n        font-size: 10px;\n        z-index: 1000;\n        font-family: monospace;\n      }\n\n      @keyframes skeleton-loading {\n        0% {\n          background-position: 200% 0;\n        }\n        100% {\n          background-position: -200% 0;\n        }\n      }\n    `,\n  ],\n})\nexport class AdSlotComponent implements OnInit, OnDestroy {\n  readonly adsService = inject(AdsService);\n  private readonly platformId = inject(PLATFORM_ID);\n\n  // ===========================================================================\n  // INPUTS (Configuracion del slot)\n  // ===========================================================================\n\n  /** ID unico del slot */\n  @Input({ required: true }) slotId!: string;\n\n  /** Ad unit path (sin el network ID) */\n  @Input({ required: true }) adUnitPath!: string;\n\n  /** Tamano del ad */\n  @Input() size: AdSlotSize | AdSlotSize[] | [number, number] | [number, number][] = 'medium-rectangle';\n\n  /** Mapping responsivo */\n  @Input() sizeMapping?: SizeMapping[];\n\n  /** Targeting especifico */\n  @Input() targeting?: Record<string, string | string[]>;\n\n  /** Colapsar si vacio */\n  @Input() collapseEmpty = true;\n\n  /** Es native ad */\n  @Input() isNative = false;\n\n  /** CSS class adicional */\n  @Input() cssClass = '';\n\n  /** Altura minima */\n  @Input() minHeight = '90px';\n\n  /** Mostrar skeleton */\n  @Input() showSkeleton = true;\n\n  // ===========================================================================\n  // ESTADO\n  // ===========================================================================\n\n  private readonly _state = signal<AdSlotState>('idle');\n  readonly state = this._state.asReadonly();\n\n  /** Indica si el componente debe renderizarse */\n  readonly shouldRender = computed(() => {\n    // No renderizar en SSR\n    if (!isPlatformBrowser(this.platformId)) {\n      return false;\n    }\n\n    // No renderizar si usuario es premium\n    if (this.adsService.isPremiumUser()) {\n      return false;\n    }\n\n    // No renderizar si ads estan deshabilitados y aun no inicializado\n    if (!this.adsService.isInitialized()) {\n      return false;\n    }\n\n    return true;\n  });\n\n  /** Ancho del skeleton */\n  readonly skeletonWidth = computed(() => {\n    const firstSize = this.getFirstSize();\n    return firstSize ? `${firstSize[0]}px` : '100%';\n  });\n\n  /** Altura del skeleton */\n  readonly skeletonHeight = computed(() => {\n    const firstSize = this.getFirstSize();\n    return firstSize ? `${firstSize[1]}px` : this.minHeight;\n  });\n\n  // ===========================================================================\n  // LIFECYCLE\n  // ===========================================================================\n\n  async ngOnInit(): Promise<void> {\n    if (!this.shouldRender()) {\n      return;\n    }\n\n    await this.initializeSlot();\n  }\n\n  ngOnDestroy(): void {\n    if (isPlatformBrowser(this.platformId)) {\n      this.adsService.destroySlot(this.slotId);\n    }\n  }\n\n  // ===========================================================================\n  // PRIVATE\n  // ===========================================================================\n\n  private async initializeSlot(): Promise<void> {\n    this._state.set('loading');\n\n    const config: AdSlotConfig = {\n      slotId: this.slotId,\n      adUnitPath: this.adUnitPath,\n      size: this.size,\n      sizeMapping: this.sizeMapping,\n      targeting: this.targeting,\n      collapseEmpty: this.collapseEmpty,\n      isNative: this.isNative,\n      cssClass: this.cssClass,\n      minHeight: this.minHeight,\n      showSkeleton: this.showSkeleton,\n    };\n\n    const result = await this.adsService.defineSlot(config);\n\n    if (result) {\n      this._state.set('rendered');\n    } else {\n      this._state.set(this.adsService.getSlotState(this.slotId));\n    }\n  }\n\n  private getFirstSize(): [number, number] | null {\n    if (typeof this.size === 'string') {\n      const mapped = AD_SIZE_MAP[this.size as AdSlotSize];\n      return mapped === 'fluid' ? null : mapped;\n    }\n\n    if (Array.isArray(this.size)) {\n      if (typeof this.size[0] === 'number') {\n        return this.size as [number, number];\n      }\n      if (typeof this.size[0] === 'string') {\n        const first = AD_SIZE_MAP[this.size[0] as AdSlotSize];\n        return first === 'fluid' ? null : first;\n      }\n      if (Array.isArray(this.size[0])) {\n        return this.size[0] as [number, number];\n      }\n    }\n\n    return null;\n  }\n}\n"]}
|
|
217
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ad-slot.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/ad-slot/ad-slot.component.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Ad Slot Component\n *\n * Componente standalone para mostrar ads de Google AdSense.\n * Se integra automaticamente con el servicio de Ads y respeta consent + premium.\n *\n * @example\n * ```html\n * <!-- Banner basico -->\n * <val-ad-slot\n *   slotId=\"homepage-top\"\n *   format=\"horizontal\"\n *   [fullWidth]=\"true\"\n * />\n *\n * <!-- Rectangle ad -->\n * <val-ad-slot\n *   slotId=\"sidebar-ad\"\n *   adSlot=\"1234567890\"\n *   format=\"rectangle\"\n * />\n *\n * <!-- Auto format (Google decides) -->\n * <val-ad-slot\n *   slotId=\"article-ad\"\n *   format=\"auto\"\n *   [fullWidth]=\"true\"\n * />\n * ```\n */\n\nimport {\n  Component,\n  Input,\n  OnInit,\n  OnDestroy,\n  inject,\n  signal,\n  computed,\n  PLATFORM_ID,\n  ChangeDetectionStrategy,\n  ElementRef,\n  AfterViewInit,\n} from '@angular/core';\nimport { CommonModule, isPlatformBrowser } from '@angular/common';\nimport { AdsService } from '../../../services/ads/ads.service';\nimport { AdFormat, AdSlotState } from '../../../services/ads/types';\n\n@Component({\n  selector: 'val-ad-slot',\n  standalone: true,\n  imports: [CommonModule],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  template: `\n    @if (shouldRender()) {\n      <div\n        class=\"val-ad-slot\"\n        [class.val-ad-slot--loading]=\"state() === 'loading'\"\n        [class.val-ad-slot--rendered]=\"state() === 'rendered'\"\n        [class.val-ad-slot--empty]=\"state() === 'empty'\"\n        [class.val-ad-slot--hidden]=\"state() === 'hidden'\"\n        [class]=\"cssClass\"\n        [style.min-height]=\"minHeight\"\n      >\n        <!-- Skeleton mientras carga -->\n        @if (showSkeleton && state() === 'loading') {\n          <div\n            class=\"val-ad-slot__skeleton\"\n            [style.height]=\"minHeight\"\n          ></div>\n        }\n\n        <!-- AdSense ins element -->\n        <ins\n          class=\"adsbygoogle val-ad-slot__container\"\n          [class.val-ad-slot__container--hidden]=\"state() === 'loading' && showSkeleton\"\n          [style.display]=\"'block'\"\n          [attr.data-ad-client]=\"adsService.adClient()\"\n          [attr.data-ad-slot]=\"adSlot || null\"\n          [attr.data-ad-format]=\"format\"\n          [attr.data-full-width-responsive]=\"fullWidth ? 'true' : null\"\n        ></ins>\n\n        <!-- Debug info -->\n        @if (adsService.isDebugMode()) {\n          <div class=\"val-ad-slot__debug\">\n            <small>{{ slotId }} | {{ format }} | {{ state() }}</small>\n          </div>\n        }\n      </div>\n    }\n  `,\n  styles: [\n    `\n      .val-ad-slot {\n        position: relative;\n        display: flex;\n        justify-content: center;\n        align-items: center;\n        overflow: hidden;\n        width: 100%;\n      }\n\n      .val-ad-slot--loading {\n        background-color: var(--ion-color-light, #f4f4f4);\n      }\n\n      .val-ad-slot--empty,\n      .val-ad-slot--hidden {\n        display: none !important;\n      }\n\n      .val-ad-slot__skeleton {\n        width: 100%;\n        background: linear-gradient(\n          90deg,\n          var(--ion-color-light, #f4f4f4) 25%,\n          var(--ion-color-light-shade, #e0e0e0) 50%,\n          var(--ion-color-light, #f4f4f4) 75%\n        );\n        background-size: 200% 100%;\n        animation: skeleton-loading 1.5s infinite;\n        border-radius: 4px;\n      }\n\n      .val-ad-slot__container {\n        width: 100%;\n      }\n\n      .val-ad-slot__container--hidden {\n        visibility: hidden;\n        position: absolute;\n      }\n\n      .val-ad-slot__debug {\n        position: absolute;\n        bottom: 0;\n        left: 0;\n        background: rgba(0, 0, 0, 0.7);\n        color: white;\n        padding: 2px 6px;\n        font-size: 10px;\n        z-index: 1000;\n        font-family: monospace;\n      }\n\n      @keyframes skeleton-loading {\n        0% {\n          background-position: 200% 0;\n        }\n        100% {\n          background-position: -200% 0;\n        }\n      }\n    `,\n  ],\n})\nexport class AdSlotComponent implements OnInit, OnDestroy, AfterViewInit {\n  readonly adsService = inject(AdsService);\n  private readonly platformId = inject(PLATFORM_ID);\n  private readonly elementRef = inject(ElementRef);\n\n  // ===========================================================================\n  // INPUTS (Configuracion del slot)\n  // ===========================================================================\n\n  /** ID unico del slot (para tracking interno) */\n  @Input({ required: true }) slotId!: string;\n\n  /** Ad Slot ID de AdSense (opcional, para unidades manuales) */\n  @Input() adSlot?: string;\n\n  /** Formato del ad */\n  @Input() format: AdFormat = 'auto';\n\n  /** Full width responsive */\n  @Input() fullWidth = true;\n\n  /** CSS class adicional */\n  @Input() cssClass = '';\n\n  /** Altura minima */\n  @Input() minHeight = '90px';\n\n  /** Mostrar skeleton */\n  @Input() showSkeleton = true;\n\n  // ===========================================================================\n  // ESTADO\n  // ===========================================================================\n\n  private readonly _state = signal<AdSlotState>('idle');\n  readonly state = this._state.asReadonly();\n  private adInitialized = false;\n\n  /** Indica si el componente debe renderizarse */\n  readonly shouldRender = computed(() => {\n    // No renderizar en SSR\n    if (!isPlatformBrowser(this.platformId)) {\n      return false;\n    }\n\n    // No renderizar si usuario es premium\n    if (this.adsService.isPremiumUser()) {\n      return false;\n    }\n\n    // No renderizar si ads estan deshabilitados y aun no inicializado\n    if (!this.adsService.isInitialized()) {\n      return false;\n    }\n\n    return true;\n  });\n\n  // ===========================================================================\n  // LIFECYCLE\n  // ===========================================================================\n\n  async ngOnInit(): Promise<void> {\n    if (!this.shouldRender()) {\n      return;\n    }\n\n    this._state.set('loading');\n  }\n\n  async ngAfterViewInit(): Promise<void> {\n    if (!this.shouldRender() || this.adInitialized) {\n      return;\n    }\n\n    await this.initializeAd();\n  }\n\n  ngOnDestroy(): void {\n    // AdSense no requiere cleanup manual como GPT\n    this.adInitialized = false;\n  }\n\n  // ===========================================================================\n  // PRIVATE\n  // ===========================================================================\n\n  private async initializeAd(): Promise<void> {\n    try {\n      // Registrar slot y cargar AdSense si no esta cargado\n      const success = await this.adsService.registerSlot(this.slotId);\n\n      if (success) {\n        this._state.set('rendered');\n        this.adInitialized = true;\n\n        if (this.adsService.isDebugMode()) {\n          console.log(`[ValtechAds] Ad slot '${this.slotId}' rendered`, {\n            format: this.format,\n            adSlot: this.adSlot,\n            fullWidth: this.fullWidth,\n          });\n        }\n      } else {\n        this._state.set('empty');\n      }\n    } catch (error) {\n      console.error(`[ValtechAds] Error initializing ad slot '${this.slotId}':`, error);\n      this._state.set('error');\n    }\n  }\n}\n"]}
|
|
@@ -1,27 +1,25 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Ads Loader Service
|
|
3
3
|
*
|
|
4
|
-
* Maneja la carga lazy del script
|
|
4
|
+
* Maneja la carga lazy del script de Google AdSense.
|
|
5
5
|
* Solo carga el script cuando se necesita el primer ad.
|
|
6
6
|
*/
|
|
7
7
|
import { Injectable, Inject, PLATFORM_ID, signal } from '@angular/core';
|
|
8
8
|
import { isPlatformBrowser } from '@angular/common';
|
|
9
9
|
import { VALTECH_ADS_CONFIG } from './config';
|
|
10
10
|
import * as i0 from "@angular/core";
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
const GPT_SCRIPT_URL = 'https://securepubads.g.doubleclick.net/tag/js/gpt.js';
|
|
11
|
+
/** URL del script AdSense */
|
|
12
|
+
const ADSENSE_SCRIPT_URL = 'https://pagead2.googlesyndication.com/pagead/js/adsbygoogle.js';
|
|
14
13
|
/**
|
|
15
|
-
* Servicio para cargar el script de Google
|
|
14
|
+
* Servicio para cargar el script de Google AdSense.
|
|
16
15
|
*
|
|
17
16
|
* Implementa lazy loading: el script solo se carga cuando
|
|
18
17
|
* se solicita renderizar el primer ad slot.
|
|
19
18
|
*/
|
|
20
19
|
export class AdsLoaderService {
|
|
21
|
-
constructor(config, platformId
|
|
20
|
+
constructor(config, platformId) {
|
|
22
21
|
this.config = config;
|
|
23
22
|
this.platformId = platformId;
|
|
24
|
-
this.consentService = consentService;
|
|
25
23
|
this._isLoading = signal(false);
|
|
26
24
|
this._isLoaded = signal(false);
|
|
27
25
|
this._error = signal(null);
|
|
@@ -34,19 +32,19 @@ export class AdsLoaderService {
|
|
|
34
32
|
this.loadPromise = null;
|
|
35
33
|
}
|
|
36
34
|
/**
|
|
37
|
-
* Carga el script
|
|
38
|
-
* Retorna
|
|
35
|
+
* Carga el script AdSense de forma lazy.
|
|
36
|
+
* Retorna true si se cargo correctamente, false si fallo.
|
|
39
37
|
*
|
|
40
|
-
* @returns Promise
|
|
38
|
+
* @returns Promise<boolean>
|
|
41
39
|
*/
|
|
42
|
-
|
|
40
|
+
loadAdSense() {
|
|
43
41
|
// SSR check
|
|
44
42
|
if (!isPlatformBrowser(this.platformId)) {
|
|
45
|
-
return Promise.resolve(
|
|
43
|
+
return Promise.resolve(false);
|
|
46
44
|
}
|
|
47
45
|
// Ya cargado
|
|
48
|
-
if (this._isLoaded()
|
|
49
|
-
return Promise.resolve(
|
|
46
|
+
if (this._isLoaded()) {
|
|
47
|
+
return Promise.resolve(true);
|
|
50
48
|
}
|
|
51
49
|
// Ya hay una carga en progreso
|
|
52
50
|
if (this.loadPromise) {
|
|
@@ -55,96 +53,58 @@ export class AdsLoaderService {
|
|
|
55
53
|
this._isLoading.set(true);
|
|
56
54
|
this._error.set(null);
|
|
57
55
|
this.loadPromise = new Promise((resolve) => {
|
|
58
|
-
// Inicializar
|
|
59
|
-
window.
|
|
60
|
-
const googletag = window.googletag;
|
|
56
|
+
// Inicializar adsbygoogle array
|
|
57
|
+
window.adsbygoogle = window.adsbygoogle || [];
|
|
61
58
|
// Crear script
|
|
62
59
|
const script = document.createElement('script');
|
|
63
60
|
script.async = true;
|
|
64
|
-
script.src =
|
|
61
|
+
script.src = ADSENSE_SCRIPT_URL;
|
|
62
|
+
script.crossOrigin = 'anonymous';
|
|
63
|
+
// Agregar data-ad-client (Publisher ID)
|
|
64
|
+
script.setAttribute('data-ad-client', this.config.adClient);
|
|
65
|
+
// Test mode para desarrollo
|
|
66
|
+
if (this.config.testMode) {
|
|
67
|
+
script.setAttribute('data-adtest', 'on');
|
|
68
|
+
}
|
|
65
69
|
script.onload = () => {
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
|
|
70
|
+
this._isLoaded.set(true);
|
|
71
|
+
this._isLoading.set(false);
|
|
72
|
+
if (this.config.debugMode) {
|
|
73
|
+
console.log('[ValtechAds] Script AdSense cargado', {
|
|
74
|
+
adClient: this.config.adClient,
|
|
75
|
+
testMode: this.config.testMode,
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
resolve(true);
|
|
75
79
|
};
|
|
76
80
|
script.onerror = (error) => {
|
|
77
|
-
console.error('[ValtechAds] Error cargando
|
|
78
|
-
this._error.set(new Error('Error cargando Google
|
|
81
|
+
console.error('[ValtechAds] Error cargando AdSense:', error);
|
|
82
|
+
this._error.set(new Error('Error cargando Google AdSense'));
|
|
79
83
|
this._isLoading.set(false);
|
|
80
84
|
this.loadPromise = null;
|
|
81
|
-
resolve(
|
|
85
|
+
resolve(false);
|
|
82
86
|
};
|
|
83
|
-
// Insertar script
|
|
84
|
-
|
|
85
|
-
firstScript.parentNode?.insertBefore(script, firstScript);
|
|
87
|
+
// Insertar script en head
|
|
88
|
+
document.head.appendChild(script);
|
|
86
89
|
});
|
|
87
90
|
return this.loadPromise;
|
|
88
91
|
}
|
|
89
92
|
/**
|
|
90
|
-
*
|
|
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.
|
|
93
|
+
* Verifica si el script AdSense esta disponible.
|
|
134
94
|
*/
|
|
135
|
-
|
|
136
|
-
return isPlatformBrowser(this.platformId) && !!window.
|
|
95
|
+
isAdSenseAvailable() {
|
|
96
|
+
return isPlatformBrowser(this.platformId) && !!window.adsbygoogle && this._isLoaded();
|
|
137
97
|
}
|
|
138
98
|
/**
|
|
139
|
-
*
|
|
99
|
+
* Ejecuta un comando AdSense (push a la queue).
|
|
100
|
+
* Usado internamente para activar ad slots.
|
|
140
101
|
*/
|
|
141
|
-
|
|
142
|
-
if (this.
|
|
143
|
-
|
|
102
|
+
pushAdCommand() {
|
|
103
|
+
if (this.isAdSenseAvailable() && window.adsbygoogle) {
|
|
104
|
+
window.adsbygoogle.push({});
|
|
144
105
|
}
|
|
145
|
-
return null;
|
|
146
106
|
}
|
|
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 }
|
|
107
|
+
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 }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
148
108
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AdsLoaderService, providedIn: 'root' }); }
|
|
149
109
|
}
|
|
150
110
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AdsLoaderService, decorators: [{
|
|
@@ -156,5 +116,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
156
116
|
}] }, { type: Object, decorators: [{
|
|
157
117
|
type: Inject,
|
|
158
118
|
args: [PLATFORM_ID]
|
|
159
|
-
}] }
|
|
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"]}
|
|
119
|
+
}] }] });
|
|
120
|
+
//# 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 de Google AdSense.\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 } from './types';\n\n/** URL del script AdSense */\nconst ADSENSE_SCRIPT_URL = 'https://pagead2.googlesyndication.com/pagead/js/adsbygoogle.js';\n\n/**\n * Servicio para cargar el script de Google AdSense.\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<boolean> | null = null;\n\n  constructor(\n    @Inject(VALTECH_ADS_CONFIG) private config: ValtechAdsConfig,\n    @Inject(PLATFORM_ID) private platformId: Object\n  ) {}\n\n  /**\n   * Carga el script AdSense de forma lazy.\n   * Retorna true si se cargo correctamente, false si fallo.\n   *\n   * @returns Promise<boolean>\n   */\n  loadAdSense(): Promise<boolean> {\n    // SSR check\n    if (!isPlatformBrowser(this.platformId)) {\n      return Promise.resolve(false);\n    }\n\n    // Ya cargado\n    if (this._isLoaded()) {\n      return Promise.resolve(true);\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<boolean>((resolve) => {\n      // Inicializar adsbygoogle array\n      window.adsbygoogle = window.adsbygoogle || [];\n\n      // Crear script\n      const script = document.createElement('script');\n      script.async = true;\n      script.src = ADSENSE_SCRIPT_URL;\n      script.crossOrigin = 'anonymous';\n\n      // Agregar data-ad-client (Publisher ID)\n      script.setAttribute('data-ad-client', this.config.adClient);\n\n      // Test mode para desarrollo\n      if (this.config.testMode) {\n        script.setAttribute('data-adtest', 'on');\n      }\n\n      script.onload = () => {\n        this._isLoaded.set(true);\n        this._isLoading.set(false);\n\n        if (this.config.debugMode) {\n          console.log('[ValtechAds] Script AdSense cargado', {\n            adClient: this.config.adClient,\n            testMode: this.config.testMode,\n          });\n        }\n\n        resolve(true);\n      };\n\n      script.onerror = (error) => {\n        console.error('[ValtechAds] Error cargando AdSense:', error);\n        this._error.set(new Error('Error cargando Google AdSense'));\n        this._isLoading.set(false);\n        this.loadPromise = null;\n        resolve(false);\n      };\n\n      // Insertar script en head\n      document.head.appendChild(script);\n    });\n\n    return this.loadPromise;\n  }\n\n  /**\n   * Verifica si el script AdSense esta disponible.\n   */\n  isAdSenseAvailable(): boolean {\n    return isPlatformBrowser(this.platformId) && !!window.adsbygoogle && this._isLoaded();\n  }\n\n  /**\n   * Ejecuta un comando AdSense (push a la queue).\n   * Usado internamente para activar ad slots.\n   */\n  pushAdCommand(): void {\n    if (this.isAdSenseAvailable() && window.adsbygoogle) {\n      window.adsbygoogle.push({});\n    }\n  }\n}\n"]}
|