valtech-components 2.0.414 → 2.0.416

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,136 +1,85 @@
1
1
  import { CommonModule } from '@angular/common';
2
- import { Component, EventEmitter, inject, Input, Output } from '@angular/core';
2
+ import { Component, EventEmitter, Input, Output } from '@angular/core';
3
3
  import { IonText } from '@ionic/angular/standalone';
4
- import { of } from 'rxjs';
5
- import { LangService } from '../../../services/lang-provider/lang-provider.service';
6
4
  import { BoxComponent } from '../../atoms/box/box.component';
7
5
  import { HrefComponent } from '../../atoms/href/href.component';
8
6
  import { ButtonGroupComponent } from '../button-group/button-group.component';
9
7
  import * as i0 from "@angular/core";
10
8
  import * as i1 from "@angular/common";
11
- /**
12
- * val-prompter
13
- *
14
- * Displays a prompt/info box with text, actions (buttons/links), and custom styles.
15
- * Supports reactive i18n content via contentClass and contentKey props.
16
- *
17
- * @example Static content:
18
- * <val-prompter [props]="{ content: 'Message', buttons: [...], color: 'primary', size: 'medium' }"></val-prompter>
19
- *
20
- * @example Reactive i18n content:
21
- * <val-prompter [props]="{
22
- * contentClass: 'homePage',
23
- * contentKey: 'cookieMessage',
24
- * content: 'Fallback message',
25
- * buttons: [...],
26
- * color: 'primary',
27
- * size: 'medium'
28
- * }"></val-prompter>
29
- *
30
- * @input props: PrompterMetadata - Configuration for the prompter (content, buttons, hrefs, color, size, etc.)
31
- * @output onClick - Emits when an action is clicked
32
- */
33
9
  export class PrompterComponent {
34
10
  constructor() {
35
- this.langService = inject(LangService);
36
- /** Observable for reactive content - follows val-text/val-button pattern */
37
- this.displayContent$ = of('');
38
11
  this.onClick = new EventEmitter();
39
12
  }
40
- ngOnInit() {
41
- this.setupContent();
42
- }
43
- /**
44
- * Set up content Observable.
45
- * Follows the same pattern as val-text and val-button for consistency.
46
- */
47
- setupContent() {
48
- if (!this.props) {
49
- return;
50
- }
51
- if (this.props.contentClass && this.props.contentKey) {
52
- // Reactive content - Observable connected to currentLang$
53
- this.displayContent$ = this.langService.getContent(this.props.contentClass, this.props.contentKey, this.props.content);
54
- }
55
- else {
56
- // Static content wrapped in Observable
57
- this.displayContent$ = of(this.props.content);
58
- }
59
- }
60
13
  clickHandler(token) {
61
14
  this.onClick.emit(token);
62
15
  }
63
16
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PrompterComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
64
17
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: PrompterComponent, isStandalone: true, selector: "val-prompter", inputs: { props: "props" }, outputs: { onClick: "onClick" }, ngImport: i0, template: `
65
- @if (props) {
66
- <val-box
67
- class="teleprompter-container"
68
- [props]="{ icon: '', bordered: props.bordered, color: props.color, leftBorder: false, padding: props.padding }"
69
- (onClick)="clickHandler()"
70
- >
71
- <div class="container" [class.teleprompter-content]="props.teleprompter" body>
72
- <div>
73
- <ion-text>
74
- <p [class]="props.size" class="bold">{{ displayContent$ | async }}</p>
75
- </ion-text>
76
- </div>
77
- @if (props.buttons || props.hrefs) {
78
- <div class="actions-container">
79
- <val-button-group
80
- class="buttons-container"
81
- [props]="{ buttons: props.buttons, position: 'right', columned: false }"
82
- (onClick)="clickHandler($event)"
83
- ></val-button-group>
84
- <val-href
85
- class="link"
86
- *ngFor="let l of props.hrefs"
87
- [props]="l"
88
- (onClick)="clickHandler($event)"
89
- ></val-href>
90
- </div>
91
- }
18
+ <val-box
19
+ class="teleprompter-container"
20
+ [props]="{ icon: '', bordered: props.bordered, color: props.color, leftBorder: false, padding: props.padding }"
21
+ (onClick)="clickHandler()"
22
+ >
23
+ <div class="container" [class.teleprompter-content]="props.teleprompter" body>
24
+ <div>
25
+ <ion-text>
26
+ <p [class]="props.size" class="bold">{{ props.content }}</p>
27
+ </ion-text>
92
28
  </div>
93
- </val-box>
94
- }
95
- `, isInline: true, styles: ["@charset \"UTF-8\";:root{--ion-color-primary: #7026df;--ion-color-primary-rgb: 112, 38, 223;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #6321c4;--ion-color-primary-tint: #7e3ce2;--ion-color-secondary: #e2ccff;--ion-color-secondary-rgb: 226, 204, 255;--ion-color-secondary-contrast: #000000;--ion-color-secondary-contrast-rgb: 0, 0, 0;--ion-color-secondary-shade: #c7b4e0;--ion-color-secondary-tint: #e5d1ff;--ion-color-texti: #354c69;--ion-color-texti-rgb: 53, 76, 105;--ion-color-texti-contrast: #ffffff;--ion-color-texti-contrast-rgb: 255, 255, 255;--ion-color-texti-shade: #2f435c;--ion-color-texti-tint: #495e78;--ion-color-darki: #090f1b;--ion-color-darki-rgb: 9, 15, 27;--ion-color-darki-contrast: #ffffff;--ion-color-darki-contrast-rgb: 255, 255, 255;--ion-color-darki-shade: #080d18;--ion-color-darki-tint: #222732;--ion-color-medium: #9e9e9e;--ion-color-medium-rgb: 158, 158, 158;--ion-color-medium-contrast: #000000;--ion-color-medium-contrast-rgb: 0, 0, 0;--ion-color-medium-shade: #8b8b8b;--ion-color-medium-tint: #a8a8a8;--swiper-pagination-color: var(--ion-color-primary);--swiper-navigation-color: var(--ion-color-primary);--swiper-pagination-bullet-inactive-color: var(--ion-color-medium)}@media (prefers-color-scheme: dark){:root{--ion-color-texti: #8fc1ff;--ion-color-texti-rgb: 143, 193, 255;--ion-color-texti-contrast: #000000;--ion-color-texti-contrast-rgb: 0, 0, 0;--ion-color-texti-shade: #7eaae0;--ion-color-texti-tint: #9ac7ff;--ion-color-darki: #ffffff;--ion-color-darki-rgb: 255, 255, 255;--ion-color-darki-contrast: #000000;--ion-color-darki-contrast-rgb: 0, 0, 0;--ion-color-darki-shade: #e0e0e0;--ion-color-darki-tint: #ffffff;--ion-color-primary: #8f49f8;--ion-color-primary-rgb: 143, 73, 248;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #7e40da;--ion-color-primary-tint: #9a5bf9}}.ion-color-texti{--ion-color-base: var(--ion-color-texti);--ion-color-base-rgb: var(--ion-color-texti-rgb);--ion-color-contrast: var(--ion-color-texti-contrast);--ion-color-contrast-rgb: var(--ion-color-texti-contrast-rgb);--ion-color-shade: var(--ion-color-texti-shade);--ion-color-tint: var(--ion-color-texti-tint)}.ion-color-darki{--ion-color-base: var(--ion-color-darki);--ion-color-base-rgb: var(--ion-color-darki-rgb);--ion-color-contrast: var(--ion-color-darki-contrast);--ion-color-contrast-rgb: var(--ion-color-darki-contrast-rgb);--ion-color-shade: var(--ion-color-darki-shade);--ion-color-tint: var(--ion-color-darki-tint)}.container{display:flex;align-items:center;justify-content:space-between;flex-direction:column}@media (min-width: 768px){.container{flex-direction:row}}.teleprompter-container{overflow:hidden;width:100%}.teleprompter-content{white-space:nowrap;animation:scroll-left 15s linear infinite}@keyframes scroll-left{0%{transform:translate(100%)}to{transform:translate(-100%)}}.link{margin:0 .5rem;width:max-content}.actions-container{display:flex;align-items:center}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "component", type: IonText, selector: "ion-text", inputs: ["color", "mode"] }, { kind: "component", type: BoxComponent, selector: "val-box", inputs: ["props"], outputs: ["onClick"] }, { kind: "component", type: ButtonGroupComponent, selector: "val-button-group", inputs: ["props"], outputs: ["onClick"] }, { kind: "component", type: HrefComponent, selector: "val-href", inputs: ["props"], outputs: ["onClick"] }] }); }
29
+ @if (props.buttons || props.hrefs) {
30
+ <div class="actions-container">
31
+ <val-button-group
32
+ class="buttons-container"
33
+ [props]="{ buttons: props.buttons, position: 'right', columned: false }"
34
+ (onClick)="clickHandler($event)"
35
+ ></val-button-group>
36
+ <val-href
37
+ class="link"
38
+ *ngFor="let l of props.hrefs"
39
+ [props]="l"
40
+ (onClick)="clickHandler($event)"
41
+ ></val-href>
42
+ </div>
43
+ }
44
+ </div>
45
+ </val-box>
46
+ `, isInline: true, styles: ["@charset \"UTF-8\";:root{--ion-color-primary: #7026df;--ion-color-primary-rgb: 112, 38, 223;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #6321c4;--ion-color-primary-tint: #7e3ce2;--ion-color-secondary: #e2ccff;--ion-color-secondary-rgb: 226, 204, 255;--ion-color-secondary-contrast: #000000;--ion-color-secondary-contrast-rgb: 0, 0, 0;--ion-color-secondary-shade: #c7b4e0;--ion-color-secondary-tint: #e5d1ff;--ion-color-texti: #354c69;--ion-color-texti-rgb: 53, 76, 105;--ion-color-texti-contrast: #ffffff;--ion-color-texti-contrast-rgb: 255, 255, 255;--ion-color-texti-shade: #2f435c;--ion-color-texti-tint: #495e78;--ion-color-darki: #090f1b;--ion-color-darki-rgb: 9, 15, 27;--ion-color-darki-contrast: #ffffff;--ion-color-darki-contrast-rgb: 255, 255, 255;--ion-color-darki-shade: #080d18;--ion-color-darki-tint: #222732;--ion-color-medium: #9e9e9e;--ion-color-medium-rgb: 158, 158, 158;--ion-color-medium-contrast: #000000;--ion-color-medium-contrast-rgb: 0, 0, 0;--ion-color-medium-shade: #8b8b8b;--ion-color-medium-tint: #a8a8a8;--swiper-pagination-color: var(--ion-color-primary);--swiper-navigation-color: var(--ion-color-primary);--swiper-pagination-bullet-inactive-color: var(--ion-color-medium)}@media (prefers-color-scheme: dark){:root{--ion-color-texti: #8fc1ff;--ion-color-texti-rgb: 143, 193, 255;--ion-color-texti-contrast: #000000;--ion-color-texti-contrast-rgb: 0, 0, 0;--ion-color-texti-shade: #7eaae0;--ion-color-texti-tint: #9ac7ff;--ion-color-darki: #ffffff;--ion-color-darki-rgb: 255, 255, 255;--ion-color-darki-contrast: #000000;--ion-color-darki-contrast-rgb: 0, 0, 0;--ion-color-darki-shade: #e0e0e0;--ion-color-darki-tint: #ffffff;--ion-color-primary: #8f49f8;--ion-color-primary-rgb: 143, 73, 248;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #7e40da;--ion-color-primary-tint: #9a5bf9}}.ion-color-texti{--ion-color-base: var(--ion-color-texti);--ion-color-base-rgb: var(--ion-color-texti-rgb);--ion-color-contrast: var(--ion-color-texti-contrast);--ion-color-contrast-rgb: var(--ion-color-texti-contrast-rgb);--ion-color-shade: var(--ion-color-texti-shade);--ion-color-tint: var(--ion-color-texti-tint)}.ion-color-darki{--ion-color-base: var(--ion-color-darki);--ion-color-base-rgb: var(--ion-color-darki-rgb);--ion-color-contrast: var(--ion-color-darki-contrast);--ion-color-contrast-rgb: var(--ion-color-darki-contrast-rgb);--ion-color-shade: var(--ion-color-darki-shade);--ion-color-tint: var(--ion-color-darki-tint)}.container{display:flex;align-items:center;justify-content:space-between;flex-direction:column}@media (min-width: 768px){.container{flex-direction:row}}.teleprompter-container{overflow:hidden;width:100%}.teleprompter-content{white-space:nowrap;animation:scroll-left 15s linear infinite}@keyframes scroll-left{0%{transform:translate(100%)}to{transform:translate(-100%)}}.link{margin:0 .5rem;width:max-content}.actions-container{display:flex;align-items:center}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: IonText, selector: "ion-text", inputs: ["color", "mode"] }, { kind: "component", type: BoxComponent, selector: "val-box", inputs: ["props"], outputs: ["onClick"] }, { kind: "component", type: ButtonGroupComponent, selector: "val-button-group", inputs: ["props"], outputs: ["onClick"] }, { kind: "component", type: HrefComponent, selector: "val-href", inputs: ["props"], outputs: ["onClick"] }] }); }
96
47
  }
97
48
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PrompterComponent, decorators: [{
98
49
  type: Component,
99
50
  args: [{ selector: 'val-prompter', standalone: true, imports: [CommonModule, IonText, BoxComponent, ButtonGroupComponent, HrefComponent], template: `
100
- @if (props) {
101
- <val-box
102
- class="teleprompter-container"
103
- [props]="{ icon: '', bordered: props.bordered, color: props.color, leftBorder: false, padding: props.padding }"
104
- (onClick)="clickHandler()"
105
- >
106
- <div class="container" [class.teleprompter-content]="props.teleprompter" body>
107
- <div>
108
- <ion-text>
109
- <p [class]="props.size" class="bold">{{ displayContent$ | async }}</p>
110
- </ion-text>
111
- </div>
112
- @if (props.buttons || props.hrefs) {
113
- <div class="actions-container">
114
- <val-button-group
115
- class="buttons-container"
116
- [props]="{ buttons: props.buttons, position: 'right', columned: false }"
117
- (onClick)="clickHandler($event)"
118
- ></val-button-group>
119
- <val-href
120
- class="link"
121
- *ngFor="let l of props.hrefs"
122
- [props]="l"
123
- (onClick)="clickHandler($event)"
124
- ></val-href>
125
- </div>
126
- }
51
+ <val-box
52
+ class="teleprompter-container"
53
+ [props]="{ icon: '', bordered: props.bordered, color: props.color, leftBorder: false, padding: props.padding }"
54
+ (onClick)="clickHandler()"
55
+ >
56
+ <div class="container" [class.teleprompter-content]="props.teleprompter" body>
57
+ <div>
58
+ <ion-text>
59
+ <p [class]="props.size" class="bold">{{ props.content }}</p>
60
+ </ion-text>
127
61
  </div>
128
- </val-box>
129
- }
62
+ @if (props.buttons || props.hrefs) {
63
+ <div class="actions-container">
64
+ <val-button-group
65
+ class="buttons-container"
66
+ [props]="{ buttons: props.buttons, position: 'right', columned: false }"
67
+ (onClick)="clickHandler($event)"
68
+ ></val-button-group>
69
+ <val-href
70
+ class="link"
71
+ *ngFor="let l of props.hrefs"
72
+ [props]="l"
73
+ (onClick)="clickHandler($event)"
74
+ ></val-href>
75
+ </div>
76
+ }
77
+ </div>
78
+ </val-box>
130
79
  `, styles: ["@charset \"UTF-8\";:root{--ion-color-primary: #7026df;--ion-color-primary-rgb: 112, 38, 223;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #6321c4;--ion-color-primary-tint: #7e3ce2;--ion-color-secondary: #e2ccff;--ion-color-secondary-rgb: 226, 204, 255;--ion-color-secondary-contrast: #000000;--ion-color-secondary-contrast-rgb: 0, 0, 0;--ion-color-secondary-shade: #c7b4e0;--ion-color-secondary-tint: #e5d1ff;--ion-color-texti: #354c69;--ion-color-texti-rgb: 53, 76, 105;--ion-color-texti-contrast: #ffffff;--ion-color-texti-contrast-rgb: 255, 255, 255;--ion-color-texti-shade: #2f435c;--ion-color-texti-tint: #495e78;--ion-color-darki: #090f1b;--ion-color-darki-rgb: 9, 15, 27;--ion-color-darki-contrast: #ffffff;--ion-color-darki-contrast-rgb: 255, 255, 255;--ion-color-darki-shade: #080d18;--ion-color-darki-tint: #222732;--ion-color-medium: #9e9e9e;--ion-color-medium-rgb: 158, 158, 158;--ion-color-medium-contrast: #000000;--ion-color-medium-contrast-rgb: 0, 0, 0;--ion-color-medium-shade: #8b8b8b;--ion-color-medium-tint: #a8a8a8;--swiper-pagination-color: var(--ion-color-primary);--swiper-navigation-color: var(--ion-color-primary);--swiper-pagination-bullet-inactive-color: var(--ion-color-medium)}@media (prefers-color-scheme: dark){:root{--ion-color-texti: #8fc1ff;--ion-color-texti-rgb: 143, 193, 255;--ion-color-texti-contrast: #000000;--ion-color-texti-contrast-rgb: 0, 0, 0;--ion-color-texti-shade: #7eaae0;--ion-color-texti-tint: #9ac7ff;--ion-color-darki: #ffffff;--ion-color-darki-rgb: 255, 255, 255;--ion-color-darki-contrast: #000000;--ion-color-darki-contrast-rgb: 0, 0, 0;--ion-color-darki-shade: #e0e0e0;--ion-color-darki-tint: #ffffff;--ion-color-primary: #8f49f8;--ion-color-primary-rgb: 143, 73, 248;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #7e40da;--ion-color-primary-tint: #9a5bf9}}.ion-color-texti{--ion-color-base: var(--ion-color-texti);--ion-color-base-rgb: var(--ion-color-texti-rgb);--ion-color-contrast: var(--ion-color-texti-contrast);--ion-color-contrast-rgb: var(--ion-color-texti-contrast-rgb);--ion-color-shade: var(--ion-color-texti-shade);--ion-color-tint: var(--ion-color-texti-tint)}.ion-color-darki{--ion-color-base: var(--ion-color-darki);--ion-color-base-rgb: var(--ion-color-darki-rgb);--ion-color-contrast: var(--ion-color-darki-contrast);--ion-color-contrast-rgb: var(--ion-color-darki-contrast-rgb);--ion-color-shade: var(--ion-color-darki-shade);--ion-color-tint: var(--ion-color-darki-tint)}.container{display:flex;align-items:center;justify-content:space-between;flex-direction:column}@media (min-width: 768px){.container{flex-direction:row}}.teleprompter-container{overflow:hidden;width:100%}.teleprompter-content{white-space:nowrap;animation:scroll-left 15s linear infinite}@keyframes scroll-left{0%{transform:translate(100%)}to{transform:translate(-100%)}}.link{margin:0 .5rem;width:max-content}.actions-container{display:flex;align-items:center}\n"] }]
131
80
  }], propDecorators: { props: [{
132
81
  type: Input
133
82
  }], onClick: [{
134
83
  type: Output
135
84
  }] } });
136
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvbXB0ZXIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9jb21wb25lbnRzL21vbGVjdWxlcy9wcm9tcHRlci9wcm9tcHRlci5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsWUFBWSxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQVUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3ZGLE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUNwRCxPQUFPLEVBQWMsRUFBRSxFQUFFLE1BQU0sTUFBTSxDQUFDO0FBQ3RDLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSx1REFBdUQsQ0FBQztBQUNwRixPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sK0JBQStCLENBQUM7QUFDN0QsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLGlDQUFpQyxDQUFDO0FBQ2hFLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLHdDQUF3QyxDQUFDOzs7QUF5QzlFOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FxQkc7QUFDSCxNQUFNLE9BQU8saUJBQWlCO0lBNUQ5QjtRQTZEVSxnQkFBVyxHQUFHLE1BQU0sQ0FBQyxXQUFXLENBQUMsQ0FBQztRQUUxQyw0RUFBNEU7UUFDNUUsb0JBQWUsR0FBdUIsRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDO1FBUzdDLFlBQU8sR0FBRyxJQUFJLFlBQVksRUFBVSxDQUFDO0tBOEJ0QztJQTVCQyxRQUFRO1FBQ04sSUFBSSxDQUFDLFlBQVksRUFBRSxDQUFDO0lBQ3RCLENBQUM7SUFFRDs7O09BR0c7SUFDSyxZQUFZO1FBQ2xCLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxFQUFFLENBQUM7WUFDaEIsT0FBTztRQUNULENBQUM7UUFDRCxJQUFJLElBQUksQ0FBQyxLQUFLLENBQUMsWUFBWSxJQUFJLElBQUksQ0FBQyxLQUFLLENBQUMsVUFBVSxFQUFFLENBQUM7WUFDckQsMERBQTBEO1lBQzFELElBQUksQ0FBQyxlQUFlLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxVQUFVLENBQ2hELElBQUksQ0FBQyxLQUFLLENBQUMsWUFBWSxFQUN2QixJQUFJLENBQUMsS0FBSyxDQUFDLFVBQVUsRUFDckIsSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQ25CLENBQUM7UUFDSixDQUFDO2FBQU0sQ0FBQztZQUNOLHVDQUF1QztZQUN2QyxJQUFJLENBQUMsZUFBZSxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBQ2hELENBQUM7SUFDSCxDQUFDO0lBRUQsWUFBWSxDQUFDLEtBQWM7UUFDekIsSUFBSSxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDM0IsQ0FBQzsrR0ExQ1UsaUJBQWlCO21HQUFqQixpQkFBaUIscUlBeERsQjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQStCVCw2aUdBaENTLFlBQVksaU5BQUUsT0FBTyxnRkFBRSxZQUFZLDZGQUFFLG9CQUFvQixzR0FBRSxhQUFhOzs0RkF5RHZFLGlCQUFpQjtrQkE1RDdCLFNBQVM7K0JBQ0UsY0FBYyxjQUNaLElBQUksV0FDUCxDQUFDLFlBQVksRUFBRSxPQUFPLEVBQUUsWUFBWSxFQUFFLG9CQUFvQixFQUFFLGFBQWEsQ0FBQyxZQUN6RTs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQStCVDs4QkFtQ0QsS0FBSztzQkFESixLQUFLO2dCQUlOLE9BQU87c0JBRE4sTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBDb21wb25lbnQsIEV2ZW50RW1pdHRlciwgaW5qZWN0LCBJbnB1dCwgT25Jbml0LCBPdXRwdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IElvblRleHQgfSBmcm9tICdAaW9uaWMvYW5ndWxhci9zdGFuZGFsb25lJztcbmltcG9ydCB7IE9ic2VydmFibGUsIG9mIH0gZnJvbSAncnhqcyc7XG5pbXBvcnQgeyBMYW5nU2VydmljZSB9IGZyb20gJy4uLy4uLy4uL3NlcnZpY2VzL2xhbmctcHJvdmlkZXIvbGFuZy1wcm92aWRlci5zZXJ2aWNlJztcbmltcG9ydCB7IEJveENvbXBvbmVudCB9IGZyb20gJy4uLy4uL2F0b21zL2JveC9ib3guY29tcG9uZW50JztcbmltcG9ydCB7IEhyZWZDb21wb25lbnQgfSBmcm9tICcuLi8uLi9hdG9tcy9ocmVmL2hyZWYuY29tcG9uZW50JztcbmltcG9ydCB7IEJ1dHRvbkdyb3VwQ29tcG9uZW50IH0gZnJvbSAnLi4vYnV0dG9uLWdyb3VwL2J1dHRvbi1ncm91cC5jb21wb25lbnQnO1xuaW1wb3J0IHsgUHJvbXB0ZXJNZXRhZGF0YSB9IGZyb20gJy4vdHlwZXMnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICd2YWwtcHJvbXB0ZXInLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlLCBJb25UZXh0LCBCb3hDb21wb25lbnQsIEJ1dHRvbkdyb3VwQ29tcG9uZW50LCBIcmVmQ29tcG9uZW50XSxcbiAgdGVtcGxhdGU6IGBcbiAgICBAaWYgKHByb3BzKSB7XG4gICAgICA8dmFsLWJveFxuICAgICAgICBjbGFzcz1cInRlbGVwcm9tcHRlci1jb250YWluZXJcIlxuICAgICAgICBbcHJvcHNdPVwieyBpY29uOiAnJywgYm9yZGVyZWQ6IHByb3BzLmJvcmRlcmVkLCBjb2xvcjogcHJvcHMuY29sb3IsIGxlZnRCb3JkZXI6IGZhbHNlLCBwYWRkaW5nOiBwcm9wcy5wYWRkaW5nIH1cIlxuICAgICAgICAob25DbGljayk9XCJjbGlja0hhbmRsZXIoKVwiXG4gICAgICA+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJjb250YWluZXJcIiBbY2xhc3MudGVsZXByb21wdGVyLWNvbnRlbnRdPVwicHJvcHMudGVsZXByb21wdGVyXCIgYm9keT5cbiAgICAgICAgICA8ZGl2PlxuICAgICAgICAgICAgPGlvbi10ZXh0PlxuICAgICAgICAgICAgICA8cCBbY2xhc3NdPVwicHJvcHMuc2l6ZVwiIGNsYXNzPVwiYm9sZFwiPnt7IGRpc3BsYXlDb250ZW50JCB8IGFzeW5jIH19PC9wPlxuICAgICAgICAgICAgPC9pb24tdGV4dD5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgICBAaWYgKHByb3BzLmJ1dHRvbnMgfHwgcHJvcHMuaHJlZnMpIHtcbiAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJhY3Rpb25zLWNvbnRhaW5lclwiPlxuICAgICAgICAgICAgICA8dmFsLWJ1dHRvbi1ncm91cFxuICAgICAgICAgICAgICAgIGNsYXNzPVwiYnV0dG9ucy1jb250YWluZXJcIlxuICAgICAgICAgICAgICAgIFtwcm9wc109XCJ7IGJ1dHRvbnM6IHByb3BzLmJ1dHRvbnMsIHBvc2l0aW9uOiAncmlnaHQnLCBjb2x1bW5lZDogZmFsc2UgfVwiXG4gICAgICAgICAgICAgICAgKG9uQ2xpY2spPVwiY2xpY2tIYW5kbGVyKCRldmVudClcIlxuICAgICAgICAgICAgICA+PC92YWwtYnV0dG9uLWdyb3VwPlxuICAgICAgICAgICAgICA8dmFsLWhyZWZcbiAgICAgICAgICAgICAgICBjbGFzcz1cImxpbmtcIlxuICAgICAgICAgICAgICAgICpuZ0Zvcj1cImxldCBsIG9mIHByb3BzLmhyZWZzXCJcbiAgICAgICAgICAgICAgICBbcHJvcHNdPVwibFwiXG4gICAgICAgICAgICAgICAgKG9uQ2xpY2spPVwiY2xpY2tIYW5kbGVyKCRldmVudClcIlxuICAgICAgICAgICAgICA+PC92YWwtaHJlZj5cbiAgICAgICAgICAgIDwvZGl2PlxuICAgICAgICAgIH1cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L3ZhbC1ib3g+XG4gICAgfVxuICBgLFxuICBzdHlsZVVybHM6IFsnLi9wcm9tcHRlci5jb21wb25lbnQuc2NzcyddLFxufSlcbi8qKlxuICogdmFsLXByb21wdGVyXG4gKlxuICogRGlzcGxheXMgYSBwcm9tcHQvaW5mbyBib3ggd2l0aCB0ZXh0LCBhY3Rpb25zIChidXR0b25zL2xpbmtzKSwgYW5kIGN1c3RvbSBzdHlsZXMuXG4gKiBTdXBwb3J0cyByZWFjdGl2ZSBpMThuIGNvbnRlbnQgdmlhIGNvbnRlbnRDbGFzcyBhbmQgY29udGVudEtleSBwcm9wcy5cbiAqXG4gKiBAZXhhbXBsZSBTdGF0aWMgY29udGVudDpcbiAqIDx2YWwtcHJvbXB0ZXIgW3Byb3BzXT1cInsgY29udGVudDogJ01lc3NhZ2UnLCBidXR0b25zOiBbLi4uXSwgY29sb3I6ICdwcmltYXJ5Jywgc2l6ZTogJ21lZGl1bScgfVwiPjwvdmFsLXByb21wdGVyPlxuICpcbiAqIEBleGFtcGxlIFJlYWN0aXZlIGkxOG4gY29udGVudDpcbiAqIDx2YWwtcHJvbXB0ZXIgW3Byb3BzXT1cIntcbiAqICAgY29udGVudENsYXNzOiAnaG9tZVBhZ2UnLFxuICogICBjb250ZW50S2V5OiAnY29va2llTWVzc2FnZScsXG4gKiAgIGNvbnRlbnQ6ICdGYWxsYmFjayBtZXNzYWdlJyxcbiAqICAgYnV0dG9uczogWy4uLl0sXG4gKiAgIGNvbG9yOiAncHJpbWFyeScsXG4gKiAgIHNpemU6ICdtZWRpdW0nXG4gKiB9XCI+PC92YWwtcHJvbXB0ZXI+XG4gKlxuICogQGlucHV0IHByb3BzOiBQcm9tcHRlck1ldGFkYXRhIC0gQ29uZmlndXJhdGlvbiBmb3IgdGhlIHByb21wdGVyIChjb250ZW50LCBidXR0b25zLCBocmVmcywgY29sb3IsIHNpemUsIGV0Yy4pXG4gKiBAb3V0cHV0IG9uQ2xpY2sgLSBFbWl0cyB3aGVuIGFuIGFjdGlvbiBpcyBjbGlja2VkXG4gKi9cbmV4cG9ydCBjbGFzcyBQcm9tcHRlckNvbXBvbmVudCBpbXBsZW1lbnRzIE9uSW5pdCB7XG4gIHByaXZhdGUgbGFuZ1NlcnZpY2UgPSBpbmplY3QoTGFuZ1NlcnZpY2UpO1xuXG4gIC8qKiBPYnNlcnZhYmxlIGZvciByZWFjdGl2ZSBjb250ZW50IC0gZm9sbG93cyB2YWwtdGV4dC92YWwtYnV0dG9uIHBhdHRlcm4gKi9cbiAgZGlzcGxheUNvbnRlbnQkOiBPYnNlcnZhYmxlPHN0cmluZz4gPSBvZignJyk7XG5cbiAgLyoqXG4gICAqIFByb21wdGVyIGNvbmZpZ3VyYXRpb24gb2JqZWN0LlxuICAgKi9cbiAgQElucHV0KClcbiAgcHJvcHM6IFByb21wdGVyTWV0YWRhdGE7XG5cbiAgQE91dHB1dCgpXG4gIG9uQ2xpY2sgPSBuZXcgRXZlbnRFbWl0dGVyPHN0cmluZz4oKTtcblxuICBuZ09uSW5pdCgpIHtcbiAgICB0aGlzLnNldHVwQ29udGVudCgpO1xuICB9XG5cbiAgLyoqXG4gICAqIFNldCB1cCBjb250ZW50IE9ic2VydmFibGUuXG4gICAqIEZvbGxvd3MgdGhlIHNhbWUgcGF0dGVybiBhcyB2YWwtdGV4dCBhbmQgdmFsLWJ1dHRvbiBmb3IgY29uc2lzdGVuY3kuXG4gICAqL1xuICBwcml2YXRlIHNldHVwQ29udGVudCgpOiB2b2lkIHtcbiAgICBpZiAoIXRoaXMucHJvcHMpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgaWYgKHRoaXMucHJvcHMuY29udGVudENsYXNzICYmIHRoaXMucHJvcHMuY29udGVudEtleSkge1xuICAgICAgLy8gUmVhY3RpdmUgY29udGVudCAtIE9ic2VydmFibGUgY29ubmVjdGVkIHRvIGN1cnJlbnRMYW5nJFxuICAgICAgdGhpcy5kaXNwbGF5Q29udGVudCQgPSB0aGlzLmxhbmdTZXJ2aWNlLmdldENvbnRlbnQoXG4gICAgICAgIHRoaXMucHJvcHMuY29udGVudENsYXNzLFxuICAgICAgICB0aGlzLnByb3BzLmNvbnRlbnRLZXksXG4gICAgICAgIHRoaXMucHJvcHMuY29udGVudFxuICAgICAgKTtcbiAgICB9IGVsc2Uge1xuICAgICAgLy8gU3RhdGljIGNvbnRlbnQgd3JhcHBlZCBpbiBPYnNlcnZhYmxlXG4gICAgICB0aGlzLmRpc3BsYXlDb250ZW50JCA9IG9mKHRoaXMucHJvcHMuY29udGVudCk7XG4gICAgfVxuICB9XG5cbiAgY2xpY2tIYW5kbGVyKHRva2VuPzogc3RyaW5nKSB7XG4gICAgdGhpcy5vbkNsaWNrLmVtaXQodG9rZW4pO1xuICB9XG59XG4iXX0=
85
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvbXB0ZXIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9jb21wb25lbnRzL21vbGVjdWxlcy9wcm9tcHRlci9wcm9tcHRlci5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsWUFBWSxFQUFFLEtBQUssRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDdkUsT0FBTyxFQUFFLE9BQU8sRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBQ3BELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSwrQkFBK0IsQ0FBQztBQUM3RCxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0saUNBQWlDLENBQUM7QUFDaEUsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sd0NBQXdDLENBQUM7OztBQXVDOUUsTUFBTSxPQUFPLGlCQUFpQjtJQXBDOUI7UUF5Q0UsWUFBTyxHQUFHLElBQUksWUFBWSxFQUFVLENBQUM7S0FLdEM7SUFIQyxZQUFZLENBQUMsS0FBYztRQUN6QixJQUFJLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUMzQixDQUFDOytHQVRVLGlCQUFpQjttR0FBakIsaUJBQWlCLHFJQWhDbEI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBNkJULDZpR0E5QlMsWUFBWSw0SkFBRSxPQUFPLGdGQUFFLFlBQVksNkZBQUUsb0JBQW9CLHNHQUFFLGFBQWE7OzRGQWlDdkUsaUJBQWlCO2tCQXBDN0IsU0FBUzsrQkFDRSxjQUFjLGNBQ1osSUFBSSxXQUNQLENBQUMsWUFBWSxFQUFFLE9BQU8sRUFBRSxZQUFZLEVBQUUsb0JBQW9CLEVBQUUsYUFBYSxDQUFDLFlBQ3pFOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQTZCVDs4QkFLRCxLQUFLO3NCQURKLEtBQUs7Z0JBSU4sT0FBTztzQkFETixNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IENvbXBvbmVudCwgRXZlbnRFbWl0dGVyLCBJbnB1dCwgT3V0cHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBJb25UZXh0IH0gZnJvbSAnQGlvbmljL2FuZ3VsYXIvc3RhbmRhbG9uZSc7XG5pbXBvcnQgeyBCb3hDb21wb25lbnQgfSBmcm9tICcuLi8uLi9hdG9tcy9ib3gvYm94LmNvbXBvbmVudCc7XG5pbXBvcnQgeyBIcmVmQ29tcG9uZW50IH0gZnJvbSAnLi4vLi4vYXRvbXMvaHJlZi9ocmVmLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBCdXR0b25Hcm91cENvbXBvbmVudCB9IGZyb20gJy4uL2J1dHRvbi1ncm91cC9idXR0b24tZ3JvdXAuY29tcG9uZW50JztcbmltcG9ydCB7IFByb21wdGVyTWV0YWRhdGEgfSBmcm9tICcuL3R5cGVzJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAndmFsLXByb21wdGVyJyxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgSW9uVGV4dCwgQm94Q29tcG9uZW50LCBCdXR0b25Hcm91cENvbXBvbmVudCwgSHJlZkNvbXBvbmVudF0sXG4gIHRlbXBsYXRlOiBgXG4gICAgPHZhbC1ib3hcbiAgICAgIGNsYXNzPVwidGVsZXByb21wdGVyLWNvbnRhaW5lclwiXG4gICAgICBbcHJvcHNdPVwieyBpY29uOiAnJywgYm9yZGVyZWQ6IHByb3BzLmJvcmRlcmVkLCBjb2xvcjogcHJvcHMuY29sb3IsIGxlZnRCb3JkZXI6IGZhbHNlLCBwYWRkaW5nOiBwcm9wcy5wYWRkaW5nIH1cIlxuICAgICAgKG9uQ2xpY2spPVwiY2xpY2tIYW5kbGVyKClcIlxuICAgID5cbiAgICAgIDxkaXYgY2xhc3M9XCJjb250YWluZXJcIiBbY2xhc3MudGVsZXByb21wdGVyLWNvbnRlbnRdPVwicHJvcHMudGVsZXByb21wdGVyXCIgYm9keT5cbiAgICAgICAgPGRpdj5cbiAgICAgICAgICA8aW9uLXRleHQ+XG4gICAgICAgICAgICA8cCBbY2xhc3NdPVwicHJvcHMuc2l6ZVwiIGNsYXNzPVwiYm9sZFwiPnt7IHByb3BzLmNvbnRlbnQgfX08L3A+XG4gICAgICAgICAgPC9pb24tdGV4dD5cbiAgICAgICAgPC9kaXY+XG4gICAgICAgIEBpZiAocHJvcHMuYnV0dG9ucyB8fCBwcm9wcy5ocmVmcykge1xuICAgICAgICAgIDxkaXYgY2xhc3M9XCJhY3Rpb25zLWNvbnRhaW5lclwiPlxuICAgICAgICAgICAgPHZhbC1idXR0b24tZ3JvdXBcbiAgICAgICAgICAgICAgY2xhc3M9XCJidXR0b25zLWNvbnRhaW5lclwiXG4gICAgICAgICAgICAgIFtwcm9wc109XCJ7IGJ1dHRvbnM6IHByb3BzLmJ1dHRvbnMsIHBvc2l0aW9uOiAncmlnaHQnLCBjb2x1bW5lZDogZmFsc2UgfVwiXG4gICAgICAgICAgICAgIChvbkNsaWNrKT1cImNsaWNrSGFuZGxlcigkZXZlbnQpXCJcbiAgICAgICAgICAgID48L3ZhbC1idXR0b24tZ3JvdXA+XG4gICAgICAgICAgICA8dmFsLWhyZWZcbiAgICAgICAgICAgICAgY2xhc3M9XCJsaW5rXCJcbiAgICAgICAgICAgICAgKm5nRm9yPVwibGV0IGwgb2YgcHJvcHMuaHJlZnNcIlxuICAgICAgICAgICAgICBbcHJvcHNdPVwibFwiXG4gICAgICAgICAgICAgIChvbkNsaWNrKT1cImNsaWNrSGFuZGxlcigkZXZlbnQpXCJcbiAgICAgICAgICAgID48L3ZhbC1ocmVmPlxuICAgICAgICAgIDwvZGl2PlxuICAgICAgICB9XG4gICAgICA8L2Rpdj5cbiAgICA8L3ZhbC1ib3g+XG4gIGAsXG4gIHN0eWxlVXJsczogWycuL3Byb21wdGVyLmNvbXBvbmVudC5zY3NzJ10sXG59KVxuZXhwb3J0IGNsYXNzIFByb21wdGVyQ29tcG9uZW50IHtcbiAgQElucHV0KClcbiAgcHJvcHM6IFByb21wdGVyTWV0YWRhdGE7XG5cbiAgQE91dHB1dCgpXG4gIG9uQ2xpY2sgPSBuZXcgRXZlbnRFbWl0dGVyPHN0cmluZz4oKTtcblxuICBjbGlja0hhbmRsZXIodG9rZW4/OiBzdHJpbmcpIHtcbiAgICB0aGlzLm9uQ2xpY2suZW1pdCh0b2tlbik7XG4gIH1cbn1cbiJdfQ==
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL3Byb21wdGVyL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb2xvciB9IGZyb20gJ0Bpb25pYy9jb3JlJztcbmltcG9ydCB7IEhyZWZNZXRhZGF0YSB9IGZyb20gJy4uLy4uL2F0b21zL2hyZWYvdHlwZXMnO1xuaW1wb3J0IHsgQnV0dG9uTWV0YWRhdGEgfSBmcm9tICcuLi8uLi90eXBlcyc7XG5cbi8qKlxuICogUHJvcHMgZm9yIHZhbC1wcm9tcHRlciBjb21wb25lbnQuXG4gKlxuICogQHByb3BlcnR5IHRlbGVwcm9tcHRlciAtIFdoZXRoZXIgdG8gdXNlIHRlbGVwcm9tcHRlciBzdHlsZS5cbiAqIEBwcm9wZXJ0eSBib3JkZXJlZCAtIFdoZXRoZXIgdGhlIGJveCBoYXMgYSBib3JkZXIuXG4gKiBAcHJvcGVydHkgY29sb3IgLSBUaGUgYmFja2dyb3VuZCBjb2xvciBvZiB0aGUgYm94IChvcHRpb25hbCkuXG4gKiBAcHJvcGVydHkgY29udGVudCAtIFRoZSBtYWluIHRleHQgY29udGVudCAodXNlZCBhcyBmYWxsYmFjayBpZiBpMThuIGtleXMgcHJvdmlkZWQpLlxuICogQHByb3BlcnR5IHNpemUgLSBUaGUgc2l6ZSBvZiB0aGUgdGV4dCBhbmQgYm94ICgnc21hbGwnIHwgJ21lZGl1bScgfCAnbGFyZ2UnIHwgJ3hsYXJnZScpLlxuICogQHByb3BlcnR5IGJ1dHRvbnMgLSBBcnJheSBvZiBidXR0b24gY29uZmlndXJhdGlvbnMgKG9wdGlvbmFsKS5cbiAqIEBwcm9wZXJ0eSBocmVmcyAtIEFycmF5IG9mIGxpbmsgY29uZmlndXJhdGlvbnMgKG9wdGlvbmFsKS5cbiAqIEBwcm9wZXJ0eSBwYWRkaW5nIC0gQ3VzdG9tIHBhZGRpbmcgZm9yIHRoZSBib3ggKG9wdGlvbmFsKS5cbiAqIEBwcm9wZXJ0eSBjb250ZW50Q2xhc3MgLSBpMThuIGNvbnRlbnQgY2xhc3MgbmFtZSAob3B0aW9uYWwsIGZvciByZWFjdGl2ZSB0cmFuc2xhdGlvbnMpLlxuICogQHByb3BlcnR5IGNvbnRlbnRLZXkgLSBpMThuIGNvbnRlbnQga2V5IChvcHRpb25hbCwgZm9yIHJlYWN0aXZlIHRyYW5zbGF0aW9ucykuXG4gKi9cbmV4cG9ydCB0eXBlIFByb21wdGVyTWV0YWRhdGEgPSB7XG4gIHRlbGVwcm9tcHRlcjogYm9vbGVhbjtcbiAgYm9yZGVyZWQ6IGJvb2xlYW47XG4gIGNvbG9yPzogQ29sb3I7XG4gIGNvbnRlbnQ6IHN0cmluZztcbiAgc2l6ZTogJ3NtYWxsJyB8ICdtZWRpdW0nIHwgJ2xhcmdlJyB8ICd4bGFyZ2UnO1xuICBidXR0b25zPzogQnV0dG9uTWV0YWRhdGFbXTtcbiAgaHJlZnM/OiBIcmVmTWV0YWRhdGFbXTtcbiAgcGFkZGluZz86IHN0cmluZztcbiAgLyoqIGkxOG4gY29udGVudCBjbGFzcyBuYW1lIGZvciByZWFjdGl2ZSB0cmFuc2xhdGlvbnMgKi9cbiAgY29udGVudENsYXNzPzogc3RyaW5nO1xuICAvKiogaTE4biBjb250ZW50IGtleSBmb3IgcmVhY3RpdmUgdHJhbnNsYXRpb25zICovXG4gIGNvbnRlbnRLZXk/OiBzdHJpbmc7XG59O1xuIl19
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL3Byb21wdGVyL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb2xvciB9IGZyb20gJ0Bpb25pYy9jb3JlJztcbmltcG9ydCB7IEhyZWZNZXRhZGF0YSB9IGZyb20gJy4uLy4uL2F0b21zL2hyZWYvdHlwZXMnO1xuaW1wb3J0IHsgQnV0dG9uTWV0YWRhdGEgfSBmcm9tICcuLi8uLi90eXBlcyc7XG5cbi8qKlxuICogUHJvcHMgZm9yIHZhbC1wcm9tcHRlciBjb21wb25lbnQuXG4gKlxuICogQHByb3BlcnR5IHRlbGVwcm9tcHRlciAtIFdoZXRoZXIgdG8gdXNlIHRlbGVwcm9tcHRlciBzdHlsZS5cbiAqIEBwcm9wZXJ0eSBib3JkZXJlZCAtIFdoZXRoZXIgdGhlIGJveCBoYXMgYSBib3JkZXIuXG4gKiBAcHJvcGVydHkgY29sb3IgLSBUaGUgYmFja2dyb3VuZCBjb2xvciBvZiB0aGUgYm94IChvcHRpb25hbCkuXG4gKiBAcHJvcGVydHkgY29udGVudCAtIFRoZSBtYWluIHRleHQgY29udGVudC5cbiAqIEBwcm9wZXJ0eSBzaXplIC0gVGhlIHNpemUgb2YgdGhlIHRleHQgYW5kIGJveCAoJ3NtYWxsJyB8ICdtZWRpdW0nIHwgJ2xhcmdlJyB8ICd4bGFyZ2UnKS5cbiAqIEBwcm9wZXJ0eSBidXR0b25zIC0gQXJyYXkgb2YgYnV0dG9uIGNvbmZpZ3VyYXRpb25zIChvcHRpb25hbCkuXG4gKiBAcHJvcGVydHkgaHJlZnMgLSBBcnJheSBvZiBsaW5rIGNvbmZpZ3VyYXRpb25zIChvcHRpb25hbCkuXG4gKiBAcHJvcGVydHkgcGFkZGluZyAtIEN1c3RvbSBwYWRkaW5nIGZvciB0aGUgYm94IChvcHRpb25hbCkuXG4gKi9cbmV4cG9ydCB0eXBlIFByb21wdGVyTWV0YWRhdGEgPSB7XG4gIHRlbGVwcm9tcHRlcjogYm9vbGVhbjtcbiAgYm9yZGVyZWQ6IGJvb2xlYW47XG4gIGNvbG9yPzogQ29sb3I7XG4gIGNvbnRlbnQ6IHN0cmluZztcbiAgc2l6ZTogJ3NtYWxsJyB8ICdtZWRpdW0nIHwgJ2xhcmdlJyB8ICd4bGFyZ2UnO1xuICBidXR0b25zPzogQnV0dG9uTWV0YWRhdGFbXTtcbiAgaHJlZnM/OiBIcmVmTWV0YWRhdGFbXTtcbiAgcGFkZGluZz86IHN0cmluZztcbn07XG4iXX0=
@@ -44,7 +44,8 @@ export const VALTECH_FIREBASE_CONFIG = new InjectionToken('ValtechFirebaseConfig
44
44
  * ```
45
45
  */
46
46
  export function provideValtechFirebase(config) {
47
- return makeEnvironmentProviders([
47
+ // Construir array de providers base
48
+ const providers = [
48
49
  // Guardar configuración para uso en servicios
49
50
  { provide: VALTECH_FIREBASE_CONFIG, useValue: config },
50
51
  // Inicializar Firebase App
@@ -87,9 +88,13 @@ export function provideValtechFirebase(config) {
87
88
  }
88
89
  return storage;
89
90
  }),
90
- // Messaging (FCM) - solo si está soportado
91
- provideMessaging(() => getMessaging()),
92
- ]);
91
+ ];
92
+ // Messaging (FCM) - solo si está explícitamente habilitado
93
+ // Requiere Service Worker configurado, puede congelar la app si no está disponible
94
+ if (config.enableMessaging) {
95
+ providers.push(provideMessaging(() => getMessaging()));
96
+ }
97
+ return makeEnvironmentProviders(providers);
93
98
  }
