valtech-components 2.0.501 → 2.0.503

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.
@@ -1,7 +1,8 @@
1
1
  import { CommonModule } from '@angular/common';
2
2
  import { Component, EventEmitter, inject, Input, Output } from '@angular/core';
3
+ import { IonButton, IonIcon, IonPopover, IonList, IonItem, IonLabel } from '@ionic/angular/standalone';
3
4
  import { addIcons } from 'ionicons';
4
- import { language } from 'ionicons/icons';
5
+ import { language, globeOutline, checkmark } from 'ionicons/icons';
5
6
  import { I18nService } from '../../../services/i18n';
6
7
  import { PopoverSelectorComponent } from '../popover-selector/popover-selector.component';
7
8
  import * as i0 from "@angular/core";
@@ -12,11 +13,16 @@ import * as i0 from "@angular/core";
12
13
  * By default, language changes are reactive (no page reload).
13
14
  * Set forceReload: true in props to reload on change.
14
15
  *
15
- * @example
16
+ * @example Default mode (dropdown with label):
16
17
  * <val-language-selector
17
18
  * [props]="{ showLabel: true, showFlags: true }">
18
19
  * </val-language-selector>
19
20
  *
21
+ * @example Icon mode (compact for headers/toolbars):
22
+ * <val-language-selector
23
+ * [props]="{ mode: 'icon', color: 'primary' }">
24
+ * </val-language-selector>
25
+ *
20
26
  * @input props: LanguageSelectorMetadata - Configuration for the language selector
21
27
  * @output languageChange: EventEmitter<string> - Emitted when language changes
22
28
  */
