valtech-components 2.0.505 → 2.0.507

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.
@@ -69,6 +69,22 @@ export class AdSlotComponent {
69
69
  }
70
70
  return true;
71
71
  });
72
+ /**
73
+ * Indica si debe mostrar placeholder en vez de ad real.
74
+ * Activo en localhost o con Publisher ID placeholder.
75
+ */
76
+ this.isPlaceholderMode = computed(() => {
77
+ if (!isPlatformBrowser(this.platformId)) {
78
+ return false;
79
+ }
80
+ const adClient = this.adsService.adClient();
81
+ const isLocalhost = window.location.hostname === 'localhost' ||
82
+ window.location.hostname === '127.0.0.1';
83
+ const isPlaceholderClient = !adClient ||
84
+ adClient === 'ca-pub-0000000000000000' ||
85
+ adClient.includes('0000000000');
86
+ return isLocalhost || isPlaceholderClient;
87
+ });
72
88
  }
73
89
  // ===========================================================================
74
90
  // LIFECYCLE
@@ -83,6 +99,15 @@ export class AdSlotComponent {
83
99
  if (!this.shouldRender() || this.adInitialized) {
84
100
  return;
85
101
  }
102
+ // En modo placeholder, solo marcar como rendered
103
+ if (this.isPlaceholderMode()) {
104
+ this._state.set('rendered');
105
+ this.adInitialized = true;
106
+ if (this.adsService.isDebugMode()) {
107
+ console.log(`[ValtechAds] Ad slot '${this.slotId}' in PLACEHOLDER mode (localhost or invalid adClient)`);
108
+ }
109
+ return;
110
+ }
86
111
  await this.initializeAd();
87
112
  }