94
99
  /**
95
100
  * Verifica si los emuladores están configurados.
@@ -100,4 +105,4 @@ export function provideValtechFirebase(config) {
100
105
  export function hasEmulators(config) {
101
106
  return !!(config.emulator?.firestore || config.emulator?.auth || config.emulator?.storage);
102
107
  }
103
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"config.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/config.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Firebase Configuration\n *\n * Configuración e inicialización de Firebase para aplicaciones Angular.\n * Usa provideValtechFirebase() en el bootstrap de tu aplicación.\n */\n\nimport { EnvironmentProviders, InjectionToken, makeEnvironmentProviders } from '@angular/core';\nimport { initializeApp, provideFirebaseApp } from '@angular/fire/app';\nimport { connectAuthEmulator, getAuth, provideAuth } from '@angular/fire/auth';\nimport {\n  connectFirestoreEmulator,\n  enableIndexedDbPersistence,\n  getFirestore,\n  provideFirestore,\n} from '@angular/fire/firestore';\nimport { getMessaging, provideMessaging } from '@angular/fire/messaging';\nimport { connectStorageEmulator, getStorage, provideStorage } from '@angular/fire/storage';\n\nimport { ValtechFirebaseConfig } from './types';\n\n/**\n * Token de inyección para la configuración de Firebase.\n * Usado internamente por los servicios de Firebase.\n */\nexport const VALTECH_FIREBASE_CONFIG = new InjectionToken<ValtechFirebaseConfig>(\n  'ValtechFirebaseConfig'\n);\n\n/**\n * Provee Firebase a la aplicación Angular.\n *\n * @param config - Configuración de Firebase\n * @returns EnvironmentProviders para usar en bootstrapApplication\n *\n * @example\n * ```typescript\n * // main.ts\n * import { bootstrapApplication } from '@angular/platform-browser';\n * import { provideValtechFirebase } from 'valtech-components';\n * import { environment } from './environments/environment';\n *\n * bootstrapApplication(AppComponent, {\n *   providers: [\n *     provideValtechFirebase({\n *       firebase: environment.firebase,\n *       persistence: true,\n *       emulator: environment.useEmulators ? {\n *         firestore: { host: 'localhost', port: 8080 },\n *         auth: { host: 'localhost', port: 9099 },\n *         storage: { host: 'localhost', port: 9199 },\n *       } : undefined,\n *     }),\n *   ],\n * });\n * ```\n */\nexport function provideValtechFirebase(config: ValtechFirebaseConfig): EnvironmentProviders {\n  return makeEnvironmentProviders([\n    // Guardar configuración para uso en servicios\n    { provide: VALTECH_FIREBASE_CONFIG, useValue: config },\n\n    // Inicializar Firebase App\n    provideFirebaseApp(() => initializeApp(config.firebase)),\n\n    // Firestore con soporte para emuladores y persistencia\n    provideFirestore(() => {\n      const firestore = getFirestore();\n\n      // Conectar a emulador si está configurado\n      if (config.emulator?.firestore) {\n        connectFirestoreEmulator(\n          firestore,\n          config.emulator.firestore.host,\n          config.emulator.firestore.port\n        );\n      }\n\n      // Habilitar persistencia offline si está configurada\n      if (config.persistence) {\n        enableIndexedDbPersistence(firestore).catch((err) => {\n          if (err.code === 'failed-precondition') {\n            console.warn(\n              '[ValtechFirebase] Persistencia no disponible: múltiples pestañas abiertas'\n            );\n          } else if (err.code === 'unimplemented') {\n            console.warn('[ValtechFirebase] Persistencia no soportada en este navegador');\n          }\n        });\n      }\n\n      return firestore;\n    }),\n\n    // Auth con soporte para emulador\n    provideAuth(() => {\n      const auth = getAuth();\n\n      // Conectar a emulador si está configurado\n      if (config.emulator?.auth) {\n        connectAuthEmulator(\n          auth,\n          `http://${config.emulator.auth.host}:${config.emulator.auth.port}`,\n          { disableWarnings: true }\n        );\n      }\n\n      return auth;\n    }),\n\n    // Storage con soporte para emulador\n    provideStorage(() => {\n      const storage = getStorage();\n\n      // Conectar a emulador si está configurado\n      if (config.emulator?.storage) {\n        connectStorageEmulator(\n          storage,\n          config.emulator.storage.host,\n          config.emulator.storage.port\n        );\n      }\n\n      return storage;\n    }),\n\n    // Messaging (FCM) - solo si está soportado\n    provideMessaging(() => getMessaging()),\n  ]);\n}\n\n/**\n * Verifica si los emuladores están configurados.\n *\n * @param config - Configuración de Firebase\n * @returns true si hay al menos un emulador configurado\n */\nexport function hasEmulators(config: ValtechFirebaseConfig): boolean {\n  return !!(config.emulator?.firestore || config.emulator?.auth || config.emulator?.storage);\n}\n"]}
108
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"config.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/config.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Firebase Configuration\n *\n * Configuración e inicialización de Firebase para aplicaciones Angular.\n * Usa provideValtechFirebase() en el bootstrap de tu aplicación.\n */\n\nimport { EnvironmentProviders, InjectionToken, makeEnvironmentProviders } from '@angular/core';\nimport { initializeApp, provideFirebaseApp } from '@angular/fire/app';\nimport { connectAuthEmulator, getAuth, provideAuth } from '@angular/fire/auth';\nimport {\n  connectFirestoreEmulator,\n  enableIndexedDbPersistence,\n  getFirestore,\n  provideFirestore,\n} from '@angular/fire/firestore';\nimport { getMessaging, provideMessaging } from '@angular/fire/messaging';\nimport { connectStorageEmulator, getStorage, provideStorage } from '@angular/fire/storage';\n\nimport { ValtechFirebaseConfig } from './types';\n\n/**\n * Token de inyección para la configuración de Firebase.\n * Usado internamente por los servicios de Firebase.\n */\nexport const VALTECH_FIREBASE_CONFIG = new InjectionToken<ValtechFirebaseConfig>(\n  'ValtechFirebaseConfig'\n);\n\n/**\n * Provee Firebase a la aplicación Angular.\n *\n * @param config - Configuración de Firebase\n * @returns EnvironmentProviders para usar en bootstrapApplication\n *\n * @example\n * ```typescript\n * // main.ts\n * import { bootstrapApplication } from '@angular/platform-browser';\n * import { provideValtechFirebase } from 'valtech-components';\n * import { environment } from './environments/environment';\n *\n * bootstrapApplication(AppComponent, {\n *   providers: [\n *     provideValtechFirebase({\n *       firebase: environment.firebase,\n *       persistence: true,\n *       emulator: environment.useEmulators ? {\n *         firestore: { host: 'localhost', port: 8080 },\n *         auth: { host: 'localhost', port: 9099 },\n *         storage: { host: 'localhost', port: 9199 },\n *       } : undefined,\n *     }),\n *   ],\n * });\n * ```\n */\nexport function provideValtechFirebase(config: ValtechFirebaseConfig): EnvironmentProviders {\n  // Construir array de providers base\n  const providers: any[] = [\n    // Guardar configuración para uso en servicios\n    { provide: VALTECH_FIREBASE_CONFIG, useValue: config },\n\n    // Inicializar Firebase App\n    provideFirebaseApp(() => initializeApp(config.firebase)),\n\n    // Firestore con soporte para emuladores y persistencia\n    provideFirestore(() => {\n      const firestore = getFirestore();\n\n      // Conectar a emulador si está configurado\n      if (config.emulator?.firestore) {\n        connectFirestoreEmulator(\n          firestore,\n          config.emulator.firestore.host,\n          config.emulator.firestore.port\n        );\n      }\n\n      // Habilitar persistencia offline si está configurada\n      if (config.persistence) {\n        enableIndexedDbPersistence(firestore).catch((err) => {\n          if (err.code === 'failed-precondition') {\n            console.warn(\n              '[ValtechFirebase] Persistencia no disponible: múltiples pestañas abiertas'\n            );\n          } else if (err.code === 'unimplemented') {\n            console.warn('[ValtechFirebase] Persistencia no soportada en este navegador');\n          }\n        });\n      }\n\n      return firestore;\n    }),\n\n    // Auth con soporte para emulador\n    provideAuth(() => {\n      const auth = getAuth();\n\n      // Conectar a emulador si está configurado\n      if (config.emulator?.auth) {\n        connectAuthEmulator(\n          auth,\n          `http://${config.emulator.auth.host}:${config.emulator.auth.port}`,\n          { disableWarnings: true }\n        );\n      }\n\n      return auth;\n    }),\n\n    // Storage con soporte para emulador\n    provideStorage(() => {\n      const storage = getStorage();\n\n      // Conectar a emulador si está configurado\n      if (config.emulator?.storage) {\n        connectStorageEmulator(\n          storage,\n          config.emulator.storage.host,\n          config.emulator.storage.port\n        );\n      }\n\n      return storage;\n    }),\n  ];\n\n  // Messaging (FCM) - solo si está explícitamente habilitado\n  // Requiere Service Worker configurado, puede congelar la app si no está disponible\n  if (config.enableMessaging) {\n    providers.push(provideMessaging(() => getMessaging()));\n  }\n\n  return makeEnvironmentProviders(providers);\n}\n\n/**\n * Verifica si los emuladores están configurados.\n *\n * @param config - Configuración de Firebase\n * @returns true si hay al menos un emulador configurado\n */\nexport function hasEmulators(config: ValtechFirebaseConfig): boolean {\n  return !!(config.emulator?.firestore || config.emulator?.auth || config.emulator?.storage);\n}\n"]}
@@ -5,4 +5,4 @@
5
5
  * Todos los modelos de Firestore deben extender FirestoreDocument.
6
6
  */
7
7
  export {};
8
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/types.ts"],"names":[],"mappings":"AAAA;;;;;GAKG","sourcesContent":["/**\n * Firebase Types\n *\n * Tipos e interfaces para la integración de Firebase en valtech-components.\n * Todos los modelos de Firestore deben extender FirestoreDocument.\n */\n\n// ============================================================================\n// CONFIGURACIÓN\n// ============================================================================\n\n/**\n * Configuración de Firebase (valores de firebaseConfig)\n */\nexport interface FirebaseConfig {\n  apiKey: string;\n  authDomain: string;\n  projectId: string;\n  storageBucket: string;\n  messagingSenderId: string;\n  appId: string;\n  measurementId?: string;\n}\n\n/**\n * Configuración de emuladores para desarrollo local\n */\nexport interface EmulatorConfig {\n  firestore?: {\n    host: string;\n    port: number;\n  };\n  auth?: {\n    host: string;\n    port: number;\n  };\n  storage?: {\n    host: string;\n    port: number;\n  };\n}\n\n/**\n * Configuración completa de Valtech Firebase\n */\nexport interface ValtechFirebaseConfig {\n  /** Configuración de Firebase */\n  firebase: FirebaseConfig;\n  /** Configuración de emuladores (opcional, para desarrollo) */\n  emulator?: EmulatorConfig;\n  /** Habilitar persistencia offline de Firestore (default: false) */\n  persistence?: boolean;\n  /** VAPID key para Firebase Cloud Messaging */\n  messagingVapidKey?: string;\n}\n\n// ============================================================================\n// FIRESTORE - DOCUMENTOS\n// ============================================================================\n\n/**\n * Interface base para todos los documentos de Firestore.\n * Todos los modelos deben extender esta interface.\n *\n * @example\n * interface User extends FirestoreDocument {\n *   name: string;\n *   email: string;\n * }\n */\nexport interface FirestoreDocument {\n  /** ID del documento (asignado por Firestore) */\n  id?: string;\n  /** Fecha de creación (manejada automáticamente) */\n  createdAt?: Date;\n  /** Fecha de última actualización (manejada automáticamente) */\n  updatedAt?: Date;\n}\n\n// ============================================================================\n// FIRESTORE - QUERIES\n// ============================================================================\n\n/**\n * Operadores disponibles para cláusulas where\n */\nexport type WhereOperator =\n  | '=='\n  | '!='\n  | '<'\n  | '<='\n  | '>'\n  | '>='\n  | 'array-contains'\n  | 'array-contains-any'\n  | 'in'\n  | 'not-in';\n\n/**\n * Cláusula where para filtrar documentos\n */\nexport interface WhereClause {\n  /** Campo a filtrar */\n  field: string;\n  /** Operador de comparación */\n  operator: WhereOperator;\n  /** Valor a comparar */\n  value: unknown;\n}\n\n/**\n * Dirección de ordenamiento\n */\nexport type OrderDirection = 'asc' | 'desc';\n\n/**\n * Cláusula orderBy para ordenar resultados\n */\nexport interface OrderByClause {\n  /** Campo por el cual ordenar */\n  field: string;\n  /** Dirección del ordenamiento */\n  direction: OrderDirection;\n}\n\n/**\n * Opciones para queries de Firestore\n */\nexport interface QueryOptions {\n  /** Filtros where (AND entre todos) */\n  where?: WhereClause[];\n  /** Ordenamiento de resultados */\n  orderBy?: OrderByClause[];\n  /** Límite de documentos a retornar */\n  limit?: number;\n  /** Cursor para paginación: empezar después de este documento */\n  startAfter?: unknown;\n  /** Cursor para paginación: empezar en este documento */\n  startAt?: unknown;\n  /** Cursor para paginación: terminar antes de este documento */\n  endBefore?: unknown;\n  /** Cursor para paginación: terminar en este documento */\n  endAt?: unknown;\n}\n\n/**\n * Opciones adicionales para subscripciones real-time\n */\nexport interface SubscriptionOptions extends QueryOptions {\n  /** Incluir cambios de metadata (ej: pendingWrites) */\n  includeMetadataChanges?: boolean;\n}\n\n/**\n * Resultado de una query paginada\n */\nexport interface PaginatedResult<T> {\n  /** Documentos de la página actual */\n  data: T[];\n  /** Indica si hay más páginas disponibles */\n  hasMore: boolean;\n  /** Cursor para la siguiente página (pasar a startAfter) */\n  lastDoc: unknown;\n  /** Total de documentos (opcional, requiere query adicional) */\n  total?: number;\n}\n\n// ============================================================================\n// STORAGE\n// ============================================================================\n\n/**\n * Estado de una operación de upload\n */\nexport type UploadState = 'running' | 'paused' | 'success' | 'canceled' | 'error';\n\n/**\n * Progreso de upload de archivo\n */\nexport interface UploadProgress {\n  /** Bytes transferidos hasta ahora */\n  bytesTransferred: number;\n  /** Total de bytes a transferir */\n  totalBytes: number;\n  /** Porcentaje completado (0-100) */\n  percentage: number;\n  /** Estado actual del upload */\n  state: UploadState;\n}\n\n/**\n * Resultado de un upload completado\n */\nexport interface UploadResult {\n  /** URL de descarga del archivo */\n  downloadUrl: string;\n  /** Ruta completa en Storage */\n  fullPath: string;\n  /** Nombre del archivo */\n  name: string;\n  /** Tamaño en bytes */\n  size: number;\n  /** Tipo MIME del archivo */\n  contentType: string;\n  /** Metadata personalizada */\n  metadata: Record<string, string>;\n}\n\n/**\n * Metadata para archivos en Storage\n */\nexport interface StorageMetadata {\n  /** Tipo MIME del archivo */\n  contentType?: string;\n  /** Metadata personalizada (key-value) */\n  customMetadata?: Record<string, string>;\n  /** Control de caché HTTP */\n  cacheControl?: string;\n}\n\n/**\n * Resultado de listar archivos en Storage\n */\nexport interface StorageListResult {\n  /** Rutas de los archivos encontrados */\n  items: string[];\n  /** Token para la siguiente página (si hay más) */\n  nextPageToken?: string;\n}\n\n// ============================================================================\n// AUTH / SESSION\n// ============================================================================\n\n/**\n * Información del usuario de Firebase (simplificada)\n */\nexport interface FirebaseUser {\n  /** UID único del usuario */\n  uid: string;\n  /** Email del usuario */\n  email: string | null;\n  /** Nombre para mostrar */\n  displayName: string | null;\n  /** URL de foto de perfil */\n  photoURL: string | null;\n  /** Email verificado */\n  emailVerified: boolean;\n  /** Usuario anónimo */\n  isAnonymous: boolean;\n  /** Proveedor de autenticación */\n  providerId: string;\n}\n\n/**\n * Estado de la sesión de Firebase\n */\nexport interface SessionState {\n  /** Usuario actual (null si no autenticado) */\n  user: FirebaseUser | null;\n  /** Indica si el usuario está autenticado */\n  isAuthenticated: boolean;\n  /** Indica si se está cargando el estado de auth */\n  isLoading: boolean;\n  /** Error de autenticación (si lo hay) */\n  error: Error | null;\n}\n\n// ============================================================================\n// MESSAGING (FCM)\n// ============================================================================\n\n/**\n * Estado del permiso de notificaciones\n */\nexport type NotificationPermission = 'granted' | 'denied' | 'default';\n\n/**\n * Payload de una notificación push\n */\nexport interface NotificationPayload {\n  /** Título de la notificación */\n  title?: string;\n  /** Cuerpo del mensaje */\n  body?: string;\n  /** URL de imagen */\n  image?: string;\n  /** Datos personalizados */\n  data?: Record<string, string>;\n  /** ID del mensaje de FCM */\n  messageId?: string;\n}\n\n/**\n * Acción de navegación desde una notificación\n */\nexport interface NotificationAction {\n  /** Ruta interna de la app (ej: '/orders/123') */\n  route?: string;\n  /** URL externa (ej: 'https://example.com') */\n  url?: string;\n  /** Parámetros de query string */\n  queryParams?: Record<string, string>;\n  /** Tipo de acción personalizada */\n  actionType?: string;\n  /** Datos adicionales para la acción */\n  actionData?: Record<string, unknown>;\n}\n\n/**\n * Evento de click en una notificación\n */\nexport interface NotificationClickEvent {\n  /** Payload original de la notificación */\n  notification: NotificationPayload;\n  /** Acción de navegación extraída */\n  action: NotificationAction;\n  /** Timestamp del click */\n  timestamp: Date;\n}\n\n// ============================================================================\n// ERRORES\n// ============================================================================\n\n/**\n * Códigos de error de Firebase\n */\nexport type FirebaseErrorCode =\n  | 'permission-denied'\n  | 'not-found'\n  | 'already-exists'\n  | 'resource-exhausted'\n  | 'cancelled'\n  | 'unknown'\n  | 'invalid-argument'\n  | 'deadline-exceeded'\n  | 'unauthenticated';\n\n/**\n * Error de Firebase tipado\n */\nexport interface FirebaseError {\n  /** Código del error */\n  code: FirebaseErrorCode;\n  /** Mensaje de error (en español) */\n  message: string;\n  /** Error original de Firebase */\n  originalError?: unknown;\n}\n"]}
8
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../../src/lib/services/firebase/types.ts"],"names":[],"mappings":"AAAA;;;;;GAKG","sourcesContent":["/**\n * Firebase Types\n *\n * Tipos e interfaces para la integración de Firebase en valtech-components.\n * Todos los modelos de Firestore deben extender FirestoreDocument.\n */\n\n// ============================================================================\n// CONFIGURACIÓN\n// ============================================================================\n\n/**\n * Configuración de Firebase (valores de firebaseConfig)\n */\nexport interface FirebaseConfig {\n  apiKey: string;\n  authDomain: string;\n  projectId: string;\n  storageBucket: string;\n  messagingSenderId: string;\n  appId: string;\n  measurementId?: string;\n}\n\n/**\n * Configuración de emuladores para desarrollo local\n */\nexport interface EmulatorConfig {\n  firestore?: {\n    host: string;\n    port: number;\n  };\n  auth?: {\n    host: string;\n    port: number;\n  };\n  storage?: {\n    host: string;\n    port: number;\n  };\n}\n\n/**\n * Configuración completa de Valtech Firebase\n */\nexport interface ValtechFirebaseConfig {\n  /** Configuración de Firebase */\n  firebase: FirebaseConfig;\n  /** Configuración de emuladores (opcional, para desarrollo) */\n  emulator?: EmulatorConfig;\n  /** Habilitar persistencia offline de Firestore (default: false) */\n  persistence?: boolean;\n  /** Habilitar Firebase Cloud Messaging (default: false) - requiere Service Worker */\n  enableMessaging?: boolean;\n  /** VAPID key para Firebase Cloud Messaging */\n  messagingVapidKey?: string;\n}\n\n// ============================================================================\n// FIRESTORE - DOCUMENTOS\n// ============================================================================\n\n/**\n * Interface base para todos los documentos de Firestore.\n * Todos los modelos deben extender esta interface.\n *\n * @example\n * interface User extends FirestoreDocument {\n *   name: string;\n *   email: string;\n * }\n */\nexport interface FirestoreDocument {\n  /** ID del documento (asignado por Firestore) */\n  id?: string;\n  /** Fecha de creación (manejada automáticamente) */\n  createdAt?: Date;\n  /** Fecha de última actualización (manejada automáticamente) */\n  updatedAt?: Date;\n}\n\n// ============================================================================\n// FIRESTORE - QUERIES\n// ============================================================================\n\n/**\n * Operadores disponibles para cláusulas where\n */\nexport type WhereOperator =\n  | '=='\n  | '!='\n  | '<'\n  | '<='\n  | '>'\n  | '>='\n  | 'array-contains'\n  | 'array-contains-any'\n  | 'in'\n  | 'not-in';\n\n/**\n * Cláusula where para filtrar documentos\n */\nexport interface WhereClause {\n  /** Campo a filtrar */\n  field: string;\n  /** Operador de comparación */\n  operator: WhereOperator;\n  /** Valor a comparar */\n  value: unknown;\n}\n\n/**\n * Dirección de ordenamiento\n */\nexport type OrderDirection = 'asc' | 'desc';\n\n/**\n * Cláusula orderBy para ordenar resultados\n */\nexport interface OrderByClause {\n  /** Campo por el cual ordenar */\n  field: string;\n  /** Dirección del ordenamiento */\n  direction: OrderDirection;\n}\n\n/**\n * Opciones para queries de Firestore\n */\nexport interface QueryOptions {\n  /** Filtros where (AND entre todos) */\n  where?: WhereClause[];\n  /** Ordenamiento de resultados */\n  orderBy?: OrderByClause[];\n  /** Límite de documentos a retornar */\n  limit?: number;\n  /** Cursor para paginación: empezar después de este documento */\n  startAfter?: unknown;\n  /** Cursor para paginación: empezar en este documento */\n  startAt?: unknown;\n  /** Cursor para paginación: terminar antes de este documento */\n  endBefore?: unknown;\n  /** Cursor para paginación: terminar en este documento */\n  endAt?: unknown;\n}\n\n/**\n * Opciones adicionales para subscripciones real-time\n */\nexport interface SubscriptionOptions extends QueryOptions {\n  /** Incluir cambios de metadata (ej: pendingWrites) */\n  includeMetadataChanges?: boolean;\n}\n\n/**\n * Resultado de una query paginada\n */\nexport interface PaginatedResult<T> {\n  /** Documentos de la página actual */\n  data: T[];\n  /** Indica si hay más páginas disponibles */\n  hasMore: boolean;\n  /** Cursor para la siguiente página (pasar a startAfter) */\n  lastDoc: unknown;\n  /** Total de documentos (opcional, requiere query adicional) */\n  total?: number;\n}\n\n// ============================================================================\n// STORAGE\n// ============================================================================\n\n/**\n * Estado de una operación de upload\n */\nexport type UploadState = 'running' | 'paused' | 'success' | 'canceled' | 'error';\n\n/**\n * Progreso de upload de archivo\n */\nexport interface UploadProgress {\n  /** Bytes transferidos hasta ahora */\n  bytesTransferred: number;\n  /** Total de bytes a transferir */\n  totalBytes: number;\n  /** Porcentaje completado (0-100) */\n  percentage: number;\n  /** Estado actual del upload */\n  state: UploadState;\n}\n\n/**\n * Resultado de un upload completado\n */\nexport interface UploadResult {\n  /** URL de descarga del archivo */\n  downloadUrl: string;\n  /** Ruta completa en Storage */\n  fullPath: string;\n  /** Nombre del archivo */\n  name: string;\n  /** Tamaño en bytes */\n  size: number;\n  /** Tipo MIME del archivo */\n  contentType: string;\n  /** Metadata personalizada */\n  metadata: Record<string, string>;\n}\n\n/**\n * Metadata para archivos en Storage\n */\nexport interface StorageMetadata {\n  /** Tipo MIME del archivo */\n  contentType?: string;\n  /** Metadata personalizada (key-value) */\n  customMetadata?: Record<string, string>;\n  /** Control de caché HTTP */\n  cacheControl?: string;\n}\n\n/**\n * Resultado de listar archivos en Storage\n */\nexport interface StorageListResult {\n  /** Rutas de los archivos encontrados */\n  items: string[];\n  /** Token para la siguiente página (si hay más) */\n  nextPageToken?: string;\n}\n\n// ============================================================================\n// AUTH / SESSION\n// ============================================================================\n\n/**\n * Información del usuario de Firebase (simplificada)\n */\nexport interface FirebaseUser {\n  /** UID único del usuario */\n  uid: string;\n  /** Email del usuario */\n  email: string | null;\n  /** Nombre para mostrar */\n  displayName: string | null;\n  /** URL de foto de perfil */\n  photoURL: string | null;\n  /** Email verificado */\n  emailVerified: boolean;\n  /** Usuario anónimo */\n  isAnonymous: boolean;\n  /** Proveedor de autenticación */\n  providerId: string;\n}\n\n/**\n * Estado de la sesión de Firebase\n */\nexport interface SessionState {\n  /** Usuario actual (null si no autenticado) */\n  user: FirebaseUser | null;\n  /** Indica si el usuario está autenticado */\n  isAuthenticated: boolean;\n  /** Indica si se está cargando el estado de auth */\n  isLoading: boolean;\n  /** Error de autenticación (si lo hay) */\n  error: Error | null;\n}\n\n// ============================================================================\n// MESSAGING (FCM)\n// ============================================================================\n\n/**\n * Estado del permiso de notificaciones\n */\nexport type NotificationPermission = 'granted' | 'denied' | 'default';\n\n/**\n * Payload de una notificación push\n */\nexport interface NotificationPayload {\n  /** Título de la notificación */\n  title?: string;\n  /** Cuerpo del mensaje */\n  body?: string;\n  /** URL de imagen */\n  image?: string;\n  /** Datos personalizados */\n  data?: Record<string, string>;\n  /** ID del mensaje de FCM */\n  messageId?: string;\n}\n\n/**\n * Acción de navegación desde una notificación\n */\nexport interface NotificationAction {\n  /** Ruta interna de la app (ej: '/orders/123') */\n  route?: string;\n  /** URL externa (ej: 'https://example.com') */\n  url?: string;\n  /** Parámetros de query string */\n  queryParams?: Record<string, string>;\n  /** Tipo de acción personalizada */\n  actionType?: string;\n  /** Datos adicionales para la acción */\n  actionData?: Record<string, unknown>;\n}\n\n/**\n * Evento de click en una notificación\n */\nexport interface NotificationClickEvent {\n  /** Payload original de la notificación */\n  notification: NotificationPayload;\n  /** Acción de navegación extraída */\n  action: NotificationAction;\n  /** Timestamp del click */\n  timestamp: Date;\n}\n\n// ============================================================================\n// ERRORES\n// ============================================================================\n\n/**\n * Códigos de error de Firebase\n */\nexport type FirebaseErrorCode =\n  | 'permission-denied'\n  | 'not-found'\n  | 'already-exists'\n  | 'resource-exhausted'\n  | 'cancelled'\n  | 'unknown'\n  | 'invalid-argument'\n  | 'deadline-exceeded'\n  | 'unauthenticated';\n\n/**\n * Error de Firebase tipado\n */\nexport interface FirebaseError {\n  /** Código del error */\n  code: FirebaseErrorCode;\n  /** Mensaje de error (en español) */\n  message: string;\n  /** Error original de Firebase */\n  originalError?: unknown;\n}\n"]}
@@ -7517,125 +7517,76 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
7517
7517
  type: Input
7518
7518
  }] } });