@@ -31,6 +37,12 @@ export class LanguageSelectorComponent {
31
37
  * Emits the selected language code.
32
38
  */
33
39
  this.languageChange = new EventEmitter();
40
+ /** Unique ID for the icon mode popover trigger */
41
+ this.popoverId = `lang-selector-${Math.random().toString(36).substring(2, 9)}`;
42
+ /** Available languages (exposed for icon mode template) */
43
+ this.availableLanguages = [];
44
+ /** Current language (exposed for icon mode template) */
45
+ this.currentLanguage = '';
34
46
  this.i18n = inject(I18nService);
35
47
  /** Default language display names */
36
48
  this.defaultLanguageNames = {
@@ -48,9 +60,11 @@ export class LanguageSelectorComponent {
48
60
  fr: '🇫🇷',
49
61
  de: '🇩🇪',
50
62
  };
51
- addIcons({ language });
63
+ addIcons({ language, globeOutline, checkmark });
52
64
  }
53
65
  ngOnInit() {
66
+ this.currentLanguage = this.i18n.lang();
67
+ this.availableLanguages = this.props.availableLanguages || this.i18n.supportedLanguages();
54
68
  this.initializePopoverProps();
55
69
  }
56
70
  initializePopoverProps() {
@@ -81,6 +95,7 @@ export class LanguageSelectorComponent {
81
95
  okText: 'Aceptar',
82
96
  };
83
97
  }
98
+ /** Get display name for a language code (public for template access) */
84
99
  getLanguageDisplayName(languageCode) {
85
100
  // Use custom names if provided
86
101
  if (this.props.customLanguageNames?.[languageCode]) {
@@ -100,29 +115,105 @@ export class LanguageSelectorComponent {
100
115
  onLanguageChange(selectedLanguage) {
101
116
  if (typeof selectedLanguage === 'string') {
102
117
  const newLang = selectedLanguage;
118
+ // Update current language for icon mode
119
+ this.currentLanguage = newLang;
103
120
  // Emit the change event
104
121
  this.languageChange.emit(selectedLanguage);
105
122
  // Set the new language (reactive by default, reload if forceReload is true)
106
123
  this.i18n.setLanguage(newLang, this.props.forceReload);
107
- // Update popover selected value for reactive UI
124
+ // Update popover selected value for reactive UI (default mode)
108
125
  if (this.popoverProps) {
109
126
  this.popoverProps = { ...this.popoverProps, selectedValue: newLang };
110
127
  }
111
128
  }
112
129
  }
113
130
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: LanguageSelectorComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
114
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: LanguageSelectorComponent, isStandalone: true, selector: "val-language-selector", inputs: { props: "props" }, outputs: { languageChange: "languageChange" }, ngImport: i0, template: `
115
- <val-popover-selector [props]="popoverProps" (selectionChange)="onLanguageChange($event)"> </val-popover-selector>
116
- `, isInline: true, styles: [":host{display:inline-block;width:auto}val-popover-selector .popover-selector-container{display:inline-block;width:auto}val-popover-selector .selector-trigger .trigger-text{display:inline-flex;align-items:center;gap:8px;font-weight:700}val-popover-selector .selector-trigger .trigger-text .flag-emoji{font-size:1.2em;line-height:1;filter:drop-shadow(0 1px 2px rgba(0,0,0,.1));transition:transform .2s ease}val-popover-selector .selector-trigger.has-flag .trigger-text{letter-spacing:.025em}val-popover-selector .selector-trigger:hover:not([disabled]) .trigger-text .flag-emoji{transform:scale(1.1)}val-popover-selector .option-content{gap:10px;padding:8px 0}val-popover-selector .option-content .flag-emoji{font-size:1.1em;filter:drop-shadow(0 1px 2px rgba(0,0,0,.08))}val-popover-selector .option-content span{font-weight:600;letter-spacing:.01em}.language-flag{font-size:1.2em;margin-right:6px;vertical-align:middle;line-height:1;filter:drop-shadow(0 1px 3px rgba(0,0,0,.1));transition:all .2s ease}@media (max-width: 768px){val-popover-selector .selector-trigger .trigger-text{font-size:13px;gap:6px}val-popover-selector .selector-trigger .trigger-text .flag-emoji{font-size:1.1em}}val-popover-selector .selector-trigger:hover:not([disabled]) .trigger-text .flag-emoji{transform:scale(1.1) rotate(3deg);transition:transform .25s cubic-bezier(.4,0,.2,1)}val-popover-selector .selector-trigger:active .trigger-text .flag-emoji{transform:scale(1.05)}val-popover-selector .language-changing .flag-emoji{animation:languageSwitch .4s ease-in-out}@keyframes languageSwitch{0%{transform:scale(1)}50%{transform:scale(1.15) rotate(8deg)}to{transform:scale(1)}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: PopoverSelectorComponent, selector: "val-popover-selector", inputs: ["props"], outputs: ["selectionChange"] }] }); }
131
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: LanguageSelectorComponent, isStandalone: true, selector: "val-language-selector", inputs: { props: "props" }, outputs: { languageChange: "languageChange" }, ngImport: i0, template: `
132
+ <!-- Default mode: use popover-selector -->
133
+ @if (props.mode !== 'icon') {
134
+ <val-popover-selector [props]="popoverProps" (selectionChange)="onLanguageChange($event)"> </val-popover-selector>
135
+ }
136
+
137
+ <!-- Icon mode: compact globe button with popover -->
138
+ @if (props.mode === 'icon') {
139
+ <ion-button
140
+ [id]="popoverId"
141
+ [color]="props.color || 'medium'"
142
+ [fill]="props.fill || 'clear'"
143
+ [size]="props.size || 'default'"
144
+ [disabled]="props.disabled"
145
+ class="icon-mode-button"
146
+ >
147
+ <ion-icon slot="icon-only" [name]="props.icon || 'globe-outline'"></ion-icon>
148
+ </ion-button>
149
+
150
+ <ion-popover [trigger]="popoverId" [dismissOnSelect]="true">
151
+ <ng-template>
152
+ <ion-list>
153
+ @for (lang of availableLanguages; track lang) {
154
+ <ion-item
155
+ [button]="true"
156
+ [detail]="false"
157
+ (click)="onLanguageChange(lang)"
158
+ [class.selected]="lang === currentLanguage"
159
+ >
160
+ <ion-label>{{ getLanguageDisplayName(lang) }}</ion-label>
161
+ @if (lang === currentLanguage) {
162
+ <ion-icon slot="end" name="checkmark" color="primary"></ion-icon>
163
+ }
164
+ </ion-item>
165
+ }
166
+ </ion-list>
167
+ </ng-template>
168
+ </ion-popover>
169
+ }
170
+ `, isInline: true, styles: [":host{display:inline-block;width:auto}val-popover-selector .popover-selector-container{display:inline-block;width:auto}val-popover-selector .selector-trigger .trigger-text{display:inline-flex;align-items:center;gap:8px;font-weight:700}val-popover-selector .selector-trigger .trigger-text .flag-emoji{font-size:1.2em;line-height:1;filter:drop-shadow(0 1px 2px rgba(0,0,0,.1));transition:transform .2s ease}val-popover-selector .selector-trigger.has-flag .trigger-text{letter-spacing:.025em}val-popover-selector .selector-trigger:hover:not([disabled]) .trigger-text .flag-emoji{transform:scale(1.1)}val-popover-selector .option-content{gap:10px;padding:8px 0}val-popover-selector .option-content .flag-emoji{font-size:1.1em;filter:drop-shadow(0 1px 2px rgba(0,0,0,.08))}val-popover-selector .option-content span{font-weight:600;letter-spacing:.01em}.language-flag{font-size:1.2em;margin-right:6px;vertical-align:middle;line-height:1;filter:drop-shadow(0 1px 3px rgba(0,0,0,.1));transition:all .2s ease}@media (max-width: 768px){val-popover-selector .selector-trigger .trigger-text{font-size:13px;gap:6px}val-popover-selector .selector-trigger .trigger-text .flag-emoji{font-size:1.1em}}val-popover-selector .selector-trigger:hover:not([disabled]) .trigger-text .flag-emoji{transform:scale(1.1) rotate(3deg);transition:transform .25s cubic-bezier(.4,0,.2,1)}val-popover-selector .selector-trigger:active .trigger-text .flag-emoji{transform:scale(1.05)}val-popover-selector .language-changing .flag-emoji{animation:languageSwitch .4s ease-in-out}@keyframes languageSwitch{0%{transform:scale(1)}50%{transform:scale(1.15) rotate(8deg)}to{transform:scale(1)}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: PopoverSelectorComponent, selector: "val-popover-selector", inputs: ["props"], outputs: ["selectionChange"] }, { kind: "component", type: IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: IonPopover, selector: "ion-popover" }, { kind: "component", type: IonList, selector: "ion-list", inputs: ["inset", "lines", "mode"] }, { kind: "component", type: IonItem, selector: "ion-item", inputs: ["button", "color", "detail", "detailIcon", "disabled", "download", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }] }); }
117
171
  }
118
172
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: LanguageSelectorComponent, decorators: [{
119
173
  type: Component,
120
- args: [{ selector: 'val-language-selector', standalone: true, imports: [CommonModule, PopoverSelectorComponent], template: `
121
- <val-popover-selector [props]="popoverProps" (selectionChange)="onLanguageChange($event)"> </val-popover-selector>
174
+ args: [{ selector: 'val-language-selector', standalone: true, imports: [CommonModule, PopoverSelectorComponent, IonButton, IonIcon, IonPopover, IonList, IonItem, IonLabel], template: `
175
+ <!-- Default mode: use popover-selector -->
176
+ @if (props.mode !== 'icon') {
177
+ <val-popover-selector [props]="popoverProps" (selectionChange)="onLanguageChange($event)"> </val-popover-selector>
178
+ }
179
+
180
+ <!-- Icon mode: compact globe button with popover -->
181
+ @if (props.mode === 'icon') {
182
+ <ion-button
183
+ [id]="popoverId"
184
+ [color]="props.color || 'medium'"
185
+ [fill]="props.fill || 'clear'"
186
+ [size]="props.size || 'default'"
187
+ [disabled]="props.disabled"
188
+ class="icon-mode-button"
189
+ >
190
+ <ion-icon slot="icon-only" [name]="props.icon || 'globe-outline'"></ion-icon>
191
+ </ion-button>
192
+
193
+ <ion-popover [trigger]="popoverId" [dismissOnSelect]="true">
194
+ <ng-template>
195
+ <ion-list>
196
+ @for (lang of availableLanguages; track lang) {
197
+ <ion-item
198
+ [button]="true"
199
+ [detail]="false"
200
+ (click)="onLanguageChange(lang)"
201
+ [class.selected]="lang === currentLanguage"
202
+ >
203
+ <ion-label>{{ getLanguageDisplayName(lang) }}</ion-label>
204
+ @if (lang === currentLanguage) {
205
+ <ion-icon slot="end" name="checkmark" color="primary"></ion-icon>
206
+ }
207
+ </ion-item>
208
+ }
209
+ </ion-list>
210
+ </ng-template>
211
+ </ion-popover>
212
+ }
122
213
  `, styles: [":host{display:inline-block;width:auto}val-popover-selector .popover-selector-container{display:inline-block;width:auto}val-popover-selector .selector-trigger .trigger-text{display:inline-flex;align-items:center;gap:8px;font-weight:700}val-popover-selector .selector-trigger .trigger-text .flag-emoji{font-size:1.2em;line-height:1;filter:drop-shadow(0 1px 2px rgba(0,0,0,.1));transition:transform .2s ease}val-popover-selector .selector-trigger.has-flag .trigger-text{letter-spacing:.025em}val-popover-selector .selector-trigger:hover:not([disabled]) .trigger-text .flag-emoji{transform:scale(1.1)}val-popover-selector .option-content{gap:10px;padding:8px 0}val-popover-selector .option-content .flag-emoji{font-size:1.1em;filter:drop-shadow(0 1px 2px rgba(0,0,0,.08))}val-popover-selector .option-content span{font-weight:600;letter-spacing:.01em}.language-flag{font-size:1.2em;margin-right:6px;vertical-align:middle;line-height:1;filter:drop-shadow(0 1px 3px rgba(0,0,0,.1));transition:all .2s ease}@media (max-width: 768px){val-popover-selector .selector-trigger .trigger-text{font-size:13px;gap:6px}val-popover-selector .selector-trigger .trigger-text .flag-emoji{font-size:1.1em}}val-popover-selector .selector-trigger:hover:not([disabled]) .trigger-text .flag-emoji{transform:scale(1.1) rotate(3deg);transition:transform .25s cubic-bezier(.4,0,.2,1)}val-popover-selector .selector-trigger:active .trigger-text .flag-emoji{transform:scale(1.05)}val-popover-selector .language-changing .flag-emoji{animation:languageSwitch .4s ease-in-out}@keyframes languageSwitch{0%{transform:scale(1)}50%{transform:scale(1.15) rotate(8deg)}to{transform:scale(1)}}\n"] }]
123
214
  }], ctorParameters: () => [], propDecorators: { props: [{
124
215
  type: Input
125
216
  }], languageChange: [{
126
217
  type: Output
127
218
  }] } });
128
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"language-selector.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/language-selector/language-selector.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, inject, Input, OnInit, Output } from '@angular/core';\nimport { addIcons } from 'ionicons';\nimport { language } from 'ionicons/icons';\nimport { I18nService, I18nLang } from '../../../services/i18n';\nimport { PopoverSelectorComponent } from '../popover-selector/popover-selector.component';\nimport { PopoverOption, PopoverSelectorMetadata } from '../popover-selector/types';\nimport { LanguageSelectorMetadata } from './types';\n\n@Component({\n  selector: 'val-language-selector',\n  standalone: true,\n  imports: [CommonModule, PopoverSelectorComponent],\n  template: `\n    <val-popover-selector [props]=\"popoverProps\" (selectionChange)=\"onLanguageChange($event)\"> </val-popover-selector>\n  `,\n  styleUrls: ['./language-selector.component.scss'],\n})\n/**\n * val-language-selector\n *\n * A language selector component that uses I18nService.\n * By default, language changes are reactive (no page reload).\n * Set forceReload: true in props to reload on change.\n *\n * @example\n * <val-language-selector\n *   [props]=\"{ showLabel: true, showFlags: true }\">\n * </val-language-selector>\n *\n * @input props: LanguageSelectorMetadata - Configuration for the language selector\n * @output languageChange: EventEmitter<string> - Emitted when language changes\n */\nexport class LanguageSelectorComponent implements OnInit {\n  /**\n   * Language selector configuration object.\n   */\n  @Input()\n  props: LanguageSelectorMetadata = {};\n\n  /**\n   * Event emitted when the language selection changes.\n   * Emits the selected language code.\n   */\n  @Output()\n  languageChange = new EventEmitter<string>();\n\n  /** Popover selector configuration */\n  popoverProps: PopoverSelectorMetadata;\n\n  private i18n = inject(I18nService);\n\n  /** Default language display names */\n  private readonly defaultLanguageNames: Record<string, string> = {\n    es: 'Español',\n    en: 'English',\n    pt: 'Português',\n    fr: 'Français',\n    de: 'Deutsch',\n  };\n\n  /** Default flag icons for languages */\n  private readonly defaultLanguageFlags: Record<string, string> = {\n    es: '🇪🇸',\n    en: '🇺🇸',\n    pt: '🇧🇷',\n    fr: '🇫🇷',\n    de: '🇩🇪',\n  };\n\n  constructor() {\n    addIcons({ language });\n  }\n\n  ngOnInit() {\n    this.initializePopoverProps();\n  }\n\n  private initializePopoverProps() {\n    const currentLanguage = this.i18n.lang();\n    const availableLanguages = this.props.availableLanguages || this.i18n.supportedLanguages();\n\n    // Convert language codes to popover options\n    const options: PopoverOption[] = availableLanguages.map(lang => ({\n      value: lang,\n      label: this.getLanguageDisplayName(lang),\n    }));\n\n    // Create popover configuration\n    this.popoverProps = {\n      options,\n      selectedValue: currentLanguage,\n      label: this.props.showLabel !== false ? (this.props.label || 'Idioma') : undefined,\n      icon: 'language',\n      placeholder: 'Seleccionar idioma...',\n      color: this.props.color || 'medium',\n      size: this.props.size || 'default',\n      fill: this.props.fill || 'outline',\n      shape: this.props.shape,\n      expand: this.props.expand,\n      disabled: this.props.disabled || false,\n      interface: 'popover',\n      showCheckmark: true,\n      multiple: false,\n      cancelText: 'Cancelar',\n      okText: 'Aceptar',\n    };\n  }\n\n  private getLanguageDisplayName(languageCode: string): string {\n    // Use custom names if provided\n    if (this.props.customLanguageNames?.[languageCode]) {\n      const customName = this.props.customLanguageNames[languageCode];\n      const flag = this.props.showFlags ? this.defaultLanguageFlags[languageCode] : '';\n      return flag ? `${flag} ${customName}` : customName;\n    }\n\n    // Use default names\n    const defaultName = this.defaultLanguageNames[languageCode] || languageCode.toUpperCase();\n    const flag = this.props.showFlags ? this.defaultLanguageFlags[languageCode] : '';\n    return flag ? `${flag} ${defaultName}` : defaultName;\n  }\n\n  /**\n   * Handle language selection change.\n   * @param selectedLanguage - The selected language code(s)\n   */\n  onLanguageChange(selectedLanguage: string | string[]) {\n    if (typeof selectedLanguage === 'string') {\n      const newLang = selectedLanguage as I18nLang;\n\n      // Emit the change event\n      this.languageChange.emit(selectedLanguage);\n\n      // Set the new language (reactive by default, reload if forceReload is true)\n      this.i18n.setLanguage(newLang, this.props.forceReload);\n\n      // Update popover selected value for reactive UI\n      if (this.popoverProps) {\n        this.popoverProps = { ...this.popoverProps, selectedValue: newLang };\n      }\n    }\n  }\n}\n"]}
219
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"language-selector.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/language-selector/language-selector.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, inject, Input, OnInit, Output } from '@angular/core';\nimport { IonButton, IonIcon, IonPopover, IonList, IonItem, IonLabel } from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport { language, globeOutline, checkmark } from 'ionicons/icons';\nimport { I18nService, I18nLang } from '../../../services/i18n';\nimport { PopoverSelectorComponent } from '../popover-selector/popover-selector.component';\nimport { PopoverOption, PopoverSelectorMetadata } from '../popover-selector/types';\nimport { LanguageSelectorMetadata } from './types';\n\n@Component({\n  selector: 'val-language-selector',\n  standalone: true,\n  imports: [CommonModule, PopoverSelectorComponent, IonButton, IonIcon, IonPopover, IonList, IonItem, IonLabel],\n  template: `\n    <!-- Default mode: use popover-selector -->\n    @if (props.mode !== 'icon') {\n      <val-popover-selector [props]=\"popoverProps\" (selectionChange)=\"onLanguageChange($event)\"> </val-popover-selector>\n    }\n\n    <!-- Icon mode: compact globe button with popover -->\n    @if (props.mode === 'icon') {\n      <ion-button\n        [id]=\"popoverId\"\n        [color]=\"props.color || 'medium'\"\n        [fill]=\"props.fill || 'clear'\"\n        [size]=\"props.size || 'default'\"\n        [disabled]=\"props.disabled\"\n        class=\"icon-mode-button\"\n      >\n        <ion-icon slot=\"icon-only\" [name]=\"props.icon || 'globe-outline'\"></ion-icon>\n      </ion-button>\n\n      <ion-popover [trigger]=\"popoverId\" [dismissOnSelect]=\"true\">\n        <ng-template>\n          <ion-list>\n            @for (lang of availableLanguages; track lang) {\n              <ion-item\n                [button]=\"true\"\n                [detail]=\"false\"\n                (click)=\"onLanguageChange(lang)\"\n                [class.selected]=\"lang === currentLanguage\"\n              >\n                <ion-label>{{ getLanguageDisplayName(lang) }}</ion-label>\n                @if (lang === currentLanguage) {\n                  <ion-icon slot=\"end\" name=\"checkmark\" color=\"primary\"></ion-icon>\n                }\n              </ion-item>\n            }\n          </ion-list>\n        </ng-template>\n      </ion-popover>\n    }\n  `,\n  styleUrls: ['./language-selector.component.scss'],\n})\n/**\n * val-language-selector\n *\n * A language selector component that uses I18nService.\n * By default, language changes are reactive (no page reload).\n * Set forceReload: true in props to reload on change.\n *\n * @example Default mode (dropdown with label):\n * <val-language-selector\n *   [props]=\"{ showLabel: true, showFlags: true }\">\n * </val-language-selector>\n *\n * @example Icon mode (compact for headers/toolbars):\n * <val-language-selector\n *   [props]=\"{ mode: 'icon', color: 'primary' }\">\n * </val-language-selector>\n *\n * @input props: LanguageSelectorMetadata - Configuration for the language selector\n * @output languageChange: EventEmitter<string> - Emitted when language changes\n */\nexport class LanguageSelectorComponent implements OnInit {\n  /**\n   * Language selector configuration object.\n   */\n  @Input()\n  props: LanguageSelectorMetadata = {};\n\n  /**\n   * Event emitted when the language selection changes.\n   * Emits the selected language code.\n   */\n  @Output()\n  languageChange = new EventEmitter<string>();\n\n  /** Popover selector configuration (for default mode) */\n  popoverProps: PopoverSelectorMetadata;\n\n  /** Unique ID for the icon mode popover trigger */\n  popoverId = `lang-selector-${Math.random().toString(36).substring(2, 9)}`;\n\n  /** Available languages (exposed for icon mode template) */\n  availableLanguages: string[] = [];\n\n  /** Current language (exposed for icon mode template) */\n  currentLanguage: string = '';\n\n  private i18n = inject(I18nService);\n\n  /** Default language display names */\n  readonly defaultLanguageNames: Record<string, string> = {\n    es: 'Español',\n    en: 'English',\n    pt: 'Português',\n    fr: 'Français',\n    de: 'Deutsch',\n  };\n\n  /** Default flag icons for languages */\n  readonly defaultLanguageFlags: Record<string, string> = {\n    es: '🇪🇸',\n    en: '🇺🇸',\n    pt: '🇧🇷',\n    fr: '🇫🇷',\n    de: '🇩🇪',\n  };\n\n  constructor() {\n    addIcons({ language, globeOutline, checkmark });\n  }\n\n  ngOnInit() {\n    this.currentLanguage = this.i18n.lang();\n    this.availableLanguages = this.props.availableLanguages || this.i18n.supportedLanguages();\n    this.initializePopoverProps();\n  }\n\n  private initializePopoverProps() {\n    const currentLanguage = this.i18n.lang();\n    const availableLanguages = this.props.availableLanguages || this.i18n.supportedLanguages();\n\n    // Convert language codes to popover options\n    const options: PopoverOption[] = availableLanguages.map(lang => ({\n      value: lang,\n      label: this.getLanguageDisplayName(lang),\n    }));\n\n    // Create popover configuration\n    this.popoverProps = {\n      options,\n      selectedValue: currentLanguage,\n      label: this.props.showLabel !== false ? (this.props.label || 'Idioma') : undefined,\n      icon: 'language',\n      placeholder: 'Seleccionar idioma...',\n      color: this.props.color || 'medium',\n      size: this.props.size || 'default',\n      fill: this.props.fill || 'outline',\n      shape: this.props.shape,\n      expand: this.props.expand,\n      disabled: this.props.disabled || false,\n      interface: 'popover',\n      showCheckmark: true,\n      multiple: false,\n      cancelText: 'Cancelar',\n      okText: 'Aceptar',\n    };\n  }\n\n  /** Get display name for a language code (public for template access) */\n  getLanguageDisplayName(languageCode: string): string {\n    // Use custom names if provided\n    if (this.props.customLanguageNames?.[languageCode]) {\n      const customName = this.props.customLanguageNames[languageCode];\n      const flag = this.props.showFlags ? this.defaultLanguageFlags[languageCode] : '';\n      return flag ? `${flag} ${customName}` : customName;\n    }\n\n    // Use default names\n    const defaultName = this.defaultLanguageNames[languageCode] || languageCode.toUpperCase();\n    const flag = this.props.showFlags ? this.defaultLanguageFlags[languageCode] : '';\n    return flag ? `${flag} ${defaultName}` : defaultName;\n  }\n\n  /**\n   * Handle language selection change.\n   * @param selectedLanguage - The selected language code(s)\n   */\n  onLanguageChange(selectedLanguage: string | string[]) {\n    if (typeof selectedLanguage === 'string') {\n      const newLang = selectedLanguage as I18nLang;\n\n      // Update current language for icon mode\n      this.currentLanguage = newLang;\n\n      // Emit the change event\n      this.languageChange.emit(selectedLanguage);\n\n      // Set the new language (reactive by default, reload if forceReload is true)\n      this.i18n.setLanguage(newLang, this.props.forceReload);\n\n      // Update popover selected value for reactive UI (default mode)\n      if (this.popoverProps) {\n        this.popoverProps = { ...this.popoverProps, selectedValue: newLang };\n      }\n    }\n  }\n}\n"]}
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2xhbmd1YWdlLXNlbGVjdG9yL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb2xvciB9IGZyb20gJ0Bpb25pYy9jb3JlJztcblxuLyoqXG4gKiBQcm9wcyBmb3IgdmFsLWxhbmd1YWdlLXNlbGVjdG9yIGNvbXBvbmVudC5cbiAqIEEgc3BlY2lhbGl6ZWQgY29tcG9uZW50IGZvciBsYW5ndWFnZSBzZWxlY3Rpb24uXG4gKlxuICogQHByb3BlcnR5IGN1cnJlbnRMYW5ndWFnZSAtIEN1cnJlbnRseSBzZWxlY3RlZCBsYW5ndWFnZSBjb2RlLlxuICogQHByb3BlcnR5IGF2YWlsYWJsZUxhbmd1YWdlcyAtIEFycmF5IG9mIGF2YWlsYWJsZSBsYW5ndWFnZSBjb2Rlcy5cbiAqIEBwcm9wZXJ0eSBzaG93TGFiZWwgLSBXaGV0aGVyIHRvIHNob3cgdGhlIGxhYmVsLlxuICogQHByb3BlcnR5IGxhYmVsIC0gQ3VzdG9tIGxhYmVsIHRleHQgKHN0YXRpYykuXG4gKiBAcHJvcGVydHkgbGFiZWxDb25maWcgLSBSZWFjdGl2ZSBjb250ZW50IGNvbmZpZ3VyYXRpb24gZm9yIGxhYmVsLlxuICogQHByb3BlcnR5IHNob3dGbGFncyAtIFdoZXRoZXIgdG8gc2hvdyBmbGFnIGljb25zIGZvciBsYW5ndWFnZXMuXG4gKiBAcHJvcGVydHkgY29sb3IgLSBCdXR0b24gY29sb3IgKElvbmljIGNvbG9yIHN0cmluZykuXG4gKiBAcHJvcGVydHkgc2l6ZSAtIEJ1dHRvbiBzaXplICgnc21hbGwnIHwgJ2RlZmF1bHQnIHwgJ2xhcmdlJykuXG4gKiBAcHJvcGVydHkgZmlsbCAtIEJ1dHRvbiBmaWxsIHN0eWxlICgnY2xlYXInIHwgJ291dGxpbmUnIHwgJ3NvbGlkJyB8ICdkZWZhdWx0JykuXG4gKiBAcHJvcGVydHkgc2hhcGUgLSBCdXR0b24gc2hhcGUgKCdyb3VuZCcgfCB1bmRlZmluZWQpLlxuICogQHByb3BlcnR5IGV4cGFuZCAtIEJ1dHRvbiBleHBhbnNpb24gKCdmdWxsJyB8ICdibG9jaycgfCB1bmRlZmluZWQpLlxuICogQHByb3BlcnR5IGRpc2FibGVkIC0gV2hldGhlciB0aGUgc2VsZWN0b3IgaXMgZGlzYWJsZWQuXG4gKiBAcHJvcGVydHkgY3VzdG9tTGFuZ3VhZ2VOYW1lcyAtIEN1c3RvbSBkaXNwbGF5IG5hbWVzIGZvciBsYW5ndWFnZXMuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgTGFuZ3VhZ2VTZWxlY3Rvck1ldGFkYXRhIHtcbiAgLyoqIEN1cnJlbnRseSBzZWxlY3RlZCBsYW5ndWFnZSBjb2RlICovXG4gIGN1cnJlbnRMYW5ndWFnZT86IHN0cmluZztcbiAgLyoqIEFycmF5IG9mIGF2YWlsYWJsZSBsYW5ndWFnZSBjb2RlcyAqL1xuICBhdmFpbGFibGVMYW5ndWFnZXM/OiBzdHJpbmdbXTtcbiAgLyoqIFdoZXRoZXIgdG8gc2hvdyB0aGUgbGFiZWwgKi9cbiAgc2hvd0xhYmVsPzogYm9vbGVhbjtcbiAgLyoqIFN0YXRpYyBsYWJlbCB0ZXh0ICh0YWtlcyBwcmVjZWRlbmNlIG92ZXIgbGFiZWxDb25maWcpICovXG4gIGxhYmVsPzogc3RyaW5nO1xuICAvKiogUmVhY3RpdmUgY29udGVudCBjb25maWd1cmF0aW9uIGZvciBsYWJlbCAqL1xuICBsYWJlbENvbmZpZz86IHtcbiAgICBjbGFzc05hbWU/OiBzdHJpbmc7XG4gICAga2V5OiBzdHJpbmc7XG4gICAgZmFsbGJhY2s/OiBzdHJpbmc7XG4gICAgaW50ZXJwb2xhdGlvbj86IFJlY29yZDxzdHJpbmcsIGFueT47XG4gIH07XG4gIC8qKiBXaGV0aGVyIHRvIHNob3cgZmxhZyBpY29ucyBmb3IgbGFuZ3VhZ2VzICovXG4gIHNob3dGbGFncz86IGJvb2xlYW47XG4gIC8qKiBCdXR0b24gY29sb3IgKi9cbiAgY29sb3I/OiBDb2xvcjtcbiAgLyoqIEJ1dHRvbiBzaXplICovXG4gIHNpemU/OiAnc21hbGwnIHwgJ2RlZmF1bHQnIHwgJ2xhcmdlJztcbiAgLyoqIEJ1dHRvbiBmaWxsIHN0eWxlICovXG4gIGZpbGw/OiAnY2xlYXInIHwgJ291dGxpbmUnIHwgJ3NvbGlkJyB8ICdkZWZhdWx0JztcbiAgLyoqIEJ1dHRvbiBzaGFwZSAqL1xuICBzaGFwZT86ICdyb3VuZCc7XG4gIC8qKiBCdXR0b24gZXhwYW5zaW9uICovXG4gIGV4cGFuZD86ICdmdWxsJyB8ICdibG9jayc7XG4gIC8qKiBXaGV0aGVyIHRoZSBzZWxlY3RvciBpcyBkaXNhYmxlZCAqL1xuICBkaXNhYmxlZD86IGJvb2xlYW47XG4gIC8qKiBDdXN0b20gZGlzcGxheSBuYW1lcyBmb3IgbGFuZ3VhZ2VzICovXG4gIGN1c3RvbUxhbmd1YWdlTmFtZXM/OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+O1xuICAvKipcbiAgICogRm9yY2UgcGFnZSByZWxvYWQgb24gbGFuZ3VhZ2UgY2hhbmdlLlxuICAgKiBCeSBkZWZhdWx0IChmYWxzZSksIGxhbmd1YWdlIGNoYW5nZXMgYXJlIHJlYWN0aXZlIHdpdGhvdXQgcmVsb2FkLlxuICAgKiBTZXQgdG8gdHJ1ZSBmb3IgbGVnYWN5IGJlaGF2aW9yIG9yIGlmIHJlYWN0aXZlIHVwZGF0ZXMgZG9uJ3Qgd29yay5cbiAgICogQGRlZmF1bHQgZmFsc2VcbiAgICovXG4gIGZvcmNlUmVsb2FkPzogYm9vbGVhbjtcbn1cbiJdfQ==
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2xhbmd1YWdlLXNlbGVjdG9yL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb2xvciB9IGZyb20gJ0Bpb25pYy9jb3JlJztcblxuLyoqXG4gKiBEaXNwbGF5IG1vZGUgZm9yIHRoZSBsYW5ndWFnZSBzZWxlY3Rvci5cbiAqIC0gJ2RlZmF1bHQnOiBTaG93cyBidXR0b24gd2l0aCBsYWJlbCBhbmQvb3IgY3VycmVudCBsYW5ndWFnZVxuICogLSAnaWNvbic6IENvbXBhY3QgbW9kZSBzaG93aW5nIG9ubHkgYSBnbG9iZSBpY29uIChpZGVhbCBmb3IgaGVhZGVycylcbiAqL1xuZXhwb3J0IHR5cGUgTGFuZ3VhZ2VTZWxlY3Rvck1vZGUgPSAnZGVmYXVsdCcgfCAnaWNvbic7XG5cbi8qKlxuICogUHJvcHMgZm9yIHZhbC1sYW5ndWFnZS1zZWxlY3RvciBjb21wb25lbnQuXG4gKiBBIHNwZWNpYWxpemVkIGNvbXBvbmVudCBmb3IgbGFuZ3VhZ2Ugc2VsZWN0aW9uLlxuICpcbiAqIEBwcm9wZXJ0eSBtb2RlIC0gRGlzcGxheSBtb2RlOiAnZGVmYXVsdCcgb3IgJ2ljb24nIChjb21wYWN0IGZvciBoZWFkZXJzKS5cbiAqIEBwcm9wZXJ0eSBpY29uIC0gQ3VzdG9tIGljb24gbmFtZSBmb3IgaWNvbiBtb2RlIChkZWZhdWx0OiAnZ2xvYmUtb3V0bGluZScpLlxuICogQHByb3BlcnR5IGN1cnJlbnRMYW5ndWFnZSAtIEN1cnJlbnRseSBzZWxlY3RlZCBsYW5ndWFnZSBjb2RlLlxuICogQHByb3BlcnR5IGF2YWlsYWJsZUxhbmd1YWdlcyAtIEFycmF5IG9mIGF2YWlsYWJsZSBsYW5ndWFnZSBjb2Rlcy5cbiAqIEBwcm9wZXJ0eSBzaG93TGFiZWwgLSBXaGV0aGVyIHRvIHNob3cgdGhlIGxhYmVsLlxuICogQHByb3BlcnR5IGxhYmVsIC0gQ3VzdG9tIGxhYmVsIHRleHQgKHN0YXRpYykuXG4gKiBAcHJvcGVydHkgbGFiZWxDb25maWcgLSBSZWFjdGl2ZSBjb250ZW50IGNvbmZpZ3VyYXRpb24gZm9yIGxhYmVsLlxuICogQHByb3BlcnR5IHNob3dGbGFncyAtIFdoZXRoZXIgdG8gc2hvdyBmbGFnIGljb25zIGZvciBsYW5ndWFnZXMuXG4gKiBAcHJvcGVydHkgY29sb3IgLSBCdXR0b24gY29sb3IgKElvbmljIGNvbG9yIHN0cmluZykuXG4gKiBAcHJvcGVydHkgc2l6ZSAtIEJ1dHRvbiBzaXplICgnc21hbGwnIHwgJ2RlZmF1bHQnIHwgJ2xhcmdlJykuXG4gKiBAcHJvcGVydHkgZmlsbCAtIEJ1dHRvbiBmaWxsIHN0eWxlICgnY2xlYXInIHwgJ291dGxpbmUnIHwgJ3NvbGlkJyB8ICdkZWZhdWx0JykuXG4gKiBAcHJvcGVydHkgc2hhcGUgLSBCdXR0b24gc2hhcGUgKCdyb3VuZCcgfCB1bmRlZmluZWQpLlxuICogQHByb3BlcnR5IGV4cGFuZCAtIEJ1dHRvbiBleHBhbnNpb24gKCdmdWxsJyB8ICdibG9jaycgfCB1bmRlZmluZWQpLlxuICogQHByb3BlcnR5IGRpc2FibGVkIC0gV2hldGhlciB0aGUgc2VsZWN0b3IgaXMgZGlzYWJsZWQuXG4gKiBAcHJvcGVydHkgY3VzdG9tTGFuZ3VhZ2VOYW1lcyAtIEN1c3RvbSBkaXNwbGF5IG5hbWVzIGZvciBsYW5ndWFnZXMuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgTGFuZ3VhZ2VTZWxlY3Rvck1ldGFkYXRhIHtcbiAgLyoqXG4gICAqIERpc3BsYXkgbW9kZSBmb3IgdGhlIHNlbGVjdG9yLlxuICAgKiAtICdkZWZhdWx0JzogRnVsbCBidXR0b24gd2l0aCBsYWJlbC9kcm9wZG93blxuICAgKiAtICdpY29uJzogQ29tcGFjdCBnbG9iZSBpY29uIG9ubHkgKGlkZWFsIGZvciBoZWFkZXJzL3Rvb2xiYXJzKVxuICAgKiBAZGVmYXVsdCAnZGVmYXVsdCdcbiAgICovXG4gIG1vZGU/OiBMYW5ndWFnZVNlbGVjdG9yTW9kZTtcbiAgLyoqXG4gICAqIEljb24gbmFtZSBmb3IgaWNvbiBtb2RlLlxuICAgKiBAZGVmYXVsdCAnZ2xvYmUtb3V0bGluZSdcbiAgICovXG4gIGljb24/OiBzdHJpbmc7XG4gIC8qKiBDdXJyZW50bHkgc2VsZWN0ZWQgbGFuZ3VhZ2UgY29kZSAqL1xuICBjdXJyZW50TGFuZ3VhZ2U/OiBzdHJpbmc7XG4gIC8qKiBBcnJheSBvZiBhdmFpbGFibGUgbGFuZ3VhZ2UgY29kZXMgKi9cbiAgYXZhaWxhYmxlTGFuZ3VhZ2VzPzogc3RyaW5nW107XG4gIC8qKiBXaGV0aGVyIHRvIHNob3cgdGhlIGxhYmVsICovXG4gIHNob3dMYWJlbD86IGJvb2xlYW47XG4gIC8qKiBTdGF0aWMgbGFiZWwgdGV4dCAodGFrZXMgcHJlY2VkZW5jZSBvdmVyIGxhYmVsQ29uZmlnKSAqL1xuICBsYWJlbD86IHN0cmluZztcbiAgLyoqIFJlYWN0aXZlIGNvbnRlbnQgY29uZmlndXJhdGlvbiBmb3IgbGFiZWwgKi9cbiAgbGFiZWxDb25maWc/OiB7XG4gICAgY2xhc3NOYW1lPzogc3RyaW5nO1xuICAgIGtleTogc3RyaW5nO1xuICAgIGZhbGxiYWNrPzogc3RyaW5nO1xuICAgIGludGVycG9sYXRpb24/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+O1xuICB9O1xuICAvKiogV2hldGhlciB0byBzaG93IGZsYWcgaWNvbnMgZm9yIGxhbmd1YWdlcyAqL1xuICBzaG93RmxhZ3M/OiBib29sZWFuO1xuICAvKiogQnV0dG9uIGNvbG9yICovXG4gIGNvbG9yPzogQ29sb3I7XG4gIC8qKiBCdXR0b24gc2l6ZSAqL1xuICBzaXplPzogJ3NtYWxsJyB8ICdkZWZhdWx0JyB8ICdsYXJnZSc7XG4gIC8qKiBCdXR0b24gZmlsbCBzdHlsZSAqL1xuICBmaWxsPzogJ2NsZWFyJyB8ICdvdXRsaW5lJyB8ICdzb2xpZCcgfCAnZGVmYXVsdCc7XG4gIC8qKiBCdXR0b24gc2hhcGUgKi9cbiAgc2hhcGU/OiAncm91bmQnO1xuICAvKiogQnV0dG9uIGV4cGFuc2lvbiAqL1xuICBleHBhbmQ/OiAnZnVsbCcgfCAnYmxvY2snO1xuICAvKiogV2hldGhlciB0aGUgc2VsZWN0b3IgaXMgZGlzYWJsZWQgKi9cbiAgZGlzYWJsZWQ/OiBib29sZWFuO1xuICAvKiogQ3VzdG9tIGRpc3BsYXkgbmFtZXMgZm9yIGxhbmd1YWdlcyAqL1xuICBjdXN0b21MYW5ndWFnZU5hbWVzPzogUmVjb3JkPHN0cmluZywgc3RyaW5nPjtcbiAgLyoqXG4gICAqIEZvcmNlIHBhZ2UgcmVsb2FkIG9uIGxhbmd1YWdlIGNoYW5nZS5cbiAgICogQnkgZGVmYXVsdCAoZmFsc2UpLCBsYW5ndWFnZSBjaGFuZ2VzIGFyZSByZWFjdGl2ZSB3aXRob3V0IHJlbG9hZC5cbiAgICogU2V0IHRvIHRydWUgZm9yIGxlZ2FjeSBiZWhhdmlvciBvciBpZiByZWFjdGl2ZSB1cGRhdGVzIGRvbid0IHdvcmsuXG4gICAqIEBkZWZhdWx0IGZhbHNlXG4gICAqL1xuICBmb3JjZVJlbG9hZD86IGJvb2xlYW47XG59XG4iXX0=
@@ -4,11 +4,12 @@
4
4
  * Servicio principal para Google Ad Manager (GPT).
5
5
  * Integra con el sistema de consent existente y respeta usuarios premium.
6
6
  */
7
- import { Inject, Injectable, PLATFORM_ID, signal, computed } from '@angular/core';
7
+ import { Inject, Injectable, PLATFORM_ID, signal, computed, inject } from '@angular/core';
8
8
  import { isPlatformBrowser } from '@angular/common';
9
9
  import { NavigationEnd } from '@angular/router';
10
10
  import { filter } from 'rxjs/operators';
11
11
  import { VALTECH_ADS_CONFIG } from './config';
12
+ import { AuthStateService } from '../auth/auth-state.service';
12
13
  import { AD_SIZE_MAP, } from './types';
13
14
  import * as i0 from "@angular/core";
14
15
  import * as i1 from "@angular/router";
@@ -45,9 +46,9 @@ export class AdsService {
45
46
  // ===========================================================================
46
47
  // ESTADO (Signals)
47
48
  // ===========================================================================
49
+ this.authStateService = inject(AuthStateService);
48
50
  this._isInitialized = signal(false);
49
51
  this._isEnabled = signal(false);
50
- this._isPremiumUser = signal(false);
51
52
  this._isDebugMode = signal(false);
52
53
  this._slots = signal(new Map());
53
54
  this._slotStates = signal(new Map());
@@ -58,11 +59,11 @@ export class AdsService {
58
59
  this.isEnabled = computed(() => {
59
60
  return (this._isInitialized() &&
60
61
  this._isEnabled() &&
61
- !this._isPremiumUser() &&
62
+ !this.authStateService.isPremium() &&
62
63
  this.consentService.canShowAds());
63
64
  });
64
- /** Indica si el usuario es premium (no ve ads) */
65
- this.isPremiumUser = this._isPremiumUser.asReadonly();
65
+ /** Indica si el usuario es premium (no ve ads) - lee de AuthStateService */
66
+ this.isPremiumUser = this.authStateService.isPremium;
66
67
  /** Indica si esta en modo debug */
67
68
  this.isDebugMode = this._isDebugMode.asReadonly();
68
69
  /** Estado de consent para ads */
@@ -89,17 +90,8 @@ export class AdsService {
89
90
  if (!isPlatformBrowser(this.platformId)) {
90
91
  return;
91
92
  }
92
- // Verificar si usuario es premium
93
- if (this.config.isPremiumUser) {
94
- try {
95
- this._isPremiumUser.set(this.config.isPremiumUser());
96
- }
97
- catch {
98
- this._isPremiumUser.set(false);
99
- }
100
- }
101
- // Si es premium, no inicializar ads
102
- if (this._isPremiumUser()) {
93
+ // Si es premium, no inicializar ads (lee de AuthStateService)
94
+ if (this.authStateService.isPremium()) {
103
95
  if (this._isDebugMode()) {
104
96
  console.log('[ValtechAds] Usuario premium detectado - ads deshabilitados');
105
97
  }
@@ -285,32 +277,11 @@ export class AdsService {
285
277
  return this._slotStates().get(slotId) ?? 'idle';
286
278
  }
287
279
  // ===========================================================================
288
- // PREMIUM USER
280
+ // SLOT CLEANUP
289
281
  // ===========================================================================
290
- /**
291
- * Actualiza el estado premium del usuario.
292
- * Llamar cuando cambie el estado de suscripcion.
293
- *
294
- * @param isPremium - Nuevo estado premium
295
- */
296
- updatePremiumStatus(isPremium) {
297
- const wasEnabled = this.isEnabled();
298
- this._isPremiumUser.set(isPremium);
299
- if (isPremium && wasEnabled) {
300
- // Destruir todos los slots activos
301
- this.destroyAllSlots();
302
- if (this._isDebugMode()) {
303
- console.log('[ValtechAds] Usuario ahora es premium - ads deshabilitados');
304
- }
305
- }
306
- else if (!isPremium && !wasEnabled && this._isInitialized()) {
307
- if (this._isDebugMode()) {
308
- console.log('[ValtechAds] Usuario ya no es premium - ads habilitados');
309
- }
310
- }
311
- }
312
282
  /**
313
283
  * Destruye todos los slots activos.
284
+ * Llamar cuando el usuario se vuelve premium o cambia de página.
314
285
  */
315
286
  destroyAllSlots() {
316
287
  const googletag = window.googletag;
@@ -446,4 +417,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
446
417
  type: Inject,
447
418
  args: [PLATFORM_ID]
448
419
  }] }, { type: i1.Router }, { type: i2.AdsLoaderService }, { type: i3.AdsConsentService }] });
449
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ads.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/ads/ads.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Ads Service\n *\n * Servicio principal para Google Ad Manager (GPT).\n * Integra con el sistema de consent existente y respeta usuarios premium.\n */\n\nimport { Inject, Injectable, Injector, PLATFORM_ID, signal, computed, OnDestroy } from '@angular/core';\nimport { isPlatformBrowser } from '@angular/common';\nimport { Router, NavigationEnd } from '@angular/router';\nimport { filter } from 'rxjs/operators';\nimport { Subscription } from 'rxjs';\n\nimport { VALTECH_ADS_CONFIG } from './config';\nimport { AdsLoaderService } from './ads-loader.service';\nimport { AdsConsentService } from './ads-consent.service';\nimport {\n  ValtechAdsConfig,\n  AdSlotConfig,\n  AdSlotState,\n  AdEvent,\n  GPTSlot,\n  GPTNamespace,\n  SizeMapping,\n  AD_SIZE_MAP,\n  AdSlotSize,\n} from './types';\n\n/**\n * Servicio principal de Google Ad Manager.\n *\n * Maneja la creacion, destruccion y refresh de ad slots.\n * Integra automaticamente con consent mode y detecta usuarios premium.\n *\n * @example\n * ```typescript\n * @Component({...})\n * export class MyComponent {\n *   private ads = inject(AdsService);\n *\n *   // Verificar si mostrar ads\n *   showAds = this.ads.isEnabled;\n *\n *   // Verificar si usuario es premium\n *   isPremium = this.ads.isPremiumUser;\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class AdsService implements OnDestroy {\n  // ===========================================================================\n  // ESTADO (Signals)\n  // ===========================================================================\n\n  private readonly _isInitialized = signal<boolean>(false);\n  private readonly _isEnabled = signal<boolean>(false);\n  private readonly _isPremiumUser = signal<boolean>(false);\n  private readonly _isDebugMode = signal<boolean>(false);\n  private readonly _slots = signal<Map<string, GPTSlot>>(new Map());\n  private readonly _slotStates = signal<Map<string, AdSlotState>>(new Map());\n  private readonly _events = signal<AdEvent[]>([]);\n\n  /** Indica si el servicio esta inicializado */\n  readonly isInitialized = this._isInitialized.asReadonly();\n\n  /** Indica si los ads estan habilitados (consent + no premium) */\n  readonly isEnabled = computed(() => {\n    return (\n      this._isInitialized() &&\n      this._isEnabled() &&\n      !this._isPremiumUser() &&\n      this.consentService.canShowAds()\n    );\n  });\n\n  /** Indica si el usuario es premium (no ve ads) */\n  readonly isPremiumUser = this._isPremiumUser.asReadonly();\n\n  /** Indica si esta en modo debug */\n  readonly isDebugMode = this._isDebugMode.asReadonly();\n\n  /** Estado de consent para ads */\n  readonly consentState = this.consentService.adsConsentState;\n\n  /** Eventos de ads (historial) */\n  readonly events = this._events.asReadonly();\n\n  // ===========================================================================\n  // INTERNOS\n  // ===========================================================================\n\n  private activeSlots: Map<string, GPTSlot> = new Map();\n  private routerSubscription: Subscription | null = null;\n  private refreshTimerId: ReturnType<typeof setInterval> | null = null;\n\n  constructor(\n    private injector: Injector,\n    @Inject(VALTECH_ADS_CONFIG) private config: ValtechAdsConfig,\n    @Inject(PLATFORM_ID) private platformId: Object,\n    private router: Router,\n    private loaderService: AdsLoaderService,\n    private consentService: AdsConsentService\n  ) {\n    this._isDebugMode.set(config.debugMode ?? false);\n  }\n\n  // ===========================================================================\n  // INICIALIZACION\n  // ===========================================================================\n\n  /**\n   * Inicializa el servicio de ads.\n   * Llamado automaticamente por APP_INITIALIZER.\n   * NO carga el script GPT hasta que se necesite el primer ad.\n   */\n  async initialize(): Promise<void> {\n    if (!isPlatformBrowser(this.platformId)) {\n      return;\n    }\n\n    // Verificar si usuario es premium\n    if (this.config.isPremiumUser) {\n      try {\n        this._isPremiumUser.set(this.config.isPremiumUser());\n      } catch {\n        this._isPremiumUser.set(false);\n      }\n    }\n\n    // Si es premium, no inicializar ads\n    if (this._isPremiumUser()) {\n      if (this._isDebugMode()) {\n        console.log('[ValtechAds] Usuario premium detectado - ads deshabilitados');\n      }\n      this._isInitialized.set(true);\n      return;\n    }\n\n    // Verificar consent basico\n    if (!this.consentService.canShowAds() && !this.config.showNonPersonalizedAds) {\n      if (this._isDebugMode()) {\n        console.log('[ValtechAds] Sin consent para ads');\n      }\n      this._isInitialized.set(true);\n      return;\n    }\n\n    // Suscribirse a cambios de ruta para verificar exclusiones\n    this.setupRouteListener();\n\n    // Configurar auto-refresh si esta habilitado\n    if (this.config.autoRefreshInterval && this.config.autoRefreshInterval > 0) {\n      this.setupAutoRefresh();\n    }\n\n    this._isEnabled.set(true);\n    this._isInitialized.set(true);\n\n    if (this._isDebugMode()) {\n      console.log('[ValtechAds] Inicializado en modo debug', {\n        networkId: this.config.networkId,\n        lazyLoad: this.config.lazyLoad,\n        excludeRoutes: this.config.excludeRoutes,\n      });\n    }\n  }\n\n  /**\n   * Carga el script GPT de forma lazy.\n   * Llamado automaticamente cuando se renderiza el primer ad slot.\n   */\n  async ensureGPTLoaded(): Promise<GPTNamespace | null> {\n    if (!this.isEnabled()) {\n      return null;\n    }\n\n    return this.loaderService.loadGPT();\n  }\n\n  // ===========================================================================\n  // GESTION DE SLOTS\n  // ===========================================================================\n\n  /**\n   * Define y muestra un ad slot.\n   * Llamado internamente por el componente AdSlot.\n   *\n   * @param config - Configuracion del slot\n   * @returns ID del slot o null si no se puede mostrar\n   */\n  async defineSlot(config: AdSlotConfig): Promise<string | null> {\n    if (!this.isEnabled()) {\n      this.updateSlotState(config.slotId, 'hidden');\n      return null;\n    }\n\n    // Verificar si la ruta actual esta excluida\n    if (this.isRouteExcluded()) {\n      this.updateSlotState(config.slotId, 'hidden');\n      return null;\n    }\n\n    this.updateSlotState(config.slotId, 'loading');\n\n    const googletag = await this.ensureGPTLoaded();\n    if (!googletag) {\n      this.updateSlotState(config.slotId, 'error');\n      return null;\n    }\n\n    return new Promise((resolve) => {\n      googletag.cmd.push(() => {\n        try {\n          const adUnitPath = `${this.config.networkId}${config.adUnitPath}`;\n          const sizes = this.resolveSizes(config.size);\n\n          const slot = googletag.defineSlot(adUnitPath, sizes, config.slotId);\n          if (!slot) {\n            this.updateSlotState(config.slotId, 'error');\n            this.emitEvent({ type: 'error', slotId: config.slotId, error: new Error('No se pudo crear el slot') });\n            resolve(null);\n            return;\n          }\n\n          // Agregar al servicio pubads\n          slot.addService(googletag.pubads());\n\n          // Configurar size mapping responsivo\n          if (config.sizeMapping && config.sizeMapping.length > 0) {\n            const mapping = this.buildSizeMapping(googletag, config.sizeMapping);\n            slot.defineSizeMapping(mapping);\n          }\n\n          // Configurar targeting\n          this.applyTargeting(slot, config.targeting);\n\n          // Colapsar si esta vacio\n          if (config.collapseEmpty ?? this.config.defaultSlotConfig?.collapseEmpty) {\n            slot.setCollapseEmptyDiv(true, true);\n          }\n\n          // Guardar referencia\n          this.activeSlots.set(config.slotId, slot);\n          this._slots.update((slots) => new Map(slots).set(config.slotId, slot));\n\n          // Mostrar el ad\n          googletag.enableServices();\n          googletag.display(config.slotId);\n\n          this.updateSlotState(config.slotId, 'rendered');\n          this.emitEvent({ type: 'loaded', slotId: config.slotId, isEmpty: false });\n\n          if (this._isDebugMode()) {\n            console.log(`[ValtechAds] Slot definido: ${config.slotId}`, {\n              adUnitPath,\n              sizes: config.size,\n            });\n          }\n\n          resolve(config.slotId);\n        } catch (error) {\n          console.error('[ValtechAds] Error definiendo slot:', error);\n          this.updateSlotState(config.slotId, 'error');\n          this.emitEvent({ type: 'error', slotId: config.slotId, error: error as Error });\n          resolve(null);\n        }\n      });\n    });\n  }\n\n  /**\n   * Destruye un slot y libera recursos.\n   *\n   * @param slotId - ID del slot a destruir\n   */\n  destroySlot(slotId: string): void {\n    const slot = this.activeSlots.get(slotId);\n    if (!slot) return;\n\n    const googletag = window.googletag;\n    if (googletag) {\n      googletag.cmd.push(() => {\n        googletag.destroySlots([slot]);\n      });\n    }\n\n    this.activeSlots.delete(slotId);\n    this._slots.update((slots) => {\n      const newSlots = new Map(slots);\n      newSlots.delete(slotId);\n      return newSlots;\n    });\n    this._slotStates.update((states) => {\n      const newStates = new Map(states);\n      newStates.delete(slotId);\n      return newStates;\n    });\n\n    if (this._isDebugMode()) {\n      console.log(`[ValtechAds] Slot destruido: ${slotId}`);\n    }\n  }\n\n  /**\n   * Refresca un slot especifico o todos los slots activos.\n   *\n   * @param slotIds - IDs de slots a refrescar (undefined = todos)\n   */\n  refreshSlots(slotIds?: string[]): void {\n    if (!this.isEnabled()) return;\n\n    const googletag = window.googletag;\n    if (!googletag) return;\n\n    googletag.cmd.push(() => {\n      if (slotIds && slotIds.length > 0) {\n        const slots = slotIds.map((id) => this.activeSlots.get(id)).filter((s): s is GPTSlot => !!s);\n        if (slots.length > 0) {\n          googletag.pubads().refresh(slots);\n          if (this._isDebugMode()) {\n            console.log(`[ValtechAds] Slots refrescados: ${slotIds.join(', ')}`);\n          }\n        }\n      } else {\n        googletag.pubads().refresh();\n        if (this._isDebugMode()) {\n          console.log('[ValtechAds] Todos los slots refrescados');\n        }\n      }\n    });\n  }\n\n  /**\n   * Obtiene el estado de un slot.\n   *\n   * @param slotId - ID del slot\n   * @returns Estado actual del slot\n   */\n  getSlotState(slotId: string): AdSlotState {\n    return this._slotStates().get(slotId) ?? 'idle';\n  }\n\n  // ===========================================================================\n  // PREMIUM USER\n  // ===========================================================================\n\n  /**\n   * Actualiza el estado premium del usuario.\n   * Llamar cuando cambie el estado de suscripcion.\n   *\n   * @param isPremium - Nuevo estado premium\n   */\n  updatePremiumStatus(isPremium: boolean): void {\n    const wasEnabled = this.isEnabled();\n    this._isPremiumUser.set(isPremium);\n\n    if (isPremium && wasEnabled) {\n      // Destruir todos los slots activos\n      this.destroyAllSlots();\n      if (this._isDebugMode()) {\n        console.log('[ValtechAds] Usuario ahora es premium - ads deshabilitados');\n      }\n    } else if (!isPremium && !wasEnabled && this._isInitialized()) {\n      if (this._isDebugMode()) {\n        console.log('[ValtechAds] Usuario ya no es premium - ads habilitados');\n      }\n    }\n  }\n\n  /**\n   * Destruye todos los slots activos.\n   */\n  destroyAllSlots(): void {\n    const googletag = window.googletag;\n    if (googletag) {\n      googletag.cmd.push(() => {\n        googletag.destroySlots();\n      });\n    }\n    this.activeSlots.clear();\n    this._slots.set(new Map());\n    this._slotStates.set(new Map());\n\n    if (this._isDebugMode()) {\n      console.log('[ValtechAds] Todos los slots destruidos');\n    }\n  }\n\n  // ===========================================================================\n  // PRIVATE METHODS\n  // ===========================================================================\n\n  private resolveSizes(size: AdSlotConfig['size']): unknown {\n    if (typeof size === 'string') {\n      // Single preset size\n      const mapped = AD_SIZE_MAP[size as AdSlotSize];\n      return mapped === 'fluid' ? 'fluid' : [mapped];\n    }\n\n    if (Array.isArray(size)) {\n      if (typeof size[0] === 'string') {\n        // Array of preset sizes\n        return (size as AdSlotSize[]).map((s) => {\n          const mapped = AD_SIZE_MAP[s];\n          return mapped === 'fluid' ? 'fluid' : mapped;\n        });\n      }\n      // Direct size array\n      return size;\n    }\n\n    return 'fluid';\n  }\n\n  private buildSizeMapping(googletag: GPTNamespace, mapping: SizeMapping[]): unknown {\n    const builder = googletag.sizeMapping();\n\n    // Ordenar por viewport descendente\n    const sorted = [...mapping].sort((a, b) => b.viewportWidth - a.viewportWidth);\n\n    for (const m of sorted) {\n      builder.addSize([m.viewportWidth, 0], m.sizes);\n    }\n\n    return builder.build();\n  }\n\n  private applyTargeting(slot: GPTSlot, targeting?: Record<string, string | string[]>): void {\n    // Aplicar targeting global\n    if (this.config.globalTargeting) {\n      for (const [key, value] of Object.entries(this.config.globalTargeting)) {\n        slot.setTargeting(key, value);\n      }\n    }\n\n    // Aplicar targeting del slot\n    if (targeting) {\n      for (const [key, value] of Object.entries(targeting)) {\n        slot.setTargeting(key, value);\n      }\n    }\n\n    // Agregar targeting de debug si esta habilitado\n    if (this._isDebugMode()) {\n      slot.setTargeting('test', 'true');\n    }\n  }\n\n  private updateSlotState(slotId: string, state: AdSlotState): void {\n    this._slotStates.update((states) => new Map(states).set(slotId, state));\n  }\n\n  private emitEvent(event: Omit<AdEvent, 'timestamp'>): void {\n    const fullEvent: AdEvent = { ...event, timestamp: new Date() };\n    this._events.update((events) => [...events.slice(-99), fullEvent]);\n\n    // Callbacks de configuracion\n    if (event.type === 'loaded' || event.type === 'empty') {\n      this.config.onAdLoaded?.(event.slotId, event.isEmpty ?? false);\n    } else if (event.type === 'error' && event.error) {\n      this.config.onAdError?.(event.slotId, event.error);\n    }\n  }\n\n  private setupRouteListener(): void {\n    if (!this.config.excludeRoutes || this.config.excludeRoutes.length === 0) {\n      return;\n    }\n\n    this.routerSubscription = this.router.events\n      .pipe(filter((event): event is NavigationEnd => event instanceof NavigationEnd))\n      .subscribe((event: NavigationEnd) => {\n        const shouldHide = this.isRouteExcluded(event.urlAfterRedirects);\n\n        if (shouldHide) {\n          this.destroyAllSlots();\n        }\n      });\n  }\n\n  private setupAutoRefresh(): void {\n    const intervalMs = (this.config.autoRefreshInterval ?? 0) * 1000;\n    if (intervalMs <= 0) return;\n\n    this.refreshTimerId = setInterval(() => {\n      if (this.isEnabled() && this.activeSlots.size > 0) {\n        this.refreshSlots();\n      }\n    }, intervalMs);\n  }\n\n  private isRouteExcluded(url?: string): boolean {\n    if (!this.config.excludeRoutes || this.config.excludeRoutes.length === 0) {\n      return false;\n    }\n\n    const currentUrl = url ?? this.router.url;\n    return this.config.excludeRoutes.some((pattern) => {\n      const regex = new RegExp(pattern);\n      return regex.test(currentUrl);\n    });\n  }\n\n  // ===========================================================================\n  // LIFECYCLE\n  // ===========================================================================\n\n  ngOnDestroy(): void {\n    this.routerSubscription?.unsubscribe();\n\n    if (this.refreshTimerId) {\n      clearInterval(this.refreshTimerId);\n    }\n\n    this.destroyAllSlots();\n    this.consentService.destroy();\n  }\n}\n"]}
420
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ads.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/ads/ads.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Ads Service\n *\n * Servicio principal para Google Ad Manager (GPT).\n * Integra con el sistema de consent existente y respeta usuarios premium.\n */\n\nimport { Inject, Injectable, Injector, PLATFORM_ID, signal, computed, OnDestroy, inject } from '@angular/core';\nimport { isPlatformBrowser } from '@angular/common';\nimport { Router, NavigationEnd } from '@angular/router';\nimport { filter } from 'rxjs/operators';\nimport { Subscription } from 'rxjs';\n\nimport { VALTECH_ADS_CONFIG } from './config';\nimport { AdsLoaderService } from './ads-loader.service';\nimport { AdsConsentService } from './ads-consent.service';\nimport { AuthStateService } from '../auth/auth-state.service';\nimport {\n  ValtechAdsConfig,\n  AdSlotConfig,\n  AdSlotState,\n  AdEvent,\n  GPTSlot,\n  GPTNamespace,\n  SizeMapping,\n  AD_SIZE_MAP,\n  AdSlotSize,\n} from './types';\n\n/**\n * Servicio principal de Google Ad Manager.\n *\n * Maneja la creacion, destruccion y refresh de ad slots.\n * Integra automaticamente con consent mode y detecta usuarios premium.\n *\n * @example\n * ```typescript\n * @Component({...})\n * export class MyComponent {\n *   private ads = inject(AdsService);\n *\n *   // Verificar si mostrar ads\n *   showAds = this.ads.isEnabled;\n *\n *   // Verificar si usuario es premium\n *   isPremium = this.ads.isPremiumUser;\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class AdsService implements OnDestroy {\n  // ===========================================================================\n  // ESTADO (Signals)\n  // ===========================================================================\n\n  private readonly authStateService = inject(AuthStateService);\n  private readonly _isInitialized = signal<boolean>(false);\n  private readonly _isEnabled = signal<boolean>(false);\n  private readonly _isDebugMode = signal<boolean>(false);\n  private readonly _slots = signal<Map<string, GPTSlot>>(new Map());\n  private readonly _slotStates = signal<Map<string, AdSlotState>>(new Map());\n  private readonly _events = signal<AdEvent[]>([]);\n\n  /** Indica si el servicio esta inicializado */\n  readonly isInitialized = this._isInitialized.asReadonly();\n\n  /** Indica si los ads estan habilitados (consent + no premium) */\n  readonly isEnabled = computed(() => {\n    return (\n      this._isInitialized() &&\n      this._isEnabled() &&\n      !this.authStateService.isPremium() &&\n      this.consentService.canShowAds()\n    );\n  });\n\n  /** Indica si el usuario es premium (no ve ads) - lee de AuthStateService */\n  readonly isPremiumUser = this.authStateService.isPremium;\n\n  /** Indica si esta en modo debug */\n  readonly isDebugMode = this._isDebugMode.asReadonly();\n\n  /** Estado de consent para ads */\n  readonly consentState = this.consentService.adsConsentState;\n\n  /** Eventos de ads (historial) */\n  readonly events = this._events.asReadonly();\n\n  // ===========================================================================\n  // INTERNOS\n  // ===========================================================================\n\n  private activeSlots: Map<string, GPTSlot> = new Map();\n  private routerSubscription: Subscription | null = null;\n  private refreshTimerId: ReturnType<typeof setInterval> | null = null;\n\n  constructor(\n    private injector: Injector,\n    @Inject(VALTECH_ADS_CONFIG) private config: ValtechAdsConfig,\n    @Inject(PLATFORM_ID) private platformId: Object,\n    private router: Router,\n    private loaderService: AdsLoaderService,\n    private consentService: AdsConsentService\n  ) {\n    this._isDebugMode.set(config.debugMode ?? false);\n  }\n\n  // ===========================================================================\n  // INICIALIZACION\n  // ===========================================================================\n\n  /**\n   * Inicializa el servicio de ads.\n   * Llamado automaticamente por APP_INITIALIZER.\n   * NO carga el script GPT hasta que se necesite el primer ad.\n   */\n  async initialize(): Promise<void> {\n    if (!isPlatformBrowser(this.platformId)) {\n      return;\n    }\n\n    // Si es premium, no inicializar ads (lee de AuthStateService)\n    if (this.authStateService.isPremium()) {\n      if (this._isDebugMode()) {\n        console.log('[ValtechAds] Usuario premium detectado - ads deshabilitados');\n      }\n      this._isInitialized.set(true);\n      return;\n    }\n\n    // Verificar consent basico\n    if (!this.consentService.canShowAds() && !this.config.showNonPersonalizedAds) {\n      if (this._isDebugMode()) {\n        console.log('[ValtechAds] Sin consent para ads');\n      }\n      this._isInitialized.set(true);\n      return;\n    }\n\n    // Suscribirse a cambios de ruta para verificar exclusiones\n    this.setupRouteListener();\n\n    // Configurar auto-refresh si esta habilitado\n    if (this.config.autoRefreshInterval && this.config.autoRefreshInterval > 0) {\n      this.setupAutoRefresh();\n    }\n\n    this._isEnabled.set(true);\n    this._isInitialized.set(true);\n\n    if (this._isDebugMode()) {\n      console.log('[ValtechAds] Inicializado en modo debug', {\n        networkId: this.config.networkId,\n        lazyLoad: this.config.lazyLoad,\n        excludeRoutes: this.config.excludeRoutes,\n      });\n    }\n  }\n\n  /**\n   * Carga el script GPT de forma lazy.\n   * Llamado automaticamente cuando se renderiza el primer ad slot.\n   */\n  async ensureGPTLoaded(): Promise<GPTNamespace | null> {\n    if (!this.isEnabled()) {\n      return null;\n    }\n\n    return this.loaderService.loadGPT();\n  }\n\n  // ===========================================================================\n  // GESTION DE SLOTS\n  // ===========================================================================\n\n  /**\n   * Define y muestra un ad slot.\n   * Llamado internamente por el componente AdSlot.\n   *\n   * @param config - Configuracion del slot\n   * @returns ID del slot o null si no se puede mostrar\n   */\n  async defineSlot(config: AdSlotConfig): Promise<string | null> {\n    if (!this.isEnabled()) {\n      this.updateSlotState(config.slotId, 'hidden');\n      return null;\n    }\n\n    // Verificar si la ruta actual esta excluida\n    if (this.isRouteExcluded()) {\n      this.updateSlotState(config.slotId, 'hidden');\n      return null;\n    }\n\n    this.updateSlotState(config.slotId, 'loading');\n\n    const googletag = await this.ensureGPTLoaded();\n    if (!googletag) {\n      this.updateSlotState(config.slotId, 'error');\n      return null;\n    }\n\n    return new Promise((resolve) => {\n      googletag.cmd.push(() => {\n        try {\n          const adUnitPath = `${this.config.networkId}${config.adUnitPath}`;\n          const sizes = this.resolveSizes(config.size);\n\n          const slot = googletag.defineSlot(adUnitPath, sizes, config.slotId);\n          if (!slot) {\n            this.updateSlotState(config.slotId, 'error');\n            this.emitEvent({ type: 'error', slotId: config.slotId, error: new Error('No se pudo crear el slot') });\n            resolve(null);\n            return;\n          }\n\n          // Agregar al servicio pubads\n          slot.addService(googletag.pubads());\n\n          // Configurar size mapping responsivo\n          if (config.sizeMapping && config.sizeMapping.length > 0) {\n            const mapping = this.buildSizeMapping(googletag, config.sizeMapping);\n            slot.defineSizeMapping(mapping);\n          }\n\n          // Configurar targeting\n          this.applyTargeting(slot, config.targeting);\n\n          // Colapsar si esta vacio\n          if (config.collapseEmpty ?? this.config.defaultSlotConfig?.collapseEmpty) {\n            slot.setCollapseEmptyDiv(true, true);\n          }\n\n          // Guardar referencia\n          this.activeSlots.set(config.slotId, slot);\n          this._slots.update((slots) => new Map(slots).set(config.slotId, slot));\n\n          // Mostrar el ad\n          googletag.enableServices();\n          googletag.display(config.slotId);\n\n          this.updateSlotState(config.slotId, 'rendered');\n          this.emitEvent({ type: 'loaded', slotId: config.slotId, isEmpty: false });\n\n          if (this._isDebugMode()) {\n            console.log(`[ValtechAds] Slot definido: ${config.slotId}`, {\n              adUnitPath,\n              sizes: config.size,\n            });\n          }\n\n          resolve(config.slotId);\n        } catch (error) {\n          console.error('[ValtechAds] Error definiendo slot:', error);\n          this.updateSlotState(config.slotId, 'error');\n          this.emitEvent({ type: 'error', slotId: config.slotId, error: error as Error });\n          resolve(null);\n        }\n      });\n    });\n  }\n\n  /**\n   * Destruye un slot y libera recursos.\n   *\n   * @param slotId - ID del slot a destruir\n   */\n  destroySlot(slotId: string): void {\n    const slot = this.activeSlots.get(slotId);\n    if (!slot) return;\n\n    const googletag = window.googletag;\n    if (googletag) {\n      googletag.cmd.push(() => {\n        googletag.destroySlots([slot]);\n      });\n    }\n\n    this.activeSlots.delete(slotId);\n    this._slots.update((slots) => {\n      const newSlots = new Map(slots);\n      newSlots.delete(slotId);\n      return newSlots;\n    });\n    this._slotStates.update((states) => {\n      const newStates = new Map(states);\n      newStates.delete(slotId);\n      return newStates;\n    });\n\n    if (this._isDebugMode()) {\n      console.log(`[ValtechAds] Slot destruido: ${slotId}`);\n    }\n  }\n\n  /**\n   * Refresca un slot especifico o todos los slots activos.\n   *\n   * @param slotIds - IDs de slots a refrescar (undefined = todos)\n   */\n  refreshSlots(slotIds?: string[]): void {\n    if (!this.isEnabled()) return;\n\n    const googletag = window.googletag;\n    if (!googletag) return;\n\n    googletag.cmd.push(() => {\n      if (slotIds && slotIds.length > 0) {\n        const slots = slotIds.map((id) => this.activeSlots.get(id)).filter((s): s is GPTSlot => !!s);\n        if (slots.length > 0) {\n          googletag.pubads().refresh(slots);\n          if (this._isDebugMode()) {\n            console.log(`[ValtechAds] Slots refrescados: ${slotIds.join(', ')}`);\n          }\n        }\n      } else {\n        googletag.pubads().refresh();\n        if (this._isDebugMode()) {\n          console.log('[ValtechAds] Todos los slots refrescados');\n        }\n      }\n    });\n  }\n\n  /**\n   * Obtiene el estado de un slot.\n   *\n   * @param slotId - ID del slot\n   * @returns Estado actual del slot\n   */\n  getSlotState(slotId: string): AdSlotState {\n    return this._slotStates().get(slotId) ?? 'idle';\n  }\n\n  // ===========================================================================\n  // SLOT CLEANUP\n  // ===========================================================================\n\n  /**\n   * Destruye todos los slots activos.\n   * Llamar cuando el usuario se vuelve premium o cambia de página.\n   */\n  destroyAllSlots(): void {\n    const googletag = window.googletag;\n    if (googletag) {\n      googletag.cmd.push(() => {\n        googletag.destroySlots();\n      });\n    }\n    this.activeSlots.clear();\n    this._slots.set(new Map());\n    this._slotStates.set(new Map());\n\n    if (this._isDebugMode()) {\n      console.log('[ValtechAds] Todos los slots destruidos');\n    }\n  }\n\n  // ===========================================================================\n  // PRIVATE METHODS\n  // ===========================================================================\n\n  private resolveSizes(size: AdSlotConfig['size']): unknown {\n    if (typeof size === 'string') {\n      // Single preset size\n      const mapped = AD_SIZE_MAP[size as AdSlotSize];\n      return mapped === 'fluid' ? 'fluid' : [mapped];\n    }\n\n    if (Array.isArray(size)) {\n      if (typeof size[0] === 'string') {\n        // Array of preset sizes\n        return (size as AdSlotSize[]).map((s) => {\n          const mapped = AD_SIZE_MAP[s];\n          return mapped === 'fluid' ? 'fluid' : mapped;\n        });\n      }\n      // Direct size array\n      return size;\n    }\n\n    return 'fluid';\n  }\n\n  private buildSizeMapping(googletag: GPTNamespace, mapping: SizeMapping[]): unknown {\n    const builder = googletag.sizeMapping();\n\n    // Ordenar por viewport descendente\n    const sorted = [...mapping].sort((a, b) => b.viewportWidth - a.viewportWidth);\n\n    for (const m of sorted) {\n      builder.addSize([m.viewportWidth, 0], m.sizes);\n    }\n\n    return builder.build();\n  }\n\n  private applyTargeting(slot: GPTSlot, targeting?: Record<string, string | string[]>): void {\n    // Aplicar targeting global\n    if (this.config.globalTargeting) {\n      for (const [key, value] of Object.entries(this.config.globalTargeting)) {\n        slot.setTargeting(key, value);\n      }\n    }\n\n    // Aplicar targeting del slot\n    if (targeting) {\n      for (const [key, value] of Object.entries(targeting)) {\n        slot.setTargeting(key, value);\n      }\n    }\n\n    // Agregar targeting de debug si esta habilitado\n    if (this._isDebugMode()) {\n      slot.setTargeting('test', 'true');\n    }\n  }\n\n  private updateSlotState(slotId: string, state: AdSlotState): void {\n    this._slotStates.update((states) => new Map(states).set(slotId, state));\n  }\n\n  private emitEvent(event: Omit<AdEvent, 'timestamp'>): void {\n    const fullEvent: AdEvent = { ...event, timestamp: new Date() };\n    this._events.update((events) => [...events.slice(-99), fullEvent]);\n\n    // Callbacks de configuracion\n    if (event.type === 'loaded' || event.type === 'empty') {\n      this.config.onAdLoaded?.(event.slotId, event.isEmpty ?? false);\n    } else if (event.type === 'error' && event.error) {\n      this.config.onAdError?.(event.slotId, event.error);\n    }\n  }\n\n  private setupRouteListener(): void {\n    if (!this.config.excludeRoutes || this.config.excludeRoutes.length === 0) {\n      return;\n    }\n\n    this.routerSubscription = this.router.events\n      .pipe(filter((event): event is NavigationEnd => event instanceof NavigationEnd))\n      .subscribe((event: NavigationEnd) => {\n        const shouldHide = this.isRouteExcluded(event.urlAfterRedirects);\n\n        if (shouldHide) {\n          this.destroyAllSlots();\n        }\n      });\n  }\n\n  private setupAutoRefresh(): void {\n    const intervalMs = (this.config.autoRefreshInterval ?? 0) * 1000;\n    if (intervalMs <= 0) return;\n\n    this.refreshTimerId = setInterval(() => {\n      if (this.isEnabled() && this.activeSlots.size > 0) {\n        this.refreshSlots();\n      }\n    }, intervalMs);\n  }\n\n  private isRouteExcluded(url?: string): boolean {\n    if (!this.config.excludeRoutes || this.config.excludeRoutes.length === 0) {\n      return false;\n    }\n\n    const currentUrl = url ?? this.router.url;\n    return this.config.excludeRoutes.some((pattern) => {\n      const regex = new RegExp(pattern);\n      return regex.test(currentUrl);\n    });\n  }\n\n  // ===========================================================================\n  // LIFECYCLE\n  // ===========================================================================\n\n  ngOnDestroy(): void {\n    this.routerSubscription?.unsubscribe();\n\n    if (this.refreshTimerId) {\n      clearInterval(this.refreshTimerId);\n    }\n\n    this.destroyAllSlots();\n    this.consentService.destroy();\n  }\n}\n"]}
@@ -20,4 +20,4 @@ export const AD_SIZE_MAP = {
20
20
  native: 'fluid',
21
21
  custom: 'fluid',
22
22
  };
23
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../../src/lib/services/ads/types.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAwFH;;GAEG;AACH,MAAM,CAAC,MAAM,WAAW,GAAmD;IACzE,WAAW,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC;IACtB,SAAS,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;IACrB,kBAAkB,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;IAC9B,iBAAiB,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;IAC7B,WAAW,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;IACvB,UAAU,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;IACtB,eAAe,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC;IAC1B,oBAAoB,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;IAChC,KAAK,EAAE,OAAO;IACd,MAAM,EAAE,OAAO;IACf,MAAM,EAAE,OAAO;CAChB,CAAC","sourcesContent":["/**\n * Ads Types\n *\n * Tipos e interfaces para el servicio de Google Ad Manager.\n * Usa Google Publisher Tags (GPT) para renderizar ads.\n */\n\n// ============================================================================\n// CONFIGURACION\n// ============================================================================\n\n/**\n * Configuracion del servicio de Ads.\n * Pasada a provideValtechAds() en el bootstrap de la aplicacion.\n */\nexport interface ValtechAdsConfig {\n  /** Network ID de Google Ad Manager (ej: '/12345678') */\n  networkId: string;\n\n  /** Habilitar modo debug (muestra info de ads en consola) */\n  debugMode?: boolean;\n\n  /** Activar lazy loading de ads (viewport intersection) */\n  lazyLoad?: boolean;\n\n  /** Configuracion de lazy loading */\n  lazyLoadConfig?: LazyLoadConfig;\n\n  /** Funcion para determinar si el usuario es premium (sin ads) */\n  isPremiumUser?: () => boolean;\n\n  /** Configuracion por defecto para ad slots */\n  defaultSlotConfig?: Partial<AdSlotConfig>;\n\n  /** Targeting global (aplica a todos los slots) */\n  globalTargeting?: Record<string, string | string[]>;\n\n  /** Habilitar personalizacion de ads (requiere consent) */\n  enablePersonalization?: boolean;\n\n  /** Mostrar ads de fallback si no hay consent */\n  showNonPersonalizedAds?: boolean;\n\n  /** Callback cuando se carga un ad */\n  onAdLoaded?: (slotId: string, isEmpty: boolean) => void;\n\n  /** Callback cuando falla un ad */\n  onAdError?: (slotId: string, error: Error) => void;\n\n  /** Refresh automatico de ads (en segundos, 0 = deshabilitado) */\n  autoRefreshInterval?: number;\n\n  /** Rutas donde NO mostrar ads (regex patterns) */\n  excludeRoutes?: string[];\n}\n\n/**\n * Configuracion de lazy loading para GPT.\n */\nexport interface LazyLoadConfig {\n  /** Margen del viewport para cargar (ej: '200px') */\n  rootMargin?: string;\n\n  /** Threshold de interseccion (0-1) */\n  threshold?: number;\n\n  /** Fetch margin para GPT (pixels antes del viewport) */\n  fetchMarginPercent?: number;\n\n  /** Render margin para GPT */\n  renderMarginPercent?: number;\n}\n\n// ============================================================================\n// AD SLOTS\n// ============================================================================\n\n/**\n * Tamanos de ad predefinidos (IAB standard).\n */\nexport type AdSlotSize =\n  | 'leaderboard' // 728x90 - Desktop header\n  | 'billboard' // 970x250 - Large desktop\n  | 'medium-rectangle' // 300x250 - Universal\n  | 'large-rectangle' // 336x280 - Large rectangle\n  | 'half-page' // 300x600 - Sidebar\n  | 'skyscraper' // 160x600 - Sidebar\n  | 'mobile-banner' // 320x50 - Mobile\n  | 'mobile-leaderboard' // 320x100 - Mobile large\n  | 'fluid' // Responsive\n  | 'native' // Native ad format\n  | 'custom'; // Tamano custom\n\n/**\n * Mapeo de tamanos predefinidos a dimensiones.\n */\nexport const AD_SIZE_MAP: Record<AdSlotSize, [number, number] | 'fluid'> = {\n  leaderboard: [728, 90],\n  billboard: [970, 250],\n  'medium-rectangle': [300, 250],\n  'large-rectangle': [336, 280],\n  'half-page': [300, 600],\n  skyscraper: [160, 600],\n  'mobile-banner': [320, 50],\n  'mobile-leaderboard': [320, 100],\n  fluid: 'fluid',\n  native: 'fluid',\n  custom: 'fluid',\n};\n\n/**\n * Configuracion de un ad slot individual.\n */\nexport interface AdSlotConfig {\n  /** ID unico del slot (usado para el div container) */\n  slotId: string;\n\n  /** Ad unit path (ej: '/homepage/top') - se concatena con networkId */\n  adUnitPath: string;\n\n  /** Tamano del ad o tamanos responsivos */\n  size: AdSlotSize | AdSlotSize[] | [number, number] | [number, number][];\n\n  /** Mapping responsivo: [viewport-width, sizes] */\n  sizeMapping?: SizeMapping[];\n\n  /** Targeting especifico para este slot */\n  targeting?: Record<string, string | string[]>;\n\n  /** Colapsar slot si esta vacio */\n  collapseEmpty?: boolean;\n\n  /** Es un native ad */\n  isNative?: boolean;\n\n  /** CSS class adicional */\n  cssClass?: string;\n\n  /** Altura minima mientras carga */\n  minHeight?: string;\n\n  /** Mostrar skeleton mientras carga */\n  showSkeleton?: boolean;\n}\n\n/**\n * Mapping de tamanos responsivos.\n * Define que tamanos de ad mostrar segun el viewport.\n */\nexport interface SizeMapping {\n  /** Ancho minimo del viewport */\n  viewportWidth: number;\n  /** Tamanos de ad para este viewport */\n  sizes: ([number, number] | 'fluid')[];\n}\n\n// ============================================================================\n// ESTADO\n// ============================================================================\n\n/**\n * Estado de un ad slot.\n */\nexport type AdSlotState =\n  | 'idle' // No cargado\n  | 'loading' // Script/ad cargando\n  | 'rendered' // Ad visible\n  | 'empty' // Sin ad disponible\n  | 'hidden' // Usuario premium o sin consent\n  | 'error'; // Error de carga\n\n/**\n * Evento emitido por el servicio de ads.\n */\nexport interface AdEvent {\n  type: 'loaded' | 'empty' | 'error' | 'viewable' | 'clicked';\n  slotId: string;\n  timestamp: Date;\n  isEmpty?: boolean;\n  error?: Error;\n}\n\n// ============================================================================\n// CONSENT\n// ============================================================================\n\n/**\n * Estado de consent para ads (mapeado desde AnalyticsService).\n */\nexport interface AdsConsentState {\n  /** Permite almacenamiento de ads */\n  adStorage: boolean;\n\n  /** Permite personalizacion de ads */\n  adPersonalization: boolean;\n\n  /** Permite datos de usuario para ads */\n  adUserData: boolean;\n}\n\n// ============================================================================\n// GPT (Google Publisher Tag) Types\n// ============================================================================\n\n/** GPT Slot reference */\nexport interface GPTSlot {\n  getSlotElementId(): string;\n  getAdUnitPath(): string;\n  addService(service: unknown): GPTSlot;\n  defineSizeMapping(mapping: unknown): GPTSlot;\n  setTargeting(key: string, value: string | string[]): GPTSlot;\n  setCollapseEmptyDiv(collapse: boolean, collapseBeforeAdFetch?: boolean): GPTSlot;\n}\n\n/** GPT global namespace (window.googletag) */\nexport interface GPTNamespace {\n  cmd: Array<() => void>;\n  defineSlot(adUnitPath: string, size: unknown, divId: string): GPTSlot | null;\n  pubads(): GPTPubadsService;\n  enableServices(): void;\n  display(divId: string): void;\n  destroySlots(slots?: GPTSlot[]): boolean;\n  sizeMapping(): GPTSizeMappingBuilder;\n}\n\n/** GPT Pubads Service */\nexport interface GPTPubadsService {\n  enableSingleRequest(): void;\n  enableLazyLoad(config?: object): void;\n  setTargeting(key: string, value: string | string[]): void;\n  setRequestNonPersonalizedAds(value: 0 | 1): void;\n  addEventListener(event: string, callback: (event: GPTEvent) => void): void;\n  refresh(slots?: GPTSlot[]): void;\n  collapseEmptyDivs(collapseBeforeAdFetch?: boolean): void;\n}\n\n/** GPT Size Mapping Builder */\nexport interface GPTSizeMappingBuilder {\n  addSize(viewport: [number, number], sizes: unknown): GPTSizeMappingBuilder;\n  build(): unknown;\n}\n\n/** GPT Event */\nexport interface GPTEvent {\n  slot: GPTSlot;\n  isEmpty?: boolean;\n}\n\n// Extend Window interface for googletag\ndeclare global {\n  interface Window {\n    googletag?: GPTNamespace;\n  }\n}\n"]}
23
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../../src/lib/services/ads/types.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAqFH;;GAEG;AACH,MAAM,CAAC,MAAM,WAAW,GAAmD;IACzE,WAAW,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC;IACtB,SAAS,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;IACrB,kBAAkB,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;IAC9B,iBAAiB,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;IAC7B,WAAW,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;IACvB,UAAU,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;IACtB,eAAe,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC;IAC1B,oBAAoB,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;IAChC,KAAK,EAAE,OAAO;IACd,MAAM,EAAE,OAAO;IACf,MAAM,EAAE,OAAO;CAChB,CAAC","sourcesContent":["/**\n * Ads Types\n *\n * Tipos e interfaces para el servicio de Google Ad Manager.\n * Usa Google Publisher Tags (GPT) para renderizar ads.\n */\n\n// ============================================================================\n// CONFIGURACION\n// ============================================================================\n\n/**\n * Configuracion del servicio de Ads.\n * Pasada a provideValtechAds() en el bootstrap de la aplicacion.\n */\nexport interface ValtechAdsConfig {\n  /** Network ID de Google Ad Manager (ej: '/12345678') */\n  networkId: string;\n\n  /** Habilitar modo debug (muestra info de ads en consola) */\n  debugMode?: boolean;\n\n  /** Activar lazy loading de ads (viewport intersection) */\n  lazyLoad?: boolean;\n\n  /** Configuracion de lazy loading */\n  lazyLoadConfig?: LazyLoadConfig;\n\n  /** Configuracion por defecto para ad slots */\n  defaultSlotConfig?: Partial<AdSlotConfig>;\n\n  /** Targeting global (aplica a todos los slots) */\n  globalTargeting?: Record<string, string | string[]>;\n\n  /** Habilitar personalizacion de ads (requiere consent) */\n  enablePersonalization?: boolean;\n\n  /** Mostrar ads de fallback si no hay consent */\n  showNonPersonalizedAds?: boolean;\n\n  /** Callback cuando se carga un ad */\n  onAdLoaded?: (slotId: string, isEmpty: boolean) => void;\n\n  /** Callback cuando falla un ad */\n  onAdError?: (slotId: string, error: Error) => void;\n\n  /** Refresh automatico de ads (en segundos, 0 = deshabilitado) */\n  autoRefreshInterval?: number;\n\n  /** Rutas donde NO mostrar ads (regex patterns) */\n  excludeRoutes?: string[];\n}\n\n/**\n * Configuracion de lazy loading para GPT.\n */\nexport interface LazyLoadConfig {\n  /** Margen del viewport para cargar (ej: '200px') */\n  rootMargin?: string;\n\n  /** Threshold de interseccion (0-1) */\n  threshold?: number;\n\n  /** Fetch margin para GPT (pixels antes del viewport) */\n  fetchMarginPercent?: number;\n\n  /** Render margin para GPT */\n  renderMarginPercent?: number;\n}\n\n// ============================================================================\n// AD SLOTS\n// ============================================================================\n\n/**\n * Tamanos de ad predefinidos (IAB standard).\n */\nexport type AdSlotSize =\n  | 'leaderboard' // 728x90 - Desktop header\n  | 'billboard' // 970x250 - Large desktop\n  | 'medium-rectangle' // 300x250 - Universal\n  | 'large-rectangle' // 336x280 - Large rectangle\n  | 'half-page' // 300x600 - Sidebar\n  | 'skyscraper' // 160x600 - Sidebar\n  | 'mobile-banner' // 320x50 - Mobile\n  | 'mobile-leaderboard' // 320x100 - Mobile large\n  | 'fluid' // Responsive\n  | 'native' // Native ad format\n  | 'custom'; // Tamano custom\n\n/**\n * Mapeo de tamanos predefinidos a dimensiones.\n */\nexport const AD_SIZE_MAP: Record<AdSlotSize, [number, number] | 'fluid'> = {\n  leaderboard: [728, 90],\n  billboard: [970, 250],\n  'medium-rectangle': [300, 250],\n  'large-rectangle': [336, 280],\n  'half-page': [300, 600],\n  skyscraper: [160, 600],\n  'mobile-banner': [320, 50],\n  'mobile-leaderboard': [320, 100],\n  fluid: 'fluid',\n  native: 'fluid',\n  custom: 'fluid',\n};\n\n/**\n * Configuracion de un ad slot individual.\n */\nexport interface AdSlotConfig {\n  /** ID unico del slot (usado para el div container) */\n  slotId: string;\n\n  /** Ad unit path (ej: '/homepage/top') - se concatena con networkId */\n  adUnitPath: string;\n\n  /** Tamano del ad o tamanos responsivos */\n  size: AdSlotSize | AdSlotSize[] | [number, number] | [number, number][];\n\n  /** Mapping responsivo: [viewport-width, sizes] */\n  sizeMapping?: SizeMapping[];\n\n  /** Targeting especifico para este slot */\n  targeting?: Record<string, string | string[]>;\n\n  /** Colapsar slot si esta vacio */\n  collapseEmpty?: boolean;\n\n  /** Es un native ad */\n  isNative?: boolean;\n\n  /** CSS class adicional */\n  cssClass?: string;\n\n  /** Altura minima mientras carga */\n  minHeight?: string;\n\n  /** Mostrar skeleton mientras carga */\n  showSkeleton?: boolean;\n}\n\n/**\n * Mapping de tamanos responsivos.\n * Define que tamanos de ad mostrar segun el viewport.\n */\nexport interface SizeMapping {\n  /** Ancho minimo del viewport */\n  viewportWidth: number;\n  /** Tamanos de ad para este viewport */\n  sizes: ([number, number] | 'fluid')[];\n}\n\n// ============================================================================\n// ESTADO\n// ============================================================================\n\n/**\n * Estado de un ad slot.\n */\nexport type AdSlotState =\n  | 'idle' // No cargado\n  | 'loading' // Script/ad cargando\n  | 'rendered' // Ad visible\n  | 'empty' // Sin ad disponible\n  | 'hidden' // Usuario premium o sin consent\n  | 'error'; // Error de carga\n\n/**\n * Evento emitido por el servicio de ads.\n */\nexport interface AdEvent {\n  type: 'loaded' | 'empty' | 'error' | 'viewable' | 'clicked';\n  slotId: string;\n  timestamp: Date;\n  isEmpty?: boolean;\n  error?: Error;\n}\n\n// ============================================================================\n// CONSENT\n// ============================================================================\n\n/**\n * Estado de consent para ads (mapeado desde AnalyticsService).\n */\nexport interface AdsConsentState {\n  /** Permite almacenamiento de ads */\n  adStorage: boolean;\n\n  /** Permite personalizacion de ads */\n  adPersonalization: boolean;\n\n  /** Permite datos de usuario para ads */\n  adUserData: boolean;\n}\n\n// ============================================================================\n// GPT (Google Publisher Tag) Types\n// ============================================================================\n\n/** GPT Slot reference */\nexport interface GPTSlot {\n  getSlotElementId(): string;\n  getAdUnitPath(): string;\n  addService(service: unknown): GPTSlot;\n  defineSizeMapping(mapping: unknown): GPTSlot;\n  setTargeting(key: string, value: string | string[]): GPTSlot;\n  setCollapseEmptyDiv(collapse: boolean, collapseBeforeAdFetch?: boolean): GPTSlot;\n}\n\n/** GPT global namespace (window.googletag) */\nexport interface GPTNamespace {\n  cmd: Array<() => void>;\n  defineSlot(adUnitPath: string, size: unknown, divId: string): GPTSlot | null;\n  pubads(): GPTPubadsService;\n  enableServices(): void;\n  display(divId: string): void;\n  destroySlots(slots?: GPTSlot[]): boolean;\n  sizeMapping(): GPTSizeMappingBuilder;\n}\n\n/** GPT Pubads Service */\nexport interface GPTPubadsService {\n  enableSingleRequest(): void;\n  enableLazyLoad(config?: object): void;\n  setTargeting(key: string, value: string | string[]): void;\n  setRequestNonPersonalizedAds(value: 0 | 1): void;\n  addEventListener(event: string, callback: (event: GPTEvent) => void): void;\n  refresh(slots?: GPTSlot[]): void;\n  collapseEmptyDivs(collapseBeforeAdFetch?: boolean): void;\n}\n\n/** GPT Size Mapping Builder */\nexport interface GPTSizeMappingBuilder {\n  addSize(viewport: [number, number], sizes: unknown): GPTSizeMappingBuilder;\n  build(): unknown;\n}\n\n/** GPT Event */\nexport interface GPTEvent {\n  slot: GPTSlot;\n  isEmpty?: boolean;\n}\n\n// Extend Window interface for googletag\ndeclare global {\n  interface Window {\n    googletag?: GPTNamespace;\n  }\n}\n"]}