valtech-components 2.0.506 → 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":"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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.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"]}
@@ -28958,6 +28958,22 @@ class AdSlotComponent {
28958
28958
  }
28959
28959
  return true;
28960
28960
  });
28961
+ /**
28962
+ * Indica si debe mostrar placeholder en vez de ad real.
28963
+ * Activo en localhost o con Publisher ID placeholder.
28964
+ */
28965
+ this.isPlaceholderMode = computed(() => {
28966
+ if (!isPlatformBrowser(this.platformId)) {
28967
+ return false;
28968
+ }
28969
+ const adClient = this.adsService.adClient();
28970
+ const isLocalhost = window.location.hostname === 'localhost' ||
28971
+ window.location.hostname === '127.0.0.1';
28972
+ const isPlaceholderClient = !adClient ||
28973
+ adClient === 'ca-pub-0000000000000000' ||
28974
+ adClient.includes('0000000000');
28975
+ return isLocalhost || isPlaceholderClient;
28976
+ });
28961
28977
  }
28962
28978
  // ===========================================================================
28963
28979
  // LIFECYCLE
@@ -28972,6 +28988,15 @@ class AdSlotComponent {
28972
28988
  if (!this.shouldRender() || this.adInitialized) {
28973
28989
  return;
28974
28990
  }
28991
+ // En modo placeholder, solo marcar como rendered
28992
+ if (this.isPlaceholderMode()) {
28993
+ this._state.set('rendered');
28994
+ this.adInitialized = true;
28995
+ if (this.adsService.isDebugMode()) {
28996
+ console.log(`[ValtechAds] Ad slot '${this.slotId}' in PLACEHOLDER mode (localhost or invalid adClient)`);
28997
+ }
28998
+ return;
28999
+ }
28975
29000
  await this.initializeAd();
28976
29001
  }
28977
29002
  ngOnDestroy() {
@@ -29014,37 +29039,53 @@ class AdSlotComponent {
29014
29039
  [class.val-ad-slot--rendered]="state() === 'rendered'"
29015
29040
  [class.val-ad-slot--empty]="state() === 'empty'"
29016
29041
  [class.val-ad-slot--hidden]="state() === 'hidden'"
29042
+ [class.val-ad-slot--placeholder]="isPlaceholderMode()"
29017
29043
  [class]="cssClass"
29018
29044
  [style.min-height]="minHeight"
29019
29045
  >
29020
- <!-- Skeleton mientras carga -->
29021
- @if (showSkeleton && state() === 'loading') {
29022
- <div
29023
- class="val-ad-slot__skeleton"
29024
- [style.height]="minHeight"
29025
- ></div>
29026
- }
29027
-
29028
- <!-- AdSense ins element -->
29029
- <ins
29030
- class="adsbygoogle val-ad-slot__container"
29031
- [class.val-ad-slot__container--hidden]="state() === 'loading' && showSkeleton"
29032
- [style.display]="'block'"
29033
- [attr.data-ad-client]="adsService.adClient()"
29034
- [attr.data-ad-slot]="adSlot || null"
29035
- [attr.data-ad-format]="format"
29036
- [attr.data-full-width-responsive]="fullWidth ? 'true' : null"
29037
- ></ins>
29046
+ <!-- Placeholder mode (desarrollo local) -->
29047
+ @if (isPlaceholderMode()) {
29048
+ <div class="val-ad-slot__placeholder" [style.height]="minHeight">
29049
+ <div class="val-ad-slot__placeholder-content">
29050
+ <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
29051
+ <rect x="3" y="3" width="18" height="18" rx="2" ry="2"/>
29052
+ <line x1="3" y1="9" x2="21" y2="9"/>
29053
+ <line x1="9" y1="21" x2="9" y2="9"/>
29054
+ </svg>
29055
+ <span class="val-ad-slot__placeholder-label">Ad Placeholder</span>
29056
+ <span class="val-ad-slot__placeholder-info">{{ slotId }} | {{ format }}</span>
29057
+ </div>
29058
+ </div>
29059
+ } @else {
29060
+ <!-- Skeleton mientras carga -->
29061
+ @if (showSkeleton && state() === 'loading') {
29062
+ <div
29063
+ class="val-ad-slot__skeleton"
29064
+ [style.height]="minHeight"
29065
+ ></div>
29066
+ }
29067
+
29068
+ <!-- AdSense ins element -->
29069
+ <ins
29070
+ class="adsbygoogle val-ad-slot__container"
29071
+ [class.val-ad-slot__container--hidden]="state() === 'loading' && showSkeleton"
29072
+ [style.display]="'block'"
29073
+ [attr.data-ad-client]="adsService.adClient()"
29074
+ [attr.data-ad-slot]="adSlot || null"
29075
+ [attr.data-ad-format]="format"
29076
+ [attr.data-full-width-responsive]="fullWidth ? 'true' : null"
29077
+ ></ins>
29078
+ }
29038
29079
 
29039
29080
  <!-- Debug info -->
29040
29081
  @if (adsService.isDebugMode()) {
29041
29082
  <div class="val-ad-slot__debug">
29042
- <small>{{ slotId }} | {{ format }} | {{ state() }}</small>
29083
+ <small>{{ slotId }} | {{ format }} | {{ state() }}{{ isPlaceholderMode() ? ' | PLACEHOLDER' : '' }}</small>
29043
29084
  </div>
29044
29085
  }
29045
29086
  </div>
29046
29087
  }
29047
- `, 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 }); }
29088
+ `, 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 }); }
29048
29089
  }
