valtech-components 2.0.277 → 2.0.278
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/README.md +149 -6
- package/esm2022/lib/components/_examples/global-content-example-content.mjs +23 -0
- package/esm2022/lib/components/_examples/global-content-example.component.mjs +504 -0
- package/esm2022/lib/components/_examples/reactive-content-example-content.mjs +43 -0
- package/esm2022/lib/components/_examples/reactive-content-example.component.mjs +347 -0
- package/esm2022/lib/components/atoms/text/text.component.mjs +143 -15
- package/esm2022/lib/components/atoms/text/types.mjs +1 -1
- package/esm2022/lib/services/content.service.mjs +327 -0
- package/esm2022/lib/services/icons.service.mjs +3 -2
- package/esm2022/lib/services/lang-provider/content.mjs +136 -1
- package/esm2022/lib/services/lang-provider/lang-provider.service.mjs +118 -8
- package/esm2022/lib/shared/utils/content.mjs +186 -0
- package/esm2022/public-api.mjs +11 -5
- package/fesm2022/valtech-components.mjs +2938 -1359
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/_examples/global-content-example-content.d.ts +9 -0
- package/lib/components/_examples/global-content-example.component.d.ts +73 -0
- package/lib/components/_examples/reactive-content-example-content.d.ts +32 -0
- package/lib/components/_examples/reactive-content-example.component.d.ts +47 -0
- package/lib/components/atoms/text/text.component.d.ts +57 -8
- package/lib/components/atoms/text/types.d.ts +26 -5
- package/lib/services/content.service.d.ts +296 -0
- package/lib/services/lang-provider/lang-provider.service.d.ts +87 -7
- package/lib/shared/utils/content.d.ts +199 -0
- package/package.json +1 -1
- package/public-api.d.ts +9 -4
|
@@ -0,0 +1,347 @@
|
|
|
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,34 +1,162 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { AsyncPipe } from '@angular/common';
|
|
2
|
+
import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
|
|
2
3
|
import { IonText } from '@ionic/angular/standalone';
|
|
4
|
+
import { of, Subscription } from 'rxjs';
|
|
3
5
|
import * as i0 from "@angular/core";
|
|
6
|
+
import * as i1 from "../../../services/content.service";
|
|
4
7
|
/**
|
|
5
8
|
* val-text
|
|
6
9
|
*
|
|
7
|
-
*
|
|
10
|
+
* Enhanced text component that supports both static content and reactive content from the language service.
|
|
11
|
+
* The component automatically updates when the language changes if using reactive content.
|
|
8
12
|
*
|
|
9
|
-
* @example
|
|
10
|
-
*
|
|
13
|
+
* @example Static content:
|
|
14
|
+
* ```html
|
|
15
|
+
* <val-text [props]="{
|
|
16
|
+
* content: 'Static text',
|
|
17
|
+
* color: 'primary',
|
|
18
|
+
* size: 'medium',
|
|
19
|
+
* bold: false
|
|
20
|
+
* }"></val-text>
|
|
21
|
+
* ```
|
|
22
|
+
*
|
|
23
|
+
* @example Reactive content:
|
|
24
|
+
* ```html
|
|
25
|
+
* <val-text [props]="{
|
|
26
|
+
* contentKey: 'welcomeMessage',
|
|
27
|
+
* contentClass: 'HomeComponent',
|
|
28
|
+
* contentFallback: 'Welcome!',
|
|
29
|
+
* color: 'primary',
|
|
30
|
+
* size: 'large',
|
|
31
|
+
* bold: true
|
|
32
|
+
* }"></val-text>
|
|
33
|
+
* ```
|
|
34
|
+
*
|
|
35
|
+
* @example Reactive content with interpolation:
|
|
36
|
+
* ```html
|
|
37
|
+
* <val-text [props]="{
|
|
38
|
+
* contentKey: 'greeting',
|
|
39
|
+
* contentClass: 'UserComponent',
|
|
40
|
+
* contentInterpolation: { name: 'John', count: 5 },
|
|
41
|
+
* color: 'secondary',
|
|
42
|
+
* size: 'medium',
|
|
43
|
+
* bold: false
|
|
44
|
+
* }"></val-text>
|
|
45
|
+
* ```
|
|
11
46
|
*
|
|
12
|
-
* @
|
|
47
|
+
* @example Using ContentService helper:
|
|
48
|
+
* ```typescript
|
|
49
|
+
* // In component
|
|
50
|
+
* content = inject(ContentService);
|
|
51
|
+
* componentContent = this.content.forComponent('MyComponent');
|
|
52
|
+
*
|
|
53
|
+
* textProps = {
|
|
54
|
+
* content: this.componentContent.getText('title'), // sync
|
|
55
|
+
* color: 'primary',
|
|
56
|
+
* size: 'large',
|
|
57
|
+
* bold: true
|
|
58
|
+
* };
|
|
59
|
+
* // Or with reactive binding:
|
|
60
|
+
* title$ = this.componentContent.get('title');
|
|
61
|
+
* ```
|
|
62
|
+
*
|
|
63
|
+
* @input props: TextMetadata - Configuration for the text (content, styling, and reactive content options)
|
|
13
64
|
*/
|
|
14
65
|
export class TextComponent {
|
|
15
|
-
constructor() {
|
|
16
|
-
|
|
17
|
-
|
|
66
|
+
constructor(contentService) {
|
|
67
|
+
this.contentService = contentService;
|
|
68
|
+
this.subscription = new Subscription();
|
|
69
|
+
}
|
|
70
|
+
ngOnInit() {
|
|
71
|
+
this.setupDisplayContent();
|
|
72
|
+
}
|
|
73
|
+
ngOnDestroy() {
|
|
74
|
+
this.subscription.unsubscribe();
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Set up the content observable based on the props configuration.
|
|
78
|
+
* Priority: static content > reactive content with interpolation > reactive content
|
|
79
|
+
*/
|
|
80
|
+
setupDisplayContent() {
|
|
81
|
+
if (this.props.content) {
|
|
82
|
+
// Static content takes precedence
|
|
83
|
+
this.displayContent$ = of(this.props.content);
|
|
84
|
+
}
|
|
85
|
+
else if (this.props.contentKey && this.props.contentClass) {
|
|
86
|
+
// Reactive content from language service
|
|
87
|
+
if (this.props.contentInterpolation) {
|
|
88
|
+
// With interpolation
|
|
89
|
+
this.displayContent$ = this.contentService.fromContentWithInterpolation({
|
|
90
|
+
className: this.props.contentClass,
|
|
91
|
+
key: this.props.contentKey,
|
|
92
|
+
fallback: this.props.contentFallback,
|
|
93
|
+
interpolation: this.props.contentInterpolation,
|
|
94
|
+
});
|
|
95
|
+
}
|
|
96
|
+
else {
|
|
97
|
+
// Simple reactive content
|
|
98
|
+
this.displayContent$ = this.contentService.fromContent({
|
|
99
|
+
className: this.props.contentClass,
|
|
100
|
+
key: this.props.contentKey,
|
|
101
|
+
fallback: this.props.contentFallback,
|
|
102
|
+
});
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
else {
|
|
106
|
+
// Fallback to empty string if no valid content configuration
|
|
107
|
+
console.warn('val-text: No valid content configuration provided. Use either "content" for static text or "contentKey" + "contentClass" for reactive content.');
|
|
108
|
+
this.displayContent$ = of(this.props.contentFallback || '');
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TextComponent, deps: [{ token: i1.ContentService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
18
112
|
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: TextComponent, isStandalone: true, selector: "val-text", inputs: { props: "props" }, ngImport: i0, template: `
|
|
19
113
|
<ion-text [color]="props.color">
|
|
20
|
-
<p [class]="props.size" [class.bold]="props.bold">{{
|
|
114
|
+
<p [class]="props.size" [class.bold]="props.bold">{{ displayContent$ | async }}</p>
|
|
21
115
|
</ion-text>
|
|
22
|
-
`, isInline: true, styles: [":root{--ion-color-primary: #7026df;--ion-color-primary-rgb: 112, 38, 223;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #6321c4;--ion-color-primary-tint: #7e3ce2;--ion-color-secondary: #e2ccff;--ion-color-secondary-rgb: 226, 204, 255;--ion-color-secondary-contrast: #000000;--ion-color-secondary-contrast-rgb: 0, 0, 0;--ion-color-secondary-shade: #c7b4e0;--ion-color-secondary-tint: #e5d1ff;--ion-color-texti: #354c69;--ion-color-texti-rgb: 53, 76, 105;--ion-color-texti-contrast: #ffffff;--ion-color-texti-contrast-rgb: 255, 255, 255;--ion-color-texti-shade: #2f435c;--ion-color-texti-tint: #495e78;--ion-color-darki: #090f1b;--ion-color-darki-rgb: 9, 15, 27;--ion-color-darki-contrast: #ffffff;--ion-color-darki-contrast-rgb: 255, 255, 255;--ion-color-darki-shade: #080d18;--ion-color-darki-tint: #222732;--ion-color-medium: #9e9e9e;--ion-color-medium-rgb: 158, 158, 158;--ion-color-medium-contrast: #000000;--ion-color-medium-contrast-rgb: 0, 0, 0;--ion-color-medium-shade: #8b8b8b;--ion-color-medium-tint: #a8a8a8}@media (prefers-color-scheme: dark){:root{--ion-color-texti: #8fc1ff;--ion-color-texti-rgb: 143, 193, 255;--ion-color-texti-contrast: #000000;--ion-color-texti-contrast-rgb: 0, 0, 0;--ion-color-texti-shade: #7eaae0;--ion-color-texti-tint: #9ac7ff;--ion-color-darki: #ffffff;--ion-color-darki-rgb: 255, 255, 255;--ion-color-darki-contrast: #000000;--ion-color-darki-contrast-rgb: 0, 0, 0;--ion-color-darki-shade: #e0e0e0;--ion-color-darki-tint: #ffffff;--ion-color-primary: #8f49f8;--ion-color-primary-rgb: 143,73,248;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255,255,255;--ion-color-primary-shade: #7e40da;--ion-color-primary-tint: #9a5bf9}}.ion-color-texti{--ion-color-base: var(--ion-color-texti);--ion-color-base-rgb: var(--ion-color-texti-rgb);--ion-color-contrast: var(--ion-color-texti-contrast);--ion-color-contrast-rgb: var(--ion-color-texti-contrast-rgb);--ion-color-shade: var(--ion-color-texti-shade);--ion-color-tint: var(--ion-color-texti-tint)}.ion-color-darki{--ion-color-base: var(--ion-color-darki);--ion-color-base-rgb: var(--ion-color-darki-rgb);--ion-color-contrast: var(--ion-color-darki-contrast);--ion-color-contrast-rgb: var(--ion-color-darki-contrast-rgb);--ion-color-shade: var(--ion-color-darki-shade);--ion-color-tint: var(--ion-color-darki-tint)}.small{font-size:.75rem;line-height:1.25rem;font-weight:400}.small.bold{font-size:.75rem;line-height:1.25rem;font-weight:700}.medium{font-size:.875rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.medium{font-size:1rem;line-height:1.5rem}}.medium.bold{font-size:.875rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.medium.bold{font-size:1rem;line-height:1.5rem}}.large{font-size:1rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.large{font-size:1.125rem;line-height:1.5rem}}.large.bold{font-size:1rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.large.bold{font-size:1.125rem;line-height:1.5rem}}.xlarge{font-size:1.125rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.xlarge{font-size:1.5rem;line-height:2rem}}.xlarge.bold{font-size:1.125rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.xlarge.bold{font-size:1.5rem;line-height:2rem}}\n"], dependencies: [{ kind: "component", type: IonText, selector: "ion-text", inputs: ["color", "mode"] }] }); }
|
|
116
|
+
`, isInline: true, styles: [":root{--ion-color-primary: #7026df;--ion-color-primary-rgb: 112, 38, 223;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #6321c4;--ion-color-primary-tint: #7e3ce2;--ion-color-secondary: #e2ccff;--ion-color-secondary-rgb: 226, 204, 255;--ion-color-secondary-contrast: #000000;--ion-color-secondary-contrast-rgb: 0, 0, 0;--ion-color-secondary-shade: #c7b4e0;--ion-color-secondary-tint: #e5d1ff;--ion-color-texti: #354c69;--ion-color-texti-rgb: 53, 76, 105;--ion-color-texti-contrast: #ffffff;--ion-color-texti-contrast-rgb: 255, 255, 255;--ion-color-texti-shade: #2f435c;--ion-color-texti-tint: #495e78;--ion-color-darki: #090f1b;--ion-color-darki-rgb: 9, 15, 27;--ion-color-darki-contrast: #ffffff;--ion-color-darki-contrast-rgb: 255, 255, 255;--ion-color-darki-shade: #080d18;--ion-color-darki-tint: #222732;--ion-color-medium: #9e9e9e;--ion-color-medium-rgb: 158, 158, 158;--ion-color-medium-contrast: #000000;--ion-color-medium-contrast-rgb: 0, 0, 0;--ion-color-medium-shade: #8b8b8b;--ion-color-medium-tint: #a8a8a8}@media (prefers-color-scheme: dark){:root{--ion-color-texti: #8fc1ff;--ion-color-texti-rgb: 143, 193, 255;--ion-color-texti-contrast: #000000;--ion-color-texti-contrast-rgb: 0, 0, 0;--ion-color-texti-shade: #7eaae0;--ion-color-texti-tint: #9ac7ff;--ion-color-darki: #ffffff;--ion-color-darki-rgb: 255, 255, 255;--ion-color-darki-contrast: #000000;--ion-color-darki-contrast-rgb: 0, 0, 0;--ion-color-darki-shade: #e0e0e0;--ion-color-darki-tint: #ffffff;--ion-color-primary: #8f49f8;--ion-color-primary-rgb: 143,73,248;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255,255,255;--ion-color-primary-shade: #7e40da;--ion-color-primary-tint: #9a5bf9}}.ion-color-texti{--ion-color-base: var(--ion-color-texti);--ion-color-base-rgb: var(--ion-color-texti-rgb);--ion-color-contrast: var(--ion-color-texti-contrast);--ion-color-contrast-rgb: var(--ion-color-texti-contrast-rgb);--ion-color-shade: var(--ion-color-texti-shade);--ion-color-tint: var(--ion-color-texti-tint)}.ion-color-darki{--ion-color-base: var(--ion-color-darki);--ion-color-base-rgb: var(--ion-color-darki-rgb);--ion-color-contrast: var(--ion-color-darki-contrast);--ion-color-contrast-rgb: var(--ion-color-darki-contrast-rgb);--ion-color-shade: var(--ion-color-darki-shade);--ion-color-tint: var(--ion-color-darki-tint)}.small{font-size:.75rem;line-height:1.25rem;font-weight:400}.small.bold{font-size:.75rem;line-height:1.25rem;font-weight:700}.medium{font-size:.875rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.medium{font-size:1rem;line-height:1.5rem}}.medium.bold{font-size:.875rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.medium.bold{font-size:1rem;line-height:1.5rem}}.large{font-size:1rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.large{font-size:1.125rem;line-height:1.5rem}}.large.bold{font-size:1rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.large.bold{font-size:1.125rem;line-height:1.5rem}}.xlarge{font-size:1.125rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.xlarge{font-size:1.5rem;line-height:2rem}}.xlarge.bold{font-size:1.125rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.xlarge.bold{font-size:1.5rem;line-height:2rem}}\n"], dependencies: [{ kind: "component", type: IonText, selector: "ion-text", inputs: ["color", "mode"] }, { kind: "pipe", type: AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
23
117
|
}
|
|
24
118
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TextComponent, decorators: [{
|
|
25
119
|
type: Component,
|
|
26
|
-
args: [{ selector: 'val-text', standalone: true, imports: [IonText], template: `
|
|
120
|
+
args: [{ selector: 'val-text', standalone: true, imports: [IonText, AsyncPipe], template: `
|
|
27
121
|
<ion-text [color]="props.color">
|
|
28
|
-
<p [class]="props.size" [class.bold]="props.bold">{{
|
|
122
|
+
<p [class]="props.size" [class.bold]="props.bold">{{ displayContent$ | async }}</p>
|
|
29
123
|
</ion-text>
|
|
30
|
-
`, styles: [":root{--ion-color-primary: #7026df;--ion-color-primary-rgb: 112, 38, 223;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #6321c4;--ion-color-primary-tint: #7e3ce2;--ion-color-secondary: #e2ccff;--ion-color-secondary-rgb: 226, 204, 255;--ion-color-secondary-contrast: #000000;--ion-color-secondary-contrast-rgb: 0, 0, 0;--ion-color-secondary-shade: #c7b4e0;--ion-color-secondary-tint: #e5d1ff;--ion-color-texti: #354c69;--ion-color-texti-rgb: 53, 76, 105;--ion-color-texti-contrast: #ffffff;--ion-color-texti-contrast-rgb: 255, 255, 255;--ion-color-texti-shade: #2f435c;--ion-color-texti-tint: #495e78;--ion-color-darki: #090f1b;--ion-color-darki-rgb: 9, 15, 27;--ion-color-darki-contrast: #ffffff;--ion-color-darki-contrast-rgb: 255, 255, 255;--ion-color-darki-shade: #080d18;--ion-color-darki-tint: #222732;--ion-color-medium: #9e9e9e;--ion-color-medium-rgb: 158, 158, 158;--ion-color-medium-contrast: #000000;--ion-color-medium-contrast-rgb: 0, 0, 0;--ion-color-medium-shade: #8b8b8b;--ion-color-medium-tint: #a8a8a8}@media (prefers-color-scheme: dark){:root{--ion-color-texti: #8fc1ff;--ion-color-texti-rgb: 143, 193, 255;--ion-color-texti-contrast: #000000;--ion-color-texti-contrast-rgb: 0, 0, 0;--ion-color-texti-shade: #7eaae0;--ion-color-texti-tint: #9ac7ff;--ion-color-darki: #ffffff;--ion-color-darki-rgb: 255, 255, 255;--ion-color-darki-contrast: #000000;--ion-color-darki-contrast-rgb: 0, 0, 0;--ion-color-darki-shade: #e0e0e0;--ion-color-darki-tint: #ffffff;--ion-color-primary: #8f49f8;--ion-color-primary-rgb: 143,73,248;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255,255,255;--ion-color-primary-shade: #7e40da;--ion-color-primary-tint: #9a5bf9}}.ion-color-texti{--ion-color-base: var(--ion-color-texti);--ion-color-base-rgb: var(--ion-color-texti-rgb);--ion-color-contrast: var(--ion-color-texti-contrast);--ion-color-contrast-rgb: var(--ion-color-texti-contrast-rgb);--ion-color-shade: var(--ion-color-texti-shade);--ion-color-tint: var(--ion-color-texti-tint)}.ion-color-darki{--ion-color-base: var(--ion-color-darki);--ion-color-base-rgb: var(--ion-color-darki-rgb);--ion-color-contrast: var(--ion-color-darki-contrast);--ion-color-contrast-rgb: var(--ion-color-darki-contrast-rgb);--ion-color-shade: var(--ion-color-darki-shade);--ion-color-tint: var(--ion-color-darki-tint)}.small{font-size:.75rem;line-height:1.25rem;font-weight:400}.small.bold{font-size:.75rem;line-height:1.25rem;font-weight:700}.medium{font-size:.875rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.medium{font-size:1rem;line-height:1.5rem}}.medium.bold{font-size:.875rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.medium.bold{font-size:1rem;line-height:1.5rem}}.large{font-size:1rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.large{font-size:1.125rem;line-height:1.5rem}}.large.bold{font-size:1rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.large.bold{font-size:1.125rem;line-height:1.5rem}}.xlarge{font-size:1.125rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.xlarge{font-size:1.5rem;line-height:2rem}}.xlarge.bold{font-size:1.125rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.xlarge.bold{font-size:1.5rem;line-height:2rem}}\n"] }]
|
|
31
|
-
}], ctorParameters: () => [], propDecorators: { props: [{
|
|
124
|
+
`, changeDetection: ChangeDetectionStrategy.OnPush, styles: [":root{--ion-color-primary: #7026df;--ion-color-primary-rgb: 112, 38, 223;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #6321c4;--ion-color-primary-tint: #7e3ce2;--ion-color-secondary: #e2ccff;--ion-color-secondary-rgb: 226, 204, 255;--ion-color-secondary-contrast: #000000;--ion-color-secondary-contrast-rgb: 0, 0, 0;--ion-color-secondary-shade: #c7b4e0;--ion-color-secondary-tint: #e5d1ff;--ion-color-texti: #354c69;--ion-color-texti-rgb: 53, 76, 105;--ion-color-texti-contrast: #ffffff;--ion-color-texti-contrast-rgb: 255, 255, 255;--ion-color-texti-shade: #2f435c;--ion-color-texti-tint: #495e78;--ion-color-darki: #090f1b;--ion-color-darki-rgb: 9, 15, 27;--ion-color-darki-contrast: #ffffff;--ion-color-darki-contrast-rgb: 255, 255, 255;--ion-color-darki-shade: #080d18;--ion-color-darki-tint: #222732;--ion-color-medium: #9e9e9e;--ion-color-medium-rgb: 158, 158, 158;--ion-color-medium-contrast: #000000;--ion-color-medium-contrast-rgb: 0, 0, 0;--ion-color-medium-shade: #8b8b8b;--ion-color-medium-tint: #a8a8a8}@media (prefers-color-scheme: dark){:root{--ion-color-texti: #8fc1ff;--ion-color-texti-rgb: 143, 193, 255;--ion-color-texti-contrast: #000000;--ion-color-texti-contrast-rgb: 0, 0, 0;--ion-color-texti-shade: #7eaae0;--ion-color-texti-tint: #9ac7ff;--ion-color-darki: #ffffff;--ion-color-darki-rgb: 255, 255, 255;--ion-color-darki-contrast: #000000;--ion-color-darki-contrast-rgb: 0, 0, 0;--ion-color-darki-shade: #e0e0e0;--ion-color-darki-tint: #ffffff;--ion-color-primary: #8f49f8;--ion-color-primary-rgb: 143,73,248;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255,255,255;--ion-color-primary-shade: #7e40da;--ion-color-primary-tint: #9a5bf9}}.ion-color-texti{--ion-color-base: var(--ion-color-texti);--ion-color-base-rgb: var(--ion-color-texti-rgb);--ion-color-contrast: var(--ion-color-texti-contrast);--ion-color-contrast-rgb: var(--ion-color-texti-contrast-rgb);--ion-color-shade: var(--ion-color-texti-shade);--ion-color-tint: var(--ion-color-texti-tint)}.ion-color-darki{--ion-color-base: var(--ion-color-darki);--ion-color-base-rgb: var(--ion-color-darki-rgb);--ion-color-contrast: var(--ion-color-darki-contrast);--ion-color-contrast-rgb: var(--ion-color-darki-contrast-rgb);--ion-color-shade: var(--ion-color-darki-shade);--ion-color-tint: var(--ion-color-darki-tint)}.small{font-size:.75rem;line-height:1.25rem;font-weight:400}.small.bold{font-size:.75rem;line-height:1.25rem;font-weight:700}.medium{font-size:.875rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.medium{font-size:1rem;line-height:1.5rem}}.medium.bold{font-size:.875rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.medium.bold{font-size:1rem;line-height:1.5rem}}.large{font-size:1rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.large{font-size:1.125rem;line-height:1.5rem}}.large.bold{font-size:1rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.large.bold{font-size:1.125rem;line-height:1.5rem}}.xlarge{font-size:1.125rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.xlarge{font-size:1.5rem;line-height:2rem}}.xlarge.bold{font-size:1.125rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.xlarge.bold{font-size:1.5rem;line-height:2rem}}\n"] }]
|
|
125
|
+
}], ctorParameters: () => [{ type: i1.ContentService }], propDecorators: { props: [{
|
|
32
126
|
type: Input
|
|
33
127
|
}] } });
|
|
34
|
-
|
|
128
|
+
/**
|
|
129
|
+
* Helper function to create reactive text props from content configuration.
|
|
130
|
+
* This provides a convenient way to create val-text props with reactive content.
|
|
131
|
+
*
|
|
132
|
+
* @param contentConfig - Content configuration
|
|
133
|
+
* @param styleConfig - Optional style configuration
|
|
134
|
+
* @returns Partial TextMetadata with content properties set
|
|
135
|
+
*
|
|
136
|
+
* @example
|
|
137
|
+
* ```typescript
|
|
138
|
+
* // In component
|
|
139
|
+
* titleProps: TextMetadata = {
|
|
140
|
+
* ...createTextProps({
|
|
141
|
+
* contentKey: 'title',
|
|
142
|
+
* contentClass: 'HeaderComponent'
|
|
143
|
+
* }, {
|
|
144
|
+
* color: 'primary',
|
|
145
|
+
* size: 'large',
|
|
146
|
+
* bold: true
|
|
147
|
+
* })
|
|
148
|
+
* };
|
|
149
|
+
* ```
|
|
150
|
+
*/
|
|
151
|
+
export function createTextProps(contentConfig, styleConfig = {}) {
|
|
152
|
+
return {
|
|
153
|
+
contentKey: contentConfig.contentKey,
|
|
154
|
+
contentClass: contentConfig.contentClass,
|
|
155
|
+
contentFallback: contentConfig.contentFallback,
|
|
156
|
+
contentInterpolation: contentConfig.contentInterpolation,
|
|
157
|
+
color: styleConfig.color || 'dark',
|
|
158
|
+
size: styleConfig.size || 'medium',
|
|
159
|
+
bold: styleConfig.bold || false,
|
|
160
|
+
};
|
|
161
|
+
}
|
|
162
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"text.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/atoms/text/text.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { AsyncPipe } from '@angular/common';\nimport { ChangeDetectionStrategy, Component, Input, OnDestroy, OnInit } from '@angular/core';\nimport { IonText } from '@ionic/angular/standalone';\nimport { Observable, of, Subscription } from 'rxjs';\nimport { ContentService } from '../../../services/content.service';\nimport { TextContentConfig, TextMetadata } from './types';\n\n@Component({\n  selector: 'val-text',\n  standalone: true,\n  imports: [IonText, AsyncPipe],\n  template: `\n    <ion-text [color]=\"props.color\">\n      <p [class]=\"props.size\" [class.bold]=\"props.bold\">{{ displayContent$ | async }}</p>\n    </ion-text>\n  `,\n  styleUrls: ['./text.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\n/**\n * val-text\n *\n * Enhanced text component that supports both static content and reactive content from the language service.\n * The component automatically updates when the language changes if using reactive content.\n *\n * @example Static content:\n * ```html\n * <val-text [props]=\"{\n *   content: 'Static text',\n *   color: 'primary',\n *   size: 'medium',\n *   bold: false\n * }\"></val-text>\n * ```\n *\n * @example Reactive content:\n * ```html\n * <val-text [props]=\"{\n *   contentKey: 'welcomeMessage',\n *   contentClass: 'HomeComponent',\n *   contentFallback: 'Welcome!',\n *   color: 'primary',\n *   size: 'large',\n *   bold: true\n * }\"></val-text>\n * ```\n *\n * @example Reactive content with interpolation:\n * ```html\n * <val-text [props]=\"{\n *   contentKey: 'greeting',\n *   contentClass: 'UserComponent',\n *   contentInterpolation: { name: 'John', count: 5 },\n *   color: 'secondary',\n *   size: 'medium',\n *   bold: false\n * }\"></val-text>\n * ```\n *\n * @example Using ContentService helper:\n * ```typescript\n * // In component\n * content = inject(ContentService);\n * componentContent = this.content.forComponent('MyComponent');\n *\n * textProps = {\n *   content: this.componentContent.getText('title'), // sync\n *   color: 'primary',\n *   size: 'large',\n *   bold: true\n * };\n * // Or with reactive binding:\n * title$ = this.componentContent.get('title');\n * ```\n *\n * @input props: TextMetadata - Configuration for the text (content, styling, and reactive content options)\n */\nexport class TextComponent implements OnInit, OnDestroy {\n  /**\n   * Text configuration object.\n   * @type {TextMetadata}\n   *\n   * For static content:\n   * @property content - The text to display (takes precedence over reactive content)\n   *\n   * For reactive content:\n   * @property contentKey - The content key to retrieve from language service\n   * @property contentClass - The component class name for content lookup\n   * @property contentFallback - Optional fallback text if content is not found\n   * @property contentInterpolation - Optional values to interpolate into content\n   *\n   * For styling:\n   * @property color - The text color (Ionic color string)\n   * @property size - The text size ('small' | 'medium' | 'large' | 'xlarge')\n   * @property bold - Whether the text is bold\n   */\n  @Input()\n  props: TextMetadata;\n\n  /**\n   * Observable that provides the content to display.\n   * This will be either static content or reactive content from the language service.\n   */\n  displayContent$: Observable<string>;\n\n  private subscription = new Subscription();\n\n  constructor(private contentService: ContentService) {}\n\n  ngOnInit() {\n    this.setupDisplayContent();\n  }\n\n  ngOnDestroy() {\n    this.subscription.unsubscribe();\n  }\n\n  /**\n   * Set up the content observable based on the props configuration.\n   * Priority: static content > reactive content with interpolation > reactive content\n   */\n  private setupDisplayContent(): void {\n    if (this.props.content) {\n      // Static content takes precedence\n      this.displayContent$ = of(this.props.content);\n    } else if (this.props.contentKey && this.props.contentClass) {\n      // Reactive content from language service\n      if (this.props.contentInterpolation) {\n        // With interpolation\n        this.displayContent$ = this.contentService.fromContentWithInterpolation({\n          className: this.props.contentClass,\n          key: this.props.contentKey,\n          fallback: this.props.contentFallback,\n          interpolation: this.props.contentInterpolation,\n        });\n      } else {\n        // Simple reactive content\n        this.displayContent$ = this.contentService.fromContent({\n          className: this.props.contentClass,\n          key: this.props.contentKey,\n          fallback: this.props.contentFallback,\n        });\n      }\n    } else {\n      // Fallback to empty string if no valid content configuration\n      console.warn(\n        'val-text: No valid content configuration provided. Use either \"content\" for static text or \"contentKey\" + \"contentClass\" for reactive content.'\n      );\n      this.displayContent$ = of(this.props.contentFallback || '');\n    }\n  }\n}\n\n/**\n * Helper function to create reactive text props from content configuration.\n * This provides a convenient way to create val-text props with reactive content.\n *\n * @param contentConfig - Content configuration\n * @param styleConfig - Optional style configuration\n * @returns Partial TextMetadata with content properties set\n *\n * @example\n * ```typescript\n * // In component\n * titleProps: TextMetadata = {\n *   ...createTextProps({\n *     contentKey: 'title',\n *     contentClass: 'HeaderComponent'\n *   }, {\n *     color: 'primary',\n *     size: 'large',\n *     bold: true\n *   })\n * };\n * ```\n */\nexport function createTextProps(\n  contentConfig: TextContentConfig,\n  styleConfig: Partial<Pick<TextMetadata, 'color' | 'size' | 'bold'>> = {}\n): Partial<TextMetadata> {\n  return {\n    contentKey: contentConfig.contentKey,\n    contentClass: contentConfig.contentClass,\n    contentFallback: contentConfig.contentFallback,\n    contentInterpolation: contentConfig.contentInterpolation,\n    color: styleConfig.color || 'dark',\n    size: styleConfig.size || 'medium',\n    bold: styleConfig.bold || false,\n  };\n}\n"]}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy92YWx0ZWNoLWNvbXBvbmVudHMvc3JjL2xpYi9jb21wb25lbnRzL2F0b21zL3RleHQvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbG9yIH0gZnJvbSAnQGlvbmljL2NvcmUnO1xuXG4vKipcbiAqIFByb3BzIGZvciB2YWwtdGV4dCBjb21wb25lbnQuXG4gKlxuICogVGhlIGNvbXBvbmVudCBzdXBwb3J0cyBib3RoIHN0YXRpYyBjb250ZW50IGFuZCByZWFjdGl2ZSBjb250ZW50IGZyb20gdGhlIGxhbmd1YWdlIHNlcnZpY2UuXG4gKiBVc2UgZWl0aGVyIGBjb250ZW50YCBmb3Igc3RhdGljIHRleHQgb3IgYGNvbnRlbnRLZXlgICsgYGNvbnRlbnRDbGFzc2AgZm9yIHJlYWN0aXZlIHRleHQuXG4gKlxuICogQHByb3BlcnR5IGNvbnRlbnQgLSBTdGF0aWMgdGV4dCB0byBkaXNwbGF5ICh0YWtlcyBwcmVjZWRlbmNlIG92ZXIgY29udGVudEtleSlcbiAqIEBwcm9wZXJ0eSBjb250ZW50S2V5IC0gS2V5IGZvciBkeW5hbWljIGNvbnRlbnQgZnJvbSBsYW5ndWFnZSBzZXJ2aWNlXG4gKiBAcHJvcGVydHkgY29udGVudENsYXNzIC0gQ29tcG9uZW50IGNsYXNzIG5hbWUgZm9yIGNvbnRlbnQgbG9va3VwIChyZXF1aXJlZCB3aXRoIGNvbnRlbnRLZXkpXG4gKiBAcHJvcGVydHkgY29udGVudEZhbGxiYWNrIC0gRmFsbGJhY2sgdGV4dCBpZiBjb250ZW50S2V5IGlzIG5vdCBmb3VuZFxuICogQHByb3BlcnR5IGNvbnRlbnRJbnRlcnBvbGF0aW9uIC0gVmFsdWVzIHRvIGludGVycG9sYXRlIGludG8gdGhlIGNvbnRlbnQgc3RyaW5nXG4gKiBAcHJvcGVydHkgY29sb3IgLSBUaGUgdGV4dCBjb2xvciAoSW9uaWMgY29sb3Igc3RyaW5nKVxuICogQHByb3BlcnR5IHNpemUgLSBUaGUgdGV4dCBzaXplICgnc21hbGwnIHwgJ21lZGl1bScgfCAnbGFyZ2UnIHwgJ3hsYXJnZScpXG4gKiBAcHJvcGVydHkgYm9sZCAtIFdoZXRoZXIgdGhlIHRleHQgaXMgYm9sZFxuICovXG5leHBvcnQgaW50ZXJmYWNlIFRleHRNZXRhZGF0YSB7XG4gIHNpemU6ICdzbWFsbCcgfCAnbWVkaXVtJyB8ICdsYXJnZScgfCAneGxhcmdlJztcbiAgY29sb3I6IENvbG9yO1xuICBjb250ZW50Pzogc3RyaW5nO1xuICBib2xkOiBib29sZWFuO1xuXG4gIC8vIFJlYWN0aXZlIGNvbnRlbnQgcHJvcGVydGllc1xuICBjb250ZW50S2V5Pzogc3RyaW5nO1xuICBjb250ZW50Q2xhc3M/OiBzdHJpbmc7XG4gIGNvbnRlbnRGYWxsYmFjaz86IHN0cmluZztcbiAgY29udGVudEludGVycG9sYXRpb24/OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmcgfCBudW1iZXI+O1xufVxuXG4vKipcbiAqIENvbmZpZ3VyYXRpb24gZm9yIHJlYWN0aXZlIGNvbnRlbnQgaW4gdmFsLXRleHQgY29tcG9uZW50LlxuICogVXNlIHRoaXMgaW50ZXJmYWNlIHdoZW4geW91IG9ubHkgbmVlZCB0byBzcGVjaWZ5IGNvbnRlbnQtcmVsYXRlZCBwcm9wZXJ0aWVzLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFRleHRDb250ZW50Q29uZmlnIHtcbiAgY29udGVudEtleTogc3RyaW5nO1xuICBjb250ZW50Q2xhc3M6IHN0cmluZztcbiAgY29udGVudEZhbGxiYWNrPzogc3RyaW5nO1xuICBjb250ZW50SW50ZXJwb2xhdGlvbj86IFJlY29yZDxzdHJpbmcsIHN0cmluZyB8IG51bWJlcj47XG59XG4iXX0=
|