88
113
  ngOnDestroy() {
@@ -125,37 +150,53 @@ export class AdSlotComponent {
125
150
  [class.val-ad-slot--rendered]="state() === 'rendered'"
126
151
  [class.val-ad-slot--empty]="state() === 'empty'"
127
152
  [class.val-ad-slot--hidden]="state() === 'hidden'"
153
+ [class.val-ad-slot--placeholder]="isPlaceholderMode()"
128
154
  [class]="cssClass"
129
155
  [style.min-height]="minHeight"
130
156
  >
131
- <!-- Skeleton mientras carga -->
132
- @if (showSkeleton && state() === 'loading') {
133
- <div
134
- class="val-ad-slot__skeleton"
135
- [style.height]="minHeight"
136
- ></div>
137
- }
157
+ <!-- Placeholder mode (desarrollo local) -->
158
+ @if (isPlaceholderMode()) {
159
+ <div class="val-ad-slot__placeholder" [style.height]="minHeight">
160
+ <div class="val-ad-slot__placeholder-content">
161
+ <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
162
+ <rect x="3" y="3" width="18" height="18" rx="2" ry="2"/>
163
+ <line x1="3" y1="9" x2="21" y2="9"/>
164
+ <line x1="9" y1="21" x2="9" y2="9"/>
165
+ </svg>
166
+ <span class="val-ad-slot__placeholder-label">Ad Placeholder</span>
167
+ <span class="val-ad-slot__placeholder-info">{{ slotId }} | {{ format }}</span>
168
+ </div>
169
+ </div>
170
+ } @else {
171
+ <!-- Skeleton mientras carga -->
172
+ @if (showSkeleton && state() === 'loading') {
173
+ <div
174
+ class="val-ad-slot__skeleton"
175
+ [style.height]="minHeight"
176
+ ></div>
177
+ }
138
178
 
139
- <!-- AdSense ins element -->
140
- <ins
141
- class="adsbygoogle val-ad-slot__container"
142
- [class.val-ad-slot__container--hidden]="state() === 'loading' && showSkeleton"
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>
179
+ <!-- AdSense ins element -->
180
+ <ins
181
+ class="adsbygoogle val-ad-slot__container"
182
+ [class.val-ad-slot__container--hidden]="state() === 'loading' && showSkeleton"
183
+ [style.display]="'block'"
184
+ [attr.data-ad-client]="adsService.adClient()"
185
+ [attr.data-ad-slot]="adSlot || null"
186
+ [attr.data-ad-format]="format"
187
+ [attr.data-full-width-responsive]="fullWidth ? 'true' : null"
188
+ ></ins>
189
+ }
149
190
 
150
191
  <!-- Debug info -->
151
192
  @if (adsService.isDebugMode()) {
152
193
  <div class="val-ad-slot__debug">
153
- <small>{{ slotId }} | {{ format }} | {{ state() }}</small>
194
+ <small>{{ slotId }} | {{ format }} | {{ state() }}{{ isPlaceholderMode() ? ' | PLACEHOLDER' : '' }}</small>
154
195
  </div>
155
196
  }
156
197
  </div>
157
198
  }
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 }); }
199
+ `, 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__placeholder{width:100%;background:linear-gradient(135deg,#667eea,#764ba2);border-radius:8px;display:flex;align-items:center;justify-content:center;color:#fff;border:2px dashed rgba(255,255,255,.4)}.val-ad-slot__placeholder-content{display:flex;flex-direction:column;align-items:center;gap:8px;padding:16px;text-align:center}.val-ad-slot__placeholder-content svg{opacity:.8}.val-ad-slot__placeholder-label{font-weight:600;font-size:14px;letter-spacing:.5px}.val-ad-slot__placeholder-info{font-size:11px;opacity:.7;font-family:monospace}.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 }); }
159
200
  }
160
201
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AdSlotComponent, decorators: [{
161
202
  type: Component,
@@ -167,37 +208,53 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
167
208
  [class.val-ad-slot--rendered]="state() === 'rendered'"
168
209
  [class.val-ad-slot--empty]="state() === 'empty'"
169
210
  [class.val-ad-slot--hidden]="state() === 'hidden'"
211
+ [class.val-ad-slot--placeholder]="isPlaceholderMode()"
170
212
  [class]="cssClass"
171
213
  [style.min-height]="minHeight"
172
214
  >
173
- <!-- Skeleton mientras carga -->
174
- @if (showSkeleton && state() === 'loading') {
175
- <div
176
- class="val-ad-slot__skeleton"
177
- [style.height]="minHeight"
178
- ></div>
179
- }
215
+ <!-- Placeholder mode (desarrollo local) -->
216
+ @if (isPlaceholderMode()) {
217
+ <div class="val-ad-slot__placeholder" [style.height]="minHeight">
218
+ <div class="val-ad-slot__placeholder-content">
219
+ <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
220
+ <rect x="3" y="3" width="18" height="18" rx="2" ry="2"/>
221
+ <line x1="3" y1="9" x2="21" y2="9"/>
222
+ <line x1="9" y1="21" x2="9" y2="9"/>
223
+ </svg>
224
+ <span class="val-ad-slot__placeholder-label">Ad Placeholder</span>
225
+ <span class="val-ad-slot__placeholder-info">{{ slotId }} | {{ format }}</span>
226
+ </div>
227
+ </div>
228
+ } @else {
229
+ <!-- Skeleton mientras carga -->
230
+ @if (showSkeleton && state() === 'loading') {
231
+ <div
232
+ class="val-ad-slot__skeleton"
233
+ [style.height]="minHeight"
234
+ ></div>
235
+ }
180
236
 
181
- <!-- AdSense ins element -->
182
- <ins
183
- class="adsbygoogle val-ad-slot__container"
184
- [class.val-ad-slot__container--hidden]="state() === 'loading' && showSkeleton"
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>
237
+ <!-- AdSense ins element -->
238
+ <ins
239
+ class="adsbygoogle val-ad-slot__container"
240
+ [class.val-ad-slot__container--hidden]="state() === 'loading' && showSkeleton"
241
+ [style.display]="'block'"
242
+ [attr.data-ad-client]="adsService.adClient()"
243
+ [attr.data-ad-slot]="adSlot || null"
244
+ [attr.data-ad-format]="format"
245
+ [attr.data-full-width-responsive]="fullWidth ? 'true' : null"
246
+ ></ins>
247
+ }
191
248
 
192
249
  <!-- Debug info -->
193
250
  @if (adsService.isDebugMode()) {
194
251
  <div class="val-ad-slot__debug">
195
- <small>{{ slotId }} | {{ format }} | {{ state() }}</small>
252
+ <small>{{ slotId }} | {{ format }} | {{ state() }}{{ isPlaceholderMode() ? ' | PLACEHOLDER' : '' }}</small>
196
253
  </div>
197
254
  }
198
255
  </div>
199
256
  }
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"] }]
257
+ `, 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__placeholder{width:100%;background:linear-gradient(135deg,#667eea,#764ba2);border-radius:8px;display:flex;align-items:center;justify-content:center;color:#fff;border:2px dashed rgba(255,255,255,.4)}.val-ad-slot__placeholder-content{display:flex;flex-direction:column;align-items:center;gap:8px;padding:16px;text-align:center}.val-ad-slot__placeholder-content svg{opacity:.8}.val-ad-slot__placeholder-label{font-weight:600;font-size:14px;letter-spacing:.5px}.val-ad-slot__placeholder-info{font-size:11px;opacity:.7;font-family:monospace}.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"] }]
201
258
  }], propDecorators: { slotId: [{
202
259
  type: Input,
203
260
  args: [{ required: true }]
@@ -214,4 +271,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
214
271
  }], showSkeleton: [{
215
272
  type: Input
216
273
  }] } });
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"]}
274
+ //# 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;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6BG;AAEH,OAAO,EACL,SAAS,EACT,KAAK,EAGL,MAAM,EACN,MAAM,EACN,QAAQ,EACR,WAAW,EACX,uBAAuB,EACvB,UAAU,GAEX,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,YAAY,EAAE,iBAAiB,EAAE,MAAM,iBAAiB,CAAC;AAClE,OAAO,EAAE,UAAU,EAAE,MAAM,mCAAmC,CAAC;;AAqK/D,MAAM,OAAO,eAAe;IAlK5B;QAmKW,eAAU,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;QACxB,eAAU,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;QACjC,eAAU,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;QAYjD,qBAAqB;QACZ,WAAM,GAAa,MAAM,CAAC;QAEnC,4BAA4B;QACnB,cAAS,GAAG,IAAI,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;QAClC,kBAAa,GAAG,KAAK,CAAC;QAE9B,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;;;WAGG;QACM,sBAAiB,GAAG,QAAQ,CAAC,GAAG,EAAE;YACzC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;gBACxC,OAAO,KAAK,CAAC;YACf,CAAC;YAED,MAAM,QAAQ,GAAG,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC;YAC5C,MAAM,WAAW,GAAG,MAAM,CAAC,QAAQ,CAAC,QAAQ,KAAK,WAAW;gBACxC,MAAM,CAAC,QAAQ,CAAC,QAAQ,KAAK,WAAW,CAAC;YAC7D,MAAM,mBAAmB,GAAG,CAAC,QAAQ;gBACR,QAAQ,KAAK,yBAAyB;gBACtC,QAAQ,CAAC,QAAQ,CAAC,YAAY,CAAC,CAAC;YAE7D,OAAO,WAAW,IAAI,mBAAmB,CAAC;QAC5C,CAAC,CAAC,CAAC;KAkEJ;IAhEC,8EAA8E;IAC9E,YAAY;IACZ,8EAA8E;IAE9E,KAAK,CAAC,QAAQ;QACZ,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC;YACzB,OAAO;QACT,CAAC;QAED,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;IAC7B,CAAC;IAED,KAAK,CAAC,eAAe;QACnB,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YAC/C,OAAO;QACT,CAAC;QAED,iDAAiD;QACjD,IAAI,IAAI,CAAC,iBAAiB,EAAE,EAAE,CAAC;YAC7B,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;YAC5B,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;YAE1B,IAAI,IAAI,CAAC,UAAU,CAAC,WAAW,EAAE,EAAE,CAAC;gBAClC,OAAO,CAAC,GAAG,CAAC,yBAAyB,IAAI,CAAC,MAAM,uDAAuD,CAAC,CAAC;YAC3G,CAAC;YACD,OAAO;QACT,CAAC;QAED,MAAM,IAAI,CAAC,YAAY,EAAE,CAAC;IAC5B,CAAC;IAED,WAAW;QACT,8CAA8C;QAC9C,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;IAC7B,CAAC;IAED,8EAA8E;IAC9E,UAAU;IACV,8EAA8E;IAEtE,KAAK,CAAC,YAAY;QACxB,IAAI,CAAC;YACH,qDAAqD;YACrD,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YAEhE,IAAI,OAAO,EAAE,CAAC;gBACZ,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;gBAC5B,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;gBAE1B,IAAI,IAAI,CAAC,UAAU,CAAC,WAAW,EAAE,EAAE,CAAC;oBAClC,OAAO,CAAC,GAAG,CAAC,yBAAyB,IAAI,CAAC,MAAM,YAAY,EAAE;wBAC5D,MAAM,EAAE,IAAI,CAAC,MAAM;wBACnB,MAAM,EAAE,IAAI,CAAC,MAAM;wBACnB,SAAS,EAAE,IAAI,CAAC,SAAS;qBAC1B,CAAC,CAAC;gBACL,CAAC;YACH,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;YAC3B,CAAC;QACH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,4CAA4C,IAAI,CAAC,MAAM,IAAI,EAAE,KAAK,CAAC,CAAC;YAClF,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;QAC3B,CAAC;IACH,CAAC;+GA5IU,eAAe;mGAAf,eAAe,6OA7JhB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsDT,48CAxDS,YAAY;;4FA+JX,eAAe;kBAlK3B,SAAS;+BACE,aAAa,cACX,IAAI,WACP,CAAC,YAAY,CAAC,mBACN,uBAAuB,CAAC,MAAM,YACrC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsDT;8BAiH0B,MAAM;sBAAhC,KAAK;uBAAC,EAAE,QAAQ,EAAE,IAAI,EAAE;gBAGhB,MAAM;sBAAd,KAAK;gBAGG,MAAM;sBAAd,KAAK;gBAGG,SAAS;sBAAjB,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 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.val-ad-slot--placeholder]=\"isPlaceholderMode()\"\n        [class]=\"cssClass\"\n        [style.min-height]=\"minHeight\"\n      >\n        <!-- Placeholder mode (desarrollo local) -->\n        @if (isPlaceholderMode()) {\n          <div class=\"val-ad-slot__placeholder\" [style.height]=\"minHeight\">\n            <div class=\"val-ad-slot__placeholder-content\">\n              <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\">\n                <rect x=\"3\" y=\"3\" width=\"18\" height=\"18\" rx=\"2\" ry=\"2\"/>\n                <line x1=\"3\" y1=\"9\" x2=\"21\" y2=\"9\"/>\n                <line x1=\"9\" y1=\"21\" x2=\"9\" y2=\"9\"/>\n              </svg>\n              <span class=\"val-ad-slot__placeholder-label\">Ad Placeholder</span>\n              <span class=\"val-ad-slot__placeholder-info\">{{ slotId }} | {{ format }}</span>\n            </div>\n          </div>\n        } @else {\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\n        <!-- Debug info -->\n        @if (adsService.isDebugMode()) {\n          <div class=\"val-ad-slot__debug\">\n            <small>{{ slotId }} | {{ format }} | {{ state() }}{{ isPlaceholderMode() ? ' | PLACEHOLDER' : '' }}</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      /* Placeholder styles */\n      .val-ad-slot__placeholder {\n        width: 100%;\n        background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);\n        border-radius: 8px;\n        display: flex;\n        align-items: center;\n        justify-content: center;\n        color: white;\n        border: 2px dashed rgba(255, 255, 255, 0.4);\n      }\n\n      .val-ad-slot__placeholder-content {\n        display: flex;\n        flex-direction: column;\n        align-items: center;\n        gap: 8px;\n        padding: 16px;\n        text-align: center;\n      }\n\n      .val-ad-slot__placeholder-content svg {\n        opacity: 0.8;\n      }\n\n      .val-ad-slot__placeholder-label {\n        font-weight: 600;\n        font-size: 14px;\n        letter-spacing: 0.5px;\n      }\n\n      .val-ad-slot__placeholder-info {\n        font-size: 11px;\n        opacity: 0.7;\n        font-family: monospace;\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   * Indica si debe mostrar placeholder en vez de ad real.\n   * Activo en localhost o con Publisher ID placeholder.\n   */\n  readonly isPlaceholderMode = computed(() => {\n    if (!isPlatformBrowser(this.platformId)) {\n      return false;\n    }\n\n    const adClient = this.adsService.adClient();\n    const isLocalhost = window.location.hostname === 'localhost' ||\n                        window.location.hostname === '127.0.0.1';\n    const isPlaceholderClient = !adClient ||\n                                 adClient === 'ca-pub-0000000000000000' ||\n                                 adClient.includes('0000000000');\n\n    return isLocalhost || isPlaceholderClient;\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    // En modo placeholder, solo marcar como rendered\n    if (this.isPlaceholderMode()) {\n      this._state.set('rendered');\n      this.adInitialized = true;\n\n      if (this.adsService.isDebugMode()) {\n        console.log(`[ValtechAds] Ad slot '${this.slotId}' in PLACEHOLDER mode (localhost or invalid adClient)`);\n      }\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"]}
@@ -4,4 +4,4 @@
4
4
  * Tipos e interfaces para el servicio de Firebase Analytics (GA4).
