valtech-components 2.0.621 → 2.0.623

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.
Files changed (31) hide show
  1. package/esm2022/lib/components/molecules/docs-code-example/docs-code-example.component.mjs +3 -3
  2. package/esm2022/lib/components/molecules/footer-links/footer-links.component.mjs +5 -5
  3. package/esm2022/lib/components/molecules/update-banner/update-banner.component.mjs +112 -0
  4. package/esm2022/lib/components/organisms/menu/menu.component.mjs +3 -3
  5. package/esm2022/lib/components/templates/maintenance-page/maintenance-page.component.mjs +153 -0
  6. package/esm2022/lib/components/templates/maintenance-page/types.mjs +2 -0
  7. package/esm2022/lib/components/templates/page-content/page-content.component.mjs +21 -5
  8. package/esm2022/lib/components/templates/page-content/types.mjs +1 -1
  9. package/esm2022/lib/services/app-config/app-config.service.mjs +209 -0
  10. package/esm2022/lib/services/app-config/config.mjs +47 -0
  11. package/esm2022/lib/services/app-config/index.mjs +39 -0
  12. package/esm2022/lib/services/app-config/types.mjs +13 -0
  13. package/esm2022/lib/services/i18n/default-content.mjs +21 -1
  14. package/esm2022/public-api.mjs +8 -1
  15. package/fesm2022/valtech-components.mjs +8180 -7595
  16. package/fesm2022/valtech-components.mjs.map +1 -1
  17. package/lib/components/atoms/rights-footer/rights-footer.component.d.ts +1 -1
  18. package/lib/components/molecules/features-list/features-list.component.d.ts +2 -2
  19. package/lib/components/molecules/update-banner/update-banner.component.d.ts +42 -0
  20. package/lib/components/organisms/article/article.component.d.ts +2 -2
  21. package/lib/components/organisms/toolbar/toolbar.component.d.ts +1 -1
  22. package/lib/components/templates/maintenance-page/maintenance-page.component.d.ts +57 -0
  23. package/lib/components/templates/maintenance-page/types.d.ts +12 -0
  24. package/lib/components/templates/page-content/page-content.component.d.ts +6 -0
  25. package/lib/components/templates/page-content/types.d.ts +6 -0
  26. package/lib/services/app-config/app-config.service.d.ts +115 -0
  27. package/lib/services/app-config/config.d.ts +31 -0
  28. package/lib/services/app-config/index.d.ts +38 -0
  29. package/lib/services/app-config/types.d.ts +54 -0
  30. package/package.json +1 -1
  31. package/public-api.d.ts +4 -0
