valtech-components 2.0.291 → 2.0.293
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/components/atoms/avatar/avatar.component.mjs +3 -3
- package/esm2022/lib/components/atoms/box/box.component.mjs +3 -3
- package/esm2022/lib/components/atoms/button/button.component.mjs +45 -12
- package/esm2022/lib/components/atoms/display/display.component.mjs +47 -11
- package/esm2022/lib/components/atoms/display/types.mjs +1 -1
- package/esm2022/lib/components/atoms/divider/divider.component.mjs +3 -3
- package/esm2022/lib/components/atoms/href/href.component.mjs +3 -3
- package/esm2022/lib/components/atoms/icon/icon.component.mjs +3 -3
- package/esm2022/lib/components/atoms/image/image.component.mjs +3 -3
- package/esm2022/lib/components/atoms/progress-bar/progress-bar.component.mjs +3 -3
- package/esm2022/lib/components/atoms/text/text.component.mjs +3 -3
- package/esm2022/lib/components/atoms/title/title.component.mjs +77 -23
- package/esm2022/lib/components/atoms/title/types.mjs +30 -2
- package/esm2022/lib/components/molecules/alert-box/alert-box.component.mjs +57 -10
- package/esm2022/lib/components/molecules/alert-box/types.mjs +1 -1
- package/esm2022/lib/components/molecules/button-group/button-group.component.mjs +3 -3
- package/esm2022/lib/components/molecules/card/card.component.mjs +3 -3
- package/esm2022/lib/components/molecules/content-loader/content-loader.component.mjs +3 -3
- package/esm2022/lib/components/molecules/date-input/date-input.component.mjs +3 -3
- package/esm2022/lib/components/molecules/file-input/file-input.component.mjs +3 -3
- package/esm2022/lib/components/molecules/hint/hint.component.mjs +3 -3
- package/esm2022/lib/components/molecules/layered-card/layered-card.component.mjs +3 -3
- package/esm2022/lib/components/molecules/link/link.component.mjs +3 -3
- package/esm2022/lib/components/molecules/links-cake/links-cake.component.mjs +3 -3
- package/esm2022/lib/components/molecules/notes-box/notes-box.component.mjs +3 -3
- package/esm2022/lib/components/molecules/password-input/password-input.component.mjs +3 -3
- package/esm2022/lib/components/molecules/pin-input/pin-input.component.mjs +3 -3
- package/esm2022/lib/components/molecules/progress-status/progress-status.component.mjs +3 -3
- package/esm2022/lib/components/molecules/prompter/prompter.component.mjs +3 -3
- package/esm2022/lib/components/molecules/searchbar/searchbar.component.mjs +3 -3
- package/esm2022/lib/components/molecules/text-input/text-input.component.mjs +3 -3
- package/esm2022/lib/components/molecules/title-block/title-block.component.mjs +3 -3
- package/esm2022/lib/components/organisms/banner/banner.component.mjs +3 -3
- package/esm2022/lib/components/organisms/form/form.component.mjs +3 -3
- package/esm2022/lib/components/organisms/header/header.component.mjs +3 -3
- package/esm2022/lib/components/organisms/item-list/item-list.component.mjs +3 -3
- package/esm2022/lib/components/organisms/item-list/types.mjs +1 -1
- package/esm2022/lib/components/organisms/no-content/no-content.component.mjs +3 -3
- package/esm2022/lib/components/organisms/toolbar/toolbar.component.mjs +3 -3
- package/esm2022/lib/components/organisms/wizard/wizard.component.mjs +3 -3
- package/esm2022/lib/components/templates/layout/layout.component.mjs +2 -2
- package/esm2022/lib/components/types.mjs +1 -1
- package/esm2022/lib/examples/custom-content-demo.component.mjs +3 -3
- package/esm2022/lib/examples/display-demo.component.mjs +518 -0
- package/esm2022/lib/examples/display-simple-example.component.mjs +202 -0
- package/esm2022/lib/examples/multi-language-demo.component.mjs +304 -0
- package/esm2022/lib/services/lang-provider/content.mjs +34 -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/shared/utils/reactive-content.mjs +117 -0
- package/esm2022/public-api.mjs +5 -1
- package/fesm2022/valtech-components.mjs +3895 -2358
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/atoms/button/button.component.d.ts +10 -3
- package/lib/components/atoms/display/display.component.d.ts +12 -4
- package/lib/components/atoms/display/types.d.ts +13 -3
- package/lib/components/atoms/title/title.component.d.ts +6 -1
- package/lib/components/atoms/title/types.d.ts +29 -3
- package/lib/components/molecules/alert-box/alert-box.component.d.ts +17 -9
- package/lib/components/molecules/alert-box/types.d.ts +29 -0
- package/lib/components/organisms/item-list/types.d.ts +54 -9
- package/lib/components/types.d.ts +12 -3
- package/lib/examples/display-demo.component.d.ts +59 -0
- package/lib/examples/display-simple-example.component.d.ts +23 -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/shared/utils/reactive-content.d.ts +109 -0
- package/package.json +4 -2
- package/public-api.d.ts +4 -0
- package/src/lib/components/styles/overrides.scss +583 -0
- package/src/lib/components/styles/variables.scss +7 -3
|
@@ -0,0 +1,202 @@
|
|
|
1
|
+
import { Component } from '@angular/core';
|
|
2
|
+
import { DisplayComponent } from '../components/atoms/display/display.component';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
/**
|
|
5
|
+
* Ejemplo rápido de uso del componente val-display con contenido reactivo
|
|
6
|
+
*/
|
|
7
|
+
export class DisplayExampleComponent {
|
|
8
|
+
constructor() {
|
|
9
|
+
// ✅ Ejemplos de contenido estático
|
|
10
|
+
this.staticTitle = {
|
|
11
|
+
content: 'Este es un título estático',
|
|
12
|
+
color: 'primary',
|
|
13
|
+
size: 'xlarge',
|
|
14
|
+
};
|
|
15
|
+
this.staticSubtitle = {
|
|
16
|
+
content: 'Este es un subtítulo que no cambia',
|
|
17
|
+
color: 'medium',
|
|
18
|
+
size: 'large',
|
|
19
|
+
};
|
|
20
|
+
// ✅ Ejemplos de contenido reactivo (requiere configuración de i18n)
|
|
21
|
+
this.reactiveWelcome = {
|
|
22
|
+
contentConfig: {
|
|
23
|
+
className: 'displayExample',
|
|
24
|
+
key: 'welcome.title',
|
|
25
|
+
fallback: 'Bienvenido a la aplicación',
|
|
26
|
+
},
|
|
27
|
+
color: 'primary',
|
|
28
|
+
size: 'xlarge',
|
|
29
|
+
};
|
|
30
|
+
this.reactiveDescription = {
|
|
31
|
+
contentConfig: {
|
|
32
|
+
className: 'displayExample',
|
|
33
|
+
key: 'welcome.description',
|
|
34
|
+
fallback: 'Esta descripción cambia según el idioma',
|
|
35
|
+
},
|
|
36
|
+
color: 'dark',
|
|
37
|
+
size: 'medium',
|
|
38
|
+
};
|
|
39
|
+
// ✅ Ejemplos con interpolación de variables
|
|
40
|
+
this.userGreeting = {
|
|
41
|
+
contentConfig: {
|
|
42
|
+
className: 'displayExample',
|
|
43
|
+
key: 'user.greeting',
|
|
44
|
+
fallback: 'Hola, {{name}}! Tienes {{messages}} mensajes.',
|
|
45
|
+
interpolation: {
|
|
46
|
+
name: 'María',
|
|
47
|
+
messages: 5,
|
|
48
|
+
},
|
|
49
|
+
},
|
|
50
|
+
color: 'tertiary',
|
|
51
|
+
size: 'large',
|
|
52
|
+
};
|
|
53
|
+
this.statusMessage = {
|
|
54
|
+
contentConfig: {
|
|
55
|
+
className: 'displayExample',
|
|
56
|
+
key: 'status.online',
|
|
57
|
+
fallback: 'Estado: {{status}} desde {{time}}',
|
|
58
|
+
interpolation: {
|
|
59
|
+
status: 'En línea',
|
|
60
|
+
time: '10:30 AM',
|
|
61
|
+
},
|
|
62
|
+
},
|
|
63
|
+
color: 'success',
|
|
64
|
+
size: 'small',
|
|
65
|
+
};
|
|
66
|
+
// ✅ Ejemplos de diferentes tamaños
|
|
67
|
+
this.smallText = {
|
|
68
|
+
content: 'Texto pequeño (small)',
|
|
69
|
+
color: 'dark',
|
|
70
|
+
size: 'small',
|
|
71
|
+
};
|
|
72
|
+
this.mediumText = {
|
|
73
|
+
content: 'Texto mediano (medium)',
|
|
74
|
+
color: 'dark',
|
|
75
|
+
size: 'medium',
|
|
76
|
+
};
|
|
77
|
+
this.largeText = {
|
|
78
|
+
content: 'Texto grande (large)',
|
|
79
|
+
color: 'dark',
|
|
80
|
+
size: 'large',
|
|
81
|
+
};
|
|
82
|
+
this.xlargeText = {
|
|
83
|
+
content: 'Texto extra grande (xlarge)',
|
|
84
|
+
color: 'dark',
|
|
85
|
+
size: 'xlarge',
|
|
86
|
+
};
|
|
87
|
+
// ✅ Ejemplos de diferentes colores
|
|
88
|
+
this.primaryText = {
|
|
89
|
+
content: 'Texto color primary',
|
|
90
|
+
color: 'primary',
|
|
91
|
+
size: 'medium',
|
|
92
|
+
};
|
|
93
|
+
this.successText = {
|
|
94
|
+
content: 'Texto color success',
|
|
95
|
+
color: 'success',
|
|
96
|
+
size: 'medium',
|
|
97
|
+
};
|
|
98
|
+
this.warningText = {
|
|
99
|
+
content: 'Texto color warning',
|
|
100
|
+
color: 'warning',
|
|
101
|
+
size: 'medium',
|
|
102
|
+
};
|
|
103
|
+
this.dangerText = {
|
|
104
|
+
content: 'Texto color danger',
|
|
105
|
+
color: 'danger',
|
|
106
|
+
size: 'medium',
|
|
107
|
+
};
|
|
108
|
+
}
|
|
109
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: DisplayExampleComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
110
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: DisplayExampleComponent, isStandalone: true, selector: "app-display-example", ngImport: i0, template: `
|
|
111
|
+
<div class="display-examples">
|
|
112
|
+
<h2>Ejemplos de val-display</h2>
|
|
113
|
+
|
|
114
|
+
<!-- Contenido estático -->
|
|
115
|
+
<div class="section">
|
|
116
|
+
<h3>Contenido Estático</h3>
|
|
117
|
+
<val-display [props]="staticTitle"></val-display>
|
|
118
|
+
<val-display [props]="staticSubtitle"></val-display>
|
|
119
|
+
</div>
|
|
120
|
+
|
|
121
|
+
<!-- Contenido reactivo -->
|
|
122
|
+
<div class="section">
|
|
123
|
+
<h3>Contenido Reactivo</h3>
|
|
124
|
+
<val-display [props]="reactiveWelcome"></val-display>
|
|
125
|
+
<val-display [props]="reactiveDescription"></val-display>
|
|
126
|
+
</div>
|
|
127
|
+
|
|
128
|
+
<!-- Contenido con interpolación -->
|
|
129
|
+
<div class="section">
|
|
130
|
+
<h3>Contenido con Variables</h3>
|
|
131
|
+
<val-display [props]="userGreeting"></val-display>
|
|
132
|
+
<val-display [props]="statusMessage"></val-display>
|
|
133
|
+
</div>
|
|
134
|
+
|
|
135
|
+
<!-- Diferentes tamaños -->
|
|
136
|
+
<div class="section">
|
|
137
|
+
<h3>Diferentes Tamaños</h3>
|
|
138
|
+
<val-display [props]="smallText"></val-display>
|
|
139
|
+
<val-display [props]="mediumText"></val-display>
|
|
140
|
+
<val-display [props]="largeText"></val-display>
|
|
141
|
+
<val-display [props]="xlargeText"></val-display>
|
|
142
|
+
</div>
|
|
143
|
+
|
|
144
|
+
<!-- Diferentes colores -->
|
|
145
|
+
<div class="section">
|
|
146
|
+
<h3>Diferentes Colores</h3>
|
|
147
|
+
<val-display [props]="primaryText"></val-display>
|
|
148
|
+
<val-display [props]="successText"></val-display>
|
|
149
|
+
<val-display [props]="warningText"></val-display>
|
|
150
|
+
<val-display [props]="dangerText"></val-display>
|
|
151
|
+
</div>
|
|
152
|
+
</div>
|
|
153
|
+
`, isInline: true, styles: [".display-examples{padding:20px;max-width:800px;margin:0 auto}.section{margin:30px 0;padding:20px;border:1px solid var(--ion-color-light);border-radius:8px;background:var(--ion-color-step-50)}.section h3{margin:0 0 15px;color:var(--ion-color-primary)}val-display{display:block;margin:10px 0}\n"], dependencies: [{ kind: "component", type: DisplayComponent, selector: "val-display", inputs: ["props"] }] }); }
|
|
154
|
+
}
|
|
155
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: DisplayExampleComponent, decorators: [{
|
|
156
|
+
type: Component,
|
|
157
|
+
args: [{ selector: 'app-display-example', standalone: true, imports: [DisplayComponent], template: `
|
|
158
|
+
<div class="display-examples">
|
|
159
|
+
<h2>Ejemplos de val-display</h2>
|
|
160
|
+
|
|
161
|
+
<!-- Contenido estático -->
|
|
162
|
+
<div class="section">
|
|
163
|
+
<h3>Contenido Estático</h3>
|
|
164
|
+
<val-display [props]="staticTitle"></val-display>
|
|
165
|
+
<val-display [props]="staticSubtitle"></val-display>
|
|
166
|
+
</div>
|
|
167
|
+
|
|
168
|
+
<!-- Contenido reactivo -->
|
|
169
|
+
<div class="section">
|
|
170
|
+
<h3>Contenido Reactivo</h3>
|
|
171
|
+
<val-display [props]="reactiveWelcome"></val-display>
|
|
172
|
+
<val-display [props]="reactiveDescription"></val-display>
|
|
173
|
+
</div>
|
|
174
|
+
|
|
175
|
+
<!-- Contenido con interpolación -->
|
|
176
|
+
<div class="section">
|
|
177
|
+
<h3>Contenido con Variables</h3>
|
|
178
|
+
<val-display [props]="userGreeting"></val-display>
|
|
179
|
+
<val-display [props]="statusMessage"></val-display>
|
|
180
|
+
</div>
|
|
181
|
+
|
|
182
|
+
<!-- Diferentes tamaños -->
|
|
183
|
+
<div class="section">
|
|
184
|
+
<h3>Diferentes Tamaños</h3>
|
|
185
|
+
<val-display [props]="smallText"></val-display>
|
|
186
|
+
<val-display [props]="mediumText"></val-display>
|
|
187
|
+
<val-display [props]="largeText"></val-display>
|
|
188
|
+
<val-display [props]="xlargeText"></val-display>
|
|
189
|
+
</div>
|
|
190
|
+
|
|
191
|
+
<!-- Diferentes colores -->
|
|
192
|
+
<div class="section">
|
|
193
|
+
<h3>Diferentes Colores</h3>
|
|
194
|
+
<val-display [props]="primaryText"></val-display>
|
|
195
|
+
<val-display [props]="successText"></val-display>
|
|
196
|
+
<val-display [props]="warningText"></val-display>
|
|
197
|
+
<val-display [props]="dangerText"></val-display>
|
|
198
|
+
</div>
|
|
199
|
+
</div>
|
|
200
|
+
`, styles: [".display-examples{padding:20px;max-width:800px;margin:0 auto}.section{margin:30px 0;padding:20px;border:1px solid var(--ion-color-light);border-radius:8px;background:var(--ion-color-step-50)}.section h3{margin:0 0 15px;color:var(--ion-color-primary)}val-display{display:block;margin:10px 0}\n"] }]
|
|
201
|
+
}] });
|
|
202
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"display-simple-example.component.js","sourceRoot":"","sources":["../../../../../projects/valtech-components/src/lib/examples/display-simple-example.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component } from '@angular/core';\nimport { DisplayComponent } from '../components/atoms/display/display.component';\nimport { DisplayMetadata } from '../components/atoms/display/types';\n\n/**\n * Ejemplo rápido de uso del componente val-display con contenido reactivo\n */\n@Component({\n  selector: 'app-display-example',\n  standalone: true,\n  imports: [DisplayComponent],\n  template: `\n    <div class=\"display-examples\">\n      <h2>Ejemplos de val-display</h2>\n\n      <!-- Contenido estático -->\n      <div class=\"section\">\n        <h3>Contenido Estático</h3>\n        <val-display [props]=\"staticTitle\"></val-display>\n        <val-display [props]=\"staticSubtitle\"></val-display>\n      </div>\n\n      <!-- Contenido reactivo -->\n      <div class=\"section\">\n        <h3>Contenido Reactivo</h3>\n        <val-display [props]=\"reactiveWelcome\"></val-display>\n        <val-display [props]=\"reactiveDescription\"></val-display>\n      </div>\n\n      <!-- Contenido con interpolación -->\n      <div class=\"section\">\n        <h3>Contenido con Variables</h3>\n        <val-display [props]=\"userGreeting\"></val-display>\n        <val-display [props]=\"statusMessage\"></val-display>\n      </div>\n\n      <!-- Diferentes tamaños -->\n      <div class=\"section\">\n        <h3>Diferentes Tamaños</h3>\n        <val-display [props]=\"smallText\"></val-display>\n        <val-display [props]=\"mediumText\"></val-display>\n        <val-display [props]=\"largeText\"></val-display>\n        <val-display [props]=\"xlargeText\"></val-display>\n      </div>\n\n      <!-- Diferentes colores -->\n      <div class=\"section\">\n        <h3>Diferentes Colores</h3>\n        <val-display [props]=\"primaryText\"></val-display>\n        <val-display [props]=\"successText\"></val-display>\n        <val-display [props]=\"warningText\"></val-display>\n        <val-display [props]=\"dangerText\"></val-display>\n      </div>\n    </div>\n  `,\n  styles: [\n    `\n      .display-examples {\n        padding: 20px;\n        max-width: 800px;\n        margin: 0 auto;\n      }\n\n      .section {\n        margin: 30px 0;\n        padding: 20px;\n        border: 1px solid var(--ion-color-light);\n        border-radius: 8px;\n        background: var(--ion-color-step-50);\n      }\n\n      .section h3 {\n        margin: 0 0 15px 0;\n        color: var(--ion-color-primary);\n      }\n\n      val-display {\n        display: block;\n        margin: 10px 0;\n      }\n    `,\n  ],\n})\nexport class DisplayExampleComponent {\n  // ✅ Ejemplos de contenido estático\n  staticTitle: DisplayMetadata = {\n    content: 'Este es un título estático',\n    color: 'primary',\n    size: 'xlarge',\n  };\n\n  staticSubtitle: DisplayMetadata = {\n    content: 'Este es un subtítulo que no cambia',\n    color: 'medium',\n    size: 'large',\n  };\n\n  // ✅ Ejemplos de contenido reactivo (requiere configuración de i18n)\n  reactiveWelcome: DisplayMetadata = {\n    contentConfig: {\n      className: 'displayExample',\n      key: 'welcome.title',\n      fallback: 'Bienvenido a la aplicación',\n    },\n    color: 'primary',\n    size: 'xlarge',\n  };\n\n  reactiveDescription: DisplayMetadata = {\n    contentConfig: {\n      className: 'displayExample',\n      key: 'welcome.description',\n      fallback: 'Esta descripción cambia según el idioma',\n    },\n    color: 'dark',\n    size: 'medium',\n  };\n\n  // ✅ Ejemplos con interpolación de variables\n  userGreeting: DisplayMetadata = {\n    contentConfig: {\n      className: 'displayExample',\n      key: 'user.greeting',\n      fallback: 'Hola, {{name}}! Tienes {{messages}} mensajes.',\n      interpolation: {\n        name: 'María',\n        messages: 5,\n      },\n    },\n    color: 'tertiary',\n    size: 'large',\n  };\n\n  statusMessage: DisplayMetadata = {\n    contentConfig: {\n      className: 'displayExample',\n      key: 'status.online',\n      fallback: 'Estado: {{status}} desde {{time}}',\n      interpolation: {\n        status: 'En línea',\n        time: '10:30 AM',\n      },\n    },\n    color: 'success',\n    size: 'small',\n  };\n\n  // ✅ Ejemplos de diferentes tamaños\n  smallText: DisplayMetadata = {\n    content: 'Texto pequeño (small)',\n    color: 'dark',\n    size: 'small',\n  };\n\n  mediumText: DisplayMetadata = {\n    content: 'Texto mediano (medium)',\n    color: 'dark',\n    size: 'medium',\n  };\n\n  largeText: DisplayMetadata = {\n    content: 'Texto grande (large)',\n    color: 'dark',\n    size: 'large',\n  };\n\n  xlargeText: DisplayMetadata = {\n    content: 'Texto extra grande (xlarge)',\n    color: 'dark',\n    size: 'xlarge',\n  };\n\n  // ✅ Ejemplos de diferentes colores\n  primaryText: DisplayMetadata = {\n    content: 'Texto color primary',\n    color: 'primary',\n    size: 'medium',\n  };\n\n  successText: DisplayMetadata = {\n    content: 'Texto color success',\n    color: 'success',\n    size: 'medium',\n  };\n\n  warningText: DisplayMetadata = {\n    content: 'Texto color warning',\n    color: 'warning',\n    size: 'medium',\n  };\n\n  dangerText: DisplayMetadata = {\n    content: 'Texto color danger',\n    color: 'danger',\n    size: 'medium',\n  };\n}\n"]}
|
|
@@ -0,0 +1,304 @@
|
|
|
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"]}
|
|
@@ -3,7 +3,10 @@ import { TextContent } from './types';
|
|
|
3
3
|
/**
|
|
4
4
|
* Global content that can be used across all components.
|
|
5
5
|
* These are common texts like buttons, actions, states, etc.
|
|
6
|
-
* Structure: {es: {key1: 'value1', key2: 'value2'}, en: {key1: 'value1', key2: 'value2'}}
|
|
6
|
+
* Structure: {es: {key1: 'value1', key2: 'value2'}, en: {key1: 'value1', key2: 'value2'}, fr: {...}}
|
|
7
|
+
*
|
|
8
|
+
* Note: You can add any language code. The system will automatically detect available languages
|
|
9
|
+
* and provide intelligent fallbacks with warnings for missing translations.
|
|
7
10
|
*/
|
|
8
11
|
const globalContentData = {
|
|
9
12
|
es: {
|
|
@@ -33,6 +36,7 @@ const globalContentData = {
|
|
|
33
36
|
success: 'Éxito',
|
|
34
37
|
warning: 'Advertencia',
|
|
35
38
|
info: 'Información',
|
|
39
|
+
language: 'Idioma',
|
|
36
40
|
// Common confirmations
|
|
37
41
|
areYouSure: '¿Estás seguro?',
|
|
38
42
|
deleteConfirmation: '¿Estás seguro de que deseas eliminar {itemName}?',
|
|
@@ -74,6 +78,34 @@ const globalContentData = {
|
|
|
74
78
|
// Common placeholders
|
|
75
79
|
searchPlaceholder: 'Search...',
|
|
76
80
|
},
|
|
81
|
+
fr: {
|
|
82
|
+
// Common buttons - Example of partial translation (missing some keys intentionally)
|
|
83
|
+
ok: 'OK',
|
|
84
|
+
cancel: 'Annuler',
|
|
85
|
+
save: 'Sauvegarder',
|
|
86
|
+
delete: 'Supprimer',
|
|
87
|
+
edit: 'Modifier',
|
|
88
|
+
close: 'Fermer',
|
|
89
|
+
back: 'Retour',
|
|
90
|
+
// Common states and messages (intentionally incomplete to show fallback behavior)
|
|
91
|
+
loading: 'Chargement...',
|
|
92
|
+
error: 'Erreur',
|
|
93
|
+
success: 'Succès',
|
|
94
|
+
// Common confirmations
|
|
95
|
+
areYouSure: 'Êtes-vous sûr?',
|
|
96
|
+
},
|
|
97
|
+
de: {
|
|
98
|
+
// Common buttons - Another example of partial translation
|
|
99
|
+
ok: 'OK',
|
|
100
|
+
cancel: 'Abbrechen',
|
|
101
|
+
save: 'Speichern',
|
|
102
|
+
delete: 'Löschen',
|
|
103
|
+
// Common states and messages
|
|
104
|
+
loading: 'Laden...',
|
|
105
|
+
error: 'Fehler',
|
|
106
|
+
// Common confirmations
|
|
107
|
+
areYouSure: 'Sind Sie sicher?',
|
|
108
|
+
},
|
|
77
109
|
};
|
|
78
110
|
const GlobalContent = new TextContent(globalContentData);
|
|
79
111
|
const content = {
|
|
@@ -83,4 +115,4 @@ const content = {
|
|
|
83
115
|
export default content;
|
|
84
116
|
// Export named exports for user convenience
|
|
85
117
|
export { content, GlobalContent, globalContentData };
|
|
86
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
118
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29udGVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3ZhbHRlY2gtY29tcG9uZW50cy9zcmMvbGliL3NlcnZpY2VzL2xhbmctcHJvdmlkZXIvY29udGVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLFlBQVksTUFBTSw0QkFBNEIsQ0FBQztBQUN0RCxPQUFPLEVBQW9CLFdBQVcsRUFBRSxNQUFNLFNBQVMsQ0FBQztBQU14RDs7Ozs7OztHQU9HO0FBQ0gsTUFBTSxpQkFBaUIsR0FBcUI7SUFDMUMsRUFBRSxFQUFFO1FBQ0YsaUJBQWlCO1FBQ2pCLEVBQUUsRUFBRSxTQUFTO1FBQ2IsTUFBTSxFQUFFLFVBQVU7UUFDbEIsSUFBSSxFQUFFLFNBQVM7UUFDZixNQUFNLEVBQUUsVUFBVTtRQUNsQixJQUFJLEVBQUUsUUFBUTtRQUNkLEtBQUssRUFBRSxRQUFRO1FBQ2YsSUFBSSxFQUFFLFFBQVE7UUFDZCxJQUFJLEVBQUUsV0FBVztRQUNqQixRQUFRLEVBQUUsVUFBVTtRQUNwQixNQUFNLEVBQUUsV0FBVztRQUNuQixRQUFRLEVBQUUsV0FBVztRQUVyQixpQkFBaUI7UUFDakIsR0FBRyxFQUFFLFNBQVM7UUFDZCxNQUFNLEVBQUUsUUFBUTtRQUNoQixNQUFNLEVBQUUsUUFBUTtRQUNoQixNQUFNLEVBQUUsU0FBUztRQUNqQixJQUFJLEVBQUUsU0FBUztRQUNmLE9BQU8sRUFBRSxZQUFZO1FBRXJCLDZCQUE2QjtRQUM3QixPQUFPLEVBQUUsYUFBYTtRQUN0QixNQUFNLEVBQUUsMEJBQTBCO1FBQ2xDLEtBQUssRUFBRSxPQUFPO1FBQ2QsT0FBTyxFQUFFLE9BQU87UUFDaEIsT0FBTyxFQUFFLGFBQWE7UUFDdEIsSUFBSSxFQUFFLGFBQWE7UUFDbkIsUUFBUSxFQUFFLFFBQVE7UUFFbEIsdUJBQXVCO1FBQ3ZCLFVBQVUsRUFBRSxnQkFBZ0I7UUFDNUIsa0JBQWtCLEVBQUUsa0RBQWtEO1FBQ3RFLGNBQWMsRUFBRSxnREFBZ0Q7UUFFaEUsc0JBQXNCO1FBQ3RCLGlCQUFpQixFQUFFLFdBQVc7S0FDL0I7SUFDRCxFQUFFLEVBQUU7UUFDRixpQkFBaUI7UUFDakIsRUFBRSxFQUFFLElBQUk7UUFDUixNQUFNLEVBQUUsUUFBUTtRQUNoQixJQUFJLEVBQUUsTUFBTTtRQUNaLE1BQU0sRUFBRSxRQUFRO1FBQ2hCLElBQUksRUFBRSxNQUFNO1FBQ1osS0FBSyxFQUFFLE9BQU87UUFDZCxJQUFJLEVBQUUsTUFBTTtRQUNaLElBQUksRUFBRSxNQUFNO1FBQ1osUUFBUSxFQUFFLFVBQVU7UUFDcEIsTUFBTSxFQUFFLFFBQVE7UUFDaEIsUUFBUSxFQUFFLFVBQVU7UUFFcEIsaUJBQWlCO1FBQ2pCLEdBQUcsRUFBRSxLQUFLO1FBQ1YsTUFBTSxFQUFFLFFBQVE7UUFDaEIsTUFBTSxFQUFFLFFBQVE7UUFDaEIsTUFBTSxFQUFFLFFBQVE7UUFDaEIsSUFBSSxFQUFFLE1BQU07UUFDWixPQUFPLEVBQUUsU0FBUztRQUVsQiw2QkFBNkI7UUFDN0IsT0FBTyxFQUFFLFlBQVk7UUFDckIsTUFBTSxFQUFFLG1CQUFtQjtRQUMzQixLQUFLLEVBQUUsT0FBTztRQUNkLE9BQU8sRUFBRSxTQUFTO1FBQ2xCLE9BQU8sRUFBRSxTQUFTO1FBQ2xCLElBQUksRUFBRSxhQUFhO1FBRW5CLHVCQUF1QjtRQUN2QixVQUFVLEVBQUUsZUFBZTtRQUMzQixrQkFBa0IsRUFBRSw2Q0FBNkM7UUFDakUsY0FBYyxFQUFFLG9EQUFvRDtRQUVwRSxzQkFBc0I7UUFDdEIsaUJBQWlCLEVBQUUsV0FBVztLQUMvQjtJQUNELEVBQUUsRUFBRTtRQUNGLG9GQUFvRjtRQUNwRixFQUFFLEVBQUUsSUFBSTtRQUNSLE1BQU0sRUFBRSxTQUFTO1FBQ2pCLElBQUksRUFBRSxhQUFhO1FBQ25CLE1BQU0sRUFBRSxXQUFXO1FBQ25CLElBQUksRUFBRSxVQUFVO1FBQ2hCLEtBQUssRUFBRSxRQUFRO1FBQ2YsSUFBSSxFQUFFLFFBQVE7UUFFZCxrRkFBa0Y7UUFDbEYsT0FBTyxFQUFFLGVBQWU7UUFDeEIsS0FBSyxFQUFFLFFBQVE7UUFDZixPQUFPLEVBQUUsUUFBUTtRQUVqQix1QkFBdUI7UUFDdkIsVUFBVSxFQUFFLGdCQUFnQjtLQUM3QjtJQUNELEVBQUUsRUFBRTtRQUNGLDBEQUEwRDtRQUMxRCxFQUFFLEVBQUUsSUFBSTtRQUNSLE1BQU0sRUFBRSxXQUFXO1FBQ25CLElBQUksRUFBRSxXQUFXO1FBQ2pCLE1BQU0sRUFBRSxTQUFTO1FBRWpCLDZCQUE2QjtRQUM3QixPQUFPLEVBQUUsVUFBVTtRQUNuQixLQUFLLEVBQUUsUUFBUTtRQUVmLHVCQUF1QjtRQUN2QixVQUFVLEVBQUUsa0JBQWtCO0tBQy9CO0NBQ0YsQ0FBQztBQUVGLE1BQU0sYUFBYSxHQUFHLElBQUksV0FBVyxDQUFDLGlCQUFpQixDQUFDLENBQUM7QUFFekQsTUFBTSxPQUFPLEdBQWE7SUFDeEIsT0FBTyxFQUFFLGFBQWE7SUFDdEIsWUFBWTtDQUNiLENBQUM7QUFFRixlQUFlLE9BQU8sQ0FBQztBQUV2Qiw0Q0FBNEM7QUFDNUMsT0FBTyxFQUFFLE9BQU8sRUFBRSxhQUFhLEVBQUUsaUJBQWlCLEVBQUUsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBMYW5nU2V0dGluZ3MgZnJvbSAnLi9jb21wb25lbnRzL2xhbmctc2V0dGluZ3MnO1xuaW1wb3J0IHsgTGFuZ3VhZ2VzQ29udGVudCwgVGV4dENvbnRlbnQgfSBmcm9tICcuL3R5cGVzJztcblxuZXhwb3J0IGludGVyZmFjZSBQcm92aWRlciB7XG4gIFt4OiBzdHJpbmddOiBUZXh0Q29udGVudDtcbn1cblxuLyoqXG4gKiBHbG9iYWwgY29udGVudCB0aGF0IGNhbiBiZSB1c2VkIGFjcm9zcyBhbGwgY29tcG9uZW50cy5cbiAqIFRoZXNlIGFyZSBjb21tb24gdGV4dHMgbGlrZSBidXR0b25zLCBhY3Rpb25zLCBzdGF0ZXMsIGV0Yy5cbiAqIFN0cnVjdHVyZToge2VzOiB7a2V5MTogJ3ZhbHVlMScsIGtleTI6ICd2YWx1ZTInfSwgZW46IHtrZXkxOiAndmFsdWUxJywga2V5MjogJ3ZhbHVlMid9LCBmcjogey4uLn19XG4gKlxuICogTm90ZTogWW91IGNhbiBhZGQgYW55IGxhbmd1YWdlIGNvZGUuIFRoZSBzeXN0ZW0gd2lsbCBhdXRvbWF0aWNhbGx5IGRldGVjdCBhdmFpbGFibGUgbGFuZ3VhZ2VzXG4gKiBhbmQgcHJvdmlkZSBpbnRlbGxpZ2VudCBmYWxsYmFja3Mgd2l0aCB3YXJuaW5ncyBmb3IgbWlzc2luZyB0cmFuc2xhdGlvbnMuXG4gKi9cbmNvbnN0IGdsb2JhbENvbnRlbnREYXRhOiBMYW5ndWFnZXNDb250ZW50ID0ge1xuICBlczoge1xuICAgIC8vIENvbW1vbiBidXR0b25zXG4gICAgb2s6ICdBY2VwdGFyJyxcbiAgICBjYW5jZWw6ICdDYW5jZWxhcicsXG4gICAgc2F2ZTogJ0d1YXJkYXInLFxuICAgIGRlbGV0ZTogJ0VsaW1pbmFyJyxcbiAgICBlZGl0OiAnRWRpdGFyJyxcbiAgICBjbG9zZTogJ0NlcnJhcicsXG4gICAgYmFjazogJ1ZvbHZlcicsXG4gICAgbmV4dDogJ1NpZ3VpZW50ZScsXG4gICAgcHJldmlvdXM6ICdBbnRlcmlvcicsXG4gICAgZmluaXNoOiAnRmluYWxpemFyJyxcbiAgICBjb250aW51ZTogJ0NvbnRpbnVhcicsXG5cbiAgICAvLyBDb21tb24gYWN0aW9uc1xuICAgIGFkZDogJ0FncmVnYXInLFxuICAgIHJlbW92ZTogJ1F1aXRhcicsXG4gICAgc2VhcmNoOiAnQnVzY2FyJyxcbiAgICBmaWx0ZXI6ICdGaWx0cmFyJyxcbiAgICBzb3J0OiAnT3JkZW5hcicsXG4gICAgcmVmcmVzaDogJ0FjdHVhbGl6YXInLFxuXG4gICAgLy8gQ29tbW9uIHN0YXRlcyBhbmQgbWVzc2FnZXNcbiAgICBsb2FkaW5nOiAnQ2FyZ2FuZG8uLi4nLFxuICAgIG5vRGF0YTogJ05vIGhheSBkYXRvcyBkaXNwb25pYmxlcycsXG4gICAgZXJyb3I6ICdFcnJvcicsXG4gICAgc3VjY2VzczogJ8OJeGl0bycsXG4gICAgd2FybmluZzogJ0FkdmVydGVuY2lhJyxcbiAgICBpbmZvOiAnSW5mb3JtYWNpw7NuJyxcbiAgICBsYW5ndWFnZTogJ0lkaW9tYScsXG5cbiAgICAvLyBDb21tb24gY29uZmlybWF0aW9uc1xuICAgIGFyZVlvdVN1cmU6ICfCv0VzdMOhcyBzZWd1cm8/JyxcbiAgICBkZWxldGVDb25maXJtYXRpb246ICfCv0VzdMOhcyBzZWd1cm8gZGUgcXVlIGRlc2VhcyBlbGltaW5hciB7aXRlbU5hbWV9PycsXG4gICAgdW5zYXZlZENoYW5nZXM6ICdUaWVuZXMgY2FtYmlvcyBzaW4gZ3VhcmRhci4gwr9EZXNlYXMgY29udGludWFyPycsXG5cbiAgICAvLyBDb21tb24gcGxhY2Vob2xkZXJzXG4gICAgc2VhcmNoUGxhY2Vob2xkZXI6ICdCdXNjYXIuLi4nLFxuICB9LFxuICBlbjoge1xuICAgIC8vIENvbW1vbiBidXR0b25zXG4gICAgb2s6ICdPSycsXG4gICAgY2FuY2VsOiAnQ2FuY2VsJyxcbiAgICBzYXZlOiAnU2F2ZScsXG4gICAgZGVsZXRlOiAnRGVsZXRlJyxcbiAgICBlZGl0OiAnRWRpdCcsXG4gICAgY2xvc2U6ICdDbG9zZScsXG4gICAgYmFjazogJ0JhY2snLFxuICAgIG5leHQ6ICdOZXh0JyxcbiAgICBwcmV2aW91czogJ1ByZXZpb3VzJyxcbiAgICBmaW5pc2g6ICdGaW5pc2gnLFxuICAgIGNvbnRpbnVlOiAnQ29udGludWUnLFxuXG4gICAgLy8gQ29tbW9uIGFjdGlvbnNcbiAgICBhZGQ6ICdBZGQnLFxuICAgIHJlbW92ZTogJ1JlbW92ZScsXG4gICAgc2VhcmNoOiAnU2VhcmNoJyxcbiAgICBmaWx0ZXI6ICdGaWx0ZXInLFxuICAgIHNvcnQ6ICdTb3J0JyxcbiAgICByZWZyZXNoOiAnUmVmcmVzaCcsXG5cbiAgICAvLyBDb21tb24gc3RhdGVzIGFuZCBtZXNzYWdlc1xuICAgIGxvYWRpbmc6ICdMb2FkaW5nLi4uJyxcbiAgICBub0RhdGE6ICdObyBkYXRhIGF2YWlsYWJsZScsXG4gICAgZXJyb3I6ICdFcnJvcicsXG4gICAgc3VjY2VzczogJ1N1Y2Nlc3MnLFxuICAgIHdhcm5pbmc6ICdXYXJuaW5nJyxcbiAgICBpbmZvOiAnSW5mb3JtYXRpb24nLFxuXG4gICAgLy8gQ29tbW9uIGNvbmZpcm1hdGlvbnNcbiAgICBhcmVZb3VTdXJlOiAnQXJlIHlvdSBzdXJlPycsXG4gICAgZGVsZXRlQ29uZmlybWF0aW9uOiAnQXJlIHlvdSBzdXJlIHlvdSB3YW50IHRvIGRlbGV0ZSB7aXRlbU5hbWV9PycsXG4gICAgdW5zYXZlZENoYW5nZXM6ICdZb3UgaGF2ZSB1bnNhdmVkIGNoYW5nZXMuIERvIHlvdSB3YW50IHRvIGNvbnRpbnVlPycsXG5cbiAgICAvLyBDb21tb24gcGxhY2Vob2xkZXJzXG4gICAgc2VhcmNoUGxhY2Vob2xkZXI6ICdTZWFyY2guLi4nLFxuICB9LFxuICBmcjoge1xuICAgIC8vIENvbW1vbiBidXR0b25zIC0gRXhhbXBsZSBvZiBwYXJ0aWFsIHRyYW5zbGF0aW9uIChtaXNzaW5nIHNvbWUga2V5cyBpbnRlbnRpb25hbGx5KVxuICAgIG9rOiAnT0snLFxuICAgIGNhbmNlbDogJ0FubnVsZXInLFxuICAgIHNhdmU6ICdTYXV2ZWdhcmRlcicsXG4gICAgZGVsZXRlOiAnU3VwcHJpbWVyJyxcbiAgICBlZGl0OiAnTW9kaWZpZXInLFxuICAgIGNsb3NlOiAnRmVybWVyJyxcbiAgICBiYWNrOiAnUmV0b3VyJyxcblxuICAgIC8vIENvbW1vbiBzdGF0ZXMgYW5kIG1lc3NhZ2VzIChpbnRlbnRpb25hbGx5IGluY29tcGxldGUgdG8gc2hvdyBmYWxsYmFjayBiZWhhdmlvcilcbiAgICBsb2FkaW5nOiAnQ2hhcmdlbWVudC4uLicsXG4gICAgZXJyb3I6ICdFcnJldXInLFxuICAgIHN1Y2Nlc3M6ICdTdWNjw6hzJyxcblxuICAgIC8vIENvbW1vbiBjb25maXJtYXRpb25zXG4gICAgYXJlWW91U3VyZTogJ8OKdGVzLXZvdXMgc8O7cj8nLFxuICB9LFxuICBkZToge1xuICAgIC8vIENvbW1vbiBidXR0b25zIC0gQW5vdGhlciBleGFtcGxlIG9mIHBhcnRpYWwgdHJhbnNsYXRpb25cbiAgICBvazogJ09LJyxcbiAgICBjYW5jZWw6ICdBYmJyZWNoZW4nLFxuICAgIHNhdmU6ICdTcGVpY2hlcm4nLFxuICAgIGRlbGV0ZTogJ0zDtnNjaGVuJyxcblxuICAgIC8vIENvbW1vbiBzdGF0ZXMgYW5kIG1lc3NhZ2VzXG4gICAgbG9hZGluZzogJ0xhZGVuLi4uJyxcbiAgICBlcnJvcjogJ0ZlaGxlcicsXG5cbiAgICAvLyBDb21tb24gY29uZmlybWF0aW9uc1xuICAgIGFyZVlvdVN1cmU6ICdTaW5kIFNpZSBzaWNoZXI/JyxcbiAgfSxcbn07XG5cbmNvbnN0IEdsb2JhbENvbnRlbnQgPSBuZXcgVGV4dENvbnRlbnQoZ2xvYmFsQ29udGVudERhdGEpO1xuXG5jb25zdCBjb250ZW50OiBQcm92aWRlciA9IHtcbiAgX2dsb2JhbDogR2xvYmFsQ29udGVudCxcbiAgTGFuZ1NldHRpbmdzLFxufTtcblxuZXhwb3J0IGRlZmF1bHQgY29udGVudDtcblxuLy8gRXhwb3J0IG5hbWVkIGV4cG9ydHMgZm9yIHVzZXIgY29udmVuaWVuY2VcbmV4cG9ydCB7IGNvbnRlbnQsIEdsb2JhbENvbnRlbnQsIGdsb2JhbENvbnRlbnREYXRhIH07XG4iXX0=
|