valtech-components 2.0.506 → 2.0.508
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/components/atoms/qr-code/qr-code.component.mjs +6 -4
- package/esm2022/lib/components/molecules/ad-slot/ad-slot.component.mjs +96 -39
- package/esm2022/lib/components/molecules/code-display/code-display.component.mjs +4 -2
- package/esm2022/lib/components/molecules/date-input/date-input.component.mjs +18 -7
- package/esm2022/lib/components/molecules/date-range-input/date-range-input.component.mjs +41 -16
- package/esm2022/lib/components/molecules/file-input/file-input.component.mjs +17 -6
- package/esm2022/lib/components/molecules/language-selector/language-selector.component.mjs +5 -5
- package/esm2022/lib/components/molecules/multi-select-search/multi-select-search.component.mjs +53 -27
- package/esm2022/lib/components/molecules/participant-card/participant-card.component.mjs +28 -10
- package/esm2022/lib/components/molecules/popover-selector/popover-selector.component.mjs +8 -6
- package/esm2022/lib/components/molecules/searchbar/searchbar.component.mjs +21 -7
- package/esm2022/lib/components/molecules/select-input/select-input.component.mjs +11 -7
- package/esm2022/lib/components/molecules/select-search/select-search.component.mjs +21 -7
- package/esm2022/lib/components/organisms/data-table/data-table.component.mjs +52 -17
- package/esm2022/lib/services/i18n/config.mjs +64 -4
- package/esm2022/lib/services/i18n/default-content.mjs +203 -0
- package/esm2022/lib/services/i18n/index.mjs +3 -1
- package/esm2022/lib/services/i18n/types.mjs +2 -1
- package/fesm2022/valtech-components.mjs +1180 -708
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/atoms/qr-code/qr-code.component.d.ts +1 -0
- package/lib/components/molecules/ad-slot/ad-slot.component.d.ts +5 -0
- package/lib/components/molecules/code-display/code-display.component.d.ts +2 -2
- package/lib/components/molecules/date-input/date-input.component.d.ts +5 -0
- package/lib/components/molecules/date-range-input/date-range-input.component.d.ts +11 -0
- package/lib/components/molecules/file-input/file-input.component.d.ts +3 -0
- package/lib/components/molecules/multi-select-search/multi-select-search.component.d.ts +13 -0
- package/lib/components/molecules/participant-card/participant-card.component.d.ts +9 -0
- package/lib/components/molecules/popover-selector/popover-selector.component.d.ts +1 -0
- package/lib/components/molecules/searchbar/searchbar.component.d.ts +14 -1
- package/lib/components/molecules/select-input/select-input.component.d.ts +1 -0
- package/lib/components/molecules/select-search/select-search.component.d.ts +7 -0
- package/lib/components/organisms/article/article.component.d.ts +1 -1
- package/lib/components/organisms/data-table/data-table.component.d.ts +17 -2
- package/lib/services/i18n/default-content.d.ts +30 -0
- package/lib/services/i18n/index.d.ts +1 -0
- package/lib/services/i18n/types.d.ts +35 -1
- package/package.json +1 -1
|
@@ -4,6 +4,7 @@ import { IonButton, IonIcon, IonSpinner } from '@ionic/angular/standalone';
|
|
|
4
4
|
import { addIcons } from 'ionicons';
|
|
5
5
|
import { downloadOutline, copyOutline, shareOutline } from 'ionicons/icons';
|
|
6
6
|
import { QrGeneratorService } from '../../../services/qr-generator/qr-generator.service';
|
|
7
|
+
import { I18nService } from '../../../services/i18n';
|
|
7
8
|
import * as i0 from "@angular/core";
|
|
8
9
|
addIcons({ downloadOutline, copyOutline, shareOutline });
|
|
9
10
|
/**
|
|
@@ -49,6 +50,7 @@ export class QrCodeComponent {
|
|
|
49
50
|
this.canShare = false;
|
|
50
51
|
this.canCopy = false;
|
|
51
52
|
this.qrService = inject(QrGeneratorService);
|
|
53
|
+
this.i18n = inject(I18nService);
|
|
52
54
|
}
|
|
53
55
|
ngOnInit() {
|
|
54
56
|
this.canShare = this.qrService.canShare();
|
|
@@ -117,13 +119,13 @@ export class QrCodeComponent {
|
|
|
117
119
|
return !!(this.props.showDownload || this.props.showCopy || this.props.showShare);
|
|
118
120
|
}
|
|
119
121
|
getDownloadLabel() {
|
|
120
|
-
return this.props.downloadLabel || '
|
|
122
|
+
return this.props.downloadLabel || this.i18n.t('download');
|
|
121
123
|
}
|
|
122
124
|
getCopyLabel() {
|
|
123
|
-
return this.props.copyLabel || '
|
|
125
|
+
return this.props.copyLabel || this.i18n.t('copy');
|
|
124
126
|
}
|
|
125
127
|
getShareLabel() {
|
|
126
|
-
return this.props.shareLabel || '
|
|
128
|
+
return this.props.shareLabel || this.i18n.t('share');
|
|
127
129
|
}
|
|
128
130
|
async onDownload() {
|
|
129
131
|
try {
|
|
@@ -389,4 +391,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
389
391
|
}], imageError: [{
|
|
390
392
|
type: Output
|
|
391
393
|
}] } });
|
|
392
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"qr-code.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/atoms/qr-code/qr-code.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit, inject } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { IonButton, IonIcon, IonSpinner } from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport { downloadOutline, copyOutline, shareOutline } from 'ionicons/icons';\n\nimport { QrCodeMetadata, QrCodeActionEvent } from './types';\nimport { QrGeneratorService } from '../../../services/qr-generator/qr-generator.service';\n\naddIcons({ downloadOutline, copyOutline, shareOutline });\n\n@Component({\n  selector: 'val-qr-code',\n  standalone: true,\n  imports: [CommonModule, IonButton, IonIcon, IonSpinner],\n  template: `\n    <div\n      class=\"qr-code-container\"\n      [class]=\"getContainerClasses()\"\n      [style.--border-radius]=\"getBorderRadius()\"\n      [style.--padding]=\"getPadding()\"\n      [style.--container-bg]=\"getContainerBackground()\"\n      [style.--border-color]=\"props.borderColor || 'var(--ion-color-light-shade)'\"\n      [style.--shadow]=\"getShadow()\"\n      [style.--glow-color]=\"props.glowColor || '#8b5cf6'\"\n      [style.--qr-border-radius]=\"getQrBorderRadius()\"\n    >\n      @if (props.loading) {\n        <div class=\"loading-overlay\">\n          <ion-spinner name=\"crescent\"></ion-spinner>\n        </div>\n      }\n\n      @if (props.label && props.label.position === 'top') {\n        <div class=\"qr-label qr-label--top\" [style.--label-color]=\"props.label.color || 'inherit'\" [style.--label-size]=\"getLabelFontSize()\">\n          @if (props.label.icon && props.label.iconPosition !== 'end') {\n            <ion-icon [name]=\"props.label.icon\"></ion-icon>\n          }\n          <span [class]=\"'font-weight--' + (props.label.fontWeight || 'medium')\">{{ props.label.text }}</span>\n          @if (props.label.icon && props.label.iconPosition === 'end') {\n            <ion-icon [name]=\"props.label.icon\"></ion-icon>\n          }\n        </div>\n      }\n\n      <div class=\"qr-image-wrapper\" [style.--size]=\"getDisplaySize()\">\n        <img\n          [src]=\"props.qr.dataUrl\"\n          [alt]=\"props.alt || 'QR Code'\"\n          class=\"qr-image\"\n          (load)=\"onImageLoad()\"\n          (error)=\"onImageError($event)\"\n        />\n      </div>\n\n      @if (props.label && props.label.position !== 'top') {\n        <div class=\"qr-label qr-label--bottom\" [style.--label-color]=\"props.label.color || 'inherit'\" [style.--label-size]=\"getLabelFontSize()\">\n          @if (props.label.icon && props.label.iconPosition !== 'end') {\n            <ion-icon [name]=\"props.label.icon\"></ion-icon>\n          }\n          <span [class]=\"'font-weight--' + (props.label.fontWeight || 'medium')\">{{ props.label.text }}</span>\n          @if (props.label.icon && props.label.iconPosition === 'end') {\n            <ion-icon [name]=\"props.label.icon\"></ion-icon>\n          }\n        </div>\n      }\n\n      @if (hasActions()) {\n        <div class=\"qr-actions\">\n          @if (props.showDownload) {\n            <ion-button\n              fill=\"clear\"\n              size=\"small\"\n              (click)=\"onDownload()\"\n              [title]=\"getDownloadLabel()\"\n            >\n              <ion-icon name=\"download-outline\" slot=\"icon-only\"></ion-icon>\n            </ion-button>\n          }\n\n          @if (props.showCopy && canCopy) {\n            <ion-button\n              fill=\"clear\"\n              size=\"small\"\n              (click)=\"onCopy()\"\n              [title]=\"getCopyLabel()\"\n            >\n              <ion-icon name=\"copy-outline\" slot=\"icon-only\"></ion-icon>\n            </ion-button>\n          }\n\n          @if (props.showShare && canShare) {\n            <ion-button\n              fill=\"clear\"\n              size=\"small\"\n              (click)=\"onShare()\"\n              [title]=\"getShareLabel()\"\n            >\n              <ion-icon name=\"share-outline\" slot=\"icon-only\"></ion-icon>\n            </ion-button>\n          }\n        </div>\n      }\n    </div>\n  `,\n  styleUrls: ['./qr-code.component.scss'],\n})\n/**\n * val-qr-code\n *\n * A component to display QR codes generated by QrGeneratorService.\n * Provides optional action buttons for download, copy, and share.\n *\n * @example Basic usage\n * ```typescript\n * qr = await this.qrService.generate({ data: 'https://example.com' });\n * ```\n * ```html\n * <val-qr-code [props]=\"{ qr: qr }\"></val-qr-code>\n * ```\n *\n * @example With actions\n * ```html\n * <val-qr-code\n *   [props]=\"{\n *     qr: qr,\n *     showDownload: true,\n *     showCopy: true,\n *     showShare: true,\n *     displaySize: 200,\n *     showBorder: true,\n *     borderRadius: 12\n *   }\"\n *   (actionComplete)=\"onAction($event)\"\n * ></val-qr-code>\n * ```\n *\n * @input props: QrCodeMetadata - Configuration for the QR display\n * @output actionComplete - Emits when an action (download/copy/share) completes\n * @output imageLoad - Emits when the QR image loads\n * @output imageError - Emits when the QR image fails to load\n */\nexport class QrCodeComponent implements OnInit {\n  @Input() props: QrCodeMetadata;\n\n  @Output() actionComplete = new EventEmitter<QrCodeActionEvent>();\n  @Output() imageLoad = new EventEmitter<void>();\n  @Output() imageError = new EventEmitter<Error>();\n\n  canShare = false;\n  canCopy = false;\n\n  private qrService = inject(QrGeneratorService);\n\n  ngOnInit(): void {\n    this.canShare = this.qrService.canShare();\n    this.canCopy = this.qrService.canCopyToClipboard();\n  }\n\n  getDisplaySize(): string {\n    if (this.props.displaySize) {\n      return `${this.props.displaySize}px`;\n    }\n    return `${this.props.qr.config.width || 300}px`;\n  }\n\n  getBorderRadius(): string {\n    return this.props.borderRadius ? `${this.props.borderRadius}px` : '0';\n  }\n\n  getPadding(): string {\n    return this.props.padding ? `${this.props.padding}px` : '0';\n  }\n\n  getQrBorderRadius(): string {\n    return this.props.qrBorderRadius ? `${this.props.qrBorderRadius}px` : '0';\n  }\n\n  getContainerClasses(): string {\n    const classes: string[] = [];\n\n    if (this.props.cssClass) classes.push(this.props.cssClass);\n    if (this.props.showBorder) classes.push('with-border');\n    if (this.props.loading) classes.push('loading');\n    if (this.props.theme) classes.push(`theme--${this.props.theme}`);\n    if (this.props.pulseOnHover) classes.push('pulse-on-hover');\n    if (this.props.scaleOnHover) classes.push('scale-on-hover');\n    if (this.props.gradient) classes.push('has-gradient');\n\n    return classes.join(' ');\n  }\n\n  getContainerBackground(): string {\n    if (this.props.gradient) {\n      const { from, to, via, direction = 'to-bottom' } = this.props.gradient;\n      const cssDirection = direction.replace(/-/g, ' ');\n      if (via) {\n        return `linear-gradient(${cssDirection}, ${from}, ${via}, ${to})`;\n      }\n      return `linear-gradient(${cssDirection}, ${from}, ${to})`;\n    }\n    return this.props.containerBackground || 'transparent';\n  }\n\n  getShadow(): string {\n    const shadowMap: Record<string, string> = {\n      none: 'none',\n      sm: '0 1px 2px 0 rgba(0, 0, 0, 0.05)',\n      md: '0 4px 6px -1px rgba(0, 0, 0, 0.1), 0 2px 4px -1px rgba(0, 0, 0, 0.06)',\n      lg: '0 10px 15px -3px rgba(0, 0, 0, 0.1), 0 4px 6px -2px rgba(0, 0, 0, 0.05)',\n      xl: '0 20px 25px -5px rgba(0, 0, 0, 0.1), 0 10px 10px -5px rgba(0, 0, 0, 0.04)',\n    };\n\n    if (!this.props.shadow) return 'none';\n    return shadowMap[this.props.shadow] || this.props.shadow;\n  }\n\n  getLabelFontSize(): string {\n    return this.props.label?.fontSize ? `${this.props.label.fontSize}px` : '14px';\n  }\n\n  hasActions(): boolean {\n    return !!(this.props.showDownload || this.props.showCopy || this.props.showShare);\n  }\n\n  getDownloadLabel(): string {\n    return this.props.downloadLabel || 'Descargar';\n  }\n\n  getCopyLabel(): string {\n    return this.props.copyLabel || 'Copiar';\n  }\n\n  getShareLabel(): string {\n    return this.props.shareLabel || 'Compartir';\n  }\n\n  async onDownload(): Promise<void> {\n    try {\n      const options = this.props.downloadFilename\n        ? { filename: this.props.downloadFilename }\n        : undefined;\n      await this.qrService.download(this.props.qr, options);\n      this.actionComplete.emit({\n        action: 'download',\n        success: true,\n        qr: this.props.qr,\n      });\n    } catch (error) {\n      this.actionComplete.emit({\n        action: 'download',\n        success: false,\n        qr: this.props.qr,\n        error: error as Error,\n      });\n    }\n  }\n\n  async onCopy(): Promise<void> {\n    try {\n      const success = await this.qrService.copyToClipboard(this.props.qr);\n      this.actionComplete.emit({\n        action: 'copy',\n        success,\n        qr: this.props.qr,\n      });\n    } catch (error) {\n      this.actionComplete.emit({\n        action: 'copy',\n        success: false,\n        qr: this.props.qr,\n        error: error as Error,\n      });\n    }\n  }\n\n  async onShare(): Promise<void> {\n    try {\n      const options: { title?: string; text?: string; filename?: string } = {};\n      if (this.props.shareTitle) options.title = this.props.shareTitle;\n      if (this.props.shareText) options.text = this.props.shareText;\n      if (this.props.shareFilename) options.filename = this.props.shareFilename;\n\n      const success = await this.qrService.share(\n        this.props.qr,\n        Object.keys(options).length > 0 ? options : undefined\n      );\n      this.actionComplete.emit({\n        action: 'share',\n        success,\n        qr: this.props.qr,\n      });\n    } catch (error) {\n      this.actionComplete.emit({\n        action: 'share',\n        success: false,\n        qr: this.props.qr,\n        error: error as Error,\n      });\n    }\n  }\n\n  onImageLoad(): void {\n    this.imageLoad.emit();\n  }\n\n  onImageError(_event: Event): void {\n    const error = new Error('Failed to load QR code image');\n    this.imageError.emit(error);\n  }\n}\n"]}
|
|
394
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"qr-code.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/atoms/qr-code/qr-code.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit, inject } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { IonButton, IonIcon, IonSpinner } from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport { downloadOutline, copyOutline, shareOutline } from 'ionicons/icons';\n\nimport { QrCodeMetadata, QrCodeActionEvent } from './types';\nimport { QrGeneratorService } from '../../../services/qr-generator/qr-generator.service';\nimport { I18nService } from '../../../services/i18n';\n\naddIcons({ downloadOutline, copyOutline, shareOutline });\n\n@Component({\n  selector: 'val-qr-code',\n  standalone: true,\n  imports: [CommonModule, IonButton, IonIcon, IonSpinner],\n  template: `\n    <div\n      class=\"qr-code-container\"\n      [class]=\"getContainerClasses()\"\n      [style.--border-radius]=\"getBorderRadius()\"\n      [style.--padding]=\"getPadding()\"\n      [style.--container-bg]=\"getContainerBackground()\"\n      [style.--border-color]=\"props.borderColor || 'var(--ion-color-light-shade)'\"\n      [style.--shadow]=\"getShadow()\"\n      [style.--glow-color]=\"props.glowColor || '#8b5cf6'\"\n      [style.--qr-border-radius]=\"getQrBorderRadius()\"\n    >\n      @if (props.loading) {\n        <div class=\"loading-overlay\">\n          <ion-spinner name=\"crescent\"></ion-spinner>\n        </div>\n      }\n\n      @if (props.label && props.label.position === 'top') {\n        <div class=\"qr-label qr-label--top\" [style.--label-color]=\"props.label.color || 'inherit'\" [style.--label-size]=\"getLabelFontSize()\">\n          @if (props.label.icon && props.label.iconPosition !== 'end') {\n            <ion-icon [name]=\"props.label.icon\"></ion-icon>\n          }\n          <span [class]=\"'font-weight--' + (props.label.fontWeight || 'medium')\">{{ props.label.text }}</span>\n          @if (props.label.icon && props.label.iconPosition === 'end') {\n            <ion-icon [name]=\"props.label.icon\"></ion-icon>\n          }\n        </div>\n      }\n\n      <div class=\"qr-image-wrapper\" [style.--size]=\"getDisplaySize()\">\n        <img\n          [src]=\"props.qr.dataUrl\"\n          [alt]=\"props.alt || 'QR Code'\"\n          class=\"qr-image\"\n          (load)=\"onImageLoad()\"\n          (error)=\"onImageError($event)\"\n        />\n      </div>\n\n      @if (props.label && props.label.position !== 'top') {\n        <div class=\"qr-label qr-label--bottom\" [style.--label-color]=\"props.label.color || 'inherit'\" [style.--label-size]=\"getLabelFontSize()\">\n          @if (props.label.icon && props.label.iconPosition !== 'end') {\n            <ion-icon [name]=\"props.label.icon\"></ion-icon>\n          }\n          <span [class]=\"'font-weight--' + (props.label.fontWeight || 'medium')\">{{ props.label.text }}</span>\n          @if (props.label.icon && props.label.iconPosition === 'end') {\n            <ion-icon [name]=\"props.label.icon\"></ion-icon>\n          }\n        </div>\n      }\n\n      @if (hasActions()) {\n        <div class=\"qr-actions\">\n          @if (props.showDownload) {\n            <ion-button\n              fill=\"clear\"\n              size=\"small\"\n              (click)=\"onDownload()\"\n              [title]=\"getDownloadLabel()\"\n            >\n              <ion-icon name=\"download-outline\" slot=\"icon-only\"></ion-icon>\n            </ion-button>\n          }\n\n          @if (props.showCopy && canCopy) {\n            <ion-button\n              fill=\"clear\"\n              size=\"small\"\n              (click)=\"onCopy()\"\n              [title]=\"getCopyLabel()\"\n            >\n              <ion-icon name=\"copy-outline\" slot=\"icon-only\"></ion-icon>\n            </ion-button>\n          }\n\n          @if (props.showShare && canShare) {\n            <ion-button\n              fill=\"clear\"\n              size=\"small\"\n              (click)=\"onShare()\"\n              [title]=\"getShareLabel()\"\n            >\n              <ion-icon name=\"share-outline\" slot=\"icon-only\"></ion-icon>\n            </ion-button>\n          }\n        </div>\n      }\n    </div>\n  `,\n  styleUrls: ['./qr-code.component.scss'],\n})\n/**\n * val-qr-code\n *\n * A component to display QR codes generated by QrGeneratorService.\n * Provides optional action buttons for download, copy, and share.\n *\n * @example Basic usage\n * ```typescript\n * qr = await this.qrService.generate({ data: 'https://example.com' });\n * ```\n * ```html\n * <val-qr-code [props]=\"{ qr: qr }\"></val-qr-code>\n * ```\n *\n * @example With actions\n * ```html\n * <val-qr-code\n *   [props]=\"{\n *     qr: qr,\n *     showDownload: true,\n *     showCopy: true,\n *     showShare: true,\n *     displaySize: 200,\n *     showBorder: true,\n *     borderRadius: 12\n *   }\"\n *   (actionComplete)=\"onAction($event)\"\n * ></val-qr-code>\n * ```\n *\n * @input props: QrCodeMetadata - Configuration for the QR display\n * @output actionComplete - Emits when an action (download/copy/share) completes\n * @output imageLoad - Emits when the QR image loads\n * @output imageError - Emits when the QR image fails to load\n */\nexport class QrCodeComponent implements OnInit {\n  @Input() props: QrCodeMetadata;\n\n  @Output() actionComplete = new EventEmitter<QrCodeActionEvent>();\n  @Output() imageLoad = new EventEmitter<void>();\n  @Output() imageError = new EventEmitter<Error>();\n\n  canShare = false;\n  canCopy = false;\n\n  private qrService = inject(QrGeneratorService);\n  private i18n = inject(I18nService);\n\n  ngOnInit(): void {\n    this.canShare = this.qrService.canShare();\n    this.canCopy = this.qrService.canCopyToClipboard();\n  }\n\n  getDisplaySize(): string {\n    if (this.props.displaySize) {\n      return `${this.props.displaySize}px`;\n    }\n    return `${this.props.qr.config.width || 300}px`;\n  }\n\n  getBorderRadius(): string {\n    return this.props.borderRadius ? `${this.props.borderRadius}px` : '0';\n  }\n\n  getPadding(): string {\n    return this.props.padding ? `${this.props.padding}px` : '0';\n  }\n\n  getQrBorderRadius(): string {\n    return this.props.qrBorderRadius ? `${this.props.qrBorderRadius}px` : '0';\n  }\n\n  getContainerClasses(): string {\n    const classes: string[] = [];\n\n    if (this.props.cssClass) classes.push(this.props.cssClass);\n    if (this.props.showBorder) classes.push('with-border');\n    if (this.props.loading) classes.push('loading');\n    if (this.props.theme) classes.push(`theme--${this.props.theme}`);\n    if (this.props.pulseOnHover) classes.push('pulse-on-hover');\n    if (this.props.scaleOnHover) classes.push('scale-on-hover');\n    if (this.props.gradient) classes.push('has-gradient');\n\n    return classes.join(' ');\n  }\n\n  getContainerBackground(): string {\n    if (this.props.gradient) {\n      const { from, to, via, direction = 'to-bottom' } = this.props.gradient;\n      const cssDirection = direction.replace(/-/g, ' ');\n      if (via) {\n        return `linear-gradient(${cssDirection}, ${from}, ${via}, ${to})`;\n      }\n      return `linear-gradient(${cssDirection}, ${from}, ${to})`;\n    }\n    return this.props.containerBackground || 'transparent';\n  }\n\n  getShadow(): string {\n    const shadowMap: Record<string, string> = {\n      none: 'none',\n      sm: '0 1px 2px 0 rgba(0, 0, 0, 0.05)',\n      md: '0 4px 6px -1px rgba(0, 0, 0, 0.1), 0 2px 4px -1px rgba(0, 0, 0, 0.06)',\n      lg: '0 10px 15px -3px rgba(0, 0, 0, 0.1), 0 4px 6px -2px rgba(0, 0, 0, 0.05)',\n      xl: '0 20px 25px -5px rgba(0, 0, 0, 0.1), 0 10px 10px -5px rgba(0, 0, 0, 0.04)',\n    };\n\n    if (!this.props.shadow) return 'none';\n    return shadowMap[this.props.shadow] || this.props.shadow;\n  }\n\n  getLabelFontSize(): string {\n    return this.props.label?.fontSize ? `${this.props.label.fontSize}px` : '14px';\n  }\n\n  hasActions(): boolean {\n    return !!(this.props.showDownload || this.props.showCopy || this.props.showShare);\n  }\n\n  getDownloadLabel(): string {\n    return this.props.downloadLabel || this.i18n.t('download');\n  }\n\n  getCopyLabel(): string {\n    return this.props.copyLabel || this.i18n.t('copy');\n  }\n\n  getShareLabel(): string {\n    return this.props.shareLabel || this.i18n.t('share');\n  }\n\n  async onDownload(): Promise<void> {\n    try {\n      const options = this.props.downloadFilename\n        ? { filename: this.props.downloadFilename }\n        : undefined;\n      await this.qrService.download(this.props.qr, options);\n      this.actionComplete.emit({\n        action: 'download',\n        success: true,\n        qr: this.props.qr,\n      });\n    } catch (error) {\n      this.actionComplete.emit({\n        action: 'download',\n        success: false,\n        qr: this.props.qr,\n        error: error as Error,\n      });\n    }\n  }\n\n  async onCopy(): Promise<void> {\n    try {\n      const success = await this.qrService.copyToClipboard(this.props.qr);\n      this.actionComplete.emit({\n        action: 'copy',\n        success,\n        qr: this.props.qr,\n      });\n    } catch (error) {\n      this.actionComplete.emit({\n        action: 'copy',\n        success: false,\n        qr: this.props.qr,\n        error: error as Error,\n      });\n    }\n  }\n\n  async onShare(): Promise<void> {\n    try {\n      const options: { title?: string; text?: string; filename?: string } = {};\n      if (this.props.shareTitle) options.title = this.props.shareTitle;\n      if (this.props.shareText) options.text = this.props.shareText;\n      if (this.props.shareFilename) options.filename = this.props.shareFilename;\n\n      const success = await this.qrService.share(\n        this.props.qr,\n        Object.keys(options).length > 0 ? options : undefined\n      );\n      this.actionComplete.emit({\n        action: 'share',\n        success,\n        qr: this.props.qr,\n      });\n    } catch (error) {\n      this.actionComplete.emit({\n        action: 'share',\n        success: false,\n        qr: this.props.qr,\n        error: error as Error,\n      });\n    }\n  }\n\n  onImageLoad(): void {\n    this.imageLoad.emit();\n  }\n\n  onImageError(_event: Event): void {\n    const error = new Error('Failed to load QR code image');\n    this.imageError.emit(error);\n  }\n}\n"]}
|
|
@@ -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
|
-
<!--
|
|
132
|
-
@if (
|
|
133
|
-
<div
|
|
134
|
-
class="val-ad-
|
|
135
|
-
|
|
136
|
-
|
|
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
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
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
|
-
<!--
|
|
174
|
-
@if (
|
|
175
|
-
<div
|
|
176
|
-
class="val-ad-
|
|
177
|
-
|
|
178
|
-
|
|
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
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
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"]}
|
|
@@ -4,6 +4,7 @@ import { Clipboard } from '@capacitor/clipboard';
|
|
|
4
4
|
import { ToastController } from '@ionic/angular';
|
|
5
5
|
import { IonButton, IonIcon } from '@ionic/angular/standalone';
|
|
6
6
|
import * as Prism from 'prismjs';
|
|
7
|
+
import { I18nService } from '../../../services/i18n';
|
|
7
8
|
import 'prismjs/components/prism-css';
|
|
8
9
|
import 'prismjs/components/prism-javascript';
|
|
9
10
|
import 'prismjs/components/prism-markup';
|
|
@@ -14,6 +15,7 @@ export class CodeDisplayComponent {
|
|
|
14
15
|
constructor(cdr) {
|
|
15
16
|
this.cdr = cdr;
|
|
16
17
|
this.toast = inject(ToastController);
|
|
18
|
+
this.i18n = inject(I18nService);
|
|
17
19
|
this.selectedTab = 0;
|
|
18
20
|
}
|
|
19
21
|
ngOnChanges(changes) {
|
|
@@ -41,7 +43,7 @@ export class CodeDisplayComponent {
|
|
|
41
43
|
try {
|
|
42
44
|
const code = this.props.tabs.length > 0 ? this.props.tabs[this.selectedTab]?.code : this.props.code;
|
|
43
45
|
await Clipboard.write({ string: code || '' });
|
|
44
|
-
this.presentToast('
|
|
46
|
+
this.presentToast(this.i18n.t('copiedToClipboard'));
|
|
45
47
|
}
|
|
46
48
|
catch (error) {
|
|
47
49
|
console.error('Error al copiar al portapapeles:', error);
|
|
@@ -108,4 +110,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
108
110
|
}], props: [{
|
|
109
111
|
type: Input
|
|
110
112
|
}] } });
|
|
111
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
113
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"code-display.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/code-display/code-display.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  inject,\n  Input,\n  OnChanges,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { Clipboard } from '@capacitor/clipboard';\nimport { ToastController } from '@ionic/angular';\nimport { IonButton, IonIcon } from '@ionic/angular/standalone';\nimport * as Prism from 'prismjs';\nimport { I18nService } from '../../../services/i18n';\n\nimport 'prismjs/components/prism-css';\nimport 'prismjs/components/prism-javascript';\nimport 'prismjs/components/prism-markup';\nimport 'prismjs/components/prism-typescript';\nimport { CodeDisplayMetadata } from './types';\n\n@Component({\n  selector: 'val-code-display',\n  standalone: true,\n  imports: [CommonModule, IonIcon, IonButton],\n  template: `\n    <div class=\"code-viewer-container\">\n      <div class=\"tab-bar\">\n        <div\n          class=\"tab\"\n          *ngFor=\"let tab of props.tabs; let i = index\"\n          [class.active]=\"i === selectedTab\"\n          (click)=\"selectTab(i)\"\n        >\n          <span class=\"tab-label\">{{ tab.label }}</span>\n        </div>\n      </div>\n      <div class=\"code-content\">\n        <!-- <span class=\"language-badge\">{{ props.tabs[selectedTab]?.language || props.language }}</span> -->\n        <ion-button fill=\"clear\" class=\"copy-button\" (click)=\"copyToClipboard()\">\n          <ion-icon name=\"copy-outline\"></ion-icon>\n        </ion-button>\n        <pre><code [class]=\"'language-' + (props.tabs[selectedTab]?.language || props.language)\" #codeElement></code></pre>\n      </div>\n    </div>\n  `,\n\n  styleUrl: './code-display.component.scss',\n})\nexport class CodeDisplayComponent implements AfterViewInit, OnChanges {\n  @ViewChild('codeElement') codeElement!: ElementRef<HTMLElement>;\n\n  @Input() props: CodeDisplayMetadata;\n\n  private toast = inject(ToastController);\n  private i18n = inject(I18nService);\n  selectedTab: number = 0;\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['code'] || changes['language'] || changes['tabs']) {\n      this.cdr.detectChanges();\n      setTimeout(() => this.highlightCode());\n    }\n  }\n\n  ngAfterViewInit() {\n    setTimeout(() => this.highlightCode());\n  }\n\n  selectTab(i: number) {\n    this.selectedTab = i;\n    this.cdr.detectChanges();\n    setTimeout(() => this.highlightCode());\n  }\n\n  private highlightCode() {\n    if (this.codeElement) {\n      const code = this.props.tabs.length > 0 ? this.props.tabs[this.selectedTab]?.code : this.props.code;\n      this.codeElement.nativeElement.textContent = code || '';\n      Prism.highlightElement(this.codeElement.nativeElement);\n    }\n  }\n\n  async copyToClipboard() {\n    try {\n      const code = this.props.tabs.length > 0 ? this.props.tabs[this.selectedTab]?.code : this.props.code;\n      await Clipboard.write({ string: code || '' });\n      this.presentToast(this.i18n.t('copiedToClipboard'));\n    } catch (error) {\n      console.error('Error al copiar al portapapeles:', error);\n    }\n  }\n\n  async presentToast(message: string) {\n    const toast = await this.toast.create({\n      message: message,\n      duration: 2000,\n      position: 'bottom',\n      color: 'dark',\n    });\n    toast.present();\n  }\n}\n"]}
|
|
@@ -1,7 +1,8 @@
|
|
|
1
1
|
import { CommonModule } from '@angular/common';
|
|
2
|
-
import { Component, Input } from '@angular/core';
|
|
2
|
+
import { Component, inject, Input } from '@angular/core';
|
|
3
3
|
import { ReactiveFormsModule } from '@angular/forms';
|
|
4
4
|
import { IonDatetime, IonDatetimeButton, IonModal } from '@ionic/angular/standalone';
|
|
5
|
+
import { I18nService } from '../../../services/i18n';
|
|
5
6
|
import * as i0 from "@angular/core";
|
|
6
7
|
import * as i1 from "@angular/forms";
|
|
7
8
|
/**
|
|
@@ -15,7 +16,17 @@ import * as i1 from "@angular/forms";
|
|
|
15
16
|
* @input props: DateInputMetadata - Configuration for the date input (form control, hint, etc.)
|
|
16
17
|
*/
|
|
17
18
|
export class DateInputComponent {
|
|
18
|
-
|
|
19
|
+
/** Done button text - from props or i18n default */
|
|
20
|
+
get doneText() {
|
|
21
|
+
return this.props?.doneText || this.i18n.t('ok');
|
|
22
|
+
}
|
|
23
|
+
/** Cancel button text - from props or i18n default */
|
|
24
|
+
get cancelText() {
|
|
25
|
+
return this.props?.cancelText || this.i18n.t('cancel');
|
|
26
|
+
}
|
|
27
|
+
constructor() {
|
|
28
|
+
this.i18n = inject(I18nService);
|
|
29
|
+
}
|
|
19
30
|
ngOnInit() {
|
|
20
31
|
// Apply default values on initialization
|
|
21
32
|
if (this.props?.withDefault || this.props?.value) {
|
|
@@ -56,8 +67,8 @@ export class DateInputComponent {
|
|
|
56
67
|
locale="es-ES"
|
|
57
68
|
[firstDayOfWeek]="1"
|
|
58
69
|
[showDefaultButtons]="true"
|
|
59
|
-
doneText="
|
|
60
|
-
cancelText="
|
|
70
|
+
[doneText]="doneText"
|
|
71
|
+
[cancelText]="cancelText"
|
|
61
72
|
formatOptions="{
|
|
62
73
|
date: { dateStyle: 'medium' },
|
|
63
74
|
time: { timeStyle: 'short' }
|
|
@@ -84,8 +95,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
84
95
|
locale="es-ES"
|
|
85
96
|
[firstDayOfWeek]="1"
|
|
86
97
|
[showDefaultButtons]="true"
|
|
87
|
-
doneText="
|
|
88
|
-
cancelText="
|
|
98
|
+
[doneText]="doneText"
|
|
99
|
+
[cancelText]="cancelText"
|
|
89
100
|
formatOptions="{
|
|
90
101
|
date: { dateStyle: 'medium' },
|
|
91
102
|
time: { timeStyle: 'short' }
|
|
@@ -99,4 +110,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
99
110
|
}], ctorParameters: () => [], propDecorators: { props: [{
|
|
100
111
|
type: Input
|
|
101
112
|
}] } });
|
|
102
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
113
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0ZS1pbnB1dC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2RhdGUtaW5wdXQvZGF0ZS1pbnB1dC5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFFLEtBQUssRUFBVSxNQUFNLGVBQWUsQ0FBQztBQUNqRSxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUNyRCxPQUFPLEVBQUUsV0FBVyxFQUFFLGlCQUFpQixFQUFFLFFBQVEsRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBQ3JGLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQzs7O0FBa0NyRDs7Ozs7Ozs7O0dBU0c7QUFDSCxNQUFNLE9BQU8sa0JBQWtCO0lBVzdCLG9EQUFvRDtJQUNwRCxJQUFJLFFBQVE7UUFDVixPQUFPLElBQUksQ0FBQyxLQUFLLEVBQUUsUUFBUSxJQUFJLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQ25ELENBQUM7SUFFRCxzREFBc0Q7SUFDdEQsSUFBSSxVQUFVO1FBQ1osT0FBTyxJQUFJLENBQUMsS0FBSyxFQUFFLFVBQVUsSUFBSSxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxRQUFRLENBQUMsQ0FBQztJQUN6RCxDQUFDO0lBRUQ7UUFaUSxTQUFJLEdBQUcsTUFBTSxDQUFDLFdBQVcsQ0FBQyxDQUFDO0lBWXBCLENBQUM7SUFFaEIsUUFBUTtRQUNOLHlDQUF5QztRQUN6QyxJQUFJLElBQUksQ0FBQyxLQUFLLEVBQUUsV0FBVyxJQUFJLElBQUksQ0FBQyxLQUFLLEVBQUUsS0FBSyxFQUFFLENBQUM7WUFDakQsSUFBSSxDQUFDLGlCQUFpQixFQUFFLENBQUM7UUFDM0IsQ0FBQztJQUNILENBQUM7SUFFTyxpQkFBaUI7UUFDdkIsSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTztZQUFFLE9BQU87UUFFaEMsSUFBSSxZQUFZLEdBQWtCLElBQUksQ0FBQztRQUV2QyxrQ0FBa0M7UUFDbEMsSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLEtBQUssRUFBRSxDQUFDO1lBQ3JCLFlBQVksR0FBRyxJQUFJLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLEtBQUssQ0FBQyxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQzFELENBQUM7YUFBTSxJQUFJLE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQyxXQUFXLEtBQUssUUFBUSxFQUFFLENBQUM7WUFDdEQsWUFBWSxHQUFHLElBQUksSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsV0FBVyxDQUFDLENBQUMsV0FBVyxFQUFFLENBQUM7UUFDaEUsQ0FBQzthQUFNLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBQyxXQUFXLEtBQUssSUFBSSxFQUFFLENBQUM7WUFDM0MsWUFBWSxHQUFHLElBQUksSUFBSSxFQUFFLENBQUMsV0FBVyxFQUFFLENBQUM7UUFDMUMsQ0FBQztRQUVELElBQUksWUFBWSxFQUFFLENBQUM7WUFDakIsSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsUUFBUSxDQUFDLFlBQVksQ0FBQyxDQUFDO1lBQzFDLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLGNBQWMsRUFBRSxDQUFDO1lBQ3BDLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLHNCQUFzQixFQUFFLENBQUM7UUFDOUMsQ0FBQztJQUNILENBQUM7K0dBakRVLGtCQUFrQjttR0FBbEIsa0JBQWtCLHNHQXJDbkI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXdCVCxzeEZBekJTLFlBQVksOEJBQUUsbUJBQW1CLDBUQUFFLFdBQVcsb2hCQUFFLGlCQUFpQixtSEFBRSxRQUFROzs0RkFzQzFFLGtCQUFrQjtrQkF6QzlCLFNBQVM7K0JBQ0UsZ0JBQWdCLGNBQ2QsSUFBSSxXQUNQLENBQUMsWUFBWSxFQUFFLG1CQUFtQixFQUFFLFdBQVcsRUFBRSxpQkFBaUIsRUFBRSxRQUFRLENBQUMsWUFDNUU7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXdCVDt3REFvQlEsS0FBSztzQkFBYixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IENvbXBvbmVudCwgaW5qZWN0LCBJbnB1dCwgT25Jbml0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBSZWFjdGl2ZUZvcm1zTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuaW1wb3J0IHsgSW9uRGF0ZXRpbWUsIElvbkRhdGV0aW1lQnV0dG9uLCBJb25Nb2RhbCB9IGZyb20gJ0Bpb25pYy9hbmd1bGFyL3N0YW5kYWxvbmUnO1xuaW1wb3J0IHsgSTE4blNlcnZpY2UgfSBmcm9tICcuLi8uLi8uLi9zZXJ2aWNlcy9pMThuJztcbmltcG9ydCB7IERhdGVJbnB1dE1ldGFkYXRhIH0gZnJvbSAnLi90eXBlcyc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ3ZhbC1kYXRlLWlucHV0JyxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgUmVhY3RpdmVGb3Jtc01vZHVsZSwgSW9uRGF0ZXRpbWUsIElvbkRhdGV0aW1lQnV0dG9uLCBJb25Nb2RhbF0sXG4gIHRlbXBsYXRlOiBgXG4gICAgPGRpdiBjbGFzcz1cImJ1dHRvbi1jb250YWluZXJcIj5cbiAgICAgIDxpb24tZGF0ZXRpbWUtYnV0dG9uIGNsYXNzPVwiYWN0aW9uXCIgW2RhdGV0aW1lXT1cInByb3BzLnRva2VuXCI+PC9pb24tZGF0ZXRpbWUtYnV0dG9uPlxuICAgIDwvZGl2PlxuICAgIDxpb24tbW9kYWwgW2tlZXBDb250ZW50c01vdW50ZWRdPVwidHJ1ZVwiPlxuICAgICAgPG5nLXRlbXBsYXRlPlxuICAgICAgICA8aW9uLWRhdGV0aW1lXG4gICAgICAgICAgW2Zvcm1Db250cm9sXT1cInByb3BzLmNvbnRyb2xcIlxuICAgICAgICAgIFtpZF09XCJwcm9wcy50b2tlblwiXG4gICAgICAgICAgcHJlc2VudGF0aW9uPVwiZGF0ZVwiXG4gICAgICAgICAgbG9jYWxlPVwiZXMtRVNcIlxuICAgICAgICAgIFtmaXJzdERheU9mV2Vla109XCIxXCJcbiAgICAgICAgICBbc2hvd0RlZmF1bHRCdXR0b25zXT1cInRydWVcIlxuICAgICAgICAgIFtkb25lVGV4dF09XCJkb25lVGV4dFwiXG4gICAgICAgICAgW2NhbmNlbFRleHRdPVwiY2FuY2VsVGV4dFwiXG4gICAgICAgICAgZm9ybWF0T3B0aW9ucz1cIntcbiAgICAgICAgICAgIGRhdGU6IHsgZGF0ZVN0eWxlOiAnbWVkaXVtJyB9LFxuICAgICAgICAgICAgdGltZTogeyB0aW1lU3R5bGU6ICdzaG9ydCcgfVxuICAgICAgICAgIH1cIlxuICAgICAgICA+XG4gICAgICAgICAgPHNwYW4gc2xvdD1cInRpdGxlXCI+e3sgcHJvcHMuaGludCB9fTwvc3Bhbj5cbiAgICAgICAgPC9pb24tZGF0ZXRpbWU+XG4gICAgICA8L25nLXRlbXBsYXRlPlxuICAgIDwvaW9uLW1vZGFsPlxuICBgLFxuICBzdHlsZVVybHM6IFsnLi9kYXRlLWlucHV0LmNvbXBvbmVudC5zY3NzJ10sXG59KVxuLyoqXG4gKiB2YWwtZGF0ZS1pbnB1dFxuICpcbiAqIEEgZGF0ZSBwaWNrZXIgaW5wdXQgaW50ZWdyYXRlZCB3aXRoIEFuZ3VsYXIgZm9ybXMsIHVzaW5nIElvbmljJ3MgZGF0ZXRpbWUgY29tcG9uZW50LlxuICpcbiAqIEBleGFtcGxlXG4gKiA8dmFsLWRhdGUtaW5wdXQgW3Byb3BzXT1cInsgY29udHJvbDogbXlDb250cm9sLCBoaW50OiAnU2VsZWN0IGEgZGF0ZScgfVwiPjwvdmFsLWRhdGUtaW5wdXQ+XG4gKlxuICogQGlucHV0IHByb3BzOiBEYXRlSW5wdXRNZXRhZGF0YSAtIENvbmZpZ3VyYXRpb24gZm9yIHRoZSBkYXRlIGlucHV0IChmb3JtIGNvbnRyb2wsIGhpbnQsIGV0Yy4pXG4gKi9cbmV4cG9ydCBjbGFzcyBEYXRlSW5wdXRDb21wb25lbnQgaW1wbGVtZW50cyBPbkluaXQge1xuICAvKipcbiAgICogSW5wdXQgY29uZmlndXJhdGlvbiBvYmplY3QuXG4gICAqIEB0eXBlIHtEYXRlSW5wdXRNZXRhZGF0YX1cbiAgICogQHByb3BlcnR5IGNvbnRyb2wgLSBUaGUgQW5ndWxhciBGb3JtQ29udHJvbCBmb3IgdGhlIGRhdGUgaW5wdXQuXG4gICAqIEBwcm9wZXJ0eSBoaW50IC0gVGhlIGhpbnQgdGV4dCBmb3IgdGhlIGlucHV0LlxuICAgKi9cbiAgQElucHV0KCkgcHJvcHM6IERhdGVJbnB1dE1ldGFkYXRhO1xuXG4gIHByaXZhdGUgaTE4biA9IGluamVjdChJMThuU2VydmljZSk7XG5cbiAgLyoqIERvbmUgYnV0dG9uIHRleHQgLSBmcm9tIHByb3BzIG9yIGkxOG4gZGVmYXVsdCAqL1xuICBnZXQgZG9uZVRleHQoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gdGhpcy5wcm9wcz8uZG9uZVRleHQgfHwgdGhpcy5pMThuLnQoJ29rJyk7XG4gIH1cblxuICAvKiogQ2FuY2VsIGJ1dHRvbiB0ZXh0IC0gZnJvbSBwcm9wcyBvciBpMThuIGRlZmF1bHQgKi9cbiAgZ2V0IGNhbmNlbFRleHQoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gdGhpcy5wcm9wcz8uY2FuY2VsVGV4dCB8fCB0aGlzLmkxOG4udCgnY2FuY2VsJyk7XG4gIH1cblxuICBjb25zdHJ1Y3RvcigpIHt9XG5cbiAgbmdPbkluaXQoKSB7XG4gICAgLy8gQXBwbHkgZGVmYXVsdCB2YWx1ZXMgb24gaW5pdGlhbGl6YXRpb25cbiAgICBpZiAodGhpcy5wcm9wcz8ud2l0aERlZmF1bHQgfHwgdGhpcy5wcm9wcz8udmFsdWUpIHtcbiAgICAgIHRoaXMuYXBwbHlEZWZhdWx0VmFsdWUoKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIGFwcGx5RGVmYXVsdFZhbHVlKCk6IHZvaWQge1xuICAgIGlmICghdGhpcy5wcm9wcy5jb250cm9sKSByZXR1cm47XG5cbiAgICBsZXQgZGVmYXVsdFZhbHVlOiBzdHJpbmcgfCBudWxsID0gbnVsbDtcblxuICAgIC8vIEV4cGxpY2l0IHZhbHVlIHRha2VzIHByZWNlZGVuY2VcbiAgICBpZiAodGhpcy5wcm9wcy52YWx1ZSkge1xuICAgICAgZGVmYXVsdFZhbHVlID0gbmV3IERhdGUodGhpcy5wcm9wcy52YWx1ZSkudG9JU09TdHJpbmcoKTtcbiAgICB9IGVsc2UgaWYgKHR5cGVvZiB0aGlzLnByb3BzLndpdGhEZWZhdWx0ID09PSAnc3RyaW5nJykge1xuICAgICAgZGVmYXVsdFZhbHVlID0gbmV3IERhdGUodGhpcy5wcm9wcy53aXRoRGVmYXVsdCkudG9JU09TdHJpbmcoKTtcbiAgICB9IGVsc2UgaWYgKHRoaXMucHJvcHMud2l0aERlZmF1bHQgPT09IHRydWUpIHtcbiAgICAgIGRlZmF1bHRWYWx1ZSA9IG5ldyBEYXRlKCkudG9JU09TdHJpbmcoKTtcbiAgICB9XG5cbiAgICBpZiAoZGVmYXVsdFZhbHVlKSB7XG4gICAgICB0aGlzLnByb3BzLmNvbnRyb2wuc2V0VmFsdWUoZGVmYXVsdFZhbHVlKTtcbiAgICAgIHRoaXMucHJvcHMuY29udHJvbC5tYXJrQXNQcmlzdGluZSgpO1xuICAgICAgdGhpcy5wcm9wcy5jb250cm9sLnVwZGF0ZVZhbHVlQW5kVmFsaWRpdHkoKTtcbiAgICB9XG4gIH1cbn1cbiJdfQ==
|