valtech-components 2.0.286 → 2.0.289

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (30) hide show
  1. package/README.md +61 -21
  2. package/esm2022/lib/components/atoms/text/text.component.mjs +46 -10
  3. package/esm2022/lib/components/atoms/text/types.mjs +1 -1
  4. package/esm2022/lib/examples/custom-content-demo.component.mjs +291 -0
  5. package/esm2022/lib/examples/link-processing-example.component.mjs +189 -0
  6. package/esm2022/lib/services/lang-provider/content.mjs +1 -13
  7. package/esm2022/lib/services/link-processor.service.mjs +192 -0
  8. package/esm2022/lib/shared/pipes/process-links.pipe.mjs +69 -0
  9. package/esm2022/public-api.mjs +5 -4
  10. package/fesm2022/valtech-components.mjs +473 -919
  11. package/fesm2022/valtech-components.mjs.map +1 -1
  12. package/lib/components/atoms/text/text.component.d.ts +5 -1
  13. package/lib/components/atoms/text/types.d.ts +5 -0
  14. package/lib/examples/link-processing-example.component.d.ts +24 -0
  15. package/lib/services/link-processor.service.d.ts +98 -0
  16. package/lib/shared/pipes/process-links.pipe.d.ts +55 -0
  17. package/package.json +1 -1
  18. package/public-api.d.ts +4 -3
  19. package/esm2022/lib/components/_examples/custom-content-demo.component.mjs +0 -291
  20. package/esm2022/lib/components/_examples/global-content-example-content.mjs +0 -23
  21. package/esm2022/lib/components/_examples/global-content-example.component.mjs +0 -504
  22. package/esm2022/lib/components/_examples/reactive-content-example-content.mjs +0 -43
  23. package/esm2022/lib/components/_examples/reactive-content-example.component.mjs +0 -347
  24. package/esm2022/lib/services/lang-provider/components/theme-settings.mjs +0 -15
  25. package/lib/components/_examples/global-content-example-content.d.ts +0 -9
  26. package/lib/components/_examples/global-content-example.component.d.ts +0 -73
  27. package/lib/components/_examples/reactive-content-example-content.d.ts +0 -32
  28. package/lib/components/_examples/reactive-content-example.component.d.ts +0 -47
  29. package/lib/services/lang-provider/components/theme-settings.d.ts +0 -3
  30. /package/lib/{components/_examples → examples}/custom-content-demo.component.d.ts +0 -0
