valtech-components 2.0.302 → 2.0.305

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 (32) hide show
  1. package/esm2022/lib/components/atoms/button/button.component.mjs +103 -23
  2. package/esm2022/lib/components/types.mjs +1 -1
  3. package/esm2022/lib/services/lang-provider/content.mjs +1 -72
  4. package/esm2022/lib/services/lang-provider/lang-provider.service.mjs +4 -9
  5. package/esm2022/lib/shared/utils/content.mjs +2 -8
  6. package/esm2022/public-api.mjs +1 -12
  7. package/fesm2022/valtech-components.mjs +124 -2824
  8. package/fesm2022/valtech-components.mjs.map +1 -1
  9. package/lib/components/atoms/button/button.component.d.ts +33 -2
  10. package/lib/components/types.d.ts +19 -7
  11. package/package.json +1 -1
  12. package/public-api.d.ts +0 -10
  13. package/esm2022/lib/examples/comprehensive-link-test.component.mjs +0 -208
  14. package/esm2022/lib/examples/custom-content-demo.component.mjs +0 -291
  15. package/esm2022/lib/examples/display-demo.component.mjs +0 -518
  16. package/esm2022/lib/examples/display-simple-example.component.mjs +0 -202
  17. package/esm2022/lib/examples/link-processing-example.component.mjs +0 -233
  18. package/esm2022/lib/examples/multi-language-demo.component.mjs +0 -304
  19. package/esm2022/lib/examples/reactive-components-demo.component.mjs +0 -303
  20. package/esm2022/lib/examples/reactivity-test.component.mjs +0 -200
  21. package/esm2022/lib/examples/selector-examples.component.mjs +0 -234
  22. package/esm2022/lib/examples/user-issue-test.component.mjs +0 -267
  23. package/lib/examples/comprehensive-link-test.component.d.ts +0 -23
  24. package/lib/examples/custom-content-demo.component.d.ts +0 -26
  25. package/lib/examples/display-demo.component.d.ts +0 -62
  26. package/lib/examples/display-simple-example.component.d.ts +0 -23
  27. package/lib/examples/link-processing-example.component.d.ts +0 -26
  28. package/lib/examples/multi-language-demo.component.d.ts +0 -34
  29. package/lib/examples/reactive-components-demo.component.d.ts +0 -45
  30. package/lib/examples/reactivity-test.component.d.ts +0 -27
  31. package/lib/examples/selector-examples.component.d.ts +0 -21
  32. package/lib/examples/user-issue-test.component.d.ts +0 -31