7519
7519
 
7520
- /**
7521
- * val-prompter
7522
- *
7523
- * Displays a prompt/info box with text, actions (buttons/links), and custom styles.
7524
- * Supports reactive i18n content via contentClass and contentKey props.
7525
- *
7526
- * @example Static content:
7527
- * <val-prompter [props]="{ content: 'Message', buttons: [...], color: 'primary', size: 'medium' }"></val-prompter>
7528
- *
7529
- * @example Reactive i18n content:
7530
- * <val-prompter [props]="{
7531
- * contentClass: 'homePage',
7532
- * contentKey: 'cookieMessage',
7533
- * content: 'Fallback message',
7534
- * buttons: [...],
7535
- * color: 'primary',
7536
- * size: 'medium'
7537
- * }"></val-prompter>
7538
- *
7539
- * @input props: PrompterMetadata - Configuration for the prompter (content, buttons, hrefs, color, size, etc.)
7540
- * @output onClick - Emits when an action is clicked
7541
- */
7542
7520
  class PrompterComponent {
7543
7521
  constructor() {
7544
- this.langService = inject(LangService);
7545
- /** Observable for reactive content - follows val-text/val-button pattern */
7546
- this.displayContent$ = of('');
7547
7522
  this.onClick = new EventEmitter();
7548
7523
  }
7549
- ngOnInit() {
7550
- this.setupContent();
7551
- }
7552
- /**
7553
- * Set up content Observable.
7554
- * Follows the same pattern as val-text and val-button for consistency.
7555
- */
7556
- setupContent() {
7557
- if (!this.props) {
7558
- return;
7559
- }
7560
- if (this.props.contentClass && this.props.contentKey) {
7561
- // Reactive content - Observable connected to currentLang$
7562
- this.displayContent$ = this.langService.getContent(this.props.contentClass, this.props.contentKey, this.props.content);
7563
- }
7564
- else {
7565
- // Static content wrapped in Observable
7566
- this.displayContent$ = of(this.props.content);
7567
- }
7568
- }
7569
7524
  clickHandler(token) {
7570
7525
  this.onClick.emit(token);
7571
7526
  }
7572
7527
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PrompterComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
7573
7528
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: PrompterComponent, isStandalone: true, selector: "val-prompter", inputs: { props: "props" }, outputs: { onClick: "onClick" }, ngImport: i0, template: `
7574
- @if (props) {
7575
- <val-box
7576
- class="teleprompter-container"
7577
- [props]="{ icon: '', bordered: props.bordered, color: props.color, leftBorder: false, padding: props.padding }"
7578
- (onClick)="clickHandler()"
7579
- >
7580
- <div class="container" [class.teleprompter-content]="props.teleprompter" body>
7581
- <div>
7582
- <ion-text>
7583
- <p [class]="props.size" class="bold">{{ displayContent$ | async }}</p>
7584
- </ion-text>
7585
- </div>
7586
- @if (props.buttons || props.hrefs) {
7587
- <div class="actions-container">
7588
- <val-button-group
7589
- class="buttons-container"
7590
- [props]="{ buttons: props.buttons, position: 'right', columned: false }"
7591
- (onClick)="clickHandler($event)"
7592
- ></val-button-group>
7593
- <val-href
7594
- class="link"
7595
- *ngFor="let l of props.hrefs"
7596
- [props]="l"
7597
- (onClick)="clickHandler($event)"
7598
- ></val-href>
7599
- </div>
7600
- }
7529
+ <val-box
7530
+ class="teleprompter-container"
7531
+ [props]="{ icon: '', bordered: props.bordered, color: props.color, leftBorder: false, padding: props.padding }"
7532
+ (onClick)="clickHandler()"
7533
+ >
7534
+ <div class="container" [class.teleprompter-content]="props.teleprompter" body>
7535
+ <div>
7536
+ <ion-text>
7537
+ <p [class]="props.size" class="bold">{{ props.content }}</p>
7538
+ </ion-text>
7601
7539
  </div>
7602
- </val-box>
7603
- }
7604
- `, isInline: true, styles: ["@charset \"UTF-8\";:root{--ion-color-primary: #7026df;--ion-color-primary-rgb: 112, 38, 223;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #6321c4;--ion-color-primary-tint: #7e3ce2;--ion-color-secondary: #e2ccff;--ion-color-secondary-rgb: 226, 204, 255;--ion-color-secondary-contrast: #000000;--ion-color-secondary-contrast-rgb: 0, 0, 0;--ion-color-secondary-shade: #c7b4e0;--ion-color-secondary-tint: #e5d1ff;--ion-color-texti: #354c69;--ion-color-texti-rgb: 53, 76, 105;--ion-color-texti-contrast: #ffffff;--ion-color-texti-contrast-rgb: 255, 255, 255;--ion-color-texti-shade: #2f435c;--ion-color-texti-tint: #495e78;--ion-color-darki: #090f1b;--ion-color-darki-rgb: 9, 15, 27;--ion-color-darki-contrast: #ffffff;--ion-color-darki-contrast-rgb: 255, 255, 255;--ion-color-darki-shade: #080d18;--ion-color-darki-tint: #222732;--ion-color-medium: #9e9e9e;--ion-color-medium-rgb: 158, 158, 158;--ion-color-medium-contrast: #000000;--ion-color-medium-contrast-rgb: 0, 0, 0;--ion-color-medium-shade: #8b8b8b;--ion-color-medium-tint: #a8a8a8;--swiper-pagination-color: var(--ion-color-primary);--swiper-navigation-color: var(--ion-color-primary);--swiper-pagination-bullet-inactive-color: var(--ion-color-medium)}@media (prefers-color-scheme: dark){:root{--ion-color-texti: #8fc1ff;--ion-color-texti-rgb: 143, 193, 255;--ion-color-texti-contrast: #000000;--ion-color-texti-contrast-rgb: 0, 0, 0;--ion-color-texti-shade: #7eaae0;--ion-color-texti-tint: #9ac7ff;--ion-color-darki: #ffffff;--ion-color-darki-rgb: 255, 255, 255;--ion-color-darki-contrast: #000000;--ion-color-darki-contrast-rgb: 0, 0, 0;--ion-color-darki-shade: #e0e0e0;--ion-color-darki-tint: #ffffff;--ion-color-primary: #8f49f8;--ion-color-primary-rgb: 143, 73, 248;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #7e40da;--ion-color-primary-tint: #9a5bf9}}.ion-color-texti{--ion-color-base: var(--ion-color-texti);--ion-color-base-rgb: var(--ion-color-texti-rgb);--ion-color-contrast: var(--ion-color-texti-contrast);--ion-color-contrast-rgb: var(--ion-color-texti-contrast-rgb);--ion-color-shade: var(--ion-color-texti-shade);--ion-color-tint: var(--ion-color-texti-tint)}.ion-color-darki{--ion-color-base: var(--ion-color-darki);--ion-color-base-rgb: var(--ion-color-darki-rgb);--ion-color-contrast: var(--ion-color-darki-contrast);--ion-color-contrast-rgb: var(--ion-color-darki-contrast-rgb);--ion-color-shade: var(--ion-color-darki-shade);--ion-color-tint: var(--ion-color-darki-tint)}.container{display:flex;align-items:center;justify-content:space-between;flex-direction:column}@media (min-width: 768px){.container{flex-direction:row}}.teleprompter-container{overflow:hidden;width:100%}.teleprompter-content{white-space:nowrap;animation:scroll-left 15s linear infinite}@keyframes scroll-left{0%{transform:translate(100%)}to{transform:translate(-100%)}}.link{margin:0 .5rem;width:max-content}.actions-container{display:flex;align-items:center}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "component", type: IonText, selector: "ion-text", inputs: ["color", "mode"] }, { kind: "component", type: BoxComponent, selector: "val-box", inputs: ["props"], outputs: ["onClick"] }, { kind: "component", type: ButtonGroupComponent, selector: "val-button-group", inputs: ["props"], outputs: ["onClick"] }, { kind: "component", type: HrefComponent, selector: "val-href", inputs: ["props"], outputs: ["onClick"] }] }); }
7540
+ @if (props.buttons || props.hrefs) {
7541
+ <div class="actions-container">
7542
+ <val-button-group
7543
+ class="buttons-container"
7544
+ [props]="{ buttons: props.buttons, position: 'right', columned: false }"
7545
+ (onClick)="clickHandler($event)"
7546
+ ></val-button-group>
7547
+ <val-href
7548
+ class="link"
7549
+ *ngFor="let l of props.hrefs"
7550
+ [props]="l"
7551
+ (onClick)="clickHandler($event)"
7552
+ ></val-href>
7553
+ </div>
7554
+ }
7555
+ </div>
7556
+ </val-box>
7557
+ `, isInline: true, styles: ["@charset \"UTF-8\";:root{--ion-color-primary: #7026df;--ion-color-primary-rgb: 112, 38, 223;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #6321c4;--ion-color-primary-tint: #7e3ce2;--ion-color-secondary: #e2ccff;--ion-color-secondary-rgb: 226, 204, 255;--ion-color-secondary-contrast: #000000;--ion-color-secondary-contrast-rgb: 0, 0, 0;--ion-color-secondary-shade: #c7b4e0;--ion-color-secondary-tint: #e5d1ff;--ion-color-texti: #354c69;--ion-color-texti-rgb: 53, 76, 105;--ion-color-texti-contrast: #ffffff;--ion-color-texti-contrast-rgb: 255, 255, 255;--ion-color-texti-shade: #2f435c;--ion-color-texti-tint: #495e78;--ion-color-darki: #090f1b;--ion-color-darki-rgb: 9, 15, 27;--ion-color-darki-contrast: #ffffff;--ion-color-darki-contrast-rgb: 255, 255, 255;--ion-color-darki-shade: #080d18;--ion-color-darki-tint: #222732;--ion-color-medium: #9e9e9e;--ion-color-medium-rgb: 158, 158, 158;--ion-color-medium-contrast: #000000;--ion-color-medium-contrast-rgb: 0, 0, 0;--ion-color-medium-shade: #8b8b8b;--ion-color-medium-tint: #a8a8a8;--swiper-pagination-color: var(--ion-color-primary);--swiper-navigation-color: var(--ion-color-primary);--swiper-pagination-bullet-inactive-color: var(--ion-color-medium)}@media (prefers-color-scheme: dark){:root{--ion-color-texti: #8fc1ff;--ion-color-texti-rgb: 143, 193, 255;--ion-color-texti-contrast: #000000;--ion-color-texti-contrast-rgb: 0, 0, 0;--ion-color-texti-shade: #7eaae0;--ion-color-texti-tint: #9ac7ff;--ion-color-darki: #ffffff;--ion-color-darki-rgb: 255, 255, 255;--ion-color-darki-contrast: #000000;--ion-color-darki-contrast-rgb: 0, 0, 0;--ion-color-darki-shade: #e0e0e0;--ion-color-darki-tint: #ffffff;--ion-color-primary: #8f49f8;--ion-color-primary-rgb: 143, 73, 248;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #7e40da;--ion-color-primary-tint: #9a5bf9}}.ion-color-texti{--ion-color-base: var(--ion-color-texti);--ion-color-base-rgb: var(--ion-color-texti-rgb);--ion-color-contrast: var(--ion-color-texti-contrast);--ion-color-contrast-rgb: var(--ion-color-texti-contrast-rgb);--ion-color-shade: var(--ion-color-texti-shade);--ion-color-tint: var(--ion-color-texti-tint)}.ion-color-darki{--ion-color-base: var(--ion-color-darki);--ion-color-base-rgb: var(--ion-color-darki-rgb);--ion-color-contrast: var(--ion-color-darki-contrast);--ion-color-contrast-rgb: var(--ion-color-darki-contrast-rgb);--ion-color-shade: var(--ion-color-darki-shade);--ion-color-tint: var(--ion-color-darki-tint)}.container{display:flex;align-items:center;justify-content:space-between;flex-direction:column}@media (min-width: 768px){.container{flex-direction:row}}.teleprompter-container{overflow:hidden;width:100%}.teleprompter-content{white-space:nowrap;animation:scroll-left 15s linear infinite}@keyframes scroll-left{0%{transform:translate(100%)}to{transform:translate(-100%)}}.link{margin:0 .5rem;width:max-content}.actions-container{display:flex;align-items:center}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: IonText, selector: "ion-text", inputs: ["color", "mode"] }, { kind: "component", type: BoxComponent, selector: "val-box", inputs: ["props"], outputs: ["onClick"] }, { kind: "component", type: ButtonGroupComponent, selector: "val-button-group", inputs: ["props"], outputs: ["onClick"] }, { kind: "component", type: HrefComponent, selector: "val-href", inputs: ["props"], outputs: ["onClick"] }] }); }
7605
7558
  }
7606
7559
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PrompterComponent, decorators: [{
7607
7560
  type: Component,
7608
7561
  args: [{ selector: 'val-prompter', standalone: true, imports: [CommonModule, IonText, BoxComponent, ButtonGroupComponent, HrefComponent], template: `
7609
- @if (props) {
7610
- <val-box
7611
- class="teleprompter-container"
7612
- [props]="{ icon: '', bordered: props.bordered, color: props.color, leftBorder: false, padding: props.padding }"
7613
- (onClick)="clickHandler()"
7614
- >
7615
- <div class="container" [class.teleprompter-content]="props.teleprompter" body>
7616
- <div>
7617
- <ion-text>
7618
- <p [class]="props.size" class="bold">{{ displayContent$ | async }}</p>
7619
- </ion-text>
7620
- </div>
7621
- @if (props.buttons || props.hrefs) {
7622
- <div class="actions-container">
7623
- <val-button-group
7624
- class="buttons-container"
7625
- [props]="{ buttons: props.buttons, position: 'right', columned: false }"
7626
- (onClick)="clickHandler($event)"
7627
- ></val-button-group>
7628
- <val-href
7629
- class="link"
7630
- *ngFor="let l of props.hrefs"
7631
- [props]="l"
7632
- (onClick)="clickHandler($event)"
7633
- ></val-href>
7634
- </div>
7635
- }
7562
+ <val-box
7563
+ class="teleprompter-container"
7564
+ [props]="{ icon: '', bordered: props.bordered, color: props.color, leftBorder: false, padding: props.padding }"
7565
+ (onClick)="clickHandler()"
7566
+ >
7567
+ <div class="container" [class.teleprompter-content]="props.teleprompter" body>
7568
+ <div>
7569
+ <ion-text>
7570
+ <p [class]="props.size" class="bold">{{ props.content }}</p>
7571
+ </ion-text>
7636
7572
  </div>
7637
- </val-box>
7638
- }
7573
+ @if (props.buttons || props.hrefs) {
7574
+ <div class="actions-container">
7575
+ <val-button-group
7576
+ class="buttons-container"
7577
+ [props]="{ buttons: props.buttons, position: 'right', columned: false }"
7578
+ (onClick)="clickHandler($event)"
7579
+ ></val-button-group>
7580
+ <val-href
7581
+ class="link"
7582
+ *ngFor="let l of props.hrefs"
7583
+ [props]="l"
7584
+ (onClick)="clickHandler($event)"
7585
+ ></val-href>
7586
+ </div>
7587
+ }
7588
+ </div>
7589
+ </val-box>
7639
7590
  `, styles: ["@charset \"UTF-8\";:root{--ion-color-primary: #7026df;--ion-color-primary-rgb: 112, 38, 223;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #6321c4;--ion-color-primary-tint: #7e3ce2;--ion-color-secondary: #e2ccff;--ion-color-secondary-rgb: 226, 204, 255;--ion-color-secondary-contrast: #000000;--ion-color-secondary-contrast-rgb: 0, 0, 0;--ion-color-secondary-shade: #c7b4e0;--ion-color-secondary-tint: #e5d1ff;--ion-color-texti: #354c69;--ion-color-texti-rgb: 53, 76, 105;--ion-color-texti-contrast: #ffffff;--ion-color-texti-contrast-rgb: 255, 255, 255;--ion-color-texti-shade: #2f435c;--ion-color-texti-tint: #495e78;--ion-color-darki: #090f1b;--ion-color-darki-rgb: 9, 15, 27;--ion-color-darki-contrast: #ffffff;--ion-color-darki-contrast-rgb: 255, 255, 255;--ion-color-darki-shade: #080d18;--ion-color-darki-tint: #222732;--ion-color-medium: #9e9e9e;--ion-color-medium-rgb: 158, 158, 158;--ion-color-medium-contrast: #000000;--ion-color-medium-contrast-rgb: 0, 0, 0;--ion-color-medium-shade: #8b8b8b;--ion-color-medium-tint: #a8a8a8;--swiper-pagination-color: var(--ion-color-primary);--swiper-navigation-color: var(--ion-color-primary);--swiper-pagination-bullet-inactive-color: var(--ion-color-medium)}@media (prefers-color-scheme: dark){:root{--ion-color-texti: #8fc1ff;--ion-color-texti-rgb: 143, 193, 255;--ion-color-texti-contrast: #000000;--ion-color-texti-contrast-rgb: 0, 0, 0;--ion-color-texti-shade: #7eaae0;--ion-color-texti-tint: #9ac7ff;--ion-color-darki: #ffffff;--ion-color-darki-rgb: 255, 255, 255;--ion-color-darki-contrast: #000000;--ion-color-darki-contrast-rgb: 0, 0, 0;--ion-color-darki-shade: #e0e0e0;--ion-color-darki-tint: #ffffff;--ion-color-primary: #8f49f8;--ion-color-primary-rgb: 143, 73, 248;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #7e40da;--ion-color-primary-tint: #9a5bf9}}.ion-color-texti{--ion-color-base: var(--ion-color-texti);--ion-color-base-rgb: var(--ion-color-texti-rgb);--ion-color-contrast: var(--ion-color-texti-contrast);--ion-color-contrast-rgb: var(--ion-color-texti-contrast-rgb);--ion-color-shade: var(--ion-color-texti-shade);--ion-color-tint: var(--ion-color-texti-tint)}.ion-color-darki{--ion-color-base: var(--ion-color-darki);--ion-color-base-rgb: var(--ion-color-darki-rgb);--ion-color-contrast: var(--ion-color-darki-contrast);--ion-color-contrast-rgb: var(--ion-color-darki-contrast-rgb);--ion-color-shade: var(--ion-color-darki-shade);--ion-color-tint: var(--ion-color-darki-tint)}.container{display:flex;align-items:center;justify-content:space-between;flex-direction:column}@media (min-width: 768px){.container{flex-direction:row}}.teleprompter-container{overflow:hidden;width:100%}.teleprompter-content{white-space:nowrap;animation:scroll-left 15s linear infinite}@keyframes scroll-left{0%{transform:translate(100%)}to{transform:translate(-100%)}}.link{margin:0 .5rem;width:max-content}.actions-container{display:flex;align-items:center}\n"] }]
7640
7591
  }], propDecorators: { props: [{
7641
7592
  type: Input
@@ -22355,7 +22306,8 @@ const VALTECH_FIREBASE_CONFIG = new InjectionToken('ValtechFirebaseConfig');
22355
22306
  * ```
22356
22307
  */
22357
22308
  function provideValtechFirebase(config) {
22358
- return makeEnvironmentProviders([
22309
+ // Construir array de providers base
22310
+ const providers = [
22359
22311
  // Guardar configuración para uso en servicios
22360
22312
  { provide: VALTECH_FIREBASE_CONFIG, useValue: config },
22361
22313
  // Inicializar Firebase App
@@ -22398,9 +22350,13 @@ function provideValtechFirebase(config) {
22398
22350
  }
22399
22351
  return storage;
22400
22352
  }),
22401
- // Messaging (FCM) - solo si está soportado
22402
- provideMessaging(() => getMessaging()),
22403
- ]);
22353
+ ];
22354
+ // Messaging (FCM) - solo si está explícitamente habilitado
22355
+ // Requiere Service Worker configurado, puede congelar la app si no está disponible
22356
+ if (config.enableMessaging) {
22357
+ providers.push(provideMessaging(() => getMessaging()));
22358
+ }
22359
+ return makeEnvironmentProviders(providers);
22404
22360
  }
22405
22361
  /**
22406
22362
  * Verifica si los emuladores están configurados.