@@ -0,0 +1,153 @@
1
+ /**
2
+ * MaintenancePageComponent
3
+ *
4
+ * Página de mantenimiento que se muestra cuando la aplicación está en modo mantenimiento.
5
+ */
6
+ import { CommonModule } from '@angular/common';
7
+ import { Component, computed, inject, Input } from '@angular/core';
8
+ import { IonContent } from '@ionic/angular/standalone';
9
+ import { ImageComponent } from '../../atoms/image/image.component';
10
+ import { I18nService } from '../../../services/i18n';
11
+ import * as i0 from "@angular/core";
12
+ /**
13
+ * val-maintenance-page
14
+ *
15
+ * Página completa de mantenimiento con imagen, título y mensaje personalizables.
16
+ * Se integra con i18n para textos por defecto en múltiples idiomas.
17
+ *
18
+ * @example
19
+ * ```html
20
+ * <!-- Uso básico (usa textos por defecto) -->
21
+ * <val-maintenance-page />
22
+ *
23
+ * <!-- Personalizado -->
24
+ * <val-maintenance-page
25
+ * [props]="{
26
+ * title: 'En mantenimiento',
27
+ * message: 'Volvemos pronto',
28
+ * image: 'assets/maintenance.svg'
29
+ * }"
30
+ * />
31
+ * ```
32
+ *
33
+ * @example
34
+ * ```typescript
35
+ * // En app.component.ts
36
+ * @Component({
37
+ * template: \`
38
+ * @if (appConfig.isMaintenanceMode()) {
39
+ * <val-maintenance-page />
40
+ * } @else {
41
+ * <ion-router-outlet />
42
+ * }
43
+ * \`
44
+ * })
45
+ * export class AppComponent {
46
+ * appConfig = inject(AppConfigService);
47
+ * }
48
+ * ```
49
+ */
50
+ export class MaintenancePageComponent {
51
+ constructor() {
52
+ /**
53
+ * Configuración de la página de mantenimiento.
54
+ */
55
+ this.props = {};
56
+ this.i18n = inject(I18nService);
57
+ /**
58
+ * Título de la página (reactivo a cambios de idioma).
59
+ */
60
+ this.title = computed(() => this.props.title || this.i18n.t('maintenanceMode', 'AppConfig'));
61
+ /**
62
+ * Mensaje de la página (reactivo a cambios de idioma).
63
+ */
64
+ this.message = computed(() => this.props.message || this.i18n.t('maintenanceMessage', 'AppConfig'));
65
+ }
66
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: MaintenancePageComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
67
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: MaintenancePageComponent, isStandalone: true, selector: "val-maintenance-page", inputs: { props: "props" }, ngImport: i0, template: `
68
+ <ion-content class="maintenance-content">
69
+ <div class="maintenance-container">
70
+ <div class="maintenance-icon">
71
+ @if (props.image) {
72
+ <val-image
73
+ [props]="{
74
+ src: props.image,
75
+ alt: 'Maintenance',
76
+ mode: 'box',
77
+ size: 'large',
78
+ shaded: false,
79
+ bordered: false
80
+ }"
81
+ />
82
+ } @else {
83
+ <div class="default-icon">
84
+ <svg
85
+ xmlns="http://www.w3.org/2000/svg"
86
+ viewBox="0 0 24 24"
87
+ fill="currentColor"
88
+ >
89
+ <path
90
+ d="M12 6v6l4 2m6-2a10 10 0 11-20 0 10 10 0 0120 0z"
91
+ stroke="currentColor"
92
+ stroke-width="2"
93
+ fill="none"
94
+ stroke-linecap="round"
95
+ stroke-linejoin="round"
96
+ />
97
+ </svg>
98
+ </div>
99
+ }
100
+ </div>
101
+
102
+ <h1 class="maintenance-title">{{ title() }}</h1>
103
+ <p class="maintenance-message">{{ message() }}</p>
104
+ </div>
105
+ </ion-content>
106
+ `, isInline: true, styles: [".maintenance-content{--background: var(--ion-background-color, #f5f5f5)}.maintenance-container{display:flex;flex-direction:column;align-items:center;justify-content:center;min-height:100%;padding:2rem;text-align:center}.maintenance-icon{margin-bottom:2rem}.maintenance-icon .default-icon{width:120px;height:120px;color:var(--ion-color-primary, #4a1d96);opacity:.8}.maintenance-icon .default-icon svg{width:100%;height:100%}.maintenance-title{font-size:2rem;font-weight:700;color:var(--ion-text-color, #1a1a1a);margin:0 0 1rem}.maintenance-message{font-size:1.125rem;color:var(--ion-color-medium, #666);margin:0;max-width:400px;line-height:1.6}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: IonContent, selector: "ion-content", inputs: ["color", "fixedSlotPlacement", "forceOverscroll", "fullscreen", "scrollEvents", "scrollX", "scrollY"] }, { kind: "component", type: ImageComponent, selector: "val-image", inputs: ["props"] }] }); }
107
+ }
108
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: MaintenancePageComponent, decorators: [{
109
+ type: Component,
110
+ args: [{ selector: 'val-maintenance-page', standalone: true, imports: [CommonModule, IonContent, ImageComponent], template: `
111
+ <ion-content class="maintenance-content">
112
+ <div class="maintenance-container">
113
+ <div class="maintenance-icon">
114
+ @if (props.image) {
115
+ <val-image
116
+ [props]="{
117
+ src: props.image,
118
+ alt: 'Maintenance',
119
+ mode: 'box',
120
+ size: 'large',
121
+ shaded: false,
122
+ bordered: false
123
+ }"
124
+ />
125
+ } @else {
126
+ <div class="default-icon">
127
+ <svg
128
+ xmlns="http://www.w3.org/2000/svg"
129
+ viewBox="0 0 24 24"
130
+ fill="currentColor"
131
+ >
132
+ <path
133
+ d="M12 6v6l4 2m6-2a10 10 0 11-20 0 10 10 0 0120 0z"
134
+ stroke="currentColor"
135
+ stroke-width="2"
136
+ fill="none"
137
+ stroke-linecap="round"
138
+ stroke-linejoin="round"
139
+ />
140
+ </svg>
141
+ </div>
142
+ }
143
+ </div>
144
+
145
+ <h1 class="maintenance-title">{{ title() }}</h1>
146
+ <p class="maintenance-message">{{ message() }}</p>
147
+ </div>
148
+ </ion-content>
149
+ `, styles: [".maintenance-content{--background: var(--ion-background-color, #f5f5f5)}.maintenance-container{display:flex;flex-direction:column;align-items:center;justify-content:center;min-height:100%;padding:2rem;text-align:center}.maintenance-icon{margin-bottom:2rem}.maintenance-icon .default-icon{width:120px;height:120px;color:var(--ion-color-primary, #4a1d96);opacity:.8}.maintenance-icon .default-icon svg{width:100%;height:100%}.maintenance-title{font-size:2rem;font-weight:700;color:var(--ion-text-color, #1a1a1a);margin:0 0 1rem}.maintenance-message{font-size:1.125rem;color:var(--ion-color-medium, #666);margin:0;max-width:400px;line-height:1.6}\n"] }]
150
+ }], propDecorators: { props: [{
151
+ type: Input
152
+ }] } });
153
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWFpbnRlbmFuY2UtcGFnZS5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvdGVtcGxhdGVzL21haW50ZW5hbmNlLXBhZ2UvbWFpbnRlbmFuY2UtcGFnZS5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7R0FJRztBQUVILE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsU0FBUyxFQUFFLFFBQVEsRUFBRSxNQUFNLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ25FLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUV2RCxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sbUNBQW1DLENBQUM7QUFDbkUsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLHdCQUF3QixDQUFDOztBQUdyRDs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXFDRztBQStDSCxNQUFNLE9BQU8sd0JBQXdCO0lBOUNyQztRQStDRTs7V0FFRztRQUNNLFVBQUssR0FBNEIsRUFBRSxDQUFDO1FBRXJDLFNBQUksR0FBRyxNQUFNLENBQUMsV0FBVyxDQUFDLENBQUM7UUFFbkM7O1dBRUc7UUFDSCxVQUFLLEdBQUcsUUFBUSxDQUNkLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsS0FBSyxJQUFJLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLGlCQUFpQixFQUFFLFdBQVcsQ0FBQyxDQUN0RSxDQUFDO1FBRUY7O1dBRUc7UUFDSCxZQUFPLEdBQUcsUUFBUSxDQUNoQixHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sSUFBSSxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxvQkFBb0IsRUFBRSxXQUFXLENBQUMsQ0FDM0UsQ0FBQztLQUNIOytHQXJCWSx3QkFBd0I7bUdBQXhCLHdCQUF3Qiw0R0ExQ3pCOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0F1Q1QsOHNCQXhDUyxZQUFZLCtCQUFFLFVBQVUsd0tBQUUsY0FBYzs7NEZBMkN2Qyx3QkFBd0I7a0JBOUNwQyxTQUFTOytCQUNFLHNCQUFzQixjQUNwQixJQUFJLFdBQ1AsQ0FBQyxZQUFZLEVBQUUsVUFBVSxFQUFFLGNBQWMsQ0FBQyxZQUN6Qzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBdUNUOzhCQU9RLEtBQUs7c0JBQWIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogTWFpbnRlbmFuY2VQYWdlQ29tcG9uZW50XG4gKlxuICogUMOhZ2luYSBkZSBtYW50ZW5pbWllbnRvIHF1ZSBzZSBtdWVzdHJhIGN1YW5kbyBsYSBhcGxpY2FjacOzbiBlc3TDoSBlbiBtb2RvIG1hbnRlbmltaWVudG8uXG4gKi9cblxuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IENvbXBvbmVudCwgY29tcHV0ZWQsIGluamVjdCwgSW5wdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IElvbkNvbnRlbnQgfSBmcm9tICdAaW9uaWMvYW5ndWxhci9zdGFuZGFsb25lJztcblxuaW1wb3J0IHsgSW1hZ2VDb21wb25lbnQgfSBmcm9tICcuLi8uLi9hdG9tcy9pbWFnZS9pbWFnZS5jb21wb25lbnQnO1xuaW1wb3J0IHsgSTE4blNlcnZpY2UgfSBmcm9tICcuLi8uLi8uLi9zZXJ2aWNlcy9pMThuJztcbmltcG9ydCB7IE1haW50ZW5hbmNlUGFnZU1ldGFkYXRhIH0gZnJvbSAnLi90eXBlcyc7XG5cbi8qKlxuICogdmFsLW1haW50ZW5hbmNlLXBhZ2VcbiAqXG4gKiBQw6FnaW5hIGNvbXBsZXRhIGRlIG1hbnRlbmltaWVudG8gY29uIGltYWdlbiwgdMOtdHVsbyB5IG1lbnNhamUgcGVyc29uYWxpemFibGVzLlxuICogU2UgaW50ZWdyYSBjb24gaTE4biBwYXJhIHRleHRvcyBwb3IgZGVmZWN0byBlbiBtw7psdGlwbGVzIGlkaW9tYXMuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYGh0bWxcbiAqIDwhLS0gVXNvIGLDoXNpY28gKHVzYSB0ZXh0b3MgcG9yIGRlZmVjdG8pIC0tPlxuICogPHZhbC1tYWludGVuYW5jZS1wYWdlIC8+XG4gKlxuICogPCEtLSBQZXJzb25hbGl6YWRvIC0tPlxuICogPHZhbC1tYWludGVuYW5jZS1wYWdlXG4gKiAgIFtwcm9wc109XCJ7XG4gKiAgICAgdGl0bGU6ICdFbiBtYW50ZW5pbWllbnRvJyxcbiAqICAgICBtZXNzYWdlOiAnVm9sdmVtb3MgcHJvbnRvJyxcbiAqICAgICBpbWFnZTogJ2Fzc2V0cy9tYWludGVuYW5jZS5zdmcnXG4gKiAgIH1cIlxuICogLz5cbiAqIGBgYFxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiAvLyBFbiBhcHAuY29tcG9uZW50LnRzXG4gKiBAQ29tcG9uZW50KHtcbiAqICAgdGVtcGxhdGU6IFxcYFxuICogICAgIEBpZiAoYXBwQ29uZmlnLmlzTWFpbnRlbmFuY2VNb2RlKCkpIHtcbiAqICAgICAgIDx2YWwtbWFpbnRlbmFuY2UtcGFnZSAvPlxuICogICAgIH0gQGVsc2Uge1xuICogICAgICAgPGlvbi1yb3V0ZXItb3V0bGV0IC8+XG4gKiAgICAgfVxuICogICBcXGBcbiAqIH0pXG4gKiBleHBvcnQgY2xhc3MgQXBwQ29tcG9uZW50IHtcbiAqICAgYXBwQ29uZmlnID0gaW5qZWN0KEFwcENvbmZpZ1NlcnZpY2UpO1xuICogfVxuICogYGBgXG4gKi9cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ3ZhbC1tYWludGVuYW5jZS1wYWdlJyxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgSW9uQ29udGVudCwgSW1hZ2VDb21wb25lbnRdLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxpb24tY29udGVudCBjbGFzcz1cIm1haW50ZW5hbmNlLWNvbnRlbnRcIj5cbiAgICAgIDxkaXYgY2xhc3M9XCJtYWludGVuYW5jZS1jb250YWluZXJcIj5cbiAgICAgICAgPGRpdiBjbGFzcz1cIm1haW50ZW5hbmNlLWljb25cIj5cbiAgICAgICAgICBAaWYgKHByb3BzLmltYWdlKSB7XG4gICAgICAgICAgICA8dmFsLWltYWdlXG4gICAgICAgICAgICAgIFtwcm9wc109XCJ7XG4gICAgICAgICAgICAgICAgc3JjOiBwcm9wcy5pbWFnZSxcbiAgICAgICAgICAgICAgICBhbHQ6ICdNYWludGVuYW5jZScsXG4gICAgICAgICAgICAgICAgbW9kZTogJ2JveCcsXG4gICAgICAgICAgICAgICAgc2l6ZTogJ2xhcmdlJyxcbiAgICAgICAgICAgICAgICBzaGFkZWQ6IGZhbHNlLFxuICAgICAgICAgICAgICAgIGJvcmRlcmVkOiBmYWxzZVxuICAgICAgICAgICAgICB9XCJcbiAgICAgICAgICAgIC8+XG4gICAgICAgICAgfSBAZWxzZSB7XG4gICAgICAgICAgICA8ZGl2IGNsYXNzPVwiZGVmYXVsdC1pY29uXCI+XG4gICAgICAgICAgICAgIDxzdmdcbiAgICAgICAgICAgICAgICB4bWxucz1cImh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnXCJcbiAgICAgICAgICAgICAgICB2aWV3Qm94PVwiMCAwIDI0IDI0XCJcbiAgICAgICAgICAgICAgICBmaWxsPVwiY3VycmVudENvbG9yXCJcbiAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgIDxwYXRoXG4gICAgICAgICAgICAgICAgICBkPVwiTTEyIDZ2Nmw0IDJtNi0yYTEwIDEwIDAgMTEtMjAgMCAxMCAxMCAwIDAxMjAgMHpcIlxuICAgICAgICAgICAgICAgICAgc3Ryb2tlPVwiY3VycmVudENvbG9yXCJcbiAgICAgICAgICAgICAgICAgIHN0cm9rZS13aWR0aD1cIjJcIlxuICAgICAgICAgICAgICAgICAgZmlsbD1cIm5vbmVcIlxuICAgICAgICAgICAgICAgICAgc3Ryb2tlLWxpbmVjYXA9XCJyb3VuZFwiXG4gICAgICAgICAgICAgICAgICBzdHJva2UtbGluZWpvaW49XCJyb3VuZFwiXG4gICAgICAgICAgICAgICAgLz5cbiAgICAgICAgICAgICAgPC9zdmc+XG4gICAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgICB9XG4gICAgICAgIDwvZGl2PlxuXG4gICAgICAgIDxoMSBjbGFzcz1cIm1haW50ZW5hbmNlLXRpdGxlXCI+e3sgdGl0bGUoKSB9fTwvaDE+XG4gICAgICAgIDxwIGNsYXNzPVwibWFpbnRlbmFuY2UtbWVzc2FnZVwiPnt7IG1lc3NhZ2UoKSB9fTwvcD5cbiAgICAgIDwvZGl2PlxuICAgIDwvaW9uLWNvbnRlbnQ+XG4gIGAsXG4gIHN0eWxlVXJsczogWycuL21haW50ZW5hbmNlLXBhZ2UuY29tcG9uZW50LnNjc3MnXSxcbn0pXG5leHBvcnQgY2xhc3MgTWFpbnRlbmFuY2VQYWdlQ29tcG9uZW50IHtcbiAgLyoqXG4gICAqIENvbmZpZ3VyYWNpw7NuIGRlIGxhIHDDoWdpbmEgZGUgbWFudGVuaW1pZW50by5cbiAgICovXG4gIEBJbnB1dCgpIHByb3BzOiBNYWludGVuYW5jZVBhZ2VNZXRhZGF0YSA9IHt9O1xuXG4gIHByaXZhdGUgaTE4biA9IGluamVjdChJMThuU2VydmljZSk7XG5cbiAgLyoqXG4gICAqIFTDrXR1bG8gZGUgbGEgcMOhZ2luYSAocmVhY3Rpdm8gYSBjYW1iaW9zIGRlIGlkaW9tYSkuXG4gICAqL1xuICB0aXRsZSA9IGNvbXB1dGVkKFxuICAgICgpID0+IHRoaXMucHJvcHMudGl0bGUgfHwgdGhpcy5pMThuLnQoJ21haW50ZW5hbmNlTW9kZScsICdBcHBDb25maWcnKVxuICApO1xuXG4gIC8qKlxuICAgKiBNZW5zYWplIGRlIGxhIHDDoWdpbmEgKHJlYWN0aXZvIGEgY2FtYmlvcyBkZSBpZGlvbWEpLlxuICAgKi9cbiAgbWVzc2FnZSA9IGNvbXB1dGVkKFxuICAgICgpID0+IHRoaXMucHJvcHMubWVzc2FnZSB8fCB0aGlzLmkxOG4udCgnbWFpbnRlbmFuY2VNZXNzYWdlJywgJ0FwcENvbmZpZycpXG4gICk7XG59XG4iXX0=
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvdGVtcGxhdGVzL21haW50ZW5hbmNlLXBhZ2UvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogUHJvcHMgcGFyYSB2YWwtbWFpbnRlbmFuY2UtcGFnZSBjb21wb25lbnQuXG4gKlxuICogQHByb3BlcnR5IHRpdGxlIC0gVMOtdHVsbyBwZXJzb25hbGl6YWRvIChvcGNpb25hbCwgdXNhIGkxOG4gcG9yIGRlZmVjdG8pXG4gKiBAcHJvcGVydHkgbWVzc2FnZSAtIE1lbnNhamUgcGVyc29uYWxpemFkbyAob3BjaW9uYWwsIHVzYSBpMThuIHBvciBkZWZlY3RvKVxuICogQHByb3BlcnR5IGltYWdlIC0gUnV0YSBhIGltYWdlbiBwZXJzb25hbGl6YWRhIChvcGNpb25hbClcbiAqL1xuZXhwb3J0IHR5cGUgTWFpbnRlbmFuY2VQYWdlTWV0YWRhdGEgPSB7XG4gIHRpdGxlPzogc3RyaW5nO1xuICBtZXNzYWdlPzogc3RyaW5nO1xuICBpbWFnZT86IHN0cmluZztcbn07XG4iXX0=
@@ -1,7 +1,9 @@
1
1
  import { CommonModule } from '@angular/common';
