valtech-components 2.0.495 → 2.0.497

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.
@@ -0,0 +1,50 @@
1
+ import { makeEnvironmentProviders, APP_INITIALIZER, } from '@angular/core';
2
+ import { DEFAULT_I18N_CONFIG } from './types';
3
+ import { I18nService } from './i18n.service';
4
+ /**
5
+ * Configura el sistema de internacionalización de Valtech Components.
6
+ *
7
+ * @param config Configuración de i18n
8
+ * @returns Providers para agregar en app.config.ts
9
+ *
10
+ * @example
11
+ * // app.config.ts
12
+ * import { provideValtechI18n } from 'valtech-components';
13
+ * import { GLOBAL_CONTENT } from './i18n/_global';
14
+ * import { LOGIN_CONTENT } from './i18n/login.i18n';
15
+ *
16
+ * export const appConfig: ApplicationConfig = {
17
+ * providers: [
18
+ * provideValtechI18n({
19
+ * defaultLanguage: 'es',
20
+ * supportedLanguages: ['es', 'en'],
21
+ * detectBrowserLanguage: true,
22
+ * content: {
23
+ * '_global': GLOBAL_CONTENT,
24
+ * 'Login': LOGIN_CONTENT,
25
+ * }
26
+ * }),
27
+ * ]
28
+ * };
29
+ */
30
+ export function provideValtechI18n(config = {}) {
31
+ const mergedConfig = { ...DEFAULT_I18N_CONFIG, ...config };
32
+ return makeEnvironmentProviders([
33
+ {
34
+ provide: APP_INITIALIZER,
35
+ useFactory: (i18n) => {
36
+ return () => {
37
+ // Configurar idiomas soportados
38
+ i18n.setI18nLanguages(mergedConfig.supportedLanguages);
39
+ // Registrar contenido inicial
40
+ if (mergedConfig.content && Object.keys(mergedConfig.content).length > 0) {
41
+ i18n.registerContentBulk(mergedConfig.content);
42
+ }
43
+ };
44
+ },
45
+ deps: [I18nService],
46
+ multi: true,
47
+ },
48
+ ]);
49
+ }
50
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9zZXJ2aWNlcy9pMThuL2NvbmZpZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBRUwsd0JBQXdCLEVBQ3hCLGVBQWUsR0FDaEIsTUFBTSxlQUFlLENBQUM7QUFDdkIsT0FBTyxFQUFjLG1CQUFtQixFQUFFLE1BQU0sU0FBUyxDQUFDO0FBQzFELE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUU3Qzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXlCRztBQUNILE1BQU0sVUFBVSxrQkFBa0IsQ0FBQyxTQUFxQixFQUFFO0lBQ3hELE1BQU0sWUFBWSxHQUFHLEVBQUUsR0FBRyxtQkFBbUIsRUFBRSxHQUFHLE1BQU0sRUFBRSxDQUFDO0lBRTNELE9BQU8sd0JBQXdCLENBQUM7UUFDOUI7WUFDRSxPQUFPLEVBQUUsZUFBZTtZQUN4QixVQUFVLEVBQUUsQ0FBQyxJQUFpQixFQUFFLEVBQUU7Z0JBQ2hDLE9BQU8sR0FBRyxFQUFFO29CQUNWLGdDQUFnQztvQkFDaEMsSUFBSSxDQUFDLGdCQUFnQixDQUFDLFlBQVksQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDO29CQUV2RCw4QkFBOEI7b0JBQzlCLElBQUksWUFBWSxDQUFDLE9BQU8sSUFBSSxNQUFNLENBQUMsSUFBSSxDQUFDLFlBQVksQ0FBQyxPQUFPLENBQUMsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxFQUFFLENBQUM7d0JBQ3pFLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxZQUFZLENBQUMsT0FBTyxDQUFDLENBQUM7b0JBQ2pELENBQUM7Z0JBQ0gsQ0FBQyxDQUFDO1lBQ0osQ0FBQztZQUNELElBQUksRUFBRSxDQUFDLFdBQVcsQ0FBQztZQUNuQixLQUFLLEVBQUUsSUFBSTtTQUNaO0tBQ0YsQ0FBQyxDQUFDO0FBQ0wsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIEVudmlyb25tZW50UHJvdmlkZXJzLFxuICBtYWtlRW52aXJvbm1lbnRQcm92aWRlcnMsXG4gIEFQUF9JTklUSUFMSVpFUixcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBJMThuQ29uZmlnLCBERUZBVUxUX0kxOE5fQ09ORklHIH0gZnJvbSAnLi90eXBlcyc7XG5pbXBvcnQgeyBJMThuU2VydmljZSB9IGZyb20gJy4vaTE4bi5zZXJ2aWNlJztcblxuLyoqXG4gKiBDb25maWd1cmEgZWwgc2lzdGVtYSBkZSBpbnRlcm5hY2lvbmFsaXphY2nDs24gZGUgVmFsdGVjaCBDb21wb25lbnRzLlxuICpcbiAqIEBwYXJhbSBjb25maWcgQ29uZmlndXJhY2nDs24gZGUgaTE4blxuICogQHJldHVybnMgUHJvdmlkZXJzIHBhcmEgYWdyZWdhciBlbiBhcHAuY29uZmlnLnRzXG4gKlxuICogQGV4YW1wbGVcbiAqIC8vIGFwcC5jb25maWcudHNcbiAqIGltcG9ydCB7IHByb3ZpZGVWYWx0ZWNoSTE4biB9IGZyb20gJ3ZhbHRlY2gtY29tcG9uZW50cyc7XG4gKiBpbXBvcnQgeyBHTE9CQUxfQ09OVEVOVCB9IGZyb20gJy4vaTE4bi9fZ2xvYmFsJztcbiAqIGltcG9ydCB7IExPR0lOX0NPTlRFTlQgfSBmcm9tICcuL2kxOG4vbG9naW4uaTE4bic7XG4gKlxuICogZXhwb3J0IGNvbnN0IGFwcENvbmZpZzogQXBwbGljYXRpb25Db25maWcgPSB7XG4gKiAgIHByb3ZpZGVyczogW1xuICogICAgIHByb3ZpZGVWYWx0ZWNoSTE4bih7XG4gKiAgICAgICBkZWZhdWx0TGFuZ3VhZ2U6ICdlcycsXG4gKiAgICAgICBzdXBwb3J0ZWRMYW5ndWFnZXM6IFsnZXMnLCAnZW4nXSxcbiAqICAgICAgIGRldGVjdEJyb3dzZXJMYW5ndWFnZTogdHJ1ZSxcbiAqICAgICAgIGNvbnRlbnQ6IHtcbiAqICAgICAgICAgJ19nbG9iYWwnOiBHTE9CQUxfQ09OVEVOVCxcbiAqICAgICAgICAgJ0xvZ2luJzogTE9HSU5fQ09OVEVOVCxcbiAqICAgICAgIH1cbiAqICAgICB9KSxcbiAqICAgXVxuICogfTtcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHByb3ZpZGVWYWx0ZWNoSTE4bihjb25maWc6IEkxOG5Db25maWcgPSB7fSk6IEVudmlyb25tZW50UHJvdmlkZXJzIHtcbiAgY29uc3QgbWVyZ2VkQ29uZmlnID0geyAuLi5ERUZBVUxUX0kxOE5fQ09ORklHLCAuLi5jb25maWcgfTtcblxuICByZXR1cm4gbWFrZUVudmlyb25tZW50UHJvdmlkZXJzKFtcbiAgICB7XG4gICAgICBwcm92aWRlOiBBUFBfSU5JVElBTElaRVIsXG4gICAgICB1c2VGYWN0b3J5OiAoaTE4bjogSTE4blNlcnZpY2UpID0+IHtcbiAgICAgICAgcmV0dXJuICgpID0+IHtcbiAgICAgICAgICAvLyBDb25maWd1cmFyIGlkaW9tYXMgc29wb3J0YWRvc1xuICAgICAgICAgIGkxOG4uc2V0STE4bkxhbmd1YWdlcyhtZXJnZWRDb25maWcuc3VwcG9ydGVkTGFuZ3VhZ2VzKTtcblxuICAgICAgICAgIC8vIFJlZ2lzdHJhciBjb250ZW5pZG8gaW5pY2lhbFxuICAgICAgICAgIGlmIChtZXJnZWRDb25maWcuY29udGVudCAmJiBPYmplY3Qua2V5cyhtZXJnZWRDb25maWcuY29udGVudCkubGVuZ3RoID4gMCkge1xuICAgICAgICAgICAgaTE4bi5yZWdpc3RlckNvbnRlbnRCdWxrKG1lcmdlZENvbmZpZy5jb250ZW50KTtcbiAgICAgICAgICB9XG4gICAgICAgIH07XG4gICAgICB9LFxuICAgICAgZGVwczogW0kxOG5TZXJ2aWNlXSxcbiAgICAgIG11bHRpOiB0cnVlLFxuICAgIH0sXG4gIF0pO1xufVxuIl19
@@ -0,0 +1,196 @@
1
+ import { Injectable, signal, computed } from '@angular/core';
2
+ import { LANG_STORAGE_KEY, DEFAULT_I18N_CONFIG, } from './types';
3
+ import * as i0 from "@angular/core";
4
+ /**
5
+ * Servicio de internacionalización basado en Angular Signals.
6
+ *
7
+ * Características:
8
+ * - Sin RxJS: usa Signals para evitar memory leaks y congelamiento
9
+ * - Namespace-based: organiza traducciones por contexto
10
+ * - Fallback multi-nivel: namespace → _global → placeholder
11
+ * - Interpolación: soporta {variable} en textos
12
+ *
13
+ * @example
14
+ * // En un componente
15
+ * i18n = inject(I18nService);
16
+ *
17
+ * // Obtener texto
18
+ * const title = this.i18n.t('title', 'Login');
19
+ *
20
+ * // Con interpolación
21
+ * const welcome = this.i18n.t('welcome', 'Login', { name: 'Juan' });
22
+ *
23
+ * // Cambiar idioma
24
+ * this.i18n.setLanguage('en');
25
+ */
26
+ export class I18nService {
27
+ constructor() {
28
+ // Estado interno con Signals
29
+ this._lang = signal(DEFAULT_I18N_CONFIG.defaultLanguage);
30
+ this._content = signal({});
31
+ this._supportedLanguages = signal(DEFAULT_I18N_CONFIG.supportedLanguages);
32
+ // Públicos readonly
33
+ this.lang = this._lang.asReadonly();
34
+ this.supportedLanguages = this._supportedLanguages.asReadonly();
35
+ // Computed para verificaciones rápidas
36
+ this.isSpanish = computed(() => this._lang() === 'es');
37
+ this.isEnglish = computed(() => this._lang() === 'en');
38
+ this.loadStoredLanguage();
39
+ }
40
+ /**
41
+ * Obtiene texto traducido (alias corto de getText)
42
+ *
43
+ * @param key Clave del texto
44
+ * @param namespace Namespace (default: '_global')
45
+ * @param data Variables para interpolación
46
+ * @returns Texto traducido o placeholder [namespace.key]
47
+ *
48
+ * @example
49
+ * i18n.t('submit'); // busca en _global
50
+ * i18n.t('title', 'Login'); // busca en Login
51
+ * i18n.t('welcome', 'Login', {name}); // con interpolación
52
+ */
53
+ t(key, namespace, data) {
54
+ return this.getText(key, namespace, data);
55
+ }
56
+ /**
57
+ * Obtiene texto traducido
58
+ *
59
+ * Fallback order:
60
+ * 1. content[namespace][lang][key]
61
+ * 2. content['_global'][lang][key]
62
+ * 3. "[namespace.key]" (placeholder)
63
+ */
64
+ getText(key, namespace, data) {
65
+ const content = this._content();
66
+ const lang = this._lang();
67
+ const ns = namespace || '_global';
68
+ // Buscar en namespace específico
69
+ let text = content[ns]?.[lang]?.[key];
70
+ // Fallback a _global
71
+ if (!text && ns !== '_global') {
72
+ text = content['_global']?.[lang]?.[key];
73
+ }
74
+ // Fallback a placeholder
75
+ if (!text) {
76
+ console.warn(`[i18n] Missing translation: ${ns}.${key} (${lang})`);
77
+ return `[${ns}.${key}]`;
78
+ }
79
+ // Aplicar interpolación si hay data
80
+ if (data) {
81
+ return this.interpolate(text, data);
82
+ }
83
+ return text;
84
+ }
85
+ /**
86
+ * Cambia el idioma de la aplicación
87
+ *
88
+ * @param lang Nuevo idioma
89
+ * @param forceReload Si true, recarga la página (fallback si reactividad falla)
90
+ */
91
+ setLanguage(lang, forceReload = false) {
92
+ if (!this._supportedLanguages().includes(lang)) {
93
+ console.warn(`[i18n] Language '${lang}' not in supported languages`);
94
+ return;
95
+ }
96
+ if (lang === this._lang()) {
97
+ return;
98
+ }
99
+ // Persistir en localStorage
100
+ localStorage.setItem(LANG_STORAGE_KEY, lang);
101
+ // Actualizar signal
102
+ this._lang.set(lang);
103
+ // Fallback: recargar si se solicita
104
+ if (forceReload) {
105
+ window.location.reload();
106
+ }
107
+ }
108
+ /**
109
+ * Registra contenido de traducciones para un namespace
110
+ *
111
+ * @param namespace Nombre del namespace
112
+ * @param content Contenido de traducciones
113
+ *
114
+ * @example
115
+ * i18n.registerContent('Login', {
116
+ * es: { title: 'Iniciar sesión' },
117
+ * en: { title: 'Sign in' }
118
+ * });
119
+ */
120
+ registerContent(namespace, content) {
121
+ this._content.update((store) => ({
122
+ ...store,
123
+ [namespace]: content,
124
+ }));
125
+ }
126
+ /**
127
+ * Registra múltiples namespaces de una vez
128
+ *
129
+ * @param contentStore Objeto con namespaces como keys
130
+ */
131
+ registerContentBulk(contentStore) {
132
+ this._content.update((store) => ({
133
+ ...store,
134
+ ...contentStore,
135
+ }));
136
+ }
137
+ /**
138
+ * Configura los idiomas soportados
139
+ */
140
+ setI18nLanguages(languages) {
141
+ this._supportedLanguages.set(languages);
142
+ }
143
+ /**
144
+ * Obtiene todos los namespaces registrados
145
+ */
146
+ getNamespaces() {
147
+ return Object.keys(this._content());
148
+ }
149
+ /**
150
+ * Verifica si un namespace tiene traducciones
151
+ */
152
+ hasNamespace(namespace) {
153
+ return namespace in this._content();
154
+ }
155
+ /**
156
+ * Carga idioma guardado en localStorage o detecta del navegador
157
+ */
158
+ loadStoredLanguage() {
159
+ const stored = localStorage.getItem(LANG_STORAGE_KEY);
160
+ if (stored && this.isValidLanguage(stored)) {
161
+ this._lang.set(stored);
162
+ return;
163
+ }
164
+ // Detectar idioma del navegador
165
+ const browserLang = navigator.language.split('-')[0];
166
+ if (this.isValidLanguage(browserLang)) {
167
+ this._lang.set(browserLang);
168
+ localStorage.setItem(LANG_STORAGE_KEY, browserLang);
169
+ }
170
+ }
171
+ /**
172
+ * Valida si un idioma está soportado
173
+ */
174
+ isValidLanguage(lang) {
175
+ return this._supportedLanguages().includes(lang);
176
+ }
177
+ /**
178
+ * Reemplaza {variable} en texto con valores de data
179
+ *
180
+ * @example
181
+ * interpolate('Hola {name}', { name: 'Juan' }) // 'Hola Juan'
182
+ */
183
+ interpolate(text, data) {
184
+ return Object.entries(data).reduce((result, [key, value]) => {
185
+ const regex = new RegExp(`\\{${key}\\}`, 'g');
186
+ return result.replace(regex, value);
187
+ }, text);
188
+ }
189
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: I18nService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
190
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: I18nService, providedIn: 'root' }); }
191
+ }
192
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: I18nService, decorators: [{
193
+ type: Injectable,
194
+ args: [{ providedIn: 'root' }]
195
+ }], ctorParameters: () => [] });
196
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"i18n.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/i18n/i18n.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, signal, computed } from '@angular/core';\nimport {\n  I18nLang,\n  LanguagesContent,\n  ContentStore,\n  LANG_STORAGE_KEY,\n  DEFAULT_I18N_CONFIG,\n} from './types';\n\n/**\n * Servicio de internacionalización basado en Angular Signals.\n *\n * Características:\n * - Sin RxJS: usa Signals para evitar memory leaks y congelamiento\n * - Namespace-based: organiza traducciones por contexto\n * - Fallback multi-nivel: namespace → _global → placeholder\n * - Interpolación: soporta {variable} en textos\n *\n * @example\n * // En un componente\n * i18n = inject(I18nService);\n *\n * // Obtener texto\n * const title = this.i18n.t('title', 'Login');\n *\n * // Con interpolación\n * const welcome = this.i18n.t('welcome', 'Login', { name: 'Juan' });\n *\n * // Cambiar idioma\n * this.i18n.setLanguage('en');\n */\n@Injectable({ providedIn: 'root' })\nexport class I18nService {\n  // Estado interno con Signals\n  private readonly _lang = signal<I18nLang>(DEFAULT_I18N_CONFIG.defaultLanguage);\n  private readonly _content = signal<ContentStore>({});\n  private readonly _supportedLanguages = signal<I18nLang[]>(\n    DEFAULT_I18N_CONFIG.supportedLanguages\n  );\n\n  // Públicos readonly\n  readonly lang = this._lang.asReadonly();\n  readonly supportedLanguages = this._supportedLanguages.asReadonly();\n\n  // Computed para verificaciones rápidas\n  readonly isSpanish = computed(() => this._lang() === 'es');\n  readonly isEnglish = computed(() => this._lang() === 'en');\n\n  constructor() {\n    this.loadStoredLanguage();\n  }\n\n  /**\n   * Obtiene texto traducido (alias corto de getText)\n   *\n   * @param key Clave del texto\n   * @param namespace Namespace (default: '_global')\n   * @param data Variables para interpolación\n   * @returns Texto traducido o placeholder [namespace.key]\n   *\n   * @example\n   * i18n.t('submit');                    // busca en _global\n   * i18n.t('title', 'Login');            // busca en Login\n   * i18n.t('welcome', 'Login', {name});  // con interpolación\n   */\n  t(key: string, namespace?: string, data?: Record<string, string>): string {\n    return this.getText(key, namespace, data);\n  }\n\n  /**\n   * Obtiene texto traducido\n   *\n   * Fallback order:\n   * 1. content[namespace][lang][key]\n   * 2. content['_global'][lang][key]\n   * 3. \"[namespace.key]\" (placeholder)\n   */\n  getText(key: string, namespace?: string, data?: Record<string, string>): string {\n    const content = this._content();\n    const lang = this._lang();\n    const ns = namespace || '_global';\n\n    // Buscar en namespace específico\n    let text = content[ns]?.[lang]?.[key];\n\n    // Fallback a _global\n    if (!text && ns !== '_global') {\n      text = content['_global']?.[lang]?.[key];\n    }\n\n    // Fallback a placeholder\n    if (!text) {\n      console.warn(`[i18n] Missing translation: ${ns}.${key} (${lang})`);\n      return `[${ns}.${key}]`;\n    }\n\n    // Aplicar interpolación si hay data\n    if (data) {\n      return this.interpolate(text, data);\n    }\n\n    return text;\n  }\n\n  /**\n   * Cambia el idioma de la aplicación\n   *\n   * @param lang Nuevo idioma\n   * @param forceReload Si true, recarga la página (fallback si reactividad falla)\n   */\n  setLanguage(lang: I18nLang, forceReload = false): void {\n    if (!this._supportedLanguages().includes(lang)) {\n      console.warn(`[i18n] Language '${lang}' not in supported languages`);\n      return;\n    }\n\n    if (lang === this._lang()) {\n      return;\n    }\n\n    // Persistir en localStorage\n    localStorage.setItem(LANG_STORAGE_KEY, lang);\n\n    // Actualizar signal\n    this._lang.set(lang);\n\n    // Fallback: recargar si se solicita\n    if (forceReload) {\n      window.location.reload();\n    }\n  }\n\n  /**\n   * Registra contenido de traducciones para un namespace\n   *\n   * @param namespace Nombre del namespace\n   * @param content Contenido de traducciones\n   *\n   * @example\n   * i18n.registerContent('Login', {\n   *   es: { title: 'Iniciar sesión' },\n   *   en: { title: 'Sign in' }\n   * });\n   */\n  registerContent(namespace: string, content: LanguagesContent): void {\n    this._content.update((store) => ({\n      ...store,\n      [namespace]: content,\n    }));\n  }\n\n  /**\n   * Registra múltiples namespaces de una vez\n   *\n   * @param contentStore Objeto con namespaces como keys\n   */\n  registerContentBulk(contentStore: ContentStore): void {\n    this._content.update((store) => ({\n      ...store,\n      ...contentStore,\n    }));\n  }\n\n  /**\n   * Configura los idiomas soportados\n   */\n  setI18nLanguages(languages: I18nLang[]): void {\n    this._supportedLanguages.set(languages);\n  }\n\n  /**\n   * Obtiene todos los namespaces registrados\n   */\n  getNamespaces(): string[] {\n    return Object.keys(this._content());\n  }\n\n  /**\n   * Verifica si un namespace tiene traducciones\n   */\n  hasNamespace(namespace: string): boolean {\n    return namespace in this._content();\n  }\n\n  /**\n   * Carga idioma guardado en localStorage o detecta del navegador\n   */\n  private loadStoredLanguage(): void {\n    const stored = localStorage.getItem(LANG_STORAGE_KEY);\n\n    if (stored && this.isValidLanguage(stored)) {\n      this._lang.set(stored as I18nLang);\n      return;\n    }\n\n    // Detectar idioma del navegador\n    const browserLang = navigator.language.split('-')[0] as I18nLang;\n    if (this.isValidLanguage(browserLang)) {\n      this._lang.set(browserLang);\n      localStorage.setItem(LANG_STORAGE_KEY, browserLang);\n    }\n  }\n\n  /**\n   * Valida si un idioma está soportado\n   */\n  private isValidLanguage(lang: string): boolean {\n    return this._supportedLanguages().includes(lang as I18nLang);\n  }\n\n  /**\n   * Reemplaza {variable} en texto con valores de data\n   *\n   * @example\n   * interpolate('Hola {name}', { name: 'Juan' }) // 'Hola Juan'\n   */\n  private interpolate(text: string, data: Record<string, string>): string {\n    return Object.entries(data).reduce((result, [key, value]) => {\n      const regex = new RegExp(`\\\\{${key}\\\\}`, 'g');\n      return result.replace(regex, value);\n    }, text);\n  }\n}\n"]}
@@ -0,0 +1,9 @@
1
+ // Types
2
+ export { LANG_STORAGE_KEY, } from './types';
3
+ // Service
4
+ export { I18nService } from './i18n.service';
5
+ // Pipe
6
+ export { TranslatePipe } from './translate.pipe';
7
+ // Config
8
+ export { provideValtechI18n } from './config';
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL2kxOG4vaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsUUFBUTtBQUNSLE9BQU8sRUFLTCxnQkFBZ0IsR0FDakIsTUFBTSxTQUFTLENBQUM7QUFFakIsVUFBVTtBQUNWLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUU3QyxPQUFPO0FBQ1AsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLGtCQUFrQixDQUFDO0FBRWpELFNBQVM7QUFDVCxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxVQUFVLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBUeXBlc1xuZXhwb3J0IHtcbiAgSTE4bkxhbmcsXG4gIExhbmd1YWdlc0NvbnRlbnQsXG4gIENvbnRlbnRTdG9yZSxcbiAgSTE4bkNvbmZpZyxcbiAgTEFOR19TVE9SQUdFX0tFWSxcbn0gZnJvbSAnLi90eXBlcyc7XG5cbi8vIFNlcnZpY2VcbmV4cG9ydCB7IEkxOG5TZXJ2aWNlIH0gZnJvbSAnLi9pMThuLnNlcnZpY2UnO1xuXG4vLyBQaXBlXG5leHBvcnQgeyBUcmFuc2xhdGVQaXBlIH0gZnJvbSAnLi90cmFuc2xhdGUucGlwZSc7XG5cbi8vIENvbmZpZ1xuZXhwb3J0IHsgcHJvdmlkZVZhbHRlY2hJMThuIH0gZnJvbSAnLi9jb25maWcnO1xuIl19
@@ -0,0 +1,50 @@
1
+ import { Pipe, inject } from '@angular/core';
2
+ import { I18nService } from './i18n.service';
3
+ import * as i0 from "@angular/core";
4
+ /**
5
+ * Pipe de traducción para templates.
6
+ *
7
+ * NOTA: Es impure para detectar cambios de idioma.
8
+ * El costo es mínimo porque I18nService usa Signals internamente.
9
+ *
10
+ * @example
11
+ * <!-- Busca en _global -->
12
+ * {{ 'submit' | t }}
13
+ *
14
+ * <!-- Busca en namespace específico -->
15
+ * {{ 'title' | t:'Login' }}
16
+ *
17
+ * <!-- Con interpolación -->
18
+ * {{ 'welcome' | t:'Login':{ name: userName } }}
19
+ *
20
+ * <!-- En atributos -->
21
+ * <val-button [label]="'submit' | t"></val-button>
22
+ * <val-input [label]="'email' | t:'Login'"></val-input>
23
+ */
24
+ export class TranslatePipe {
25
+ constructor() {
26
+ this.i18n = inject(I18nService);
27
+ }
28
+ /**
29
+ * Transforma una key de traducción a su valor
30
+ *
31
+ * @param key Clave del texto
32
+ * @param namespace Namespace opcional (default: '_global')
33
+ * @param data Variables para interpolación opcional
34
+ * @returns Texto traducido
35
+ */
36
+ transform(key, namespace, data) {
37
+ return this.i18n.t(key, namespace, data);
38
+ }
39
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TranslatePipe, deps: [], target: i0.ɵɵFactoryTarget.Pipe }); }
40
+ static { this.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "18.2.14", ngImport: i0, type: TranslatePipe, isStandalone: true, name: "t", pure: false }); }
41
+ }
42
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: TranslatePipe, decorators: [{
43
+ type: Pipe,
44
+ args: [{
45
+ name: 't',
46
+ standalone: true,
47
+ pure: false, // Impure para detectar cambios de idioma
48
+ }]
49
+ }] });
50
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJhbnNsYXRlLnBpcGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL2kxOG4vdHJhbnNsYXRlLnBpcGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLElBQUksRUFBaUIsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzVELE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQzs7QUFFN0M7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FtQkc7QUFNSCxNQUFNLE9BQU8sYUFBYTtJQUwxQjtRQU1tQixTQUFJLEdBQUcsTUFBTSxDQUFDLFdBQVcsQ0FBQyxDQUFDO0tBYTdDO0lBWEM7Ozs7Ozs7T0FPRztJQUNILFNBQVMsQ0FBQyxHQUFXLEVBQUUsU0FBa0IsRUFBRSxJQUE2QjtRQUN0RSxPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEdBQUcsRUFBRSxTQUFTLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDM0MsQ0FBQzsrR0FiVSxhQUFhOzZHQUFiLGFBQWE7OzRGQUFiLGFBQWE7a0JBTHpCLElBQUk7bUJBQUM7b0JBQ0osSUFBSSxFQUFFLEdBQUc7b0JBQ1QsVUFBVSxFQUFFLElBQUk7b0JBQ2hCLElBQUksRUFBRSxLQUFLLEVBQUUseUNBQXlDO2lCQUN2RCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFBpcGUsIFBpcGVUcmFuc2Zvcm0sIGluamVjdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgSTE4blNlcnZpY2UgfSBmcm9tICcuL2kxOG4uc2VydmljZSc7XG5cbi8qKlxuICogUGlwZSBkZSB0cmFkdWNjacOzbiBwYXJhIHRlbXBsYXRlcy5cbiAqXG4gKiBOT1RBOiBFcyBpbXB1cmUgcGFyYSBkZXRlY3RhciBjYW1iaW9zIGRlIGlkaW9tYS5cbiAqIEVsIGNvc3RvIGVzIG3DrW5pbW8gcG9ycXVlIEkxOG5TZXJ2aWNlIHVzYSBTaWduYWxzIGludGVybmFtZW50ZS5cbiAqXG4gKiBAZXhhbXBsZVxuICogPCEtLSBCdXNjYSBlbiBfZ2xvYmFsIC0tPlxuICoge3sgJ3N1Ym1pdCcgfCB0IH19XG4gKlxuICogPCEtLSBCdXNjYSBlbiBuYW1lc3BhY2UgZXNwZWPDrWZpY28gLS0+XG4gKiB7eyAndGl0bGUnIHwgdDonTG9naW4nIH19XG4gKlxuICogPCEtLSBDb24gaW50ZXJwb2xhY2nDs24gLS0+XG4gKiB7eyAnd2VsY29tZScgfCB0OidMb2dpbic6eyBuYW1lOiB1c2VyTmFtZSB9IH19XG4gKlxuICogPCEtLSBFbiBhdHJpYnV0b3MgLS0+XG4gKiA8dmFsLWJ1dHRvbiBbbGFiZWxdPVwiJ3N1Ym1pdCcgfCB0XCI+PC92YWwtYnV0dG9uPlxuICogPHZhbC1pbnB1dCBbbGFiZWxdPVwiJ2VtYWlsJyB8IHQ6J0xvZ2luJ1wiPjwvdmFsLWlucHV0PlxuICovXG5AUGlwZSh7XG4gIG5hbWU6ICd0JyxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgcHVyZTogZmFsc2UsIC8vIEltcHVyZSBwYXJhIGRldGVjdGFyIGNhbWJpb3MgZGUgaWRpb21hXG59KVxuZXhwb3J0IGNsYXNzIFRyYW5zbGF0ZVBpcGUgaW1wbGVtZW50cyBQaXBlVHJhbnNmb3JtIHtcbiAgcHJpdmF0ZSByZWFkb25seSBpMThuID0gaW5qZWN0KEkxOG5TZXJ2aWNlKTtcblxuICAvKipcbiAgICogVHJhbnNmb3JtYSB1bmEga2V5IGRlIHRyYWR1Y2Npw7NuIGEgc3UgdmFsb3JcbiAgICpcbiAgICogQHBhcmFtIGtleSBDbGF2ZSBkZWwgdGV4dG9cbiAgICogQHBhcmFtIG5hbWVzcGFjZSBOYW1lc3BhY2Ugb3BjaW9uYWwgKGRlZmF1bHQ6ICdfZ2xvYmFsJylcbiAgICogQHBhcmFtIGRhdGEgVmFyaWFibGVzIHBhcmEgaW50ZXJwb2xhY2nDs24gb3BjaW9uYWxcbiAgICogQHJldHVybnMgVGV4dG8gdHJhZHVjaWRvXG4gICAqL1xuICB0cmFuc2Zvcm0oa2V5OiBzdHJpbmcsIG5hbWVzcGFjZT86IHN0cmluZywgZGF0YT86IFJlY29yZDxzdHJpbmcsIHN0cmluZz4pOiBzdHJpbmcge1xuICAgIHJldHVybiB0aGlzLmkxOG4udChrZXksIG5hbWVzcGFjZSwgZGF0YSk7XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,14 @@
1
+ /**
2
+ * Valores por defecto de configuración
3
+ */
4
+ export const DEFAULT_I18N_CONFIG = {
5
+ defaultLanguage: 'es',
6
+ supportedLanguages: ['es', 'en'],
7
+ detectBrowserLanguage: true,
8
+ content: {},
9
+ };
10
+ /**
11
+ * Clave para persistir idioma en localStorage
12
+ */
13
+ export const LANG_STORAGE_KEY = 'app_lang';
14
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL2kxOG4vdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBMERBOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sbUJBQW1CLEdBQXlCO0lBQ3ZELGVBQWUsRUFBRSxJQUFJO0lBQ3JCLGtCQUFrQixFQUFFLENBQUMsSUFBSSxFQUFFLElBQUksQ0FBQztJQUNoQyxxQkFBcUIsRUFBRSxJQUFJO0lBQzNCLE9BQU8sRUFBRSxFQUFFO0NBQ1osQ0FBQztBQUVGOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sZ0JBQWdCLEdBQUcsVUFBVSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBJZGlvbWFzIHNvcG9ydGFkb3MgcG9yIGVsIHNpc3RlbWEgaTE4blxuICovXG5leHBvcnQgdHlwZSBJMThuTGFuZyA9ICdlcycgfCAnZW4nIHwgJ3B0JyB8ICdmcicgfCAnZGUnO1xuXG4vKipcbiAqIENvbnRlbmlkbyBkZSB0cmFkdWNjaW9uZXMgcGFyYSB1biBuYW1lc3BhY2VcbiAqXG4gKiBAZXhhbXBsZVxuICogY29uc3QgbG9naW5Db250ZW50OiBMYW5ndWFnZXNDb250ZW50ID0ge1xuICogICBlczogeyB0aXRsZTogJ0luaWNpYXIgc2VzacOzbicsIHN1Ym1pdDogJ0VudHJhcicgfSxcbiAqICAgZW46IHsgdGl0bGU6ICdTaWduIGluJywgc3VibWl0OiAnU2lnbiBpbicgfVxuICogfTtcbiAqL1xuZXhwb3J0IHR5cGUgTGFuZ3VhZ2VzQ29udGVudCA9IHtcbiAgW2xhbmcgaW4gSTE4bkxhbmddPzogUmVjb3JkPHN0cmluZywgc3RyaW5nPjtcbn07XG5cbi8qKlxuICogQWxtYWPDqW4gZGUgY29udGVuaWRvIHBvciBuYW1lc3BhY2VcbiAqXG4gKiBAZXhhbXBsZVxuICoge1xuICogICAnX2dsb2JhbCc6IHsgZXM6IHsuLi59LCBlbjogey4uLn0gfSxcbiAqICAgJ0xvZ2luJzogeyBlczogey4uLn0sIGVuOiB7Li4ufSB9LFxuICogICAnUHJvZmlsZSc6IHsgZXM6IHsuLi59LCBlbjogey4uLn0gfVxuICogfVxuICovXG5leHBvcnQgdHlwZSBDb250ZW50U3RvcmUgPSBSZWNvcmQ8c3RyaW5nLCBMYW5ndWFnZXNDb250ZW50PjtcblxuLyoqXG4gKiBDb25maWd1cmFjacOzbiBwYXJhIHByb3ZpZGVWYWx0ZWNoSTE4bigpXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgSTE4bkNvbmZpZyB7XG4gIC8qKlxuICAgKiBJZGlvbWEgcG9yIGRlZmVjdG8gc2kgbm8gaGF5IHByZWZlcmVuY2lhIGd1YXJkYWRhXG4gICAqIEBkZWZhdWx0ICdlcydcbiAgICovXG4gIGRlZmF1bHRMYW5ndWFnZT86IEkxOG5MYW5nO1xuXG4gIC8qKlxuICAgKiBMaXN0YSBkZSBpZGlvbWFzIGhhYmlsaXRhZG9zIGVuIGxhIGFwcFxuICAgKiBAZGVmYXVsdCBbJ2VzJywgJ2VuJ11cbiAgICovXG4gIHN1cHBvcnRlZExhbmd1YWdlcz86IEkxOG5MYW5nW107XG5cbiAgLyoqXG4gICAqIERldGVjdGFyIGlkaW9tYSBkZWwgbmF2ZWdhZG9yIGFsIGluaWNpYXJcbiAgICogQGRlZmF1bHQgdHJ1ZVxuICAgKi9cbiAgZGV0ZWN0QnJvd3Nlckxhbmd1YWdlPzogYm9vbGVhbjtcblxuICAvKipcbiAgICogQ29udGVuaWRvIGluaWNpYWwgZGUgdHJhZHVjY2lvbmVzIHBvciBuYW1lc3BhY2VcbiAgICovXG4gIGNvbnRlbnQ/OiBDb250ZW50U3RvcmU7XG59XG5cbi8qKlxuICogVmFsb3JlcyBwb3IgZGVmZWN0byBkZSBjb25maWd1cmFjacOzblxuICovXG5leHBvcnQgY29uc3QgREVGQVVMVF9JMThOX0NPTkZJRzogUmVxdWlyZWQ8STE4bkNvbmZpZz4gPSB7XG4gIGRlZmF1bHRMYW5ndWFnZTogJ2VzJyxcbiAgc3VwcG9ydGVkTGFuZ3VhZ2VzOiBbJ2VzJywgJ2VuJ10sXG4gIGRldGVjdEJyb3dzZXJMYW5ndWFnZTogdHJ1ZSxcbiAgY29udGVudDoge30sXG59O1xuXG4vKipcbiAqIENsYXZlIHBhcmEgcGVyc2lzdGlyIGlkaW9tYSBlbiBsb2NhbFN0b3JhZ2VcbiAqL1xuZXhwb3J0IGNvbnN0IExBTkdfU1RPUkFHRV9LRVkgPSAnYXBwX2xhbmcnO1xuIl19
@@ -0,0 +1,46 @@
1
+ import { makeEnvironmentProviders, APP_INITIALIZER, } from '@angular/core';
2
+ import { PresetService } from './preset.service';
3
+ /**
4
+ * Configura el sistema de presets de Valtech Components.
5
+ *
6
+ * @param presets Configuración de presets por componente
7
+ * @returns Providers para agregar en app.config.ts
8
+ *
9
+ * @example
10
+ * // app.config.ts
11
+ * import { provideValtechPresets } from 'valtech-components';
12
+ *
13
+ * export const appConfig: ApplicationConfig = {
14
+ * providers: [
15
+ * provideValtechPresets({
16
+ * button: {
17
+ * 'primary-action': { size: 'large', color: 'primary', fill: 'solid' },
18
+ * 'secondary': { size: 'medium', color: 'secondary', fill: 'outline' },
19
+ * 'danger': { size: 'medium', color: 'danger', fill: 'solid' },
20
+ * },
21
+ * card: {
22
+ * 'feature': { variant: 'elevated', padding: 'large' },
23
+ * 'compact': { variant: 'flat', padding: 'small' },
24
+ * },
25
+ * input: {
26
+ * 'form-field': { size: 'medium', fill: 'outline', labelPosition: 'floating' },
27
+ * }
28
+ * }),
29
+ * ]
30
+ * };
31
+ */
32
+ export function provideValtechPresets(presets) {
33
+ return makeEnvironmentProviders([
34
+ {
35
+ provide: APP_INITIALIZER,
36
+ useFactory: (presetService) => {
37
+ return () => {
38
+ presetService.registerPresets(presets);
39
+ };
40
+ },
41
+ deps: [PresetService],
42
+ multi: true,
43
+ },
44
+ ]);
45
+ }
46
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9zZXJ2aWNlcy9wcmVzZXRzL2NvbmZpZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBRUwsd0JBQXdCLEVBQ3hCLGVBQWUsR0FDaEIsTUFBTSxlQUFlLENBQUM7QUFFdkIsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLGtCQUFrQixDQUFDO0FBRWpEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBNEJHO0FBQ0gsTUFBTSxVQUFVLHFCQUFxQixDQUFDLE9BQXFCO0lBQ3pELE9BQU8sd0JBQXdCLENBQUM7UUFDOUI7WUFDRSxPQUFPLEVBQUUsZUFBZTtZQUN4QixVQUFVLEVBQUUsQ0FBQyxhQUE0QixFQUFFLEVBQUU7Z0JBQzNDLE9BQU8sR0FBRyxFQUFFO29CQUNWLGFBQWEsQ0FBQyxlQUFlLENBQUMsT0FBTyxDQUFDLENBQUM7Z0JBQ3pDLENBQUMsQ0FBQztZQUNKLENBQUM7WUFDRCxJQUFJLEVBQUUsQ0FBQyxhQUFhLENBQUM7WUFDckIsS0FBSyxFQUFFLElBQUk7U0FDWjtLQUNGLENBQUMsQ0FBQztBQUNMLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBFbnZpcm9ubWVudFByb3ZpZGVycyxcbiAgbWFrZUVudmlyb25tZW50UHJvdmlkZXJzLFxuICBBUFBfSU5JVElBTElaRVIsXG59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgUHJlc2V0Q29uZmlnIH0gZnJvbSAnLi90eXBlcyc7XG5pbXBvcnQgeyBQcmVzZXRTZXJ2aWNlIH0gZnJvbSAnLi9wcmVzZXQuc2VydmljZSc7XG5cbi8qKlxuICogQ29uZmlndXJhIGVsIHNpc3RlbWEgZGUgcHJlc2V0cyBkZSBWYWx0ZWNoIENvbXBvbmVudHMuXG4gKlxuICogQHBhcmFtIHByZXNldHMgQ29uZmlndXJhY2nDs24gZGUgcHJlc2V0cyBwb3IgY29tcG9uZW50ZVxuICogQHJldHVybnMgUHJvdmlkZXJzIHBhcmEgYWdyZWdhciBlbiBhcHAuY29uZmlnLnRzXG4gKlxuICogQGV4YW1wbGVcbiAqIC8vIGFwcC5jb25maWcudHNcbiAqIGltcG9ydCB7IHByb3ZpZGVWYWx0ZWNoUHJlc2V0cyB9IGZyb20gJ3ZhbHRlY2gtY29tcG9uZW50cyc7XG4gKlxuICogZXhwb3J0IGNvbnN0IGFwcENvbmZpZzogQXBwbGljYXRpb25Db25maWcgPSB7XG4gKiAgIHByb3ZpZGVyczogW1xuICogICAgIHByb3ZpZGVWYWx0ZWNoUHJlc2V0cyh7XG4gKiAgICAgICBidXR0b246IHtcbiAqICAgICAgICAgJ3ByaW1hcnktYWN0aW9uJzogeyBzaXplOiAnbGFyZ2UnLCBjb2xvcjogJ3ByaW1hcnknLCBmaWxsOiAnc29saWQnIH0sXG4gKiAgICAgICAgICdzZWNvbmRhcnknOiB7IHNpemU6ICdtZWRpdW0nLCBjb2xvcjogJ3NlY29uZGFyeScsIGZpbGw6ICdvdXRsaW5lJyB9LFxuICogICAgICAgICAnZGFuZ2VyJzogeyBzaXplOiAnbWVkaXVtJywgY29sb3I6ICdkYW5nZXInLCBmaWxsOiAnc29saWQnIH0sXG4gKiAgICAgICB9LFxuICogICAgICAgY2FyZDoge1xuICogICAgICAgICAnZmVhdHVyZSc6IHsgdmFyaWFudDogJ2VsZXZhdGVkJywgcGFkZGluZzogJ2xhcmdlJyB9LFxuICogICAgICAgICAnY29tcGFjdCc6IHsgdmFyaWFudDogJ2ZsYXQnLCBwYWRkaW5nOiAnc21hbGwnIH0sXG4gKiAgICAgICB9LFxuICogICAgICAgaW5wdXQ6IHtcbiAqICAgICAgICAgJ2Zvcm0tZmllbGQnOiB7IHNpemU6ICdtZWRpdW0nLCBmaWxsOiAnb3V0bGluZScsIGxhYmVsUG9zaXRpb246ICdmbG9hdGluZycgfSxcbiAqICAgICAgIH1cbiAqICAgICB9KSxcbiAqICAgXVxuICogfTtcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHByb3ZpZGVWYWx0ZWNoUHJlc2V0cyhwcmVzZXRzOiBQcmVzZXRDb25maWcpOiBFbnZpcm9ubWVudFByb3ZpZGVycyB7XG4gIHJldHVybiBtYWtlRW52aXJvbm1lbnRQcm92aWRlcnMoW1xuICAgIHtcbiAgICAgIHByb3ZpZGU6IEFQUF9JTklUSUFMSVpFUixcbiAgICAgIHVzZUZhY3Rvcnk6IChwcmVzZXRTZXJ2aWNlOiBQcmVzZXRTZXJ2aWNlKSA9PiB7XG4gICAgICAgIHJldHVybiAoKSA9PiB7XG4gICAgICAgICAgcHJlc2V0U2VydmljZS5yZWdpc3RlclByZXNldHMocHJlc2V0cyk7XG4gICAgICAgIH07XG4gICAgICB9LFxuICAgICAgZGVwczogW1ByZXNldFNlcnZpY2VdLFxuICAgICAgbXVsdGk6IHRydWUsXG4gICAgfSxcbiAgXSk7XG59XG4iXX0=
@@ -0,0 +1,5 @@
1
+ // Service
2
+ export { PresetService } from './preset.service';
3
+ // Config
4
+ export { provideValtechPresets } from './config';
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL3ByZXNldHMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBR0EsVUFBVTtBQUNWLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxrQkFBa0IsQ0FBQztBQUVqRCxTQUFTO0FBQ1QsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0sVUFBVSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gVHlwZXNcbmV4cG9ydCB7IFByZXNldENvbmZpZywgQ29tcG9uZW50UHJlc2V0cywgUHJlc2V0c09wdGlvbnMgfSBmcm9tICcuL3R5cGVzJztcblxuLy8gU2VydmljZVxuZXhwb3J0IHsgUHJlc2V0U2VydmljZSB9IGZyb20gJy4vcHJlc2V0LnNlcnZpY2UnO1xuXG4vLyBDb25maWdcbmV4cG9ydCB7IHByb3ZpZGVWYWx0ZWNoUHJlc2V0cyB9IGZyb20gJy4vY29uZmlnJztcbiJdfQ==
@@ -0,0 +1,104 @@
1
+ import { Injectable, signal } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ /**
4
+ * Servicio para gestionar presets de componentes.
5
+ *
6
+ * Los presets permiten definir configuraciones reutilizables
7
+ * de componentes (tamaño, color, variante, etc.) que se pueden
8
+ * aplicar con un nombre semántico.
9
+ *
10
+ * @example
11
+ * // En un componente
12
+ * presets = inject(PresetService);
13
+ *
14
+ * // Obtener preset
15
+ * const buttonProps = this.presets.get<ButtonMetadata>('button', 'primary-action');
16
+ * // { size: 'large', color: 'primary', fill: 'solid' }
17
+ */
18
+ export class PresetService {
19
+ constructor() {
20
+ this._presets = signal({});
21
+ }
22
+ /**
23
+ * Obtiene un preset específico para un componente
24
+ *
25
+ * @param component Tipo de componente (ej: 'button', 'card', 'input')
26
+ * @param presetName Nombre del preset (ej: 'primary-action', 'compact')
27
+ * @returns Propiedades del preset o objeto vacío si no existe
28
+ *
29
+ * @example
30
+ * // Obtener preset de botón
31
+ * const props = presets.get<ButtonMetadata>('button', 'primary-action');
32
+ *
33
+ * // Usar en componente
34
+ * <val-button [props]="props"></val-button>
35
+ */
36
+ get(component, presetName) {
37
+ const componentPresets = this._presets()[component];
38
+ if (!componentPresets) {
39
+ console.warn(`[presets] No presets registered for component: ${component}`);
40
+ return {};
41
+ }
42
+ const preset = componentPresets[presetName];
43
+ if (!preset) {
44
+ console.warn(`[presets] Preset '${presetName}' not found for component: ${component}`);
45
+ return {};
46
+ }
47
+ return preset;
48
+ }
49
+ /**
50
+ * Verifica si existe un preset
51
+ */
52
+ has(component, presetName) {
53
+ return !!this._presets()[component]?.[presetName];
54
+ }
55
+ /**
56
+ * Registra presets de la aplicación
57
+ *
58
+ * @param presets Configuración de presets
59
+ *
60
+ * @example
61
+ * presets.registerPresets({
62
+ * button: {
63
+ * 'primary-action': { size: 'large', color: 'primary' },
64
+ * },
65
+ * card: {
66
+ * 'feature': { variant: 'elevated' },
67
+ * }
68
+ * });
69
+ */
70
+ registerPresets(presets) {
71
+ this._presets.set(presets);
72
+ }
73
+ /**
74
+ * Agrega presets para un componente específico (merge con existentes)
75
+ */
76
+ registerComponentPresets(component, presets) {
77
+ this._presets.update((current) => ({
78
+ ...current,
79
+ [component]: {
80
+ ...current[component],
81
+ ...presets,
82
+ },
83
+ }));
84
+ }
85
+ /**
86
+ * Obtiene todos los nombres de presets para un componente
87
+ */
88
+ getPresetNames(component) {
89
+ return Object.keys(this._presets()[component] || {});
90
+ }
91
+ /**
92
+ * Obtiene todos los componentes con presets registrados
93
+ */
94
+ getRegisteredComponents() {
95
+ return Object.keys(this._presets());
96
+ }
97
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PresetService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
98
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PresetService, providedIn: 'root' }); }
99
+ }
100
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: PresetService, decorators: [{
101
+ type: Injectable,
102
+ args: [{ providedIn: 'root' }]
103
+ }] });
104
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJlc2V0LnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL3ByZXNldHMvcHJlc2V0LnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBR25EOzs7Ozs7Ozs7Ozs7OztHQWNHO0FBRUgsTUFBTSxPQUFPLGFBQWE7SUFEMUI7UUFFbUIsYUFBUSxHQUFHLE1BQU0sQ0FBZSxFQUFFLENBQUMsQ0FBQztLQW9GdEQ7SUFsRkM7Ozs7Ozs7Ozs7Ozs7T0FhRztJQUNILEdBQUcsQ0FBb0MsU0FBaUIsRUFBRSxVQUFrQjtRQUMxRSxNQUFNLGdCQUFnQixHQUFHLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUNwRCxJQUFJLENBQUMsZ0JBQWdCLEVBQUUsQ0FBQztZQUN0QixPQUFPLENBQUMsSUFBSSxDQUFDLGtEQUFrRCxTQUFTLEVBQUUsQ0FBQyxDQUFDO1lBQzVFLE9BQU8sRUFBZ0IsQ0FBQztRQUMxQixDQUFDO1FBRUQsTUFBTSxNQUFNLEdBQUcsZ0JBQWdCLENBQUMsVUFBVSxDQUFDLENBQUM7UUFDNUMsSUFBSSxDQUFDLE1BQU0sRUFBRSxDQUFDO1lBQ1osT0FBTyxDQUFDLElBQUksQ0FBQyxxQkFBcUIsVUFBVSw4QkFBOEIsU0FBUyxFQUFFLENBQUMsQ0FBQztZQUN2RixPQUFPLEVBQWdCLENBQUM7UUFDMUIsQ0FBQztRQUVELE9BQU8sTUFBb0IsQ0FBQztJQUM5QixDQUFDO0lBRUQ7O09BRUc7SUFDSCxHQUFHLENBQUMsU0FBaUIsRUFBRSxVQUFrQjtRQUN2QyxPQUFPLENBQUMsQ0FBQyxJQUFJLENBQUMsUUFBUSxFQUFFLENBQUMsU0FBUyxDQUFDLEVBQUUsQ0FBQyxVQUFVLENBQUMsQ0FBQztJQUNwRCxDQUFDO0lBRUQ7Ozs7Ozs7Ozs7Ozs7O09BY0c7SUFDSCxlQUFlLENBQUMsT0FBcUI7UUFDbkMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxHQUFHLENBQUMsT0FBTyxDQUFDLENBQUM7SUFDN0IsQ0FBQztJQUVEOztPQUVHO0lBQ0gsd0JBQXdCLENBQUMsU0FBaUIsRUFBRSxPQUF5QjtRQUNuRSxJQUFJLENBQUMsUUFBUSxDQUFDLE1BQU0sQ0FBQyxDQUFDLE9BQU8sRUFBRSxFQUFFLENBQUMsQ0FBQztZQUNqQyxHQUFHLE9BQU87WUFDVixDQUFDLFNBQVMsQ0FBQyxFQUFFO2dCQUNYLEdBQUcsT0FBTyxDQUFDLFNBQVMsQ0FBQztnQkFDckIsR0FBRyxPQUFPO2FBQ1g7U0FDRixDQUFDLENBQUMsQ0FBQztJQUNOLENBQUM7SUFFRDs7T0FFRztJQUNILGNBQWMsQ0FBQyxTQUFpQjtRQUM5QixPQUFPLE1BQU0sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDLFNBQVMsQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDO0lBQ3ZELENBQUM7SUFFRDs7T0FFRztJQUNILHVCQUF1QjtRQUNyQixPQUFPLE1BQU0sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDLENBQUM7SUFDdEMsQ0FBQzsrR0FwRlUsYUFBYTttSEFBYixhQUFhLGNBREEsTUFBTTs7NEZBQ25CLGFBQWE7a0JBRHpCLFVBQVU7bUJBQUMsRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0YWJsZSwgc2lnbmFsIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBQcmVzZXRDb25maWcsIENvbXBvbmVudFByZXNldHMgfSBmcm9tICcuL3R5cGVzJztcblxuLyoqXG4gKiBTZXJ2aWNpbyBwYXJhIGdlc3Rpb25hciBwcmVzZXRzIGRlIGNvbXBvbmVudGVzLlxuICpcbiAqIExvcyBwcmVzZXRzIHBlcm1pdGVuIGRlZmluaXIgY29uZmlndXJhY2lvbmVzIHJldXRpbGl6YWJsZXNcbiAqIGRlIGNvbXBvbmVudGVzICh0YW1hw7FvLCBjb2xvciwgdmFyaWFudGUsIGV0Yy4pIHF1ZSBzZSBwdWVkZW5cbiAqIGFwbGljYXIgY29uIHVuIG5vbWJyZSBzZW3DoW50aWNvLlxuICpcbiAqIEBleGFtcGxlXG4gKiAvLyBFbiB1biBjb21wb25lbnRlXG4gKiBwcmVzZXRzID0gaW5qZWN0KFByZXNldFNlcnZpY2UpO1xuICpcbiAqIC8vIE9idGVuZXIgcHJlc2V0XG4gKiBjb25zdCBidXR0b25Qcm9wcyA9IHRoaXMucHJlc2V0cy5nZXQ8QnV0dG9uTWV0YWRhdGE+KCdidXR0b24nLCAncHJpbWFyeS1hY3Rpb24nKTtcbiAqIC8vIHsgc2l6ZTogJ2xhcmdlJywgY29sb3I6ICdwcmltYXJ5JywgZmlsbDogJ3NvbGlkJyB9XG4gKi9cbkBJbmplY3RhYmxlKHsgcHJvdmlkZWRJbjogJ3Jvb3QnIH0pXG5leHBvcnQgY2xhc3MgUHJlc2V0U2VydmljZSB7XG4gIHByaXZhdGUgcmVhZG9ubHkgX3ByZXNldHMgPSBzaWduYWw8UHJlc2V0Q29uZmlnPih7fSk7XG5cbiAgLyoqXG4gICAqIE9idGllbmUgdW4gcHJlc2V0IGVzcGVjw61maWNvIHBhcmEgdW4gY29tcG9uZW50ZVxuICAgKlxuICAgKiBAcGFyYW0gY29tcG9uZW50IFRpcG8gZGUgY29tcG9uZW50ZSAoZWo6ICdidXR0b24nLCAnY2FyZCcsICdpbnB1dCcpXG4gICAqIEBwYXJhbSBwcmVzZXROYW1lIE5vbWJyZSBkZWwgcHJlc2V0IChlajogJ3ByaW1hcnktYWN0aW9uJywgJ2NvbXBhY3QnKVxuICAgKiBAcmV0dXJucyBQcm9waWVkYWRlcyBkZWwgcHJlc2V0IG8gb2JqZXRvIHZhY8OtbyBzaSBubyBleGlzdGVcbiAgICpcbiAgICogQGV4YW1wbGVcbiAgICogLy8gT2J0ZW5lciBwcmVzZXQgZGUgYm90w7NuXG4gICAqIGNvbnN0IHByb3BzID0gcHJlc2V0cy5nZXQ8QnV0dG9uTWV0YWRhdGE+KCdidXR0b24nLCAncHJpbWFyeS1hY3Rpb24nKTtcbiAgICpcbiAgICogLy8gVXNhciBlbiBjb21wb25lbnRlXG4gICAqIDx2YWwtYnV0dG9uIFtwcm9wc109XCJwcm9wc1wiPjwvdmFsLWJ1dHRvbj5cbiAgICovXG4gIGdldDxUIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4+KGNvbXBvbmVudDogc3RyaW5nLCBwcmVzZXROYW1lOiBzdHJpbmcpOiBQYXJ0aWFsPFQ+IHtcbiAgICBjb25zdCBjb21wb25lbnRQcmVzZXRzID0gdGhpcy5fcHJlc2V0cygpW2NvbXBvbmVudF07XG4gICAgaWYgKCFjb21wb25lbnRQcmVzZXRzKSB7XG4gICAgICBjb25zb2xlLndhcm4oYFtwcmVzZXRzXSBObyBwcmVzZXRzIHJlZ2lzdGVyZWQgZm9yIGNvbXBvbmVudDogJHtjb21wb25lbnR9YCk7XG4gICAgICByZXR1cm4ge30gYXMgUGFydGlhbDxUPjtcbiAgICB9XG5cbiAgICBjb25zdCBwcmVzZXQgPSBjb21wb25lbnRQcmVzZXRzW3ByZXNldE5hbWVdO1xuICAgIGlmICghcHJlc2V0KSB7XG4gICAgICBjb25zb2xlLndhcm4oYFtwcmVzZXRzXSBQcmVzZXQgJyR7cHJlc2V0TmFtZX0nIG5vdCBmb3VuZCBmb3IgY29tcG9uZW50OiAke2NvbXBvbmVudH1gKTtcbiAgICAgIHJldHVybiB7fSBhcyBQYXJ0aWFsPFQ+O1xuICAgIH1cblxuICAgIHJldHVybiBwcmVzZXQgYXMgUGFydGlhbDxUPjtcbiAgfVxuXG4gIC8qKlxuICAgKiBWZXJpZmljYSBzaSBleGlzdGUgdW4gcHJlc2V0XG4gICAqL1xuICBoYXMoY29tcG9uZW50OiBzdHJpbmcsIHByZXNldE5hbWU6IHN0cmluZyk6IGJvb2xlYW4ge1xuICAgIHJldHVybiAhIXRoaXMuX3ByZXNldHMoKVtjb21wb25lbnRdPy5bcHJlc2V0TmFtZV07XG4gIH1cblxuICAvKipcbiAgICogUmVnaXN0cmEgcHJlc2V0cyBkZSBsYSBhcGxpY2FjacOzblxuICAgKlxuICAgKiBAcGFyYW0gcHJlc2V0cyBDb25maWd1cmFjacOzbiBkZSBwcmVzZXRzXG4gICAqXG4gICAqIEBleGFtcGxlXG4gICAqIHByZXNldHMucmVnaXN0ZXJQcmVzZXRzKHtcbiAgICogICBidXR0b246IHtcbiAgICogICAgICdwcmltYXJ5LWFjdGlvbic6IHsgc2l6ZTogJ2xhcmdlJywgY29sb3I6ICdwcmltYXJ5JyB9LFxuICAgKiAgIH0sXG4gICAqICAgY2FyZDoge1xuICAgKiAgICAgJ2ZlYXR1cmUnOiB7IHZhcmlhbnQ6ICdlbGV2YXRlZCcgfSxcbiAgICogICB9XG4gICAqIH0pO1xuICAgKi9cbiAgcmVnaXN0ZXJQcmVzZXRzKHByZXNldHM6IFByZXNldENvbmZpZyk6IHZvaWQge1xuICAgIHRoaXMuX3ByZXNldHMuc2V0KHByZXNldHMpO1xuICB9XG5cbiAgLyoqXG4gICAqIEFncmVnYSBwcmVzZXRzIHBhcmEgdW4gY29tcG9uZW50ZSBlc3BlY8OtZmljbyAobWVyZ2UgY29uIGV4aXN0ZW50ZXMpXG4gICAqL1xuICByZWdpc3RlckNvbXBvbmVudFByZXNldHMoY29tcG9uZW50OiBzdHJpbmcsIHByZXNldHM6IENvbXBvbmVudFByZXNldHMpOiB2b2lkIHtcbiAgICB0aGlzLl9wcmVzZXRzLnVwZGF0ZSgoY3VycmVudCkgPT4gKHtcbiAgICAgIC4uLmN1cnJlbnQsXG4gICAgICBbY29tcG9uZW50XToge1xuICAgICAgICAuLi5jdXJyZW50W2NvbXBvbmVudF0sXG4gICAgICAgIC4uLnByZXNldHMsXG4gICAgICB9LFxuICAgIH0pKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBPYnRpZW5lIHRvZG9zIGxvcyBub21icmVzIGRlIHByZXNldHMgcGFyYSB1biBjb21wb25lbnRlXG4gICAqL1xuICBnZXRQcmVzZXROYW1lcyhjb21wb25lbnQ6IHN0cmluZyk6IHN0cmluZ1tdIHtcbiAgICByZXR1cm4gT2JqZWN0LmtleXModGhpcy5fcHJlc2V0cygpW2NvbXBvbmVudF0gfHwge30pO1xuICB9XG5cbiAgLyoqXG4gICAqIE9idGllbmUgdG9kb3MgbG9zIGNvbXBvbmVudGVzIGNvbiBwcmVzZXRzIHJlZ2lzdHJhZG9zXG4gICAqL1xuICBnZXRSZWdpc3RlcmVkQ29tcG9uZW50cygpOiBzdHJpbmdbXSB7XG4gICAgcmV0dXJuIE9iamVjdC5rZXlzKHRoaXMuX3ByZXNldHMoKSk7XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL3ByZXNldHMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogUHJlc2V0cyBwYXJhIHVuIHRpcG8gZGUgY29tcG9uZW50ZVxuICpcbiAqIEBleGFtcGxlXG4gKiBjb25zdCBidXR0b25QcmVzZXRzOiBDb21wb25lbnRQcmVzZXRzID0ge1xuICogICAncHJpbWFyeS1hY3Rpb24nOiB7IHNpemU6ICdsYXJnZScsIGNvbG9yOiAncHJpbWFyeScsIGZpbGw6ICdzb2xpZCcgfSxcbiAqICAgJ3NlY29uZGFyeSc6IHsgc2l6ZTogJ21lZGl1bScsIGNvbG9yOiAnc2Vjb25kYXJ5JywgZmlsbDogJ291dGxpbmUnIH0sXG4gKiAgICdkYW5nZXInOiB7IHNpemU6ICdtZWRpdW0nLCBjb2xvcjogJ2RhbmdlcicsIGZpbGw6ICdzb2xpZCcgfSxcbiAqIH07XG4gKi9cbmV4cG9ydCB0eXBlIENvbXBvbmVudFByZXNldHMgPSBSZWNvcmQ8c3RyaW5nLCBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPj47XG5cbi8qKlxuICogQ29uZmlndXJhY2nDs24gY29tcGxldGEgZGUgcHJlc2V0cyBwb3IgY29tcG9uZW50ZVxuICpcbiAqIEBleGFtcGxlXG4gKiBjb25zdCBhcHBQcmVzZXRzOiBQcmVzZXRDb25maWcgPSB7XG4gKiAgIGJ1dHRvbjoge1xuICogICAgICdwcmltYXJ5LWFjdGlvbic6IHsgc2l6ZTogJ2xhcmdlJywgY29sb3I6ICdwcmltYXJ5JywgZmlsbDogJ3NvbGlkJyB9LFxuICogICAgICdzZWNvbmRhcnknOiB7IHNpemU6ICdtZWRpdW0nLCBjb2xvcjogJ3NlY29uZGFyeScsIGZpbGw6ICdvdXRsaW5lJyB9LFxuICogICB9LFxuICogICBjYXJkOiB7XG4gKiAgICAgJ2ZlYXR1cmUnOiB7IHZhcmlhbnQ6ICdlbGV2YXRlZCcsIHBhZGRpbmc6ICdsYXJnZScgfSxcbiAqICAgICAnY29tcGFjdCc6IHsgdmFyaWFudDogJ2ZsYXQnLCBwYWRkaW5nOiAnc21hbGwnIH0sXG4gKiAgIH0sXG4gKiAgIGlucHV0OiB7XG4gKiAgICAgJ2Zvcm0tZmllbGQnOiB7IHNpemU6ICdtZWRpdW0nLCBmaWxsOiAnb3V0bGluZScsIGxhYmVsUG9zaXRpb246ICdmbG9hdGluZycgfSxcbiAqICAgfVxuICogfTtcbiAqL1xuZXhwb3J0IHR5cGUgUHJlc2V0Q29uZmlnID0gUmVjb3JkPHN0cmluZywgQ29tcG9uZW50UHJlc2V0cz47XG5cbi8qKlxuICogT3BjaW9uZXMgcGFyYSBwcm92aWRlVmFsdGVjaFByZXNldHMoKVxuICovXG5leHBvcnQgaW50ZXJmYWNlIFByZXNldHNPcHRpb25zIHtcbiAgLyoqXG4gICAqIFByZXNldHMgaW5pY2lhbGVzIGRlIGxhIGFwbGljYWNpw7NuXG4gICAqL1xuICBwcmVzZXRzOiBQcmVzZXRDb25maWc7XG59XG4iXX0=