5
5
  */
6
6
  export {};
7
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"analytics-types.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/analytics-types.ts"],"names":[],"mappings":"AAAA;;;;GAIG","sourcesContent":["/**\n * Analytics Types\n *\n * Tipos e interfaces para el servicio de Firebase Analytics (GA4).\n */\n\n// ============================================================================\n// CONFIGURACIÓN\n// ============================================================================\n\n/**\n * Configuración de Analytics para ValtechFirebaseConfig\n */\nexport interface AnalyticsConfig {\n  /** Habilitar tracking automático de page views via Router (default: true) */\n  enablePageViewTracking?: boolean;\n\n  /** Habilitar tracking automático de errores via ErrorHandler (default: false) */\n  enableErrorTracking?: boolean;\n\n  /** Habilitar integración con auth para userId y userProperties (default: true) */\n  enableAuthIntegration?: boolean;\n\n  /** Modo debug: no envía a Firebase, logea a consola (default: false en prod) */\n  debugMode?: boolean;\n\n  /** Rutas a excluir del page view tracking (regex patterns) */\n  excludeRoutes?: string[];\n\n  /** Consent mode inicial (GDPR) */\n  defaultConsent?: ConsentSettings;\n\n  /** Clave de localStorage para persistir consent (default: 'analytics_consent') */\n  consentStorageKey?: string;\n\n  /** Propiedades de usuario por defecto */\n  defaultUserProperties?: Record<string, string>;\n\n  /** Prefijo para eventos custom (ej: 'myapp_') */\n  eventPrefix?: string;\n\n  /** Sampling rate (0.0 - 1.0) para reducir volumen (default: 1.0) */\n  samplingRate?: number;\n}\n\n// ============================================================================\n// CONSENT (GDPR)\n// ============================================================================\n\n/**\n * Settings de consentimiento GDPR\n */\nexport interface ConsentSettings {\n  /** Permite recolección de analytics */\n  analytics?: boolean;\n  /** Permite personalización de ads */\n  advertising?: boolean;\n  /** Permite funcionalidad */\n  functionality?: boolean;\n  /** Permite seguridad */\n  security?: boolean;\n}\n\n/**\n * Estado completo de consentimiento\n */\nexport interface ConsentState {\n  /** Settings actuales */\n  settings: ConsentSettings;\n  /** Timestamp de la última actualización */\n  updatedAt: Date | null;\n  /** Si el usuario ha tomado una decisión explícita */\n  hasDecided: boolean;\n}\n\n// ============================================================================\n// EVENTOS\n// ============================================================================\n\n/**\n * Nombres de eventos GA4 recomendados + customs\n */\nexport type AnalyticsEventName =\n  // GA4 Recommended Events - Ecommerce\n  | 'add_payment_info'\n  | 'add_shipping_info'\n  | 'add_to_cart'\n  | 'add_to_wishlist'\n  | 'begin_checkout'\n  | 'purchase'\n  | 'refund'\n  | 'remove_from_cart'\n  | 'view_cart'\n  | 'view_item'\n  | 'view_item_list'\n  // GA4 Recommended Events - Engagement\n  | 'generate_lead'\n  | 'login'\n  | 'page_view'\n  | 'screen_view'\n  | 'search'\n  | 'select_content'\n  | 'select_item'\n  | 'select_promotion'\n  | 'share'\n  | 'sign_up'\n  | 'view_promotion'\n  // Custom Events (app-specific)\n  | 'feature_used'\n  | 'preference_changed'\n  | 'onboarding_step'\n  | 'error_occurred'\n  | 'performance_metric'\n  // Allow custom events\n  | string;\n\n/**\n * Parámetros tipados por evento\n */\nexport interface AnalyticsEventParams {\n  // Ecommerce\n  add_to_cart: {\n    item_id: string;\n    item_name?: string;\n    value?: number;\n    currency?: string;\n    quantity?: number;\n  };\n  purchase: {\n    transaction_id: string;\n    value: number;\n    currency?: string;\n    items?: AnalyticsItem[];\n    tax?: number;\n    shipping?: number;\n  };\n  begin_checkout: {\n    value?: number;\n    currency?: string;\n    items?: AnalyticsItem[];\n  };\n  view_item: {\n    item_id: string;\n    item_name?: string;\n    value?: number;\n    currency?: string;\n  };\n  remove_from_cart: {\n    item_id: string;\n    item_name?: string;\n    value?: number;\n  };\n\n  // Engagement\n  login: { method?: string };\n  sign_up: { method?: string };\n  search: { search_term: string };\n  share: { content_type?: string; item_id?: string; method?: string };\n  select_content: { content_type: string; item_id?: string };\n  page_view: { page_path?: string; page_title?: string; page_location?: string };\n  screen_view: { screen_name: string; screen_class?: string };\n\n  // Custom\n  feature_used: { feature_name: string; feature_category?: string };\n  preference_changed: { preference_name: string; old_value?: string; new_value?: string };\n  onboarding_step: { step_number: number; step_name?: string };\n  error_occurred: {\n    error_type: string;\n    error_message?: string;\n    error_stack?: string;\n    context?: string;\n  };\n  performance_metric: { metric_name: string; value: number; unit?: string };\n\n  // Allow custom params\n  [key: string]: Record<string, unknown> | undefined;\n}\n\n/**\n * Item para eventos de ecommerce\n */\nexport interface AnalyticsItem {\n  /** ID único del item */\n  item_id: string;\n  /** Nombre del item */\n  item_name: string;\n  /** Categoría principal */\n  item_category?: string;\n  /** Subcategorías */\n  item_category2?: string;\n  item_category3?: string;\n  /** Marca */\n  item_brand?: string;\n  /** Variante (ej: color, talla) */\n  item_variant?: string;\n  /** Precio unitario */\n  price?: number;\n  /** Cantidad */\n  quantity?: number;\n  /** Moneda ISO 4217 */\n  currency?: string;\n  /** Posición en lista */\n  index?: number;\n  /** Cupón aplicado */\n  coupon?: string;\n}\n\n// ============================================================================\n// DEBUG\n// ============================================================================\n\n/**\n * Tipo de evento de debug\n */\nexport type DebugEventType =\n  | 'event'\n  | 'page_view'\n  | 'screen_view'\n  | 'user_property'\n  | 'error'\n  | 'consent'\n  | 'timing';\n\n/**\n * Evento de debug (solo en debug mode)\n */\nexport interface AnalyticsDebugEvent {\n  /** Timestamp del evento */\n  timestamp: Date;\n  /** Tipo de evento */\n  type: DebugEventType;\n  /** Nombre del evento */\n  name: string;\n  /** Parámetros del evento */\n  params?: Record<string, unknown>;\n  /** Si fue enviado a Firebase (false en debug mode) */\n  sent: boolean;\n}\n\n// ============================================================================\n// TIMING / PERFORMANCE\n// ============================================================================\n\n/**\n * Métrica de timing/performance\n */\nexport interface TimingMetric {\n  /** Nombre de la métrica */\n  name: string;\n  /** Valor en milisegundos */\n  valueMs: number;\n  /** Categoría (ej: 'api', 'render', 'load') */\n  category?: string;\n  /** Parámetros adicionales */\n  params?: Record<string, string>;\n}\n\n// ============================================================================\n// USER\n// ============================================================================\n\n/**\n * Propiedades de usuario para segmentación\n */\nexport interface UserProperties {\n  /** Idioma preferido */\n  preferred_language?: string;\n  /** Nivel de suscripción */\n  subscription_tier?: string;\n  /** Organización activa (multi-tenant) */\n  active_organization?: string;\n  /** Propiedades custom */\n  [key: string]: string | number | boolean | undefined;\n}\n"]}
7
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"analytics-types.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/analytics-types.ts"],"names":[],"mappings":"AAAA;;;;GAIG","sourcesContent":["/**\n * Analytics Types\n *\n * Tipos e interfaces para el servicio de Firebase Analytics (GA4).\n */\n\n// ============================================================================\n// CONFIGURACIÓN\n// ============================================================================\n\n/**\n * Configuración de Analytics para ValtechFirebaseConfig\n */\nexport interface AnalyticsConfig {\n  /** Habilitar tracking automático de page views via Router (default: true) */\n  enablePageViewTracking?: boolean;\n\n  /** Habilitar tracking automático de errores via ErrorHandler (default: false) */\n  enableErrorTracking?: boolean;\n\n  /** Habilitar integración con auth para userId y userProperties (default: true) */\n  enableAuthIntegration?: boolean;\n\n  /** Modo debug: logea a consola además de enviar a Firebase (default: false) */\n  debugMode?: boolean;\n\n  /** Auto-grant consent para desarrollo/testing (default: false) */\n  defaultConsentGranted?: boolean;\n\n  /** Rutas a excluir del page view tracking (regex patterns) */\n  excludeRoutes?: string[];\n\n  /** Consent mode inicial (GDPR) */\n  defaultConsent?: ConsentSettings;\n\n  /** Clave de localStorage para persistir consent (default: 'analytics_consent') */\n  consentStorageKey?: string;\n\n  /** Propiedades de usuario por defecto */\n  defaultUserProperties?: Record<string, string>;\n\n  /** Prefijo para eventos custom (ej: 'myapp_') */\n  eventPrefix?: string;\n\n  /** Sampling rate (0.0 - 1.0) para reducir volumen (default: 1.0) */\n  samplingRate?: number;\n}\n\n// ============================================================================\n// CONSENT (GDPR)\n// ============================================================================\n\n/**\n * Settings de consentimiento GDPR\n */\nexport interface ConsentSettings {\n  /** Permite recolección de analytics */\n  analytics?: boolean;\n  /** Permite personalización de ads */\n  advertising?: boolean;\n  /** Permite funcionalidad */\n  functionality?: boolean;\n  /** Permite seguridad */\n  security?: boolean;\n}\n\n/**\n * Estado completo de consentimiento\n */\nexport interface ConsentState {\n  /** Settings actuales */\n  settings: ConsentSettings;\n  /** Timestamp de la última actualización */\n  updatedAt: Date | null;\n  /** Si el usuario ha tomado una decisión explícita */\n  hasDecided: boolean;\n}\n\n// ============================================================================\n// EVENTOS\n// ============================================================================\n\n/**\n * Nombres de eventos GA4 recomendados + customs\n */\nexport type AnalyticsEventName =\n  // GA4 Recommended Events - Ecommerce\n  | 'add_payment_info'\n  | 'add_shipping_info'\n  | 'add_to_cart'\n  | 'add_to_wishlist'\n  | 'begin_checkout'\n  | 'purchase'\n  | 'refund'\n  | 'remove_from_cart'\n  | 'view_cart'\n  | 'view_item'\n  | 'view_item_list'\n  // GA4 Recommended Events - Engagement\n  | 'generate_lead'\n  | 'login'\n  | 'page_view'\n  | 'screen_view'\n  | 'search'\n  | 'select_content'\n  | 'select_item'\n  | 'select_promotion'\n  | 'share'\n  | 'sign_up'\n  | 'view_promotion'\n  // Custom Events (app-specific)\n  | 'feature_used'\n  | 'preference_changed'\n  | 'onboarding_step'\n  | 'error_occurred'\n  | 'performance_metric'\n  // Allow custom events\n  | string;\n\n/**\n * Parámetros tipados por evento\n */\nexport interface AnalyticsEventParams {\n  // Ecommerce\n  add_to_cart: {\n    item_id: string;\n    item_name?: string;\n    value?: number;\n    currency?: string;\n    quantity?: number;\n  };\n  purchase: {\n    transaction_id: string;\n    value: number;\n    currency?: string;\n    items?: AnalyticsItem[];\n    tax?: number;\n    shipping?: number;\n  };\n  begin_checkout: {\n    value?: number;\n    currency?: string;\n    items?: AnalyticsItem[];\n  };\n  view_item: {\n    item_id: string;\n    item_name?: string;\n    value?: number;\n    currency?: string;\n  };\n  remove_from_cart: {\n    item_id: string;\n    item_name?: string;\n    value?: number;\n  };\n\n  // Engagement\n  login: { method?: string };\n  sign_up: { method?: string };\n  search: { search_term: string };\n  share: { content_type?: string; item_id?: string; method?: string };\n  select_content: { content_type: string; item_id?: string };\n  page_view: { page_path?: string; page_title?: string; page_location?: string };\n  screen_view: { screen_name: string; screen_class?: string };\n\n  // Custom\n  feature_used: { feature_name: string; feature_category?: string };\n  preference_changed: { preference_name: string; old_value?: string; new_value?: string };\n  onboarding_step: { step_number: number; step_name?: string };\n  error_occurred: {\n    error_type: string;\n    error_message?: string;\n    error_stack?: string;\n    context?: string;\n  };\n  performance_metric: { metric_name: string; value: number; unit?: string };\n\n  // Allow custom params\n  [key: string]: Record<string, unknown> | undefined;\n}\n\n/**\n * Item para eventos de ecommerce\n */\nexport interface AnalyticsItem {\n  /** ID único del item */\n  item_id: string;\n  /** Nombre del item */\n  item_name: string;\n  /** Categoría principal */\n  item_category?: string;\n  /** Subcategorías */\n  item_category2?: string;\n  item_category3?: string;\n  /** Marca */\n  item_brand?: string;\n  /** Variante (ej: color, talla) */\n  item_variant?: string;\n  /** Precio unitario */\n  price?: number;\n  /** Cantidad */\n  quantity?: number;\n  /** Moneda ISO 4217 */\n  currency?: string;\n  /** Posición en lista */\n  index?: number;\n  /** Cupón aplicado */\n  coupon?: string;\n}\n\n// ============================================================================\n// DEBUG\n// ============================================================================\n\n/**\n * Tipo de evento de debug\n */\nexport type DebugEventType =\n  | 'event'\n  | 'page_view'\n  | 'screen_view'\n  | 'user_property'\n  | 'error'\n  | 'consent'\n  | 'timing';\n\n/**\n * Evento de debug (solo en debug mode)\n */\nexport interface AnalyticsDebugEvent {\n  /** Timestamp del evento */\n  timestamp: Date;\n  /** Tipo de evento */\n  type: DebugEventType;\n  /** Nombre del evento */\n  name: string;\n  /** Parámetros del evento */\n  params?: Record<string, unknown>;\n  /** Si fue enviado a Firebase (false en debug mode) */\n  sent: boolean;\n}\n\n// ============================================================================\n// TIMING / PERFORMANCE\n// ============================================================================\n\n/**\n * Métrica de timing/performance\n */\nexport interface TimingMetric {\n  /** Nombre de la métrica */\n  name: string;\n  /** Valor en milisegundos */\n  valueMs: number;\n  /** Categoría (ej: 'api', 'render', 'load') */\n  category?: string;\n  /** Parámetros adicionales */\n  params?: Record<string, string>;\n}\n\n// ============================================================================\n// USER\n// ============================================================================\n\n/**\n * Propiedades de usuario para segmentación\n */\nexport interface UserProperties {\n  /** Idioma preferido */\n  preferred_language?: string;\n  /** Nivel de suscripción */\n  subscription_tier?: string;\n  /** Organización activa (multi-tenant) */\n  active_organization?: string;\n  /** Propiedades custom */\n  [key: string]: string | number | boolean | undefined;\n}\n"]}