2
- import { Component, EventEmitter, Input, Output } from '@angular/core';
2
+ import { Component, EventEmitter, inject, Input, Output } from '@angular/core';
3
3
  import { IonContent } from '@ionic/angular/standalone';
4
4
  import { HeaderComponent } from '../../organisms/header/header.component';
5
+ import { UpdateBannerComponent } from '../../molecules/update-banner/update-banner.component';
6
+ import { VALTECH_APP_CONFIG } from '../../../services/app-config/config';
5
7
  import { resolveColor } from '../../../shared/utils/styles';
6
8
  import * as i0 from "@angular/core";
7
9
  import * as i1 from "../../../services/theme.service";
@@ -41,6 +43,7 @@ export class PageContentComponent {
41
43
  * Page content configuration.
42
44
  */
43
45
  this.props = {};
46
+ this.appConfigEnabled = inject(VALTECH_APP_CONFIG, { optional: true });
44
47
  /**
45
48
  * Emits when a header action is clicked.
46
49
  */
@@ -80,6 +83,13 @@ export class PageContentComponent {
80
83
  },
81
84
  };
82
85
  }
86
+ /**
87
+ * Whether to show the update banner.
88
+ * Only shows if AppConfigService is configured and not disabled via props.
89
+ */
90
+ get showUpdateBanner() {
91
+ return this.appConfigEnabled !== null && this.props.showUpdateBanner !== false;
92
+ }
83
93
  /**
84
94
  * Gets header props, using cached default if not provided.
85
95
  * Injects languageSelector into toolbar when provided at page level.
@@ -122,7 +132,7 @@ export class PageContentComponent {
122
132
  }
123
133
  }
124
134
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PageContentComponent, deps: [{ token: i1.ThemeService }, { token: i2.NavigationService }], target: i0.ɵɵFactoryTarget.Component }); }
125
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: PageContentComponent, isStandalone: true, selector: "val-page-content", inputs: { props: "props" }, outputs: { onHeaderClick: "onHeaderClick" }, ngImport: i0, template: `
135
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: PageContentComponent, isStandalone: true, selector: "val-page-content", inputs: { props: "props" }, outputs: { onHeaderClick: "onHeaderClick" }, ngImport: i0, template: `
126
136
  <div class="ion-page">
127
137
  <val-header
128
138
  [props]="headerProps"
@@ -134,6 +144,9 @@ export class PageContentComponent {
134
144
  '--background': getBackground()
135
145
  }"
136
146
  >
147
+ @if (showUpdateBanner) {
148
+ <val-update-banner />
149
+ }
137
150
  <div class="page-wrapper">
138
151
  <main>
139
152
  <ng-content select="[content]"></ng-content>
@@ -143,11 +156,11 @@ export class PageContentComponent {
143
156
  </ion-content>
144
157
  <ng-content select="[extra-footer]"></ng-content>
145
158
  </div>
146
- `, isInline: true, styles: [".page-wrapper{display:flex;flex-direction:column;min-height:100%}main{flex:1}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: HeaderComponent, selector: "val-header", inputs: ["props"], outputs: ["onClick"] }, { kind: "component", type: IonContent, selector: "ion-content", inputs: ["color", "fixedSlotPlacement", "forceOverscroll", "fullscreen", "scrollEvents", "scrollX", "scrollY"] }] }); }
159
+ `, isInline: true, styles: [".page-wrapper{display:flex;flex-direction:column;min-height:100%}main{flex:1}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: HeaderComponent, selector: "val-header", inputs: ["props"], outputs: ["onClick"] }, { kind: "component", type: IonContent, selector: "ion-content", inputs: ["color", "fixedSlotPlacement", "forceOverscroll", "fullscreen", "scrollEvents", "scrollX", "scrollY"] }, { kind: "component", type: UpdateBannerComponent, selector: "val-update-banner" }] }); }
147
160
  }
148
161
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PageContentComponent, decorators: [{
149
162
  type: Component,
150
- args: [{ selector: 'val-page-content', standalone: true, imports: [CommonModule, HeaderComponent, IonContent], template: `
163
+ args: [{ selector: 'val-page-content', standalone: true, imports: [CommonModule, HeaderComponent, IonContent, UpdateBannerComponent], template: `
151
164
  <div class="ion-page">
152
165
  <val-header
153
166
  [props]="headerProps"
@@ -159,6 +172,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
159
172
  '--background': getBackground()
160
173
  }"
161
174
  >
175
+ @if (showUpdateBanner) {
176
+ <val-update-banner />
177
+ }
162
178
  <div class="page-wrapper">
163
179
  <main>
164
180
  <ng-content select="[content]"></ng-content>
@@ -174,4 +190,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
174
190
  }], onHeaderClick: [{
175
191
  type: Output
176
192
  }] } });
177
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-content.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/templates/page-content/page-content.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, Input, Output } from '@angular/core';\nimport { IonContent } from '@ionic/angular/standalone';\nimport { HeaderComponent } from '../../organisms/header/header.component';\nimport { ThemeService } from '../../../services/theme.service';\nimport { NavigationService } from '../../../services/navigation.service';\nimport { PageContentMetadata } from './types';\nimport { resolveColor } from '../../../shared/utils/styles';\n\n/**\n * val-page-content\n *\n * A page content template with corporate header, main content area,\n * and footer slots. Supports dark mode and customizable backgrounds.\n *\n * @example\n * <val-page-content\n *   [props]=\"{\n *     header: { ... },\n *     background: '--main-background',\n *     homeRoute: '/'\n *   }\"\n *   (onHeaderClick)=\"handleHeaderAction($event)\"\n * >\n *   <div content>\n *     <!-- Main page content -->\n *   </div>\n *   <div footer>\n *     <val-company-footer [props]=\"footerProps\"></val-company-footer>\n *   </div>\n * </val-page-content>\n *\n * @input props - Page content configuration\n * @output onHeaderClick - Emits when a header action is clicked\n */\n@Component({\n  selector: 'val-page-content',\n  standalone: true,\n  imports: [CommonModule, HeaderComponent, IonContent],\n  template: `\n    <div class=\"ion-page\">\n      <val-header\n        [props]=\"headerProps\"\n        (onClick)=\"onHeaderClickHandler($event)\"\n      />\n      <ion-content\n        [fullscreen]=\"true\"\n        [ngStyle]=\"{\n          '--background': getBackground()\n        }\"\n      >\n        <div class=\"page-wrapper\">\n          <main>\n            <ng-content select=\"[content]\"></ng-content>\n          </main>\n          <ng-content select=\"[footer]\"></ng-content>\n        </div>\n      </ion-content>\n      <ng-content select=\"[extra-footer]\"></ng-content>\n    </div>\n  `,\n  styles: `\n    .page-wrapper {\n      display: flex;\n      flex-direction: column;\n      min-height: 100%;\n    }\n\n    main {\n      flex: 1;\n    }\n  `,\n})\nexport class PageContentComponent {\n  /**\n   * Page content configuration.\n   */\n  @Input() props: PageContentMetadata = {};\n\n  constructor(\n    private theme: ThemeService,\n    private nav: NavigationService\n  ) {}\n\n  /**\n   * Emits when a header action is clicked.\n   */\n  @Output() onHeaderClick = new EventEmitter<string>();\n\n  /**\n   * Default header configuration (cached to avoid infinite change detection).\n   */\n  private readonly defaultHeader = {\n    bordered: true,\n    translucent: true,\n    toolbar: {\n      withBack: false,\n      withActions: true,\n      textColor: 'dark' as const,\n      withMenu: true,\n      title: '',\n      languageSelector: undefined as undefined,\n      actions: [\n        {\n          token: 'header-logo',\n          description: '',\n          position: 'left' as const,\n          type: 'IMAGE' as const,\n          image: {\n            width: 10,\n            src: '--main-logo',\n            alt: 'header logo',\n            mode: 'box' as const,\n            shaded: false,\n            bordered: false,\n            size: 'small' as const,\n            limited: false,\n            flex: true,\n          },\n        },\n      ],\n    },\n  };\n\n  /**\n   * Gets header props, using cached default if not provided.\n   * Injects languageSelector into toolbar when provided at page level.\n   */\n  get headerProps() {\n    const header = this.props.header || this.defaultHeader;\n\n    // Inject languageSelector into toolbar if provided at page level\n    if (this.props.languageSelector && !header.toolbar.languageSelector) {\n      return {\n        ...header,\n        toolbar: {\n          ...header.toolbar,\n          languageSelector: this.props.languageSelector,\n        },\n      };\n    }\n\n    return header;\n  }\n\n  /**\n   * Gets the background color based on theme.\n   */\n  getBackground(): string {\n    if (this.theme.IsDark) {\n      return 'var(--ion-background-color)';\n    }\n\n    const bg = this.props.background;\n    if (!bg) {\n      return 'var(--ion-background-color)';\n    }\n\n    return resolveColor(bg);\n  }\n\n  /**\n   * Handles header action clicks.\n   */\n  onHeaderClickHandler(token: string): void {\n    this.onHeaderClick.emit(token);\n\n    // Navigate to home route if configured and logo was clicked\n    if (token === 'header-logo' && this.props.homeRoute) {\n      this.nav.navigateByUrl(this.props.homeRoute);\n    }\n  }\n}\n"]}
193
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"page-content.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/templates/page-content/page-content.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, inject, Input, Output } from '@angular/core';\nimport { IonContent } from '@ionic/angular/standalone';\nimport { HeaderComponent } from '../../organisms/header/header.component';\nimport { UpdateBannerComponent } from '../../molecules/update-banner/update-banner.component';\nimport { ThemeService } from '../../../services/theme.service';\nimport { NavigationService } from '../../../services/navigation.service';\nimport { VALTECH_APP_CONFIG } from '../../../services/app-config/config';\nimport { PageContentMetadata } from './types';\nimport { resolveColor } from '../../../shared/utils/styles';\n\n/**\n * val-page-content\n *\n * A page content template with corporate header, main content area,\n * and footer slots. Supports dark mode and customizable backgrounds.\n *\n * @example\n * <val-page-content\n *   [props]=\"{\n *     header: { ... },\n *     background: '--main-background',\n *     homeRoute: '/'\n *   }\"\n *   (onHeaderClick)=\"handleHeaderAction($event)\"\n * >\n *   <div content>\n *     <!-- Main page content -->\n *   </div>\n *   <div footer>\n *     <val-company-footer [props]=\"footerProps\"></val-company-footer>\n *   </div>\n * </val-page-content>\n *\n * @input props - Page content configuration\n * @output onHeaderClick - Emits when a header action is clicked\n */\n@Component({\n  selector: 'val-page-content',\n  standalone: true,\n  imports: [CommonModule, HeaderComponent, IonContent, UpdateBannerComponent],\n  template: `\n    <div class=\"ion-page\">\n      <val-header\n        [props]=\"headerProps\"\n        (onClick)=\"onHeaderClickHandler($event)\"\n      />\n      <ion-content\n        [fullscreen]=\"true\"\n        [ngStyle]=\"{\n          '--background': getBackground()\n        }\"\n      >\n        @if (showUpdateBanner) {\n          <val-update-banner />\n        }\n        <div class=\"page-wrapper\">\n          <main>\n            <ng-content select=\"[content]\"></ng-content>\n          </main>\n          <ng-content select=\"[footer]\"></ng-content>\n        </div>\n      </ion-content>\n      <ng-content select=\"[extra-footer]\"></ng-content>\n    </div>\n  `,\n  styles: `\n    .page-wrapper {\n      display: flex;\n      flex-direction: column;\n      min-height: 100%;\n    }\n\n    main {\n      flex: 1;\n    }\n  `,\n})\nexport class PageContentComponent {\n  /**\n   * Page content configuration.\n   */\n  @Input() props: PageContentMetadata = {};\n\n  private appConfigEnabled = inject(VALTECH_APP_CONFIG, { optional: true });\n\n  constructor(\n    private theme: ThemeService,\n    private nav: NavigationService\n  ) {}\n\n  /**\n   * Whether to show the update banner.\n   * Only shows if AppConfigService is configured and not disabled via props.\n   */\n  get showUpdateBanner(): boolean {\n    return this.appConfigEnabled !== null && this.props.showUpdateBanner !== false;\n  }\n\n  /**\n   * Emits when a header action is clicked.\n   */\n  @Output() onHeaderClick = new EventEmitter<string>();\n\n  /**\n   * Default header configuration (cached to avoid infinite change detection).\n   */\n  private readonly defaultHeader = {\n    bordered: true,\n    translucent: true,\n    toolbar: {\n      withBack: false,\n      withActions: true,\n      textColor: 'dark' as const,\n      withMenu: true,\n      title: '',\n      languageSelector: undefined as undefined,\n      actions: [\n        {\n          token: 'header-logo',\n          description: '',\n          position: 'left' as const,\n          type: 'IMAGE' as const,\n          image: {\n            width: 10,\n            src: '--main-logo',\n            alt: 'header logo',\n            mode: 'box' as const,\n            shaded: false,\n            bordered: false,\n            size: 'small' as const,\n            limited: false,\n            flex: true,\n          },\n        },\n      ],\n    },\n  };\n\n  /**\n   * Gets header props, using cached default if not provided.\n   * Injects languageSelector into toolbar when provided at page level.\n   */\n  get headerProps() {\n    const header = this.props.header || this.defaultHeader;\n\n    // Inject languageSelector into toolbar if provided at page level\n    if (this.props.languageSelector && !header.toolbar.languageSelector) {\n      return {\n        ...header,\n        toolbar: {\n          ...header.toolbar,\n          languageSelector: this.props.languageSelector,\n        },\n      };\n    }\n\n    return header;\n  }\n\n  /**\n   * Gets the background color based on theme.\n   */\n  getBackground(): string {\n    if (this.theme.IsDark) {\n      return 'var(--ion-background-color)';\n    }\n\n    const bg = this.props.background;\n    if (!bg) {\n      return 'var(--ion-background-color)';\n    }\n\n    return resolveColor(bg);\n  }\n\n  /**\n   * Handles header action clicks.\n   */\n  onHeaderClickHandler(token: string): void {\n    this.onHeaderClick.emit(token);\n\n    // Navigate to home route if configured and logo was clicked\n    if (token === 'header-logo' && this.props.homeRoute) {\n      this.nav.navigateByUrl(this.props.homeRoute);\n    }\n  }\n}\n"]}
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvdGVtcGxhdGVzL3BhZ2UtY29udGVudC90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSGVhZGVyTWV0YWRhdGEgfSBmcm9tICcuLi8uLi9vcmdhbmlzbXMvaGVhZGVyL3R5cGVzJztcbmltcG9ydCB7IExhbmd1YWdlU2VsZWN0b3JNZXRhZGF0YSB9IGZyb20gJy4uLy4uL21vbGVjdWxlcy9sYW5ndWFnZS1zZWxlY3Rvci90eXBlcyc7XG5cbi8qKlxuICogQ29uZmlndXJhdGlvbiBmb3IgdGhlIHBhZ2UgY29udGVudCBjb21wb25lbnQuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUGFnZUNvbnRlbnRNZXRhZGF0YSB7XG4gIC8qKiBIZWFkZXIgY29uZmlndXJhdGlvbiAqL1xuICBoZWFkZXI/OiBIZWFkZXJNZXRhZGF0YTtcbiAgLyoqIEJhY2tncm91bmQgY29sb3Igb3IgQ1NTIHZhcmlhYmxlICovXG4gIGJhY2tncm91bmQ/OiBzdHJpbmc7XG4gIC8qKiBCYWNrZ3JvdW5kIGNvbG9yIGZvciBkYXJrIG1vZGUgKi9cbiAgYmFja2dyb3VuZERhcms/OiBzdHJpbmc7XG4gIC8qKiBSb3V0ZSB0byBuYXZpZ2F0ZSB0byB3aGVuIGhlYWRlciBsb2dvIGlzIGNsaWNrZWQgKi9cbiAgaG9tZVJvdXRlPzogc3RyaW5nO1xuICAvKipcbiAgICogTGFuZ3VhZ2Ugc2VsZWN0b3IgY29uZmlndXJhdGlvbi5cbiAgICogV2hlbiBwcm92aWRlZCwgZGlzcGxheXMgYSBsYW5ndWFnZSBzZWxlY3RvciBpY29uIGluIHRoZSBoZWFkZXIgKGxlZnQgb2YgbWVudSBidXR0b24pLlxuICAgKiBVc2VzICdpY29uJyBtb2RlIGJ5IGRlZmF1bHQgZm9yIGNvbXBhY3QgZGlzcGxheS5cbiAgICovXG4gIGxhbmd1YWdlU2VsZWN0b3I/OiBMYW5ndWFnZVNlbGVjdG9yTWV0YWRhdGE7XG59XG4iXX0=
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvdGVtcGxhdGVzL3BhZ2UtY29udGVudC90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSGVhZGVyTWV0YWRhdGEgfSBmcm9tICcuLi8uLi9vcmdhbmlzbXMvaGVhZGVyL3R5cGVzJztcbmltcG9ydCB7IExhbmd1YWdlU2VsZWN0b3JNZXRhZGF0YSB9IGZyb20gJy4uLy4uL21vbGVjdWxlcy9sYW5ndWFnZS1zZWxlY3Rvci90eXBlcyc7XG5cbi8qKlxuICogQ29uZmlndXJhdGlvbiBmb3IgdGhlIHBhZ2UgY29udGVudCBjb21wb25lbnQuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUGFnZUNvbnRlbnRNZXRhZGF0YSB7XG4gIC8qKiBIZWFkZXIgY29uZmlndXJhdGlvbiAqL1xuICBoZWFkZXI/OiBIZWFkZXJNZXRhZGF0YTtcbiAgLyoqIEJhY2tncm91bmQgY29sb3Igb3IgQ1NTIHZhcmlhYmxlICovXG4gIGJhY2tncm91bmQ/OiBzdHJpbmc7XG4gIC8qKiBCYWNrZ3JvdW5kIGNvbG9yIGZvciBkYXJrIG1vZGUgKi9cbiAgYmFja2dyb3VuZERhcms/OiBzdHJpbmc7XG4gIC8qKiBSb3V0ZSB0byBuYXZpZ2F0ZSB0byB3aGVuIGhlYWRlciBsb2dvIGlzIGNsaWNrZWQgKi9cbiAgaG9tZVJvdXRlPzogc3RyaW5nO1xuICAvKipcbiAgICogTGFuZ3VhZ2Ugc2VsZWN0b3IgY29uZmlndXJhdGlvbi5cbiAgICogV2hlbiBwcm92aWRlZCwgZGlzcGxheXMgYSBsYW5ndWFnZSBzZWxlY3RvciBpY29uIGluIHRoZSBoZWFkZXIgKGxlZnQgb2YgbWVudSBidXR0b24pLlxuICAgKiBVc2VzICdpY29uJyBtb2RlIGJ5IGRlZmF1bHQgZm9yIGNvbXBhY3QgZGlzcGxheS5cbiAgICovXG4gIGxhbmd1YWdlU2VsZWN0b3I/OiBMYW5ndWFnZVNlbGVjdG9yTWV0YWRhdGE7XG4gIC8qKlxuICAgKiBXaGV0aGVyIHRvIHNob3cgdGhlIHVwZGF0ZSBiYW5uZXIgd2hlbiBhIG5ldyB2ZXJzaW9uIGlzIGF2YWlsYWJsZS5cbiAgICogUmVxdWlyZXMgQXBwQ29uZmlnU2VydmljZSB0byBiZSBjb25maWd1cmVkIHdpdGggcHJvdmlkZVZhbHRlY2hBcHBDb25maWcoKS5cbiAgICogQGRlZmF1bHQgdHJ1ZVxuICAgKi9cbiAgc2hvd1VwZGF0ZUJhbm5lcj86IGJvb2xlYW47XG59XG4iXX0=
@@ -0,0 +1,209 @@
1
+ /**
2
+ * AppConfigService
3
+ *
4
+ * Servicio para gestionar configuración remota de la aplicación desde Firestore.
5
+ * Soporta feature flags, variables dinámicas, y detección de actualizaciones.
6
+ */
7
+ import { computed, inject, Injectable, signal } from '@angular/core';
8
+ import { FirestoreService } from '../firebase/firestore.service';
9
+ import { VALTECH_APP_CONFIG } from './config';
10
+ import * as i0 from "@angular/core";
11
+ /**
12
+ * Servicio de configuración remota de la aplicación.
13
+ *
14
+ * Lee la configuración desde Firestore (apps/{appId}/config/app) y provee:
15
+ * - Feature flags reactivos
16
+ * - Variables dinámicas de configuración
17
+ * - Detección de nuevas versiones
18
+ * - Modo mantenimiento
19
+ *
20
+ * @example
21
+ * ```typescript
22
+ * @Component({...})
23
+ * export class MyComponent {
24
+ * private appConfig = inject(AppConfigService);
25
+ *
26
+ * // Verificar feature flag
27
+ * showNewFeature = computed(() =>
28
+ * this.appConfig.isFeatureEnabled('newDashboard')
29
+ * );
30
+ *
31
+ * // Obtener variable
32
+ * maxSize = this.appConfig.getVariable('maxUploadSize', 5242880);
33
+ * }
34
+ * ```
35
+ */
36
+ export class AppConfigService {
37
+ constructor() {
38
+ this.firestore = inject(FirestoreService);
39
+ this.serviceConfig = inject(VALTECH_APP_CONFIG, { optional: true });
40
+ this.subscription = null;
41
+ // =========================================================================
42
+ // SIGNALS REACTIVOS
43
+ // =========================================================================
44
+ /**
45
+ * Configuración actual de la aplicación.
46
+ * null si no se ha cargado o no está configurado.
47
+ */
48
+ this.appConfig = signal(null);
49
+ /**
50
+ * Indica si la configuración está cargando.
51
+ */
52
+ this.loading = signal(true);
53
+ /**
54
+ * Error de carga, si existe.
55
+ */
56
+ this.error = signal(null);
57
+ /**
58
+ * Indica si hay una nueva versión disponible.
59
+ */
60
+ this.hasUpdate = computed(() => {
61
+ const config = this.appConfig();
62
+ const currentVersion = this.serviceConfig?.currentVersion;
63
+ if (!config || !currentVersion)
64
+ return false;
65
+ return this.isNewerVersion(config.version, currentVersion);
66
+ });
67
+ /**
68
+ * Indica si la aplicación está en modo mantenimiento.
69
+ */
70
+ this.isMaintenanceMode = computed(() => this.appConfig()?.maintenance ?? false);
71
+ /**
72
+ * Versión remota de la aplicación.
73
+ */
74
+ this.remoteVersion = computed(() => this.appConfig()?.version ?? null);
75
+ /**
76
+ * Versión actual de la aplicación (local).
77
+ */
78
+ this.currentVersion = computed(() => this.serviceConfig?.currentVersion ?? '0.0.0');
79
+ }
80
+ // =========================================================================
81
+ // MÉTODOS PÚBLICOS
82
+ // =========================================================================
83
+ /**
84
+ * Inicializa el servicio y comienza a escuchar cambios en la configuración.
85
+ * Se llama automáticamente via APP_INITIALIZER si se usa provideValtechAppConfig().
86
+ */
87
+ initialize() {
88
+ if (!this.serviceConfig) {
89
+ console.warn('[AppConfigService] No configuration provided. Use provideValtechAppConfig() in main.ts');
90
+ this.loading.set(false);
91
+ return Promise.resolve();
92
+ }
93
+ return new Promise((resolve) => {
94
+ this.subscription = this.firestore
95
+ .docChanges('config', 'app')
96
+ .subscribe({
97
+ next: (config) => {
98
+ this.appConfig.set(config);
99
+ this.loading.set(false);
100
+ this.error.set(null);
101
+ resolve();
102
+ },
103
+ error: (err) => {
104
+ console.error('[AppConfigService] Error loading config:', err);
105
+ this.error.set(err.message || 'Error al cargar configuración');
106
+ this.loading.set(false);
107
+ resolve(); // Resolve anyway to not block app startup
108
+ },
109
+ });
110
+ });
111
+ }
112
+ /**
113
+ * Verifica si un feature flag está habilitado.
114
+ *
115
+ * @param feature - Nombre del feature flag
116
+ * @param defaultValue - Valor por defecto si no existe (default: false)
117
+ * @returns true si el feature está habilitado
118
+ *
119
+ * @example
120
+ * ```typescript
121
+ * if (appConfig.isFeatureEnabled('darkMode')) {
122
+ * // Mostrar toggle de dark mode
123
+ * }
124
+ * ```
125
+ */
126
+ isFeatureEnabled(feature, defaultValue = false) {
127
+ const config = this.appConfig();
128
+ if (!config?.features)
129
+ return defaultValue;
130
+ return config.features[feature] ?? defaultValue;
131
+ }
132
+ /**
133
+ * Obtiene una variable de configuración.
134
+ *
135
+ * @param key - Clave de la variable
136
+ * @param defaultValue - Valor por defecto si no existe
137
+ * @returns Valor de la variable o el valor por defecto
138
+ *
139
+ * @example
140
+ * ```typescript
141
+ * const maxSize = appConfig.getVariable('maxUploadSize', 5242880);
142
+ * const supportEmail = appConfig.getVariable('supportEmail', 'support@example.com');
143
+ * ```
144
+ */
145
+ getVariable(key, defaultValue) {
146
+ const config = this.appConfig();
147
+ if (!config?.variables)
148
+ return defaultValue;
149
+ return config.variables[key] ?? defaultValue;
150
+ }
151
+ /**
152
+ * Recarga la configuración manualmente (one-time read).
153
+ * Útil si necesitas forzar una actualización.
154
+ */
155
+ async refresh() {
156
+ this.loading.set(true);
157
+ try {
158
+ const config = await this.firestore.getDoc('config', 'app');
159
+ this.appConfig.set(config);
160
+ this.error.set(null);
161
+ }
162
+ catch (err) {
163
+ const error = err;
164
+ console.error('[AppConfigService] Error refreshing config:', error);
165
+ this.error.set(error.message || 'Error al refrescar configuración');
166
+ }
167
+ finally {
168
+ this.loading.set(false);
169
+ }
170
+ }
171
+ /**
172
+ * Desuscribe del listener de Firestore.
173
+ * Se llama automáticamente cuando el servicio se destruye.
174
+ */
175
+ destroy() {
176
+ this.subscription?.unsubscribe();
177
+ this.subscription = null;
178
+ }
179
+ // =========================================================================
180
+ // MÉTODOS PRIVADOS
181
+ // =========================================================================
182
+ /**
183
+ * Compara dos versiones semver y retorna true si remote > current.
184
+ *
185
+ * @param remote - Versión remota (ej: "1.2.3")
186
+ * @param current - Versión actual (ej: "1.2.0")
187
+ * @returns true si remote es mayor que current
188
+ */
189
+ isNewerVersion(remote, current) {
190
+ const remoteParts = remote.split('.').map(Number);
191
+ const currentParts = current.split('.').map(Number);
192
+ for (let i = 0; i < Math.max(remoteParts.length, currentParts.length); i++) {
193
+ const remotePart = remoteParts[i] || 0;
194
+ const currentPart = currentParts[i] || 0;
195
+ if (remotePart > currentPart)
196
+ return true;
197
+ if (remotePart < currentPart)
198
+ return false;
199
+ }
200
+ return false;
201
+ }
202
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AppConfigService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
203
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AppConfigService, providedIn: 'root' }); }
204
+ }
205
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AppConfigService, decorators: [{
206
+ type: Injectable,
207
+ args: [{ providedIn: 'root' }]
208
+ }] });
209
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"app-config.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/app-config/app-config.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * AppConfigService\n *\n * Servicio para gestionar configuración remota de la aplicación desde Firestore.\n * Soporta feature flags, variables dinámicas, y detección de actualizaciones.\n */\n\nimport { computed, inject, Injectable, signal } from '@angular/core';\nimport { Subscription } from 'rxjs';\n\nimport { FirestoreService } from '../firebase/firestore.service';\nimport { VALTECH_APP_CONFIG } from './config';\nimport { AppConfig, AppConfigServiceConfig } from './types';\n\n/**\n * Servicio de configuración remota de la aplicación.\n *\n * Lee la configuración desde Firestore (apps/{appId}/config/app) y provee:\n * - Feature flags reactivos\n * - Variables dinámicas de configuración\n * - Detección de nuevas versiones\n * - Modo mantenimiento\n *\n * @example\n * ```typescript\n * @Component({...})\n * export class MyComponent {\n *   private appConfig = inject(AppConfigService);\n *\n *   // Verificar feature flag\n *   showNewFeature = computed(() =>\n *     this.appConfig.isFeatureEnabled('newDashboard')\n *   );\n *\n *   // Obtener variable\n *   maxSize = this.appConfig.getVariable('maxUploadSize', 5242880);\n * }\n * ```\n */\n@Injectable({ providedIn: 'root' })\nexport class AppConfigService {\n  private firestore = inject(FirestoreService);\n  private serviceConfig = inject(VALTECH_APP_CONFIG, { optional: true });\n\n  private subscription: Subscription | null = null;\n\n  // =========================================================================\n  // SIGNALS REACTIVOS\n  // =========================================================================\n\n  /**\n   * Configuración actual de la aplicación.\n   * null si no se ha cargado o no está configurado.\n   */\n  readonly appConfig = signal<AppConfig | null>(null);\n\n  /**\n   * Indica si la configuración está cargando.\n   */\n  readonly loading = signal(true);\n\n  /**\n   * Error de carga, si existe.\n   */\n  readonly error = signal<string | null>(null);\n\n  /**\n   * Indica si hay una nueva versión disponible.\n   */\n  readonly hasUpdate = computed(() => {\n    const config = this.appConfig();\n    const currentVersion = this.serviceConfig?.currentVersion;\n\n    if (!config || !currentVersion) return false;\n\n    return this.isNewerVersion(config.version, currentVersion);\n  });\n\n  /**\n   * Indica si la aplicación está en modo mantenimiento.\n   */\n  readonly isMaintenanceMode = computed(\n    () => this.appConfig()?.maintenance ?? false\n  );\n\n  /**\n   * Versión remota de la aplicación.\n   */\n  readonly remoteVersion = computed(() => this.appConfig()?.version ?? null);\n\n  /**\n   * Versión actual de la aplicación (local).\n   */\n  readonly currentVersion = computed(\n    () => this.serviceConfig?.currentVersion ?? '0.0.0'\n  );\n\n  // =========================================================================\n  // MÉTODOS PÚBLICOS\n  // =========================================================================\n\n  /**\n   * Inicializa el servicio y comienza a escuchar cambios en la configuración.\n   * Se llama automáticamente via APP_INITIALIZER si se usa provideValtechAppConfig().\n   */\n  initialize(): Promise<void> {\n    if (!this.serviceConfig) {\n      console.warn(\n        '[AppConfigService] No configuration provided. Use provideValtechAppConfig() in main.ts'\n      );\n      this.loading.set(false);\n      return Promise.resolve();\n    }\n\n    return new Promise((resolve) => {\n      this.subscription = this.firestore\n        .docChanges<AppConfig>('config', 'app')\n        .subscribe({\n          next: (config) => {\n            this.appConfig.set(config);\n            this.loading.set(false);\n            this.error.set(null);\n            resolve();\n          },\n          error: (err) => {\n            console.error('[AppConfigService] Error loading config:', err);\n            this.error.set(err.message || 'Error al cargar configuración');\n            this.loading.set(false);\n            resolve(); // Resolve anyway to not block app startup\n          },\n        });\n    });\n  }\n\n  /**\n   * Verifica si un feature flag está habilitado.\n   *\n   * @param feature - Nombre del feature flag\n   * @param defaultValue - Valor por defecto si no existe (default: false)\n   * @returns true si el feature está habilitado\n   *\n   * @example\n   * ```typescript\n   * if (appConfig.isFeatureEnabled('darkMode')) {\n   *   // Mostrar toggle de dark mode\n   * }\n   * ```\n   */\n  isFeatureEnabled(feature: string, defaultValue = false): boolean {\n    const config = this.appConfig();\n    if (!config?.features) return defaultValue;\n    return config.features[feature] ?? defaultValue;\n  }\n\n  /**\n   * Obtiene una variable de configuración.\n   *\n   * @param key - Clave de la variable\n   * @param defaultValue - Valor por defecto si no existe\n   * @returns Valor de la variable o el valor por defecto\n   *\n   * @example\n   * ```typescript\n   * const maxSize = appConfig.getVariable('maxUploadSize', 5242880);\n   * const supportEmail = appConfig.getVariable('supportEmail', 'support@example.com');\n   * ```\n   */\n  getVariable<T>(key: string, defaultValue: T): T {\n    const config = this.appConfig();\n    if (!config?.variables) return defaultValue;\n    return (config.variables[key] as T) ?? defaultValue;\n  }\n\n  /**\n   * Recarga la configuración manualmente (one-time read).\n   * Útil si necesitas forzar una actualización.\n   */\n  async refresh(): Promise<void> {\n    this.loading.set(true);\n\n    try {\n      const config = await this.firestore.getDoc<AppConfig>('config', 'app');\n      this.appConfig.set(config);\n      this.error.set(null);\n    } catch (err) {\n      const error = err as Error;\n      console.error('[AppConfigService] Error refreshing config:', error);\n      this.error.set(error.message || 'Error al refrescar configuración');\n    } finally {\n      this.loading.set(false);\n    }\n  }\n\n  /**\n   * Desuscribe del listener de Firestore.\n   * Se llama automáticamente cuando el servicio se destruye.\n   */\n  destroy(): void {\n    this.subscription?.unsubscribe();\n    this.subscription = null;\n  }\n\n  // =========================================================================\n  // MÉTODOS PRIVADOS\n  // =========================================================================\n\n  /**\n   * Compara dos versiones semver y retorna true si remote > current.\n   *\n   * @param remote - Versión remota (ej: \"1.2.3\")\n   * @param current - Versión actual (ej: \"1.2.0\")\n   * @returns true si remote es mayor que current\n   */\n  private isNewerVersion(remote: string, current: string): boolean {\n    const remoteParts = remote.split('.').map(Number);\n    const currentParts = current.split('.').map(Number);\n\n    for (let i = 0; i < Math.max(remoteParts.length, currentParts.length); i++) {\n      const remotePart = remoteParts[i] || 0;\n      const currentPart = currentParts[i] || 0;\n\n      if (remotePart > currentPart) return true;\n      if (remotePart < currentPart) return false;\n    }\n\n    return false;\n  }\n}\n"]}
@@ -0,0 +1,47 @@
1
+ import { APP_INITIALIZER, InjectionToken, makeEnvironmentProviders, } from '@angular/core';
2
+ import { DEFAULT_APP_CONFIG_SERVICE_CONFIG, } from './types';
3
+ import { AppConfigService } from './app-config.service';
4
+ /**
5
+ * Token de inyección para la configuración de AppConfig.
6
+ */
7
+ export const VALTECH_APP_CONFIG = new InjectionToken('ValtechAppConfig');
8
+ /**
9
+ * Provee el servicio de configuración remota a la aplicación Angular.
10
+ *
11
+ * @param config - Configuración del servicio
12
+ * @returns EnvironmentProviders para usar en bootstrapApplication
13
+ *
14
+ * @example
15
+ * ```typescript
16
+ * // main.ts
17
+ * import { bootstrapApplication } from '@angular/platform-browser';
18
+ * import { provideValtechAppConfig } from 'valtech-components';
19
+ * import { version } from '../package.json';
20
+ *
21
+ * bootstrapApplication(AppComponent, {
22
+ * providers: [
23
+ * provideValtechFirebase(environment.valtechFirebase),
24
+ * provideValtechAppConfig({
25
+ * currentVersion: version,
26
+ * showUpdateBanner: true,
27
+ * }),
28
+ * ],
29
+ * });
30
+ * ```
31
+ */
32
+ export function provideValtechAppConfig(config) {
33
+ const mergedConfig = {
34
+ ...DEFAULT_APP_CONFIG_SERVICE_CONFIG,
35
+ ...config,
36
+ };
37
+ return makeEnvironmentProviders([
38
+ { provide: VALTECH_APP_CONFIG, useValue: mergedConfig },
39
+ {
40
+ provide: APP_INITIALIZER,
41
+ useFactory: (appConfigService) => () => appConfigService.initialize(),
42
+ deps: [AppConfigService],
43
+ multi: true,
44
+ },
45
+ ]);
46
+ }
47
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9zZXJ2aWNlcy9hcHAtY29uZmlnL2NvbmZpZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQ0wsZUFBZSxFQUVmLGNBQWMsRUFDZCx3QkFBd0IsR0FDekIsTUFBTSxlQUFlLENBQUM7QUFDdkIsT0FBTyxFQUVMLGlDQUFpQyxHQUNsQyxNQUFNLFNBQVMsQ0FBQztBQUNqQixPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSxzQkFBc0IsQ0FBQztBQUV4RDs7R0FFRztBQUNILE1BQU0sQ0FBQyxNQUFNLGtCQUFrQixHQUFHLElBQUksY0FBYyxDQUNsRCxrQkFBa0IsQ0FDbkIsQ0FBQztBQUVGOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXVCRztBQUNILE1BQU0sVUFBVSx1QkFBdUIsQ0FDckMsTUFBOEI7SUFFOUIsTUFBTSxZQUFZLEdBQTJCO1FBQzNDLEdBQUcsaUNBQWlDO1FBQ3BDLEdBQUcsTUFBTTtLQUNWLENBQUM7SUFFRixPQUFPLHdCQUF3QixDQUFDO1FBQzlCLEVBQUUsT0FBTyxFQUFFLGtCQUFrQixFQUFFLFFBQVEsRUFBRSxZQUFZLEVBQUU7UUFDdkQ7WUFDRSxPQUFPLEVBQUUsZUFBZTtZQUN4QixVQUFVLEVBQUUsQ0FBQyxnQkFBa0MsRUFBRSxFQUFFLENBQUMsR0FBRyxFQUFFLENBQ3ZELGdCQUFnQixDQUFDLFVBQVUsRUFBRTtZQUMvQixJQUFJLEVBQUUsQ0FBQyxnQkFBZ0IsQ0FBQztZQUN4QixLQUFLLEVBQUUsSUFBSTtTQUNaO0tBQ0YsQ0FBQyxDQUFDO0FBQ0wsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIEFQUF9JTklUSUFMSVpFUixcbiAgRW52aXJvbm1lbnRQcm92aWRlcnMsXG4gIEluamVjdGlvblRva2VuLFxuICBtYWtlRW52aXJvbm1lbnRQcm92aWRlcnMsXG59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHtcbiAgQXBwQ29uZmlnU2VydmljZUNvbmZpZyxcbiAgREVGQVVMVF9BUFBfQ09ORklHX1NFUlZJQ0VfQ09ORklHLFxufSBmcm9tICcuL3R5cGVzJztcbmltcG9ydCB7IEFwcENvbmZpZ1NlcnZpY2UgfSBmcm9tICcuL2FwcC1jb25maWcuc2VydmljZSc7XG5cbi8qKlxuICogVG9rZW4gZGUgaW55ZWNjacOzbiBwYXJhIGxhIGNvbmZpZ3VyYWNpw7NuIGRlIEFwcENvbmZpZy5cbiAqL1xuZXhwb3J0IGNvbnN0IFZBTFRFQ0hfQVBQX0NPTkZJRyA9IG5ldyBJbmplY3Rpb25Ub2tlbjxBcHBDb25maWdTZXJ2aWNlQ29uZmlnPihcbiAgJ1ZhbHRlY2hBcHBDb25maWcnXG4pO1xuXG4vKipcbiAqIFByb3ZlZSBlbCBzZXJ2aWNpbyBkZSBjb25maWd1cmFjacOzbiByZW1vdGEgYSBsYSBhcGxpY2FjacOzbiBBbmd1bGFyLlxuICpcbiAqIEBwYXJhbSBjb25maWcgLSBDb25maWd1cmFjacOzbiBkZWwgc2VydmljaW9cbiAqIEByZXR1cm5zIEVudmlyb25tZW50UHJvdmlkZXJzIHBhcmEgdXNhciBlbiBib290c3RyYXBBcHBsaWNhdGlvblxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiAvLyBtYWluLnRzXG4gKiBpbXBvcnQgeyBib290c3RyYXBBcHBsaWNhdGlvbiB9IGZyb20gJ0Bhbmd1bGFyL3BsYXRmb3JtLWJyb3dzZXInO1xuICogaW1wb3J0IHsgcHJvdmlkZVZhbHRlY2hBcHBDb25maWcgfSBmcm9tICd2YWx0ZWNoLWNvbXBvbmVudHMnO1xuICogaW1wb3J0IHsgdmVyc2lvbiB9IGZyb20gJy4uL3BhY2thZ2UuanNvbic7XG4gKlxuICogYm9vdHN0cmFwQXBwbGljYXRpb24oQXBwQ29tcG9uZW50LCB7XG4gKiAgIHByb3ZpZGVyczogW1xuICogICAgIHByb3ZpZGVWYWx0ZWNoRmlyZWJhc2UoZW52aXJvbm1lbnQudmFsdGVjaEZpcmViYXNlKSxcbiAqICAgICBwcm92aWRlVmFsdGVjaEFwcENvbmZpZyh7XG4gKiAgICAgICBjdXJyZW50VmVyc2lvbjogdmVyc2lvbixcbiAqICAgICAgIHNob3dVcGRhdGVCYW5uZXI6IHRydWUsXG4gKiAgICAgfSksXG4gKiAgIF0sXG4gKiB9KTtcbiAqIGBgYFxuICovXG5leHBvcnQgZnVuY3Rpb24gcHJvdmlkZVZhbHRlY2hBcHBDb25maWcoXG4gIGNvbmZpZzogQXBwQ29uZmlnU2VydmljZUNvbmZpZ1xuKTogRW52aXJvbm1lbnRQcm92aWRlcnMge1xuICBjb25zdCBtZXJnZWRDb25maWc6IEFwcENvbmZpZ1NlcnZpY2VDb25maWcgPSB7XG4gICAgLi4uREVGQVVMVF9BUFBfQ09ORklHX1NFUlZJQ0VfQ09ORklHLFxuICAgIC4uLmNvbmZpZyxcbiAgfTtcblxuICByZXR1cm4gbWFrZUVudmlyb25tZW50UHJvdmlkZXJzKFtcbiAgICB7IHByb3ZpZGU6IFZBTFRFQ0hfQVBQX0NPTkZJRywgdXNlVmFsdWU6IG1lcmdlZENvbmZpZyB9LFxuICAgIHtcbiAgICAgIHByb3ZpZGU6IEFQUF9JTklUSUFMSVpFUixcbiAgICAgIHVzZUZhY3Rvcnk6IChhcHBDb25maWdTZXJ2aWNlOiBBcHBDb25maWdTZXJ2aWNlKSA9PiAoKSA9PlxuICAgICAgICBhcHBDb25maWdTZXJ2aWNlLmluaXRpYWxpemUoKSxcbiAgICAgIGRlcHM6IFtBcHBDb25maWdTZXJ2aWNlXSxcbiAgICAgIG11bHRpOiB0cnVlLFxuICAgIH0sXG4gIF0pO1xufVxuIl19
@@ -0,0 +1,39 @@
1
+ /**
2
+ * AppConfig Service Module
3
+ *
4
+ * Servicio de configuración remota para aplicaciones Angular.
5
+ * Lee configuración desde Firestore y provee feature flags, variables dinámicas,
6
+ * y detección de actualizaciones.
7
+ *
8
+ * @example
9
+ * ```typescript
10
+ * // main.ts
11
+ * import { provideValtechAppConfig } from 'valtech-components';
12
+ *
13
+ * bootstrapApplication(AppComponent, {
14
+ * providers: [
15
+ * provideValtechFirebase(environment.valtechFirebase),
16
+ * provideValtechAppConfig({
17
+ * currentVersion: '1.0.0',
18
+ * showUpdateBanner: true,
19
+ * }),
20
+ * ],
21
+ * });
22
+ *
23
+ * // component.ts
24
+ * import { AppConfigService } from 'valtech-components';
25
+ *
26
+ * @Component({...})
27
+ * export class MyComponent {
28
+ * private appConfig = inject(AppConfigService);
29
+ *
30
+ * showFeature = computed(() =>
31
+ * this.appConfig.isFeatureEnabled('newDashboard')
32
+ * );
33
+ * }
34
+ * ```
35
+ */
36
+ export * from './types';
37
+ export { VALTECH_APP_CONFIG, provideValtechAppConfig } from './config';
38
+ export { AppConfigService } from './app-config.service';
39
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL2FwcC1jb25maWcvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FrQ0c7QUFFSCxjQUFjLFNBQVMsQ0FBQztBQUN4QixPQUFPLEVBQUUsa0JBQWtCLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSxVQUFVLENBQUM7QUFDdkUsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sc0JBQXNCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEFwcENvbmZpZyBTZXJ2aWNlIE1vZHVsZVxuICpcbiAqIFNlcnZpY2lvIGRlIGNvbmZpZ3VyYWNpw7NuIHJlbW90YSBwYXJhIGFwbGljYWNpb25lcyBBbmd1bGFyLlxuICogTGVlIGNvbmZpZ3VyYWNpw7NuIGRlc2RlIEZpcmVzdG9yZSB5IHByb3ZlZSBmZWF0dXJlIGZsYWdzLCB2YXJpYWJsZXMgZGluw6FtaWNhcyxcbiAqIHkgZGV0ZWNjacOzbiBkZSBhY3R1YWxpemFjaW9uZXMuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIC8vIG1haW4udHNcbiAqIGltcG9ydCB7IHByb3ZpZGVWYWx0ZWNoQXBwQ29uZmlnIH0gZnJvbSAndmFsdGVjaC1jb21wb25lbnRzJztcbiAqXG4gKiBib290c3RyYXBBcHBsaWNhdGlvbihBcHBDb21wb25lbnQsIHtcbiAqICAgcHJvdmlkZXJzOiBbXG4gKiAgICAgcHJvdmlkZVZhbHRlY2hGaXJlYmFzZShlbnZpcm9ubWVudC52YWx0ZWNoRmlyZWJhc2UpLFxuICogICAgIHByb3ZpZGVWYWx0ZWNoQXBwQ29uZmlnKHtcbiAqICAgICAgIGN1cnJlbnRWZXJzaW9uOiAnMS4wLjAnLFxuICogICAgICAgc2hvd1VwZGF0ZUJhbm5lcjogdHJ1ZSxcbiAqICAgICB9KSxcbiAqICAgXSxcbiAqIH0pO1xuICpcbiAqIC8vIGNvbXBvbmVudC50c1xuICogaW1wb3J0IHsgQXBwQ29uZmlnU2VydmljZSB9IGZyb20gJ3ZhbHRlY2gtY29tcG9uZW50cyc7XG4gKlxuICogQENvbXBvbmVudCh7Li4ufSlcbiAqIGV4cG9ydCBjbGFzcyBNeUNvbXBvbmVudCB7XG4gKiAgIHByaXZhdGUgYXBwQ29uZmlnID0gaW5qZWN0KEFwcENvbmZpZ1NlcnZpY2UpO1xuICpcbiAqICAgc2hvd0ZlYXR1cmUgPSBjb21wdXRlZCgoKSA9PlxuICogICAgIHRoaXMuYXBwQ29uZmlnLmlzRmVhdHVyZUVuYWJsZWQoJ25ld0Rhc2hib2FyZCcpXG4gKiAgICk7XG4gKiB9XG4gKiBgYGBcbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL3R5cGVzJztcbmV4cG9ydCB7IFZBTFRFQ0hfQVBQX0NPTkZJRywgcHJvdmlkZVZhbHRlY2hBcHBDb25maWcgfSBmcm9tICcuL2NvbmZpZyc7XG5leHBvcnQgeyBBcHBDb25maWdTZXJ2aWNlIH0gZnJvbSAnLi9hcHAtY29uZmlnLnNlcnZpY2UnO1xuIl19