@@ -1,304 +0,0 @@
1
- import { CommonModule } from '@angular/common';
2
- import { Component } from '@angular/core';
3
- import { ButtonComponent } from '../components/atoms/button/button.component';
4
- import { TextComponent } from '../components/atoms/text/text.component';
5
- import { LANGUAGES } from '../services/lang-provider/types';
6
- import * as i0 from "@angular/core";
7
- import * as i1 from "../services/content.service";
8
- import * as i2 from "../services/lang-provider/lang-provider.service";
9
- import * as i3 from "@angular/common";
10
- /**
11
- * MultiLanguageDemoComponent - Demuestra el sistema de idiomas flexible.
12
- *
13
- * Este componente muestra cómo el sistema maneja múltiples idiomas,
14
- * fallbacks automáticos y warnings por traducciones faltantes.
15
- */
16
- export class MultiLanguageDemoComponent {
17
- constructor(content, langService) {
18
- this.content = content;
19
- this.langService = langService;
20
- this.availableLanguages = [];
21
- this.analysisResults = null;
22
- // Global content observables
23
- this.okButton$ = this.content.fromContent({ key: 'ok' });
24
- this.cancelButton$ = this.content.fromContent({ key: 'cancel' });
25
- this.saveButton$ = this.content.fromContent({ key: 'save' });
26
- this.deleteButton$ = this.content.fromContent({ key: 'delete' });
27
- // Content that might have missing translations
28
- this.nextButton$ = this.content.fromContent({ key: 'next', fallback: 'Next' });
29
- this.finishButton$ = this.content.fromContent({ key: 'finish', fallback: 'Finish' });
30
- this.searchPlaceholder$ = this.content.fromContent({ key: 'searchPlaceholder', fallback: 'Search...' });
31
- this.noDataMessage$ = this.content.fromContent({ key: 'noData', fallback: 'No data available' });
32
- }
33
- ngOnInit() {
34
- this.availableLanguages = this.langService.availableLangs;
35
- }
36
- switchLanguage(lang) {
37
- console.log(`Switching to language: ${lang}`);
38
- this.langService.setLang(lang);
39
- this.analysisResults = null; // Reset analysis when language changes
40
- }
41
- getLanguageName(lang) {
42
- const names = {
43
- [LANGUAGES.ES]: 'Español',
44
- [LANGUAGES.EN]: 'English',
45
- [LANGUAGES.FR]: 'Français',
46
- [LANGUAGES.DE]: 'Deutsch',
47
- pt: 'Português',
48
- };
49
- return names[lang] || lang.toUpperCase();
50
- }
51
- analyzeCurrentComponent() {
52
- const componentName = '_global'; // Analyzing global content for this demo
53
- const availableLanguages = this.langService.getAvailableLanguagesForComponent(componentName);
54
- const missingKeys = this.langService.getMissingContentKeys(componentName, this.langService.currentLang);
55
- this.analysisResults = {
56
- availableLanguages,
57
- missingKeys,
58
- };
59
- console.log('Component analysis results:', this.analysisResults);
60
- }
61
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MultiLanguageDemoComponent, deps: [{ token: i1.ContentService }, { token: i2.LangService }], target: i0.ɵɵFactoryTarget.Component }); }
62
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: MultiLanguageDemoComponent, isStandalone: true, selector: "val-multi-language-demo", ngImport: i0, template: `
63
- <div class="multi-lang-demo">
64
- <h2>Sistema de Idiomas Flexible</h2>
65
-
66
- <div class="language-info">
67
- <h3>Información del Sistema:</h3>
68
- <p><strong>Idioma actual:</strong> {{ langService.currentLang }}</p>
69
- <p><strong>Idioma por defecto:</strong> {{ langService.defaultLanguage }}</p>
70
- <p><strong>Idiomas disponibles:</strong> {{ langService.availableLangs.join(', ') }}</p>
71
- </div>
72
-
73
- <div class="language-switcher">
74
- <h3>Cambiar Idioma:</h3>
75
- <div class="button-group">
76
- <button
77
- *ngFor="let lang of availableLanguages"
78
- [class.active]="lang === langService.currentLang"
79
- (click)="switchLanguage(lang)"
80
- >
81
- {{ getLanguageName(lang) }}
82
- </button>
83
- <!-- Ejemplo de idioma no disponible -->
84
- <button (click)="switchLanguage('pt')">Português (no disponible)</button>
85
- </div>
86
- </div>
87
-
88
- <div class="content-examples">
89
- <h3>Contenido Global:</h3>
90
- <div class="example-grid">
91
- <val-text
92
- [props]="{ content: okButton$ | async, size: 'medium', color: 'dark', bold: false, processLinks: false }"
93
- ></val-text>
94
- <val-text
95
- [props]="{
96
- content: cancelButton$ | async,
97
- size: 'medium',
98
- color: 'dark',
99
- bold: false,
100
- processLinks: false,
101
- }"
102
- ></val-text>
103
- <val-text
104
- [props]="{ content: saveButton$ | async, size: 'medium', color: 'dark', bold: false, processLinks: false }"
105
- ></val-text>
106
- <val-text
107
- [props]="{
108
- content: deleteButton$ | async,
109
- size: 'medium',
110
- color: 'dark',
111
- bold: false,
112
- processLinks: false,
113
- }"
114
- ></val-text>
115
- </div>
116
-
117
- <h3>Contenido con Fallback (algunas traducciones faltantes):</h3>
118
- <div class="example-grid">
119
- <val-text
120
- [props]="{ content: nextButton$ | async, size: 'medium', color: 'dark', bold: false, processLinks: false }"
121
- ></val-text>
122
- <val-text
123
- [props]="{
124
- content: finishButton$ | async,
125
- size: 'medium',
126
- color: 'dark',
127
- bold: false,
128
- processLinks: false,
129
- }"
130
- ></val-text>
131
- <val-text
132
- [props]="{
133
- content: searchPlaceholder$ | async,
134
- size: 'medium',
135
- color: 'dark',
136
- bold: false,
137
- processLinks: false,
138
- }"
139
- ></val-text>
140
- <val-text
141
- [props]="{
142
- content: noDataMessage$ | async,
143
- size: 'medium',
144
- color: 'dark',
145
- bold: false,
146
- processLinks: false,
147
- }"
148
- ></val-text>
149
- </div>
150
- </div>
151
-
152
- <div class="warning-info">
153
- <h3>Información de Warnings:</h3>
154
- <p>
155
- Abre la consola del navegador para ver los warnings cuando cambies a idiomas con traducciones incompletas
156
- (francés, alemán).
157
- </p>
158
- <p>El sistema automáticamente usará el idioma por defecto o el primer idioma disponible como fallback.</p>
159
- </div>
160
-
161
- <div class="component-analysis">
162
- <h3>Análisis del Componente:</h3>
163
- <button (click)="analyzeCurrentComponent()">Analizar Contenido</button>
164
- <div *ngIf="analysisResults" class="analysis-results">
165
- <p>
166
- <strong>Idiomas disponibles para este componente:</strong>
167
- {{ analysisResults.availableLanguages.join(', ') }}
168
- </p>
169
- <div *ngIf="analysisResults.missingKeys.length > 0">
170
- <p>
171
- <strong>Claves faltantes en {{ langService.currentLang }}:</strong>
172
- </p>
173
- <ul>
174
- <li *ngFor="let key of analysisResults.missingKeys">{{ key }}</li>
175
- </ul>
176
- </div>
177
- </div>
178
- </div>
179
- </div>
180
- `, isInline: true, styles: [".multi-lang-demo{padding:20px;max-width:800px}.language-info,.content-examples,.warning-info,.component-analysis{margin:20px 0;padding:15px;border:1px solid var(--ion-color-light, #f4f5f8);border-radius:8px;background:var(--ion-color-light-tint, #f5f6f9)}.button-group{display:flex;gap:10px;flex-wrap:wrap}.button-group button{padding:8px 16px;border:1px solid var(--ion-color-primary, #3880ff);background:#fff;color:var(--ion-color-primary, #3880ff);border-radius:4px;cursor:pointer;transition:all .2s}.button-group button:hover{background:var(--ion-color-primary-tint, #4992ff);color:#fff}.button-group button.active{background:var(--ion-color-primary, #3880ff);color:#fff}.example-grid{display:grid;grid-template-columns:repeat(auto-fit,minmax(200px,1fr));gap:10px;margin-top:10px}.analysis-results{margin-top:10px;padding:10px;background:#fff;border-radius:4px}h2{color:var(--ion-color-primary, #3880ff)}h3{color:var(--ion-color-dark, #222428);margin-bottom:10px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }, { kind: "component", type: TextComponent, selector: "val-text", inputs: ["props"] }] }); }
181
- }
182
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MultiLanguageDemoComponent, decorators: [{
183
- type: Component,
184
- args: [{ selector: 'val-multi-language-demo', standalone: true, imports: [CommonModule, TextComponent, ButtonComponent], template: `
185
- <div class="multi-lang-demo">
186
- <h2>Sistema de Idiomas Flexible</h2>
187
-
188
- <div class="language-info">
189
- <h3>Información del Sistema:</h3>
190
- <p><strong>Idioma actual:</strong> {{ langService.currentLang }}</p>
191
- <p><strong>Idioma por defecto:</strong> {{ langService.defaultLanguage }}</p>
192
- <p><strong>Idiomas disponibles:</strong> {{ langService.availableLangs.join(', ') }}</p>
193
- </div>
194
-
195
- <div class="language-switcher">
196
- <h3>Cambiar Idioma:</h3>
197
- <div class="button-group">
198
- <button
199
- *ngFor="let lang of availableLanguages"
200
- [class.active]="lang === langService.currentLang"
201
- (click)="switchLanguage(lang)"
202
- >
203
- {{ getLanguageName(lang) }}
204
- </button>
205
- <!-- Ejemplo de idioma no disponible -->
206
- <button (click)="switchLanguage('pt')">Português (no disponible)</button>
207
- </div>
208
- </div>
209
-
210
- <div class="content-examples">
211
- <h3>Contenido Global:</h3>
212
- <div class="example-grid">
213
- <val-text
214
- [props]="{ content: okButton$ | async, size: 'medium', color: 'dark', bold: false, processLinks: false }"
215
- ></val-text>
216
- <val-text
217
- [props]="{
218
- content: cancelButton$ | async,
219
- size: 'medium',
220
- color: 'dark',
221
- bold: false,
222
- processLinks: false,
223
- }"
224
- ></val-text>
225
- <val-text
226
- [props]="{ content: saveButton$ | async, size: 'medium', color: 'dark', bold: false, processLinks: false }"
227
- ></val-text>
228
- <val-text
229
- [props]="{
230
- content: deleteButton$ | async,
231
- size: 'medium',
232
- color: 'dark',
233
- bold: false,
234
- processLinks: false,
235
- }"
236
- ></val-text>
237
- </div>
238
-
239
- <h3>Contenido con Fallback (algunas traducciones faltantes):</h3>
240
- <div class="example-grid">
241
- <val-text
242
- [props]="{ content: nextButton$ | async, size: 'medium', color: 'dark', bold: false, processLinks: false }"
243
- ></val-text>
244
- <val-text
245
- [props]="{
246
- content: finishButton$ | async,
247
- size: 'medium',
248
- color: 'dark',
249
- bold: false,
250
- processLinks: false,
251
- }"
252
- ></val-text>
253
- <val-text
254
- [props]="{
255
- content: searchPlaceholder$ | async,
256
- size: 'medium',
257
- color: 'dark',
258
- bold: false,
259
- processLinks: false,
260
- }"
261
- ></val-text>
262
- <val-text
263
- [props]="{
264
- content: noDataMessage$ | async,
265
- size: 'medium',
266
- color: 'dark',
267
- bold: false,
268
- processLinks: false,
269
- }"
270
- ></val-text>
271
- </div>
272
- </div>
273
-
274
- <div class="warning-info">
275
- <h3>Información de Warnings:</h3>
276
- <p>
277
- Abre la consola del navegador para ver los warnings cuando cambies a idiomas con traducciones incompletas
278
- (francés, alemán).
279
- </p>
280
- <p>El sistema automáticamente usará el idioma por defecto o el primer idioma disponible como fallback.</p>
281
- </div>
282
-
283
- <div class="component-analysis">
284
- <h3>Análisis del Componente:</h3>
285
- <button (click)="analyzeCurrentComponent()">Analizar Contenido</button>
286
- <div *ngIf="analysisResults" class="analysis-results">
287
- <p>
288
- <strong>Idiomas disponibles para este componente:</strong>
289
- {{ analysisResults.availableLanguages.join(', ') }}
290
- </p>
291
- <div *ngIf="analysisResults.missingKeys.length > 0">
292
- <p>
293
- <strong>Claves faltantes en {{ langService.currentLang }}:</strong>
294
- </p>
295
- <ul>
296
- <li *ngFor="let key of analysisResults.missingKeys">{{ key }}</li>
297
- </ul>
298
- </div>
299
- </div>
300
- </div>
301
- </div>
302
- `, styles: [".multi-lang-demo{padding:20px;max-width:800px}.language-info,.content-examples,.warning-info,.component-analysis{margin:20px 0;padding:15px;border:1px solid var(--ion-color-light, #f4f5f8);border-radius:8px;background:var(--ion-color-light-tint, #f5f6f9)}.button-group{display:flex;gap:10px;flex-wrap:wrap}.button-group button{padding:8px 16px;border:1px solid var(--ion-color-primary, #3880ff);background:#fff;color:var(--ion-color-primary, #3880ff);border-radius:4px;cursor:pointer;transition:all .2s}.button-group button:hover{background:var(--ion-color-primary-tint, #4992ff);color:#fff}.button-group button.active{background:var(--ion-color-primary, #3880ff);color:#fff}.example-grid{display:grid;grid-template-columns:repeat(auto-fit,minmax(200px,1fr));gap:10px;margin-top:10px}.analysis-results{margin-top:10px;padding:10px;background:#fff;border-radius:4px}h2{color:var(--ion-color-primary, #3880ff)}h3{color:var(--ion-color-dark, #222428);margin-bottom:10px}\n"] }]
303
- }], ctorParameters: () => [{ type: i1.ContentService }, { type: i2.LangService }] });
304
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multi-language-demo.component.js","sourceRoot":"","sources":["../../../../../projects/valtech-components/src/lib/examples/multi-language-demo.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, OnInit } from '@angular/core';\nimport { ButtonComponent } from '../components/atoms/button/button.component';\nimport { TextComponent } from '../components/atoms/text/text.component';\nimport { ContentService } from '../services/content.service';\nimport { LangService } from '../services/lang-provider/lang-provider.service';\nimport { LANGUAGES } from '../services/lang-provider/types';\n\n/**\n * MultiLanguageDemoComponent - Demuestra el sistema de idiomas flexible.\n *\n * Este componente muestra cómo el sistema maneja múltiples idiomas,\n * fallbacks automáticos y warnings por traducciones faltantes.\n */\n@Component({\n  selector: 'val-multi-language-demo',\n  standalone: true,\n  imports: [CommonModule, TextComponent, ButtonComponent],\n  template: `\n    <div class=\"multi-lang-demo\">\n      <h2>Sistema de Idiomas Flexible</h2>\n\n      <div class=\"language-info\">\n        <h3>Información del Sistema:</h3>\n        <p><strong>Idioma actual:</strong> {{ langService.currentLang }}</p>\n        <p><strong>Idioma por defecto:</strong> {{ langService.defaultLanguage }}</p>\n        <p><strong>Idiomas disponibles:</strong> {{ langService.availableLangs.join(', ') }}</p>\n      </div>\n\n      <div class=\"language-switcher\">\n        <h3>Cambiar Idioma:</h3>\n        <div class=\"button-group\">\n          <button\n            *ngFor=\"let lang of availableLanguages\"\n            [class.active]=\"lang === langService.currentLang\"\n            (click)=\"switchLanguage(lang)\"\n          >\n            {{ getLanguageName(lang) }}\n          </button>\n          <!-- Ejemplo de idioma no disponible -->\n          <button (click)=\"switchLanguage('pt')\">Português (no disponible)</button>\n        </div>\n      </div>\n\n      <div class=\"content-examples\">\n        <h3>Contenido Global:</h3>\n        <div class=\"example-grid\">\n          <val-text\n            [props]=\"{ content: okButton$ | async, size: 'medium', color: 'dark', bold: false, processLinks: false }\"\n          ></val-text>\n          <val-text\n            [props]=\"{\n              content: cancelButton$ | async,\n              size: 'medium',\n              color: 'dark',\n              bold: false,\n              processLinks: false,\n            }\"\n          ></val-text>\n          <val-text\n            [props]=\"{ content: saveButton$ | async, size: 'medium', color: 'dark', bold: false, processLinks: false }\"\n          ></val-text>\n          <val-text\n            [props]=\"{\n              content: deleteButton$ | async,\n              size: 'medium',\n              color: 'dark',\n              bold: false,\n              processLinks: false,\n            }\"\n          ></val-text>\n        </div>\n\n        <h3>Contenido con Fallback (algunas traducciones faltantes):</h3>\n        <div class=\"example-grid\">\n          <val-text\n            [props]=\"{ content: nextButton$ | async, size: 'medium', color: 'dark', bold: false, processLinks: false }\"\n          ></val-text>\n          <val-text\n            [props]=\"{\n              content: finishButton$ | async,\n              size: 'medium',\n              color: 'dark',\n              bold: false,\n              processLinks: false,\n            }\"\n          ></val-text>\n          <val-text\n            [props]=\"{\n              content: searchPlaceholder$ | async,\n              size: 'medium',\n              color: 'dark',\n              bold: false,\n              processLinks: false,\n            }\"\n          ></val-text>\n          <val-text\n            [props]=\"{\n              content: noDataMessage$ | async,\n              size: 'medium',\n              color: 'dark',\n              bold: false,\n              processLinks: false,\n            }\"\n          ></val-text>\n        </div>\n      </div>\n\n      <div class=\"warning-info\">\n        <h3>Información de Warnings:</h3>\n        <p>\n          Abre la consola del navegador para ver los warnings cuando cambies a idiomas con traducciones incompletas\n          (francés, alemán).\n        </p>\n        <p>El sistema automáticamente usará el idioma por defecto o el primer idioma disponible como fallback.</p>\n      </div>\n\n      <div class=\"component-analysis\">\n        <h3>Análisis del Componente:</h3>\n        <button (click)=\"analyzeCurrentComponent()\">Analizar Contenido</button>\n        <div *ngIf=\"analysisResults\" class=\"analysis-results\">\n          <p>\n            <strong>Idiomas disponibles para este componente:</strong>\n            {{ analysisResults.availableLanguages.join(', ') }}\n          </p>\n          <div *ngIf=\"analysisResults.missingKeys.length > 0\">\n            <p>\n              <strong>Claves faltantes en {{ langService.currentLang }}:</strong>\n            </p>\n            <ul>\n              <li *ngFor=\"let key of analysisResults.missingKeys\">{{ key }}</li>\n            </ul>\n          </div>\n        </div>\n      </div>\n    </div>\n  `,\n  styles: [\n    `\n      .multi-lang-demo {\n        padding: 20px;\n        max-width: 800px;\n      }\n\n      .language-info,\n      .content-examples,\n      .warning-info,\n      .component-analysis {\n        margin: 20px 0;\n        padding: 15px;\n        border: 1px solid var(--ion-color-light, #f4f5f8);\n        border-radius: 8px;\n        background: var(--ion-color-light-tint, #f5f6f9);\n      }\n\n      .button-group {\n        display: flex;\n        gap: 10px;\n        flex-wrap: wrap;\n      }\n\n      .button-group button {\n        padding: 8px 16px;\n        border: 1px solid var(--ion-color-primary, #3880ff);\n        background: white;\n        color: var(--ion-color-primary, #3880ff);\n        border-radius: 4px;\n        cursor: pointer;\n        transition: all 0.2s;\n      }\n\n      .button-group button:hover {\n        background: var(--ion-color-primary-tint, #4992ff);\n        color: white;\n      }\n\n      .button-group button.active {\n        background: var(--ion-color-primary, #3880ff);\n        color: white;\n      }\n\n      .example-grid {\n        display: grid;\n        grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));\n        gap: 10px;\n        margin-top: 10px;\n      }\n\n      .analysis-results {\n        margin-top: 10px;\n        padding: 10px;\n        background: white;\n        border-radius: 4px;\n      }\n\n      h2 {\n        color: var(--ion-color-primary, #3880ff);\n      }\n\n      h3 {\n        color: var(--ion-color-dark, #222428);\n        margin-bottom: 10px;\n      }\n    `,\n  ],\n})\nexport class MultiLanguageDemoComponent implements OnInit {\n  availableLanguages: string[] = [];\n  analysisResults: { availableLanguages: string[]; missingKeys: string[] } | null = null;\n\n  // Global content observables\n  okButton$ = this.content.fromContent({ key: 'ok' });\n  cancelButton$ = this.content.fromContent({ key: 'cancel' });\n  saveButton$ = this.content.fromContent({ key: 'save' });\n  deleteButton$ = this.content.fromContent({ key: 'delete' });\n\n  // Content that might have missing translations\n  nextButton$ = this.content.fromContent({ key: 'next', fallback: 'Next' });\n  finishButton$ = this.content.fromContent({ key: 'finish', fallback: 'Finish' });\n  searchPlaceholder$ = this.content.fromContent({ key: 'searchPlaceholder', fallback: 'Search...' });\n  noDataMessage$ = this.content.fromContent({ key: 'noData', fallback: 'No data available' });\n\n  constructor(\n    public content: ContentService,\n    public langService: LangService\n  ) {}\n\n  ngOnInit() {\n    this.availableLanguages = this.langService.availableLangs;\n  }\n\n  switchLanguage(lang: string) {\n    console.log(`Switching to language: ${lang}`);\n    this.langService.setLang(lang);\n    this.analysisResults = null; // Reset analysis when language changes\n  }\n\n  getLanguageName(lang: string): string {\n    const names: Record<string, string> = {\n      [LANGUAGES.ES]: 'Español',\n      [LANGUAGES.EN]: 'English',\n      [LANGUAGES.FR]: 'Français',\n      [LANGUAGES.DE]: 'Deutsch',\n      pt: 'Português',\n    };\n    return names[lang] || lang.toUpperCase();\n  }\n\n  analyzeCurrentComponent() {\n    const componentName = '_global'; // Analyzing global content for this demo\n    const availableLanguages = this.langService.getAvailableLanguagesForComponent(componentName);\n    const missingKeys = this.langService.getMissingContentKeys(componentName, this.langService.currentLang);\n\n    this.analysisResults = {\n      availableLanguages,\n      missingKeys,\n    };\n\n    console.log('Component analysis results:', this.analysisResults);\n  }\n}\n"]}