valtech-components 2.0.290 → 2.0.292
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/examples/comprehensive-link-test.component.mjs +208 -0
- package/esm2022/lib/examples/custom-content-demo.component.mjs +3 -3
- package/esm2022/lib/examples/link-processing-example.component.mjs +26 -4
- package/esm2022/lib/examples/multi-language-demo.component.mjs +304 -0
- package/esm2022/lib/services/lang-provider/content.mjs +33 -2
- package/esm2022/lib/services/lang-provider/lang-provider.service.mjs +199 -13
- package/esm2022/lib/services/lang-provider/types.mjs +15 -6
- package/esm2022/lib/services/link-processor.service.mjs +70 -27
- package/esm2022/public-api.mjs +3 -1
- package/fesm2022/valtech-components.mjs +839 -48
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/examples/comprehensive-link-test.component.d.ts +23 -0
- package/lib/examples/link-processing-example.component.d.ts +1 -0
- package/lib/examples/multi-language-demo.component.d.ts +34 -0
- package/lib/services/lang-provider/content.d.ts +4 -1
- package/lib/services/lang-provider/lang-provider.service.d.ts +64 -2
- package/lib/services/lang-provider/types.d.ts +19 -4
- package/lib/services/link-processor.service.d.ts +6 -0
- package/package.json +1 -1
- package/public-api.d.ts +2 -0
- package/src/lib/components/styles/overrides.scss +2 -2
|
@@ -3,7 +3,7 @@ import { BehaviorSubject, distinctUntilChanged, map } from 'rxjs';
|
|
|
3
3
|
import { LANG } from '../../shared/constants/storage';
|
|
4
4
|
import { LocalStorageService } from '../local-storage.service';
|
|
5
5
|
import { ValtechConfigService } from '../types';
|
|
6
|
-
import {
|
|
6
|
+
import { LANGUAGES } from './types';
|
|
7
7
|
import * as i0 from "@angular/core";
|
|
8
8
|
/**
|
|
9
9
|
* LangService - Reactive language and content management service.
|
|
@@ -11,6 +11,9 @@ import * as i0 from "@angular/core";
|
|
|
11
11
|
* This service provides reactive content management with Observable-based language switching.
|
|
12
12
|
* Components can subscribe to content changes and automatically update when the language changes.
|
|
13
13
|
*
|
|
14
|
+
* The service automatically detects available languages from the content configuration
|
|
15
|
+
* and provides intelligent fallbacks with console warnings for missing translations.
|
|
16
|
+
*
|
|
14
17
|
* @example Basic usage:
|
|
15
18
|
* ```typescript
|
|
16
19
|
* constructor(private langService: LangService) {}
|
|
@@ -30,12 +33,120 @@ import * as i0 from "@angular/core";
|
|
|
30
33
|
*/
|
|
31
34
|
export class LangService {
|
|
32
35
|
constructor(config) {
|
|
33
|
-
this.
|
|
34
|
-
|
|
36
|
+
this.availableLanguages = [];
|
|
37
|
+
this.warnedMissingLanguages = new Set();
|
|
38
|
+
console.log('LangService: Injected config:', config);
|
|
35
39
|
this.content = config.content;
|
|
36
40
|
this.config = config;
|
|
41
|
+
// Detect available languages from content
|
|
42
|
+
this.detectAvailableLanguages();
|
|
43
|
+
// Set default language (prefer Spanish, then English, then first available)
|
|
44
|
+
this.defaultLang = this.determineDefaultLanguage();
|
|
45
|
+
// Initialize with stored language or default
|
|
37
46
|
const current = LocalStorageService.get(LANG);
|
|
38
|
-
|
|
47
|
+
const initialLang = this.validateLanguage(current) || this.defaultLang;
|
|
48
|
+
this.selectedLang = new BehaviorSubject(initialLang);
|
|
49
|
+
console.log('LangService: Initialized with languages:', {
|
|
50
|
+
available: this.availableLanguages,
|
|
51
|
+
default: this.defaultLang,
|
|
52
|
+
current: initialLang,
|
|
53
|
+
});
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* Detect available languages from the content configuration.
|
|
57
|
+
* Scans all component content to find which languages are actually configured.
|
|
58
|
+
*/
|
|
59
|
+
detectAvailableLanguages() {
|
|
60
|
+
const languageSet = new Set();
|
|
61
|
+
Object.values(this.content).forEach(componentContent => {
|
|
62
|
+
if (componentContent?.Content) {
|
|
63
|
+
Object.keys(componentContent.Content).forEach(lang => {
|
|
64
|
+
languageSet.add(lang);
|
|
65
|
+
});
|
|
66
|
+
}
|
|
67
|
+
});
|
|
68
|
+
this.availableLanguages = Array.from(languageSet).sort();
|
|
69
|
+
if (this.availableLanguages.length === 0) {
|
|
70
|
+
console.warn('LangService: No languages detected in content configuration!');
|
|
71
|
+
this.availableLanguages = [LANGUAGES.ES]; // Fallback
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
/**
|
|
75
|
+
* Determine the best default language based on available content.
|
|
76
|
+
*/
|
|
77
|
+
determineDefaultLanguage() {
|
|
78
|
+
// Preference order: Spanish, English, then first available
|
|
79
|
+
const preferredOrder = [LANGUAGES.ES, LANGUAGES.EN];
|
|
80
|
+
for (const preferred of preferredOrder) {
|
|
81
|
+
if (this.availableLanguages.includes(preferred)) {
|
|
82
|
+
return preferred;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
return this.availableLanguages[0];
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Validate if a language is available in the content.
|
|
89
|
+
*/
|
|
90
|
+
validateLanguage(lang) {
|
|
91
|
+
if (!lang)
|
|
92
|
+
return null;
|
|
93
|
+
return this.availableLanguages.includes(lang) ? lang : null;
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Get the best available language for a component and key.
|
|
97
|
+
* Provides intelligent fallback with warnings.
|
|
98
|
+
*/
|
|
99
|
+
getBestAvailableContent(className, key, requestedLang) {
|
|
100
|
+
const componentContent = this.content[className];
|
|
101
|
+
if (!componentContent) {
|
|
102
|
+
return {
|
|
103
|
+
content: undefined,
|
|
104
|
+
actualLang: requestedLang,
|
|
105
|
+
shouldWarn: false,
|
|
106
|
+
};
|
|
107
|
+
}
|
|
108
|
+
// Try requested language first
|
|
109
|
+
const requestedContent = componentContent.Content[requestedLang];
|
|
110
|
+
if (requestedContent?.[key]) {
|
|
111
|
+
return {
|
|
112
|
+
content: requestedContent[key],
|
|
113
|
+
actualLang: requestedLang,
|
|
114
|
+
shouldWarn: false,
|
|
115
|
+
};
|
|
116
|
+
}
|
|
117
|
+
// Language not available, try fallbacks
|
|
118
|
+
const warningKey = `${className}.${key}.${requestedLang}`;
|
|
119
|
+
const shouldWarn = !this.warnedMissingLanguages.has(warningKey);
|
|
120
|
+
if (shouldWarn) {
|
|
121
|
+
this.warnedMissingLanguages.add(warningKey);
|
|
122
|
+
}
|
|
123
|
+
// Try default language
|
|
124
|
+
if (requestedLang !== this.defaultLang) {
|
|
125
|
+
const defaultContent = componentContent.Content[this.defaultLang];
|
|
126
|
+
if (defaultContent?.[key]) {
|
|
127
|
+
return {
|
|
128
|
+
content: defaultContent[key],
|
|
129
|
+
actualLang: this.defaultLang,
|
|
130
|
+
shouldWarn,
|
|
131
|
+
};
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
// Try first available language
|
|
135
|
+
for (const availableLang of this.availableLanguages) {
|
|
136
|
+
const availableContent = componentContent.Content[availableLang];
|
|
137
|
+
if (availableContent?.[key]) {
|
|
138
|
+
return {
|
|
139
|
+
content: availableContent[key],
|
|
140
|
+
actualLang: availableLang,
|
|
141
|
+
shouldWarn,
|
|
142
|
+
};
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
return {
|
|
146
|
+
content: undefined,
|
|
147
|
+
actualLang: requestedLang,
|
|
148
|
+
shouldWarn,
|
|
149
|
+
};
|
|
39
150
|
}
|
|
40
151
|
/**
|
|
41
152
|
* Observable that emits the current language whenever it changes.
|
|
@@ -50,13 +161,32 @@ export class LangService {
|
|
|
50
161
|
get currentLang() {
|
|
51
162
|
return this.selectedLang.value;
|
|
52
163
|
}
|
|
164
|
+
/**
|
|
165
|
+
* Get array of available languages detected from content.
|
|
166
|
+
*/
|
|
167
|
+
get availableLangs() {
|
|
168
|
+
return [...this.availableLanguages];
|
|
169
|
+
}
|
|
170
|
+
/**
|
|
171
|
+
* Get the default language.
|
|
172
|
+
*/
|
|
173
|
+
get defaultLanguage() {
|
|
174
|
+
return this.defaultLang;
|
|
175
|
+
}
|
|
53
176
|
/**
|
|
54
177
|
* Set the current language and persist it to localStorage.
|
|
55
178
|
* This will trigger updates in all reactive content subscriptions.
|
|
56
179
|
*
|
|
180
|
+
* Validates that the language is available and warns if not.
|
|
181
|
+
*
|
|
57
182
|
* @param lang - The language to set
|
|
58
183
|
*/
|
|
59
184
|
setLang(lang) {
|
|
185
|
+
if (!this.availableLanguages.includes(lang)) {
|
|
186
|
+
console.warn(`LangService: Language "${lang}" is not available. Available languages:`, this.availableLanguages);
|
|
187
|
+
console.warn(`LangService: Falling back to default language "${this.defaultLang}"`);
|
|
188
|
+
lang = this.defaultLang;
|
|
189
|
+
}
|
|
60
190
|
this.selectedLang.next(lang);
|
|
61
191
|
LocalStorageService.set(LANG, lang);
|
|
62
192
|
}
|
|
@@ -66,10 +196,12 @@ export class LangService {
|
|
|
66
196
|
* @deprecated Use getText() or getContent() for better type safety
|
|
67
197
|
*/
|
|
68
198
|
Text(className) {
|
|
69
|
-
|
|
199
|
+
const componentContent = this.content[className];
|
|
200
|
+
return componentContent?.Content[this.selectedLang.value] || {};
|
|
70
201
|
}
|
|
71
202
|
/**
|
|
72
203
|
* Get a single content string synchronously for the current language.
|
|
204
|
+
* Provides intelligent fallback with warnings for missing translations.
|
|
73
205
|
*
|
|
74
206
|
* @param className - The component class name
|
|
75
207
|
* @param key - The text key
|
|
@@ -77,12 +209,16 @@ export class LangService {
|
|
|
77
209
|
* @returns The text string or fallback
|
|
78
210
|
*/
|
|
79
211
|
getText(className, key, fallback) {
|
|
80
|
-
const
|
|
81
|
-
|
|
212
|
+
const result = this.getBestAvailableContent(className, key, this.selectedLang.value);
|
|
213
|
+
if (result.shouldWarn && result.actualLang !== this.selectedLang.value) {
|
|
214
|
+
console.warn(`LangService: Content "${className}.${key}" not available in "${this.selectedLang.value}".`, `Using "${result.actualLang}" instead. Available languages:`, this.availableLanguages);
|
|
215
|
+
}
|
|
216
|
+
return result.content || fallback || `[${className}.${key}]`;
|
|
82
217
|
}
|
|
83
218
|
/**
|
|
84
219
|
* Get a reactive Observable for a specific text key that updates when language changes.
|
|
85
220
|
* This is the recommended method for components that need reactive content.
|
|
221
|
+
* Provides intelligent fallback with warnings for missing translations.
|
|
86
222
|
*
|
|
87
223
|
* @param className - The component class name
|
|
88
224
|
* @param key - The text key
|
|
@@ -91,12 +227,16 @@ export class LangService {
|
|
|
91
227
|
*/
|
|
92
228
|
getContent(className, key, fallback) {
|
|
93
229
|
return this.currentLang$.pipe(map(lang => {
|
|
94
|
-
const
|
|
95
|
-
|
|
230
|
+
const result = this.getBestAvailableContent(className, key, lang);
|
|
231
|
+
if (result.shouldWarn && result.actualLang !== lang) {
|
|
232
|
+
console.warn(`LangService: Content "${className}.${key}" not available in "${lang}".`, `Using "${result.actualLang}" instead. Available languages:`, this.availableLanguages);
|
|
233
|
+
}
|
|
234
|
+
return result.content || fallback || `[${className}.${key}]`;
|
|
96
235
|
}), distinctUntilChanged());
|
|
97
236
|
}
|
|
98
237
|
/**
|
|
99
238
|
* Get reactive content for multiple keys at once.
|
|
239
|
+
* Provides intelligent fallback with warnings for missing translations.
|
|
100
240
|
*
|
|
101
241
|
* @param className - The component class name
|
|
102
242
|
* @param keys - Array of text keys to retrieve
|
|
@@ -104,16 +244,19 @@ export class LangService {
|
|
|
104
244
|
*/
|
|
105
245
|
getMultipleContent(className, keys) {
|
|
106
246
|
return this.currentLang$.pipe(map(lang => {
|
|
107
|
-
const classContent = this.content[className]?.Content[lang] || {};
|
|
108
247
|
const result = {};
|
|
109
248
|
keys.forEach(key => {
|
|
110
|
-
|
|
249
|
+
const contentResult = this.getBestAvailableContent(className, key, lang);
|
|
250
|
+
if (contentResult.shouldWarn && contentResult.actualLang !== lang) {
|
|
251
|
+
console.warn(`LangService: Content "${className}.${key}" not available in "${lang}".`, `Using "${contentResult.actualLang}" instead.`);
|
|
252
|
+
}
|
|
253
|
+
result[key] = contentResult.content || `[${className}.${key}]`;
|
|
111
254
|
});
|
|
112
255
|
return result;
|
|
113
256
|
}), distinctUntilChanged((prev, curr) => JSON.stringify(prev) === JSON.stringify(curr)));
|
|
114
257
|
}
|
|
115
258
|
/**
|
|
116
|
-
* Check if a content key exists for a component.
|
|
259
|
+
* Check if a content key exists for a component in any available language.
|
|
117
260
|
*
|
|
118
261
|
* @param className - The component class name
|
|
119
262
|
* @param key - The text key
|
|
@@ -125,6 +268,49 @@ export class LangService {
|
|
|
125
268
|
return false;
|
|
126
269
|
return Object.values(classContent.Content).some(langContent => langContent && typeof langContent[key] === 'string');
|
|
127
270
|
}
|
|
271
|
+
/**
|
|
272
|
+
* Check if a content key exists for a component in a specific language.
|
|
273
|
+
*
|
|
274
|
+
* @param className - The component class name
|
|
275
|
+
* @param key - The text key
|
|
276
|
+
* @param lang - The language to check (defaults to current language)
|
|
277
|
+
* @returns True if the key exists in the specified language
|
|
278
|
+
*/
|
|
279
|
+
hasContentInLanguage(className, key, lang) {
|
|
280
|
+
const targetLang = lang || this.currentLang;
|
|
281
|
+
const classContent = this.content[className]?.Content[targetLang];
|
|
282
|
+
return classContent && typeof classContent[key] === 'string';
|
|
283
|
+
}
|
|
284
|
+
/**
|
|
285
|
+
* Get available languages for a specific component.
|
|
286
|
+
*
|
|
287
|
+
* @param className - The component class name
|
|
288
|
+
* @returns Array of language codes available for the component
|
|
289
|
+
*/
|
|
290
|
+
getAvailableLanguagesForComponent(className) {
|
|
291
|
+
const classContent = this.content[className];
|
|
292
|
+
if (!classContent)
|
|
293
|
+
return [];
|
|
294
|
+
return Object.keys(classContent.Content).filter(lang => classContent.Content[lang] && Object.keys(classContent.Content[lang]).length > 0);
|
|
295
|
+
}
|
|
296
|
+
/**
|
|
297
|
+
* Get missing content keys for a component in a specific language.
|
|
298
|
+
* Useful for identifying incomplete translations.
|
|
299
|
+
*
|
|
300
|
+
* @param className - The component class name
|
|
301
|
+
* @param lang - The language to check
|
|
302
|
+
* @param referenceLang - The reference language to compare against (defaults to default language)
|
|
303
|
+
* @returns Array of missing keys
|
|
304
|
+
*/
|
|
305
|
+
getMissingContentKeys(className, lang, referenceLang) {
|
|
306
|
+
const refLang = referenceLang || this.defaultLang;
|
|
307
|
+
const classContent = this.content[className];
|
|
308
|
+
if (!classContent)
|
|
309
|
+
return [];
|
|
310
|
+
const referenceContent = classContent.Content[refLang] || {};
|
|
311
|
+
const targetContent = classContent.Content[lang] || {};
|
|
312
|
+
return Object.keys(referenceContent).filter(key => !targetContent[key] || typeof targetContent[key] !== 'string');
|
|
313
|
+
}
|
|
128
314
|
// Legacy getters/setters for backward compatibility
|
|
129
315
|
get Lang() {
|
|
130
316
|
return this.currentLang;
|
|
@@ -144,4 +330,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
144
330
|
type: Inject,
|
|
145
331
|
args: [ValtechConfigService]
|
|
146
332
|
}] }] });
|
|
147
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lang-provider.service.js","sourceRoot":"","sources":["../../../../../../projects/valtech-components/src/lib/services/lang-provider/lang-provider.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable } from '@angular/core';\nimport { BehaviorSubject, Observable, distinctUntilChanged, map } from 'rxjs';\nimport { LANG } from '../../shared/constants/storage';\nimport { LocalStorageService } from '../local-storage.service';\nimport { ValtechConfig, ValtechConfigService } from '../types';\nimport { Provider } from './content';\nimport { LangOption, LanguageText } from './types';\n\n/**\n * LangService - Reactive language and content management service.\n *\n * This service provides reactive content management with Observable-based language switching.\n * Components can subscribe to content changes and automatically update when the language changes.\n *\n * @example Basic usage:\n * ```typescript\n * constructor(private langService: LangService) {}\n *\n * // Get current language\n * const currentLang = this.langService.currentLang;\n *\n * // Subscribe to language changes\n * this.langService.currentLang$.subscribe(lang => console.log('Language changed:', lang));\n *\n * // Get static text\n * const text = this.langService.getText('ComponentName', 'textKey');\n *\n * // Get reactive text\n * const text$ = this.langService.getContent('ComponentName', 'textKey');\n * ```\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class LangService {\n  private content: Provider;\n  private default = LangOption.ES;\n  private selectedLang: BehaviorSubject<LangOption>;\n  private config: ValtechConfig;\n\n  constructor(@Inject(ValtechConfigService) config: ValtechConfig) {\n    console.log('injected config: ', config);\n    this.content = config.content;\n    this.config = config;\n    const current = LocalStorageService.get<LangOption>(LANG);\n    this.selectedLang = new BehaviorSubject<LangOption>(current || this.default);\n  }\n\n  /**\n   * Observable that emits the current language whenever it changes.\n   * Use this to subscribe to language changes in components.\n   */\n  get currentLang$(): Observable<LangOption> {\n    return this.selectedLang.asObservable().pipe(distinctUntilChanged());\n  }\n\n  /**\n   * Get the current language synchronously.\n   */\n  get currentLang(): LangOption {\n    return this.selectedLang.value;\n  }\n\n  /**\n   * Set the current language and persist it to localStorage.\n   * This will trigger updates in all reactive content subscriptions.\n   *\n   * @param lang - The language to set\n   */\n  setLang(lang: LangOption): void {\n    this.selectedLang.next(lang);\n    LocalStorageService.set<LangOption>(LANG, lang);\n  }\n\n  /**\n   * Get content for a component class and key (legacy method).\n   *\n   * @deprecated Use getText() or getContent() for better type safety\n   */\n  Text(className: string): LanguageText {\n    return this.content[className].Content[this.selectedLang.value];\n  }\n\n  /**\n   * Get a single content string synchronously for the current language.\n   *\n   * @param className - The component class name\n   * @param key - The text key\n   * @param fallback - Optional fallback text if key is not found\n   * @returns The text string or fallback\n   */\n  getText(className: string, key: string, fallback?: string): string {\n    const classContent = this.content[className]?.Content[this.selectedLang.value];\n    return classContent?.[key] || fallback || `[${className}.${key}]`;\n  }\n\n  /**\n   * Get a reactive Observable for a specific text key that updates when language changes.\n   * This is the recommended method for components that need reactive content.\n   *\n   * @param className - The component class name\n   * @param key - The text key\n   * @param fallback - Optional fallback text if key is not found\n   * @returns Observable that emits the text string whenever language changes\n   */\n  getContent(className: string, key: string, fallback?: string): Observable<string> {\n    return this.currentLang$.pipe(\n      map(lang => {\n        const classContent = this.content[className]?.Content[lang];\n        return classContent?.[key] || fallback || `[${className}.${key}]`;\n      }),\n      distinctUntilChanged()\n    );\n  }\n\n  /**\n   * Get reactive content for multiple keys at once.\n   *\n   * @param className - The component class name\n   * @param keys - Array of text keys to retrieve\n   * @returns Observable that emits an object with all requested keys\n   */\n  getMultipleContent(className: string, keys: string[]): Observable<Record<string, string>> {\n    return this.currentLang$.pipe(\n      map(lang => {\n        const classContent = this.content[className]?.Content[lang] || {};\n        const result: Record<string, string> = {};\n        keys.forEach(key => {\n          result[key] = classContent[key] || `[${className}.${key}]`;\n        });\n        return result;\n      }),\n      distinctUntilChanged((prev, curr) => JSON.stringify(prev) === JSON.stringify(curr))\n    );\n  }\n\n  /**\n   * Check if a content key exists for a component.\n   *\n   * @param className - The component class name\n   * @param key - The text key\n   * @returns True if the key exists in any language\n   */\n  hasContent(className: string, key: string): boolean {\n    const classContent = this.content[className];\n    if (!classContent) return false;\n\n    return Object.values(classContent.Content).some(\n      langContent => langContent && typeof langContent[key] === 'string'\n    );\n  }\n\n  // Legacy getters/setters for backward compatibility\n  get Lang(): LangOption {\n    return this.currentLang;\n  }\n\n  set Lang(lang: LangOption) {\n    this.setLang(lang);\n  }\n}\n"]}
|
|
333
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lang-provider.service.js","sourceRoot":"","sources":["../../../../../../projects/valtech-components/src/lib/services/lang-provider/lang-provider.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable } from '@angular/core';\nimport { BehaviorSubject, Observable, distinctUntilChanged, map } from 'rxjs';\nimport { LANG } from '../../shared/constants/storage';\nimport { LocalStorageService } from '../local-storage.service';\nimport { ValtechConfig, ValtechConfigService } from '../types';\nimport { Provider } from './content';\nimport { LANGUAGES, LangOption, LanguageText } from './types';\n\n/**\n * LangService - Reactive language and content management service.\n *\n * This service provides reactive content management with Observable-based language switching.\n * Components can subscribe to content changes and automatically update when the language changes.\n *\n * The service automatically detects available languages from the content configuration\n * and provides intelligent fallbacks with console warnings for missing translations.\n *\n * @example Basic usage:\n * ```typescript\n * constructor(private langService: LangService) {}\n *\n * // Get current language\n * const currentLang = this.langService.currentLang;\n *\n * // Subscribe to language changes\n * this.langService.currentLang$.subscribe(lang => console.log('Language changed:', lang));\n *\n * // Get static text\n * const text = this.langService.getText('ComponentName', 'textKey');\n *\n * // Get reactive text\n * const text$ = this.langService.getContent('ComponentName', 'textKey');\n * ```\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class LangService {\n  private content: Provider;\n  private defaultLang: LangOption;\n  private availableLanguages: LangOption[] = [];\n  private selectedLang: BehaviorSubject<LangOption>;\n  private config: ValtechConfig;\n  private warnedMissingLanguages = new Set<string>();\n\n  constructor(@Inject(ValtechConfigService) config: ValtechConfig) {\n    console.log('LangService: Injected config:', config);\n    this.content = config.content;\n    this.config = config;\n\n    // Detect available languages from content\n    this.detectAvailableLanguages();\n\n    // Set default language (prefer Spanish, then English, then first available)\n    this.defaultLang = this.determineDefaultLanguage();\n\n    // Initialize with stored language or default\n    const current = LocalStorageService.get<LangOption>(LANG);\n    const initialLang = this.validateLanguage(current) || this.defaultLang;\n\n    this.selectedLang = new BehaviorSubject<LangOption>(initialLang);\n\n    console.log('LangService: Initialized with languages:', {\n      available: this.availableLanguages,\n      default: this.defaultLang,\n      current: initialLang,\n    });\n  }\n\n  /**\n   * Detect available languages from the content configuration.\n   * Scans all component content to find which languages are actually configured.\n   */\n  private detectAvailableLanguages(): void {\n    const languageSet = new Set<LangOption>();\n\n    Object.values(this.content).forEach(componentContent => {\n      if (componentContent?.Content) {\n        Object.keys(componentContent.Content).forEach(lang => {\n          languageSet.add(lang);\n        });\n      }\n    });\n\n    this.availableLanguages = Array.from(languageSet).sort();\n\n    if (this.availableLanguages.length === 0) {\n      console.warn('LangService: No languages detected in content configuration!');\n      this.availableLanguages = [LANGUAGES.ES]; // Fallback\n    }\n  }\n\n  /**\n   * Determine the best default language based on available content.\n   */\n  private determineDefaultLanguage(): LangOption {\n    // Preference order: Spanish, English, then first available\n    const preferredOrder = [LANGUAGES.ES, LANGUAGES.EN];\n\n    for (const preferred of preferredOrder) {\n      if (this.availableLanguages.includes(preferred)) {\n        return preferred;\n      }\n    }\n\n    return this.availableLanguages[0];\n  }\n\n  /**\n   * Validate if a language is available in the content.\n   */\n  private validateLanguage(lang: LangOption | null): LangOption | null {\n    if (!lang) return null;\n    return this.availableLanguages.includes(lang) ? lang : null;\n  }\n\n  /**\n   * Get the best available language for a component and key.\n   * Provides intelligent fallback with warnings.\n   */\n  private getBestAvailableContent(\n    className: string,\n    key: string,\n    requestedLang: LangOption\n  ): {\n    content: string | undefined;\n    actualLang: LangOption;\n    shouldWarn: boolean;\n  } {\n    const componentContent = this.content[className];\n\n    if (!componentContent) {\n      return {\n        content: undefined,\n        actualLang: requestedLang,\n        shouldWarn: false,\n      };\n    }\n\n    // Try requested language first\n    const requestedContent = componentContent.Content[requestedLang];\n    if (requestedContent?.[key]) {\n      return {\n        content: requestedContent[key],\n        actualLang: requestedLang,\n        shouldWarn: false,\n      };\n    }\n\n    // Language not available, try fallbacks\n    const warningKey = `${className}.${key}.${requestedLang}`;\n    const shouldWarn = !this.warnedMissingLanguages.has(warningKey);\n\n    if (shouldWarn) {\n      this.warnedMissingLanguages.add(warningKey);\n    }\n\n    // Try default language\n    if (requestedLang !== this.defaultLang) {\n      const defaultContent = componentContent.Content[this.defaultLang];\n      if (defaultContent?.[key]) {\n        return {\n          content: defaultContent[key],\n          actualLang: this.defaultLang,\n          shouldWarn,\n        };\n      }\n    }\n\n    // Try first available language\n    for (const availableLang of this.availableLanguages) {\n      const availableContent = componentContent.Content[availableLang];\n      if (availableContent?.[key]) {\n        return {\n          content: availableContent[key],\n          actualLang: availableLang,\n          shouldWarn,\n        };\n      }\n    }\n\n    return {\n      content: undefined,\n      actualLang: requestedLang,\n      shouldWarn,\n    };\n  }\n\n  /**\n   * Observable that emits the current language whenever it changes.\n   * Use this to subscribe to language changes in components.\n   */\n  get currentLang$(): Observable<LangOption> {\n    return this.selectedLang.asObservable().pipe(distinctUntilChanged());\n  }\n\n  /**\n   * Get the current language synchronously.\n   */\n  get currentLang(): LangOption {\n    return this.selectedLang.value;\n  }\n\n  /**\n   * Get array of available languages detected from content.\n   */\n  get availableLangs(): LangOption[] {\n    return [...this.availableLanguages];\n  }\n\n  /**\n   * Get the default language.\n   */\n  get defaultLanguage(): LangOption {\n    return this.defaultLang;\n  }\n\n  /**\n   * Set the current language and persist it to localStorage.\n   * This will trigger updates in all reactive content subscriptions.\n   *\n   * Validates that the language is available and warns if not.\n   *\n   * @param lang - The language to set\n   */\n  setLang(lang: LangOption): void {\n    if (!this.availableLanguages.includes(lang)) {\n      console.warn(\n        `LangService: Language \"${lang}\" is not available. Available languages:`,\n        this.availableLanguages\n      );\n      console.warn(`LangService: Falling back to default language \"${this.defaultLang}\"`);\n      lang = this.defaultLang;\n    }\n\n    this.selectedLang.next(lang);\n    LocalStorageService.set<LangOption>(LANG, lang);\n  }\n\n  /**\n   * Get content for a component class and key (legacy method).\n   *\n   * @deprecated Use getText() or getContent() for better type safety\n   */\n  Text(className: string): LanguageText {\n    const componentContent = this.content[className];\n    return componentContent?.Content[this.selectedLang.value] || {};\n  }\n\n  /**\n   * Get a single content string synchronously for the current language.\n   * Provides intelligent fallback with warnings for missing translations.\n   *\n   * @param className - The component class name\n   * @param key - The text key\n   * @param fallback - Optional fallback text if key is not found\n   * @returns The text string or fallback\n   */\n  getText(className: string, key: string, fallback?: string): string {\n    const result = this.getBestAvailableContent(className, key, this.selectedLang.value);\n\n    if (result.shouldWarn && result.actualLang !== this.selectedLang.value) {\n      console.warn(\n        `LangService: Content \"${className}.${key}\" not available in \"${this.selectedLang.value}\".`,\n        `Using \"${result.actualLang}\" instead. Available languages:`,\n        this.availableLanguages\n      );\n    }\n\n    return result.content || fallback || `[${className}.${key}]`;\n  }\n\n  /**\n   * Get a reactive Observable for a specific text key that updates when language changes.\n   * This is the recommended method for components that need reactive content.\n   * Provides intelligent fallback with warnings for missing translations.\n   *\n   * @param className - The component class name\n   * @param key - The text key\n   * @param fallback - Optional fallback text if key is not found\n   * @returns Observable that emits the text string whenever language changes\n   */\n  getContent(className: string, key: string, fallback?: string): Observable<string> {\n    return this.currentLang$.pipe(\n      map(lang => {\n        const result = this.getBestAvailableContent(className, key, lang);\n\n        if (result.shouldWarn && result.actualLang !== lang) {\n          console.warn(\n            `LangService: Content \"${className}.${key}\" not available in \"${lang}\".`,\n            `Using \"${result.actualLang}\" instead. Available languages:`,\n            this.availableLanguages\n          );\n        }\n\n        return result.content || fallback || `[${className}.${key}]`;\n      }),\n      distinctUntilChanged()\n    );\n  }\n\n  /**\n   * Get reactive content for multiple keys at once.\n   * Provides intelligent fallback with warnings for missing translations.\n   *\n   * @param className - The component class name\n   * @param keys - Array of text keys to retrieve\n   * @returns Observable that emits an object with all requested keys\n   */\n  getMultipleContent(className: string, keys: string[]): Observable<Record<string, string>> {\n    return this.currentLang$.pipe(\n      map(lang => {\n        const result: Record<string, string> = {};\n\n        keys.forEach(key => {\n          const contentResult = this.getBestAvailableContent(className, key, lang);\n\n          if (contentResult.shouldWarn && contentResult.actualLang !== lang) {\n            console.warn(\n              `LangService: Content \"${className}.${key}\" not available in \"${lang}\".`,\n              `Using \"${contentResult.actualLang}\" instead.`\n            );\n          }\n\n          result[key] = contentResult.content || `[${className}.${key}]`;\n        });\n\n        return result;\n      }),\n      distinctUntilChanged((prev, curr) => JSON.stringify(prev) === JSON.stringify(curr))\n    );\n  }\n\n  /**\n   * Check if a content key exists for a component in any available language.\n   *\n   * @param className - The component class name\n   * @param key - The text key\n   * @returns True if the key exists in any language\n   */\n  hasContent(className: string, key: string): boolean {\n    const classContent = this.content[className];\n    if (!classContent) return false;\n\n    return Object.values(classContent.Content).some(\n      langContent => langContent && typeof langContent[key] === 'string'\n    );\n  }\n\n  /**\n   * Check if a content key exists for a component in a specific language.\n   *\n   * @param className - The component class name\n   * @param key - The text key\n   * @param lang - The language to check (defaults to current language)\n   * @returns True if the key exists in the specified language\n   */\n  hasContentInLanguage(className: string, key: string, lang?: LangOption): boolean {\n    const targetLang = lang || this.currentLang;\n    const classContent = this.content[className]?.Content[targetLang];\n    return classContent && typeof classContent[key] === 'string';\n  }\n\n  /**\n   * Get available languages for a specific component.\n   *\n   * @param className - The component class name\n   * @returns Array of language codes available for the component\n   */\n  getAvailableLanguagesForComponent(className: string): LangOption[] {\n    const classContent = this.content[className];\n    if (!classContent) return [];\n\n    return Object.keys(classContent.Content).filter(\n      lang => classContent.Content[lang] && Object.keys(classContent.Content[lang]).length > 0\n    );\n  }\n\n  /**\n   * Get missing content keys for a component in a specific language.\n   * Useful for identifying incomplete translations.\n   *\n   * @param className - The component class name\n   * @param lang - The language to check\n   * @param referenceLang - The reference language to compare against (defaults to default language)\n   * @returns Array of missing keys\n   */\n  getMissingContentKeys(className: string, lang: LangOption, referenceLang?: LangOption): string[] {\n    const refLang = referenceLang || this.defaultLang;\n    const classContent = this.content[className];\n\n    if (!classContent) return [];\n\n    const referenceContent = classContent.Content[refLang] || {};\n    const targetContent = classContent.Content[lang] || {};\n\n    return Object.keys(referenceContent).filter(\n      key => !targetContent[key] || typeof targetContent[key] !== 'string'\n    );\n  }\n\n  // Legacy getters/setters for backward compatibility\n  get Lang(): LangOption {\n    return this.currentLang;\n  }\n\n  set Lang(lang: LangOption) {\n    this.setLang(lang);\n  }\n}\n"]}
|
|
@@ -6,9 +6,18 @@ export class TextContent {
|
|
|
6
6
|
return this.text;
|
|
7
7
|
}
|
|
8
8
|
}
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
9
|
+
/**
|
|
10
|
+
* Common language constants for convenience.
|
|
11
|
+
* Users can still use any language code string directly.
|
|
12
|
+
*/
|
|
13
|
+
export const LANGUAGES = {
|
|
14
|
+
ES: 'es',
|
|
15
|
+
EN: 'en',
|
|
16
|
+
FR: 'fr',
|
|
17
|
+
DE: 'de',
|
|
18
|
+
PT: 'pt',
|
|
19
|
+
IT: 'it',
|
|
20
|
+
ZH: 'zh',
|
|
21
|
+
JA: 'ja',
|
|
22
|
+
};
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy92YWx0ZWNoLWNvbXBvbmVudHMvc3JjL2xpYi9zZXJ2aWNlcy9sYW5nLXByb3ZpZGVyL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQVFBLE1BQU0sT0FBTyxXQUFXO0lBRXRCLFlBQVksSUFBc0I7UUFDaEMsSUFBSSxDQUFDLElBQUksR0FBRyxJQUFJLENBQUM7SUFDbkIsQ0FBQztJQUVELElBQUksT0FBTztRQUNULE9BQU8sSUFBSSxDQUFDLElBQUksQ0FBQztJQUNuQixDQUFDO0NBQ0Y7QUFRRDs7O0dBR0c7QUFDSCxNQUFNLENBQUMsTUFBTSxTQUFTLEdBQUc7SUFDdkIsRUFBRSxFQUFFLElBQWE7SUFDakIsRUFBRSxFQUFFLElBQWE7SUFDakIsRUFBRSxFQUFFLElBQWE7SUFDakIsRUFBRSxFQUFFLElBQWE7SUFDakIsRUFBRSxFQUFFLElBQWE7SUFDakIsRUFBRSxFQUFFLElBQWE7SUFDakIsRUFBRSxFQUFFLElBQWE7SUFDakIsRUFBRSxFQUFFLElBQWE7Q0FDVCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHR5cGUgTGFuZ3VhZ2VUZXh0ID0ge1xuICBba2V5OiBzdHJpbmddOiBzdHJpbmc7XG59O1xuXG5leHBvcnQgdHlwZSBMYW5ndWFnZXNDb250ZW50ID0ge1xuICBba2V5OiBzdHJpbmddOiBMYW5ndWFnZVRleHQ7XG59O1xuXG5leHBvcnQgY2xhc3MgVGV4dENvbnRlbnQge1xuICB0ZXh0OiBMYW5ndWFnZXNDb250ZW50O1xuICBjb25zdHJ1Y3Rvcih0ZXh0OiBMYW5ndWFnZXNDb250ZW50KSB7XG4gICAgdGhpcy50ZXh0ID0gdGV4dDtcbiAgfVxuXG4gIGdldCBDb250ZW50KCk6IExhbmd1YWdlc0NvbnRlbnQge1xuICAgIHJldHVybiB0aGlzLnRleHQ7XG4gIH1cbn1cblxuLyoqXG4gKiBMYW5ndWFnZSBjb2RlIHR5cGUgLSBzdXBwb3J0cyBhbnkgdmFsaWQgbGFuZ3VhZ2UgY29kZSBzdHJpbmcuXG4gKiBDb21tb24gZXhhbXBsZXM6ICdlcycsICdlbicsICdmcicsICdkZScsICdwdCcsICdpdCcsICd6aCcsICdqYScsIGV0Yy5cbiAqL1xuZXhwb3J0IHR5cGUgTGFuZ09wdGlvbiA9IHN0cmluZztcblxuLyoqXG4gKiBDb21tb24gbGFuZ3VhZ2UgY29uc3RhbnRzIGZvciBjb252ZW5pZW5jZS5cbiAqIFVzZXJzIGNhbiBzdGlsbCB1c2UgYW55IGxhbmd1YWdlIGNvZGUgc3RyaW5nIGRpcmVjdGx5LlxuICovXG5leHBvcnQgY29uc3QgTEFOR1VBR0VTID0ge1xuICBFUzogJ2VzJyBhcyBjb25zdCxcbiAgRU46ICdlbicgYXMgY29uc3QsXG4gIEZSOiAnZnInIGFzIGNvbnN0LFxuICBERTogJ2RlJyBhcyBjb25zdCxcbiAgUFQ6ICdwdCcgYXMgY29uc3QsXG4gIElUOiAnaXQnIGFzIGNvbnN0LFxuICBaSDogJ3poJyBhcyBjb25zdCxcbiAgSkE6ICdqYScgYXMgY29uc3QsXG59IGFzIGNvbnN0O1xuIl19
|