29049
29090
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AdSlotComponent, decorators: [{
29050
29091
  type: Component,
@@ -29056,37 +29097,53 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
29056
29097
  [class.val-ad-slot--rendered]="state() === 'rendered'"
29057
29098
  [class.val-ad-slot--empty]="state() === 'empty'"
29058
29099
  [class.val-ad-slot--hidden]="state() === 'hidden'"
29100
+ [class.val-ad-slot--placeholder]="isPlaceholderMode()"
29059
29101
  [class]="cssClass"
29060
29102
  [style.min-height]="minHeight"
29061
29103
  >
29062
- <!-- Skeleton mientras carga -->
29063
- @if (showSkeleton && state() === 'loading') {
29064
- <div
29065
- class="val-ad-slot__skeleton"
29066
- [style.height]="minHeight"
29067
- ></div>
29068
- }
29069
-
29070
- <!-- AdSense ins element -->
29071
- <ins
29072
- class="adsbygoogle val-ad-slot__container"
29073
- [class.val-ad-slot__container--hidden]="state() === 'loading' && showSkeleton"
29074
- [style.display]="'block'"
29075
- [attr.data-ad-client]="adsService.adClient()"
29076
- [attr.data-ad-slot]="adSlot || null"
29077
- [attr.data-ad-format]="format"
29078
- [attr.data-full-width-responsive]="fullWidth ? 'true' : null"
29079
- ></ins>
29104
+ <!-- Placeholder mode (desarrollo local) -->
29105
+ @if (isPlaceholderMode()) {
29106
+ <div class="val-ad-slot__placeholder" [style.height]="minHeight">
29107
+ <div class="val-ad-slot__placeholder-content">
29108
+ <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
29109
+ <rect x="3" y="3" width="18" height="18" rx="2" ry="2"/>
29110
+ <line x1="3" y1="9" x2="21" y2="9"/>
29111
+ <line x1="9" y1="21" x2="9" y2="9"/>
29112
+ </svg>
29113
+ <span class="val-ad-slot__placeholder-label">Ad Placeholder</span>
29114
+ <span class="val-ad-slot__placeholder-info">{{ slotId }} | {{ format }}</span>
29115
+ </div>
29116
+ </div>
29117
+ } @else {
29118
+ <!-- Skeleton mientras carga -->
29119
+ @if (showSkeleton && state() === 'loading') {
29120
+ <div
29121
+ class="val-ad-slot__skeleton"
29122
+ [style.height]="minHeight"
29123
+ ></div>
29124
+ }
29125
+
29126
+ <!-- AdSense ins element -->
29127
+ <ins
29128
+ class="adsbygoogle val-ad-slot__container"
29129
+ [class.val-ad-slot__container--hidden]="state() === 'loading' && showSkeleton"
29130
+ [style.display]="'block'"
29131
+ [attr.data-ad-client]="adsService.adClient()"
29132
+ [attr.data-ad-slot]="adSlot || null"
29133
+ [attr.data-ad-format]="format"
29134
+ [attr.data-full-width-responsive]="fullWidth ? 'true' : null"
29135
+ ></ins>
29136
+ }
29080
29137
 
29081
29138
  <!-- Debug info -->
29082
29139
  @if (adsService.isDebugMode()) {
29083
29140
  <div class="val-ad-slot__debug">
29084
- <small>{{ slotId }} | {{ format }} | {{ state() }}</small>
29141
+ <small>{{ slotId }} | {{ format }} | {{ state() }}{{ isPlaceholderMode() ? ' | PLACEHOLDER' : '' }}</small>
29085
29142
  </div>
29086
29143
  }
29087
29144
  </div>
29088
29145
  }
29089
- `, 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"] }]
29146
+ `, 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"] }]
29090
29147
  }], propDecorators: { slotId: [{
29091
29148
  type: Input,
29092
29149
  args: [{ required: true }]