@@ -1,347 +0,0 @@
1
- import { AsyncPipe, NgIf } from '@angular/common';
2
- import { ChangeDetectionStrategy, Component, inject } from '@angular/core';
3
- import { IonButton, IonCard, IonCardContent, IonCardHeader, IonCardTitle } from '@ionic/angular/standalone';
4
- import { ContentService } from '../../services/content.service';
5
- import { LangOption } from '../../services/lang-provider/types';
6
- import { TextComponent } from '../atoms/text/text.component';
7
- import * as i0 from "@angular/core";
8
- /**
9
- * Reactive Content Example Component
10
- *
11
- * This component demonstrates various ways to use the new reactive content system with ContentService:
12
- * 1. Direct reactive content binding with val-text
13
- * 2. Content with interpolation (dynamic values)
14
- * 3. Multiple content keys retrieved at once
15
- * 4. Using the scoped content helper (forComponent)
16
- * 5. Language switching functionality
17
- *
18
- * The component automatically updates all text when the language changes,
19
- * using the simplified ContentService API that eliminates the need for manual
20
- * LangService injection and utility function imports.
21
- *
22
- * @example Usage patterns:
23
- * ```typescript
24
- * // Simple injection
25
- * content = inject(ContentService);
26
- *
27
- * // Scoped helper for this component
28
- * componentContent = this.content.forComponent('MyComponent');
29
- *
30
- * // Get reactive content
31
- * title$ = this.componentContent.get('title');
32
- * multipleTexts$ = this.componentContent.getMultiple(['title', 'subtitle']);
33
- * ```
34
- * without requiring manual subscriptions or change detection triggers.
35
- */
36
- export class ReactiveContentExampleComponent {
37
- constructor() {
38
- // Use the new ContentService with inject
39
- this.content = inject(ContentService);
40
- // Create a scoped content helper for this component
41
- this.componentContent = this.content.forComponent('ReactiveContentExample');
42
- // State for confirmation dialog
43
- this.showConfirmation = false;
44
- // Example 1: Simple reactive content props for val-text
45
- this.titleProps = {
46
- contentKey: 'title',
47
- contentClass: 'ReactiveContentExample',
48
- contentFallback: 'Default Title',
49
- color: 'primary',
50
- size: 'xlarge',
51
- bold: true,
52
- };
53
- // Example 2: Reactive content with interpolation
54
- this.greetingProps = {
55
- contentKey: 'greeting',
56
- contentClass: 'ReactiveContentExample',
57
- contentInterpolation: { name: 'Developer', count: 3 },
58
- contentFallback: 'Hello Developer',
59
- color: 'secondary',
60
- size: 'large',
61
- bold: false,
62
- };
63
- }
64
- ngOnInit() {
65
- // Set up reactive observables using ContentService
66
- this.currentLanguage$ = this.content.currentLang$;
67
- // Example 3: Get multiple content keys reactively - using the new API
68
- this.multipleContent$ = this.componentContent.getMultiple(['description', 'dynamicText']);
69
- // Button text reactive - using the new simplified API with options
70
- this.buttonText$ = this.componentContent.get('buttonText');
71
- // Global content observables using unified fromContent method
72
- this.saveButton$ = this.content.fromContent({ key: 'save' });
73
- this.cancelButton$ = this.content.fromContent({ key: 'cancel' });
74
- this.deleteButton$ = this.content.fromContent({ key: 'delete' });
75
- this.okButton$ = this.content.fromContent({ key: 'ok' });
76
- this.loadingText$ = this.content.fromContent({ key: 'loading' });
77
- // Global content with interpolation using unified method
78
- this.deleteConfirmationText$ = this.content.fromContent({
79
- key: 'deleteConfirmation',
80
- interpolation: { itemName: 'este elemento' },
81
- });
82
- // Example of synchronous global content access
83
- console.log('Global save text:', this.content.getText('save'));
84
- console.log('Global cancel text:', this.content.getGlobalText('cancel'));
85
- // Example of using new scoped helper with interpolation
86
- const greetingWithNewAPI$ = this.componentContent.get('greeting', {
87
- interpolation: { name: 'Developer', count: 3 },
88
- fallback: 'Hello!',
89
- });
90
- console.log('New unified API works!', greetingWithNewAPI$);
91
- }
92
- /**
93
- * Toggle between Spanish and English
94
- */
95
- toggleLanguage() {
96
- const currentLang = this.content.currentLang;
97
- const newLang = currentLang === LangOption.ES ? LangOption.EN : LangOption.ES;
98
- this.content.setLang(newLang);
99
- }
100
- /**
101
- * Show delete confirmation dialog
102
- */
103
- showDeleteConfirmation() {
104
- this.showConfirmation = true;
105
- }
106
- /**
107
- * Hide delete confirmation dialog
108
- */
109
- hideDeleteConfirmation() {
110
- this.showConfirmation = false;
111
- }
112
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ReactiveContentExampleComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
113
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: ReactiveContentExampleComponent, isStandalone: true, selector: "val-reactive-content-example", ngImport: i0, template: `
114
- <ion-card>
115
- <ion-card-header>
116
- <ion-card-title>
117
- <!-- Example 1: Direct reactive content using val-text -->
118
- <val-text [props]="titleProps"></val-text>
119
- </ion-card-title>
120
- </ion-card-header>
121
-
122
- <ion-card-content>
123
- <!-- Example 2: Reactive content with interpolation -->
124
- <val-text [props]="greetingProps"></val-text>
125
-
126
- <div class="example-divider"></div>
127
-
128
- <!-- Example 3: Multiple content keys -->
129
- <div *ngIf="multipleContent$ | async as content">
130
- <val-text
131
- [props]="{
132
- content: content['description'],
133
- color: 'medium',
134
- size: 'medium',
135
- bold: false,
136
- }"
137
- ></val-text>
138
-
139
- <br /><br />
140
-
141
- <val-text
142
- [props]="{
143
- content: content['dynamicText'],
144
- color: 'primary',
145
- size: 'small',
146
- bold: true,
147
- }"
148
- ></val-text>
149
- </div>
150
-
151
- <div class="example-divider"></div>
152
-
153
- <!-- Example 4: Using content helper -->
154
- <val-text
155
- [props]="{
156
- content: componentContent.getText('welcomeMessage'),
157
- color: 'success',
158
- size: 'large',
159
- bold: true,
160
- }"
161
- ></val-text>
162
-
163
- <div class="example-divider"></div>
164
-
165
- <!-- Example 5: Global content - buttons using global text -->
166
- <div class="button-group">
167
- <ion-button [fill]="'solid'" [color]="'primary'">
168
- {{ saveButton$ | async }}
169
- </ion-button>
170
- <ion-button [fill]="'outline'" [color]="'secondary'">
171
- {{ cancelButton$ | async }}
172
- </ion-button>
173
- <ion-button [fill]="'outline'" [color]="'danger'" (click)="showDeleteConfirmation()">
174
- {{ deleteButton$ | async }}
175
- </ion-button>
176
- </div>
177
-
178
- <div class="example-divider"></div>
179
-
180
- <!-- Example 6: Global content with interpolation -->
181
- <div *ngIf="showConfirmation" class="confirmation-message">
182
- <val-text
183
- [props]="{
184
- content: deleteConfirmationText$ | async,
185
- color: 'warning',
186
- size: 'medium',
187
- bold: true,
188
- }"
189
- ></val-text>
190
- <br /><br />
191
- <ion-button [fill]="'solid'" [color]="'danger'" size="small" (click)="hideDeleteConfirmation()">
192
- {{ okButton$ | async }}
193
- </ion-button>
194
- <ion-button [fill]="'clear'" [color]="'medium'" size="small" (click)="hideDeleteConfirmation()">
195
- {{ cancelButton$ | async }}
196
- </ion-button>
197
- </div>
198
-
199
- <div class="example-divider"></div>
200
-
201
- <!-- Example 7: Language switching button -->
202
- <ion-button expand="block" fill="outline" color="tertiary" (click)="toggleLanguage()">
203
- {{ buttonText$ | async }}
204
- </ion-button>
205
-
206
- <!-- Example 8: Current language display -->
207
- <div class="language-info">
208
- <p>
209
- Current language:
210
- <strong>{{ currentLanguage$ | async }}</strong>
211
- </p>
212
-
213
- <!-- Example of mixed global and component content -->
214
- <p>
215
- <val-text
216
- [props]="{
217
- content: loadingText$ | async,
218
- color: 'medium',
219
- size: 'small',
220
- bold: false,
221
- }"
222
- ></val-text>
223
- </p>
224
- </div>
225
- </ion-card-content>
226
- </ion-card>
227
- `, isInline: true, styles: [":host{display:block;margin:16px}.example-divider{margin:24px 0;border-bottom:1px solid var(--ion-color-light);padding-bottom:16px}.language-info{margin-top:24px;padding:16px;background-color:var(--ion-color-light);border-radius:8px;text-align:center}.language-info p{margin:0;color:var(--ion-color-dark)}.language-info strong{color:var(--ion-color-primary);text-transform:uppercase}ion-card{margin:0;border-radius:12px;box-shadow:0 4px 16px #0000001a}ion-card-header{padding-bottom:8px}ion-card-title{font-size:1.5rem;font-weight:600}ion-card-content{padding-top:16px}ion-button{margin:16px 0;--border-radius: 8px}ion-card-content{padding:20px}.language-info{margin-top:16px;padding:12px;background-color:var(--ion-color-light);border-radius:8px;text-align:center}.content-section{margin:16px 0;padding:12px;border:1px solid var(--ion-color-light);border-radius:8px}.content-section h3{margin-top:0;color:var(--ion-color-primary)}.example-divider{margin:24px 0;height:1px;background-color:var(--ion-color-light)}\n"], dependencies: [{ kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: IonCard, selector: "ion-card", inputs: ["button", "color", "disabled", "download", "href", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: IonCardContent, selector: "ion-card-content", inputs: ["mode"] }, { kind: "component", type: IonCardHeader, selector: "ion-card-header", inputs: ["color", "mode", "translucent"] }, { kind: "component", type: IonCardTitle, selector: "ion-card-title", inputs: ["color", "mode"] }, { kind: "component", type: IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: TextComponent, selector: "val-text", inputs: ["props"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
228
- }
229
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ReactiveContentExampleComponent, decorators: [{
230
- type: Component,
231
- args: [{ selector: 'val-reactive-content-example', standalone: true, imports: [AsyncPipe, NgIf, IonCard, IonCardContent, IonCardHeader, IonCardTitle, IonButton, TextComponent], template: `
232
- <ion-card>
233
- <ion-card-header>
234
- <ion-card-title>
235
- <!-- Example 1: Direct reactive content using val-text -->
236
- <val-text [props]="titleProps"></val-text>
237
- </ion-card-title>
238
- </ion-card-header>
239
-
240
- <ion-card-content>
241
- <!-- Example 2: Reactive content with interpolation -->
242
- <val-text [props]="greetingProps"></val-text>
243
-
244
- <div class="example-divider"></div>
245
-
246
- <!-- Example 3: Multiple content keys -->
247
- <div *ngIf="multipleContent$ | async as content">
248
- <val-text
249
- [props]="{
250
- content: content['description'],
251
- color: 'medium',
252
- size: 'medium',
253
- bold: false,
254
- }"
255
- ></val-text>
256
-
257
- <br /><br />
258
-
259
- <val-text
260
- [props]="{
261
- content: content['dynamicText'],
262
- color: 'primary',
263
- size: 'small',
264
- bold: true,
265
- }"
266
- ></val-text>
267
- </div>
268
-
269
- <div class="example-divider"></div>
270
-
271
- <!-- Example 4: Using content helper -->
272
- <val-text
273
- [props]="{
274
- content: componentContent.getText('welcomeMessage'),
275
- color: 'success',
276
- size: 'large',
277
- bold: true,
278
- }"
279
- ></val-text>
280
-
281
- <div class="example-divider"></div>
282
-
283
- <!-- Example 5: Global content - buttons using global text -->
284
- <div class="button-group">
285
- <ion-button [fill]="'solid'" [color]="'primary'">
286
- {{ saveButton$ | async }}
287
- </ion-button>
288
- <ion-button [fill]="'outline'" [color]="'secondary'">
289
- {{ cancelButton$ | async }}
290
- </ion-button>
291
- <ion-button [fill]="'outline'" [color]="'danger'" (click)="showDeleteConfirmation()">
292
- {{ deleteButton$ | async }}
293
- </ion-button>
294
- </div>
295
-
296
- <div class="example-divider"></div>
297
-
298
- <!-- Example 6: Global content with interpolation -->
299
- <div *ngIf="showConfirmation" class="confirmation-message">
300
- <val-text
301
- [props]="{
302
- content: deleteConfirmationText$ | async,
303
- color: 'warning',
304
- size: 'medium',
305
- bold: true,
306
- }"
307
- ></val-text>
308
- <br /><br />
309
- <ion-button [fill]="'solid'" [color]="'danger'" size="small" (click)="hideDeleteConfirmation()">
310
- {{ okButton$ | async }}
311
- </ion-button>
312
- <ion-button [fill]="'clear'" [color]="'medium'" size="small" (click)="hideDeleteConfirmation()">
313
- {{ cancelButton$ | async }}
314
- </ion-button>
315
- </div>
316
-
317
- <div class="example-divider"></div>
318
-
319
- <!-- Example 7: Language switching button -->
320
- <ion-button expand="block" fill="outline" color="tertiary" (click)="toggleLanguage()">
321
- {{ buttonText$ | async }}
322
- </ion-button>
323
-
324
- <!-- Example 8: Current language display -->
325
- <div class="language-info">
326
- <p>
327
- Current language:
328
- <strong>{{ currentLanguage$ | async }}</strong>
329
- </p>
330
-
331
- <!-- Example of mixed global and component content -->
332
- <p>
333
- <val-text
334
- [props]="{
335
- content: loadingText$ | async,
336
- color: 'medium',
337
- size: 'small',
338
- bold: false,
339
- }"
340
- ></val-text>
341
- </p>
342
- </div>
343
- </ion-card-content>
344
- </ion-card>
345
- `, changeDetection: ChangeDetectionStrategy.OnPush, styles: [":host{display:block;margin:16px}.example-divider{margin:24px 0;border-bottom:1px solid var(--ion-color-light);padding-bottom:16px}.language-info{margin-top:24px;padding:16px;background-color:var(--ion-color-light);border-radius:8px;text-align:center}.language-info p{margin:0;color:var(--ion-color-dark)}.language-info strong{color:var(--ion-color-primary);text-transform:uppercase}ion-card{margin:0;border-radius:12px;box-shadow:0 4px 16px #0000001a}ion-card-header{padding-bottom:8px}ion-card-title{font-size:1.5rem;font-weight:600}ion-card-content{padding-top:16px}ion-button{margin:16px 0;--border-radius: 8px}ion-card-content{padding:20px}.language-info{margin-top:16px;padding:12px;background-color:var(--ion-color-light);border-radius:8px;text-align:center}.content-section{margin:16px 0;padding:12px;border:1px solid var(--ion-color-light);border-radius:8px}.content-section h3{margin-top:0;color:var(--ion-color-primary)}.example-divider{margin:24px 0;height:1px;background-color:var(--ion-color-light)}\n"] }]
346
- }], ctorParameters: () => [] });
347
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"reactive-content-example.component.js","sourceRoot":"","sources":["../../../../../../projects/valtech-components/src/lib/components/_examples/reactive-content-example.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { AsyncPipe, NgIf } from '@angular/common';\nimport { ChangeDetectionStrategy, Component, OnInit, inject } from '@angular/core';\nimport { IonButton, IonCard, IonCardContent, IonCardHeader, IonCardTitle } from '@ionic/angular/standalone';\nimport { Observable } from 'rxjs';\n\nimport { ContentService } from '../../services/content.service';\nimport { LangOption } from '../../services/lang-provider/types';\nimport { TextComponent } from '../atoms/text/text.component';\nimport { TextMetadata } from '../atoms/text/types';\n\n@Component({\n  selector: 'val-reactive-content-example',\n  standalone: true,\n  imports: [AsyncPipe, NgIf, IonCard, IonCardContent, IonCardHeader, IonCardTitle, IonButton, TextComponent],\n  template: `\n    <ion-card>\n      <ion-card-header>\n        <ion-card-title>\n          <!-- Example 1: Direct reactive content using val-text -->\n          <val-text [props]=\"titleProps\"></val-text>\n        </ion-card-title>\n      </ion-card-header>\n\n      <ion-card-content>\n        <!-- Example 2: Reactive content with interpolation -->\n        <val-text [props]=\"greetingProps\"></val-text>\n\n        <div class=\"example-divider\"></div>\n\n        <!-- Example 3: Multiple content keys -->\n        <div *ngIf=\"multipleContent$ | async as content\">\n          <val-text\n            [props]=\"{\n              content: content['description'],\n              color: 'medium',\n              size: 'medium',\n              bold: false,\n            }\"\n          ></val-text>\n\n          <br /><br />\n\n          <val-text\n            [props]=\"{\n              content: content['dynamicText'],\n              color: 'primary',\n              size: 'small',\n              bold: true,\n            }\"\n          ></val-text>\n        </div>\n\n        <div class=\"example-divider\"></div>\n\n        <!-- Example 4: Using content helper -->\n        <val-text\n          [props]=\"{\n            content: componentContent.getText('welcomeMessage'),\n            color: 'success',\n            size: 'large',\n            bold: true,\n          }\"\n        ></val-text>\n\n        <div class=\"example-divider\"></div>\n\n        <!-- Example 5: Global content - buttons using global text -->\n        <div class=\"button-group\">\n          <ion-button [fill]=\"'solid'\" [color]=\"'primary'\">\n            {{ saveButton$ | async }}\n          </ion-button>\n          <ion-button [fill]=\"'outline'\" [color]=\"'secondary'\">\n            {{ cancelButton$ | async }}\n          </ion-button>\n          <ion-button [fill]=\"'outline'\" [color]=\"'danger'\" (click)=\"showDeleteConfirmation()\">\n            {{ deleteButton$ | async }}\n          </ion-button>\n        </div>\n\n        <div class=\"example-divider\"></div>\n\n        <!-- Example 6: Global content with interpolation -->\n        <div *ngIf=\"showConfirmation\" class=\"confirmation-message\">\n          <val-text\n            [props]=\"{\n              content: deleteConfirmationText$ | async,\n              color: 'warning',\n              size: 'medium',\n              bold: true,\n            }\"\n          ></val-text>\n          <br /><br />\n          <ion-button [fill]=\"'solid'\" [color]=\"'danger'\" size=\"small\" (click)=\"hideDeleteConfirmation()\">\n            {{ okButton$ | async }}\n          </ion-button>\n          <ion-button [fill]=\"'clear'\" [color]=\"'medium'\" size=\"small\" (click)=\"hideDeleteConfirmation()\">\n            {{ cancelButton$ | async }}\n          </ion-button>\n        </div>\n\n        <div class=\"example-divider\"></div>\n\n        <!-- Example 7: Language switching button -->\n        <ion-button expand=\"block\" fill=\"outline\" color=\"tertiary\" (click)=\"toggleLanguage()\">\n          {{ buttonText$ | async }}\n        </ion-button>\n\n        <!-- Example 8: Current language display -->\n        <div class=\"language-info\">\n          <p>\n            Current language:\n            <strong>{{ currentLanguage$ | async }}</strong>\n          </p>\n\n          <!-- Example of mixed global and component content -->\n          <p>\n            <val-text\n              [props]=\"{\n                content: loadingText$ | async,\n                color: 'medium',\n                size: 'small',\n                bold: false,\n              }\"\n            ></val-text>\n          </p>\n        </div>\n      </ion-card-content>\n    </ion-card>\n  `,\n  styleUrls: ['./reactive-content-example.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\n/**\n * Reactive Content Example Component\n *\n * This component demonstrates various ways to use the new reactive content system with ContentService:\n * 1. Direct reactive content binding with val-text\n * 2. Content with interpolation (dynamic values)\n * 3. Multiple content keys retrieved at once\n * 4. Using the scoped content helper (forComponent)\n * 5. Language switching functionality\n *\n * The component automatically updates all text when the language changes,\n * using the simplified ContentService API that eliminates the need for manual\n * LangService injection and utility function imports.\n *\n * @example Usage patterns:\n * ```typescript\n * // Simple injection\n * content = inject(ContentService);\n *\n * // Scoped helper for this component\n * componentContent = this.content.forComponent('MyComponent');\n *\n * // Get reactive content\n * title$ = this.componentContent.get('title');\n * multipleTexts$ = this.componentContent.getMultiple(['title', 'subtitle']);\n * ```\n * without requiring manual subscriptions or change detection triggers.\n */\nexport class ReactiveContentExampleComponent implements OnInit {\n  // Use the new ContentService with inject\n  content = inject(ContentService);\n\n  // Create a scoped content helper for this component\n  componentContent = this.content.forComponent('ReactiveContentExample');\n\n  // Observable for current language\n  currentLanguage$: Observable<LangOption>;\n\n  // Observable for button text\n  buttonText$: Observable<string>;\n\n  // Global content observables\n  saveButton$: Observable<string>;\n  cancelButton$: Observable<string>;\n  deleteButton$: Observable<string>;\n  okButton$: Observable<string>;\n  deleteConfirmationText$: Observable<string>;\n  loadingText$: Observable<string>;\n\n  // State for confirmation dialog\n  showConfirmation = false;\n\n  // Example 1: Simple reactive content props for val-text\n  titleProps: TextMetadata = {\n    contentKey: 'title',\n    contentClass: 'ReactiveContentExample',\n    contentFallback: 'Default Title',\n    color: 'primary',\n    size: 'xlarge',\n    bold: true,\n  };\n\n  // Example 2: Reactive content with interpolation\n  greetingProps: TextMetadata = {\n    contentKey: 'greeting',\n    contentClass: 'ReactiveContentExample',\n    contentInterpolation: { name: 'Developer', count: 3 },\n    contentFallback: 'Hello Developer',\n    color: 'secondary',\n    size: 'large',\n    bold: false,\n  };\n\n  // Example 3: Multiple content keys\n  multipleContent$: Observable<Record<string, string>>;\n\n  constructor() {}\n\n  ngOnInit() {\n    // Set up reactive observables using ContentService\n    this.currentLanguage$ = this.content.currentLang$;\n\n    // Example 3: Get multiple content keys reactively - using the new API\n    this.multipleContent$ = this.componentContent.getMultiple(['description', 'dynamicText']);\n\n    // Button text reactive - using the new simplified API with options\n    this.buttonText$ = this.componentContent.get('buttonText');\n\n    // Global content observables using unified fromContent method\n    this.saveButton$ = this.content.fromContent({ key: 'save' });\n    this.cancelButton$ = this.content.fromContent({ key: 'cancel' });\n    this.deleteButton$ = this.content.fromContent({ key: 'delete' });\n    this.okButton$ = this.content.fromContent({ key: 'ok' });\n    this.loadingText$ = this.content.fromContent({ key: 'loading' });\n\n    // Global content with interpolation using unified method\n    this.deleteConfirmationText$ = this.content.fromContent({\n      key: 'deleteConfirmation',\n      interpolation: { itemName: 'este elemento' },\n    });\n\n    // Example of synchronous global content access\n    console.log('Global save text:', this.content.getText('save'));\n    console.log('Global cancel text:', this.content.getGlobalText('cancel'));\n\n    // Example of using new scoped helper with interpolation\n    const greetingWithNewAPI$ = this.componentContent.get('greeting', {\n      interpolation: { name: 'Developer', count: 3 },\n      fallback: 'Hello!',\n    });\n    console.log('New unified API works!', greetingWithNewAPI$);\n  }\n\n  /**\n   * Toggle between Spanish and English\n   */\n  toggleLanguage(): void {\n    const currentLang = this.content.currentLang;\n    const newLang = currentLang === LangOption.ES ? LangOption.EN : LangOption.ES;\n    this.content.setLang(newLang);\n  }\n\n  /**\n   * Show delete confirmation dialog\n   */\n  showDeleteConfirmation(): void {\n    this.showConfirmation = true;\n  }\n\n  /**\n   * Hide delete confirmation dialog\n   */\n  hideDeleteConfirmation(): void {\n    this.showConfirmation = false;\n  }\n}\n"]}
@@ -1,15 +0,0 @@
1
- import { TextContent } from '../types';
2
- const text = {
3
- es: {
4
- light: 'Siempre claro',
5
- dark: 'Siempre oscuro',
6
- auto: 'Automático',
7
- },
8
- en: {
9
- light: 'Always light',
10
- dark: 'Always dark',
11
- auto: 'System settings',
12
- },
13
- };
14
- export default new TextContent(text);
15
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGhlbWUtc2V0dGluZ3MuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy92YWx0ZWNoLWNvbXBvbmVudHMvc3JjL2xpYi9zZXJ2aWNlcy9sYW5nLXByb3ZpZGVyL2NvbXBvbmVudHMvdGhlbWUtc2V0dGluZ3MudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFvQixXQUFXLEVBQUUsTUFBTSxVQUFVLENBQUM7QUFFekQsTUFBTSxJQUFJLEdBQXFCO0lBQzdCLEVBQUUsRUFBRTtRQUNGLEtBQUssRUFBRSxlQUFlO1FBQ3RCLElBQUksRUFBRSxnQkFBZ0I7UUFDdEIsSUFBSSxFQUFFLFlBQVk7S0FDbkI7SUFDRCxFQUFFLEVBQUU7UUFDRixLQUFLLEVBQUUsY0FBYztRQUNyQixJQUFJLEVBQUUsYUFBYTtRQUNuQixJQUFJLEVBQUUsaUJBQWlCO0tBQ3hCO0NBQ0YsQ0FBQztBQUVGLGVBQWUsSUFBSSxXQUFXLENBQUMsSUFBSSxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBMYW5ndWFnZXNDb250ZW50LCBUZXh0Q29udGVudCB9IGZyb20gJy4uL3R5cGVzJztcblxuY29uc3QgdGV4dDogTGFuZ3VhZ2VzQ29udGVudCA9IHtcbiAgZXM6IHtcbiAgICBsaWdodDogJ1NpZW1wcmUgY2xhcm8nLFxuICAgIGRhcms6ICdTaWVtcHJlIG9zY3VybycsXG4gICAgYXV0bzogJ0F1dG9tw6F0aWNvJyxcbiAgfSxcbiAgZW46IHtcbiAgICBsaWdodDogJ0Fsd2F5cyBsaWdodCcsXG4gICAgZGFyazogJ0Fsd2F5cyBkYXJrJyxcbiAgICBhdXRvOiAnU3lzdGVtIHNldHRpbmdzJyxcbiAgfSxcbn07XG5cbmV4cG9ydCBkZWZhdWx0IG5ldyBUZXh0Q29udGVudCh0ZXh0KTtcbiJdfQ==
@@ -1,9 +0,0 @@
1
- /**
2
- * Content configuration for the Global Content Example component.
3
- *
4
- * This demonstrates component-specific content that works alongside
5
- * global content to create a comprehensive content management system.
6
- */
7
- import { TextContent } from '../../services/lang-provider/types';
8
- declare const _default: TextContent;
9
- export default _default;
@@ -1,73 +0,0 @@
1
- import { TextMetadata } from '../atoms/text/types';
2
- import * as i0 from "@angular/core";
3
- /**
4
- * Comprehensive example demonstrating global content usage.
5
- *
6
- * This component shows:
7
- * 1. Global button texts
8
- * 2. Global state messages
9
- * 3. Global confirmations with interpolation
10
- * 4. Mixing global and component-specific content
11
- * 5. Language switching
12
- */
13
- export declare class GlobalContentExampleComponent {
14
- private content;
15
- private componentContent;
16
- formData: {
17
- name: string;
18
- email: string;
19
- message: string;
20
- };
21
- actionPerformed: boolean;
22
- titleProps: TextMetadata;
23
- languageLabel: TextMetadata;
24
- componentDescriptionProps: TextMetadata;
25
- saveButton$: import("rxjs").Observable<string>;
26
- editButton$: import("rxjs").Observable<string>;
27
- deleteButton$: import("rxjs").Observable<string>;
28
- cancelButton$: import("rxjs").Observable<string>;
29
- backButton$: import("rxjs").Observable<string>;
30
- nextButton$: import("rxjs").Observable<string>;
31
- okButton$: import("rxjs").Observable<string>;
32
- closeButton$: import("rxjs").Observable<string>;
33
- addButton$: import("rxjs").Observable<string>;
34
- removeButton$: import("rxjs").Observable<string>;
35
- searchButton$: import("rxjs").Observable<string>;
36
- filterButton$: import("rxjs").Observable<string>;
37
- sortButton$: import("rxjs").Observable<string>;
38
- refreshButton$: import("rxjs").Observable<string>;
39
- loadingText$: import("rxjs").Observable<string>;
40
- successText$: import("rxjs").Observable<string>;
41
- errorText$: import("rxjs").Observable<string>;
42
- warningText$: import("rxjs").Observable<string>;
43
- infoText$: import("rxjs").Observable<string>;
44
- noDataText$: import("rxjs").Observable<string>;
45
- requiredText$: import("rxjs").Observable<string>;
46
- optionalText$: import("rxjs").Observable<string>;
47
- searchPlaceholder$: import("rxjs").Observable<string>;
48
- deleteConfirmation$: import("rxjs").Observable<string>;
49
- areYouSureText$: import("rxjs").Observable<string>;
50
- unsavedChangesText$: import("rxjs").Observable<string>;
51
- /**
52
- * Set language
53
- */
54
- setLanguage(lang: 'es' | 'en'): void;
55
- /**
56
- * Perform an action to demonstrate status messages
57
- */
58
- performAction(): void;
59
- /**
60
- * Cancel action
61
- */
62
- cancelAction(): void;
63
- /**
64
- * Example of synchronous global content access
65
- */
66
- getOkTextSync(): string;
67
- /**
68
- * Example of synchronous global content access using getText
69
- */
70
- getCancelTextSync(): string;
71
- static ɵfac: i0.ɵɵFactoryDeclaration<GlobalContentExampleComponent, never>;
72
- static ɵcmp: i0.ɵɵComponentDeclaration<GlobalContentExampleComponent, "val-global-content-example", never, {}, {}, never, never, true, never>;
73
- }
@@ -1,32 +0,0 @@
1
- /**
2
- * Example content configuration for demonstrating the reactive content system.
3
- *
4
- * This file shows how to structure content for components that use the reactive
5
- * language system with the new fromContent pattern.
6
- */
7
- import { TextContent } from '../../services/lang-provider/types';
8
- declare const _default: TextContent;
9
- export default _default;
10
- /**
11
- * Type-safe content keys for the example component.
12
- * Use this interface to ensure type safety when accessing content keys.
13
- */
14
- export interface ExampleComponentContentKeys {
15
- title: string;
16
- subtitle: string;
17
- description: string;
18
- greeting: string;
19
- welcomeMessage: string;
20
- dynamicText: string;
21
- buttonText: string;
22
- loadingText: string;
23
- errorMessage: string;
24
- successMessage: string;
25
- confirmDialog: string;
26
- cancelButton: string;
27
- acceptButton: string;
28
- }
29
- /**
30
- * Utility type for getting content keys with type safety.
31
- */
32
- export type ExampleContentKey = keyof ExampleComponentContentKeys;
@@ -1,47 +0,0 @@
1
- import { OnInit } from '@angular/core';
2
- import { Observable } from 'rxjs';
3
- import { ContentService } from '../../services/content.service';
4
- import { LangOption } from '../../services/lang-provider/types';
5
- import { TextMetadata } from '../atoms/text/types';
6
- import * as i0 from "@angular/core";
7
- export declare class ReactiveContentExampleComponent implements OnInit {
8
- content: ContentService;
9
- componentContent: {
10
- get: (key: string, options?: {
11
- fallback?: string;
12
- interpolation?: Record<string, string | number>;
13
- }) => Observable<string>;
14
- getWithInterpolation: (key: string, interpolation: Record<string, string | number>, fallback?: string) => Observable<string>;
15
- getMultiple: (keys: string[]) => Observable<Record<string, string>>;
16
- getText: (key: string, fallback?: string) => string;
17
- hasContent: (key: string) => boolean;
18
- };
19
- currentLanguage$: Observable<LangOption>;
20
- buttonText$: Observable<string>;
21
- saveButton$: Observable<string>;
22
- cancelButton$: Observable<string>;
23
- deleteButton$: Observable<string>;
24
- okButton$: Observable<string>;
25
- deleteConfirmationText$: Observable<string>;
26
- loadingText$: Observable<string>;
27
- showConfirmation: boolean;
28
- titleProps: TextMetadata;
29
- greetingProps: TextMetadata;
30
- multipleContent$: Observable<Record<string, string>>;
31
- constructor();
32
- ngOnInit(): void;
33
- /**
34
- * Toggle between Spanish and English
35
- */
36
- toggleLanguage(): void;
37
- /**
38
- * Show delete confirmation dialog
39
- */
40
- showDeleteConfirmation(): void;
41
- /**
42
- * Hide delete confirmation dialog
43
- */
44
- hideDeleteConfirmation(): void;
45
- static ɵfac: i0.ɵɵFactoryDeclaration<ReactiveContentExampleComponent, never>;
46
- static ɵcmp: i0.ɵɵComponentDeclaration<ReactiveContentExampleComponent, "val-reactive-content-example", never, {}, {}, never, never, true, never>;
47
- }
@@ -1,3 +0,0 @@
1
- import { TextContent } from '../types';
2
- declare const _default: TextContent;
3
- export default _default;