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,504 +0,0 @@
1
- import { CommonModule } from '@angular/common';
2
- import { Component, inject } from '@angular/core';
3
- import { FormsModule } from '@angular/forms';
4
- import { IonButton, IonCard, IonCardContent, IonCardHeader, IonCardTitle, IonInput, IonItem, IonLabel, IonTextarea, } from '@ionic/angular/standalone';
5
- import { ContentService } from '../../services/content.service';
6
- import { TextComponent } from '../atoms/text/text.component';
7
- import * as i0 from "@angular/core";
8
- import * as i1 from "@angular/common";
9
- import * as i2 from "@angular/forms";
10
- /**
11
- * Comprehensive example demonstrating global content usage.
12
- *
13
- * This component shows:
14
- * 1. Global button texts
15
- * 2. Global state messages
16
- * 3. Global confirmations with interpolation
17
- * 4. Mixing global and component-specific content
18
- * 5. Language switching
19
- */
20
- export class GlobalContentExampleComponent {
21
- constructor() {
22
- this.content = inject(ContentService);
23
- this.componentContent = this.content.forComponent('GlobalContentExample');
24
- // Form data
25
- this.formData = {
26
- name: '',
27
- email: '',
28
- message: '',
29
- };
30
- // Action state
31
- this.actionPerformed = false;
32
- // Component-specific content properties
33
- this.titleProps = {
34
- contentKey: 'title',
35
- contentClass: 'GlobalContentExample',
36
- contentFallback: 'Global Content Example',
37
- color: 'primary',
38
- size: 'xlarge',
39
- bold: true,
40
- };
41
- this.languageLabel = {
42
- contentKey: 'languageLabel',
43
- contentClass: 'GlobalContentExample',
44
- contentFallback: 'Language:',
45
- color: 'medium',
46
- size: 'medium',
47
- bold: true,
48
- };
49
- this.componentDescriptionProps = {
50
- contentKey: 'description',
51
- contentClass: 'GlobalContentExample',
52
- contentFallback: 'This example demonstrates global and component-specific content.',
53
- color: 'dark',
54
- size: 'medium',
55
- bold: false,
56
- };
57
- // Global button content (reactive)
58
- this.saveButton$ = this.content.fromContent({ key: 'save' });
59
- this.editButton$ = this.content.fromContent({ key: 'edit' });
60
- this.deleteButton$ = this.content.fromContent({ key: 'delete' });
61
- this.cancelButton$ = this.content.fromContent({ key: 'cancel' });
62
- this.backButton$ = this.content.fromContent({ key: 'back' });
63
- this.nextButton$ = this.content.fromContent({ key: 'next' });
64
- this.okButton$ = this.content.fromContent({ key: 'ok' });
65
- this.closeButton$ = this.content.fromContent({ key: 'close' });
66
- // Global action content
67
- this.addButton$ = this.content.fromContent({ key: 'add' });
68
- this.removeButton$ = this.content.fromContent({ key: 'remove' });
69
- this.searchButton$ = this.content.fromContent({ key: 'search' });
70
- this.filterButton$ = this.content.fromContent({ key: 'filter' });
71
- this.sortButton$ = this.content.fromContent({ key: 'sort' });
72
- this.refreshButton$ = this.content.fromContent({ key: 'refresh' });
73
- // Global state content
74
- this.loadingText$ = this.content.fromContent({ key: 'loading' });
75
- this.successText$ = this.content.fromContent({ key: 'success' });
76
- this.errorText$ = this.content.fromContent({ key: 'error' });
77
- this.warningText$ = this.content.fromContent({ key: 'warning' });
78
- this.infoText$ = this.content.fromContent({ key: 'info' });
79
- this.noDataText$ = this.content.fromContent({ key: 'noData' });
80
- // Global form content
81
- this.requiredText$ = this.content.fromContent({ key: 'required' });
82
- this.optionalText$ = this.content.fromContent({ key: 'optional' });
83
- this.searchPlaceholder$ = this.content.fromContent({ key: 'searchPlaceholder' });
84
- // Global confirmation content with interpolation
85
- this.deleteConfirmation$ = this.content.fromContentWithInterpolation({
86
- key: 'deleteConfirmation',
87
- interpolation: { itemName: 'este registro' },
88
- });
89
- this.areYouSureText$ = this.content.fromContent({ key: 'areYouSure' });
90
- this.unsavedChangesText$ = this.content.fromContent({ key: 'unsavedChanges' });
91
- }
92
- /**
93
- * Set language
94
- */
95
- setLanguage(lang) {
96
- this.content.setLang(lang);
97
- }
98
- /**
99
- * Perform an action to demonstrate status messages
100
- */
101
- performAction() {
102
- this.actionPerformed = true;
103
- setTimeout(() => {
104
- this.actionPerformed = false;
105
- }, 3000);
106
- }
107
- /**
108
- * Cancel action
109
- */
110
- cancelAction() {
111
- this.formData = { name: '', email: '', message: '' };
112
- this.actionPerformed = false;
113
- }
114
- /**
115
- * Example of synchronous global content access
116
- */
117
- getOkTextSync() {
118
- return this.content.getGlobalText('ok');
119
- }
120
- /**
121
- * Example of synchronous global content access using getText
122
- */
123
- getCancelTextSync() {
124
- return this.content.getText('cancel');
125
- }
126
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: GlobalContentExampleComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
127
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: GlobalContentExampleComponent, isStandalone: true, selector: "val-global-content-example", ngImport: i0, template: `
128
- <ion-card>
129
- <ion-card-header>
130
- <ion-card-title>
131
- <val-text [props]="titleProps"></val-text>
132
- </ion-card-title>
133
- </ion-card-header>
134
-
135
- <ion-card-content>
136
- <!-- Language switcher -->
137
- <div class="language-section">
138
- <val-text [props]="languageLabel"></val-text>
139
- <ion-button fill="outline" size="small" (click)="setLanguage('es')"> Español </ion-button>
140
- <ion-button fill="outline" size="small" (click)="setLanguage('en')"> English </ion-button>
141
- </div>
142
-
143
- <div class="divider"></div>
144
-
145
- <!-- Global buttons demonstration -->
146
- <div class="section">
147
- <h3>Global Buttons</h3>
148
- <div class="button-grid">
149
- <ion-button color="primary">{{ saveButton$ | async }}</ion-button>
150
- <ion-button color="secondary">{{ editButton$ | async }}</ion-button>
151
- <ion-button color="danger">{{ deleteButton$ | async }}</ion-button>
152
- <ion-button fill="outline">{{ cancelButton$ | async }}</ion-button>
153
- <ion-button fill="outline">{{ backButton$ | async }}</ion-button>
154
- <ion-button fill="outline">{{ nextButton$ | async }}</ion-button>
155
- <ion-button color="success">{{ okButton$ | async }}</ion-button>
156
- <ion-button fill="clear">{{ closeButton$ | async }}</ion-button>
157
- </div>
158
- </div>
159
-
160
- <div class="divider"></div>
161
-
162
- <!-- Global actions demonstration -->
163
- <div class="section">
164
- <h3>Global Actions</h3>
165
- <div class="button-grid">
166
- <ion-button size="small">{{ addButton$ | async }}</ion-button>
167
- <ion-button size="small">{{ removeButton$ | async }}</ion-button>
168
- <ion-button size="small">{{ searchButton$ | async }}</ion-button>
169
- <ion-button size="small">{{ filterButton$ | async }}</ion-button>
170
- <ion-button size="small">{{ sortButton$ | async }}</ion-button>
171
- <ion-button size="small">{{ refreshButton$ | async }}</ion-button>
172
- </div>
173
- </div>
174
-
175
- <div class="divider"></div>
176
-
177
- <!-- Global states demonstration -->
178
- <div class="section">
179
- <h3>Global States</h3>
180
- <div class="state-messages">
181
- <div class="state-item loading">
182
- <val-text
183
- [props]="{ content: loadingText$ | async, color: 'medium', size: 'small', bold: false }"
184
- ></val-text>
185
- </div>
186
- <div class="state-item success">
187
- <val-text
188
- [props]="{ content: successText$ | async, color: 'success', size: 'small', bold: false }"
189
- ></val-text>
190
- </div>
191
- <div class="state-item error">
192
- <val-text
193
- [props]="{ content: errorText$ | async, color: 'danger', size: 'small', bold: false }"
194
- ></val-text>
195
- </div>
196
- <div class="state-item warning">
197
- <val-text
198
- [props]="{ content: warningText$ | async, color: 'warning', size: 'small', bold: false }"
199
- ></val-text>
200
- </div>
201
- <div class="state-item info">
202
- <val-text
203
- [props]="{ content: infoText$ | async, color: 'primary', size: 'small', bold: false }"
204
- ></val-text>
205
- </div>
206
- <div class="state-item no-data">
207
- <val-text
208
- [props]="{ content: noDataText$ | async, color: 'medium', size: 'small', bold: false }"
209
- ></val-text>
210
- </div>
211
- </div>
212
- </div>
213
-
214
- <div class="divider"></div>
215
-
216
- <!-- Form with global placeholders and labels -->
217
- <div class="section">
218
- <h3>Form with Global Content</h3>
219
- <form class="example-form">
220
- <ion-item>
221
- <ion-label position="stacked">Name ({{ requiredText$ | async }})</ion-label>
222
- <ion-input [(ngModel)]="formData.name" [placeholder]="'Enter your name' + '...'" name="name"> </ion-input>
223
- </ion-item>
224
-
225
- <ion-item>
226
- <ion-label position="stacked">Email ({{ optionalText$ | async }})</ion-label>
227
- <ion-input [(ngModel)]="formData.email" [placeholder]="'Enter your email' + '...'" name="email">
228
- </ion-input>
229
- </ion-item>
230
-
231
- <ion-item>
232
- <ion-label position="stacked">Message</ion-label>
233
- <ion-textarea [(ngModel)]="formData.message" [placeholder]="searchPlaceholder$ | async" name="message">
234
- </ion-textarea>
235
- </ion-item>
236
- </form>
237
- </div>
238
-
239
- <div class="divider"></div>
240
-
241
- <!-- Global confirmations with interpolation -->
242
- <div class="section">
243
- <h3>Confirmations with Interpolation</h3>
244
- <div class="confirmation-examples">
245
- <p><strong>Delete Confirmation:</strong></p>
246
- <val-text
247
- [props]="{ content: deleteConfirmation$ | async, color: 'danger', size: 'medium', bold: false }"
248
- ></val-text>
249
-
250
- <br /><br />
251
- <p><strong>General Confirmation:</strong></p>
252
- <val-text
253
- [props]="{ content: areYouSureText$ | async, color: 'warning', size: 'medium', bold: false }"
254
- ></val-text>
255
-
256
- <br /><br />
257
- <p><strong>Unsaved Changes:</strong></p>
258
- <val-text
259
- [props]="{ content: unsavedChangesText$ | async, color: 'medium', size: 'medium', bold: false }"
260
- ></val-text>
261
- </div>
262
- </div>
263
-
264
- <div class="divider"></div>
265
-
266
- <!-- Mixed content: Global + Component specific -->
267
- <div class="section">
268
- <h3>Mixed Content Example</h3>
269
- <div class="mixed-content">
270
- <!-- Component-specific content -->
271
- <val-text [props]="componentDescriptionProps"></val-text>
272
-
273
- <br /><br />
274
-
275
- <!-- Action buttons mixing global and component content -->
276
- <div class="action-buttons">
277
- <ion-button color="primary" (click)="performAction()">
278
- {{ saveButton$ | async }}
279
- </ion-button>
280
- <ion-button fill="outline" (click)="cancelAction()">
281
- {{ cancelButton$ | async }}
282
- </ion-button>
283
- </div>
284
-
285
- <!-- Status message using global content -->
286
- <div *ngIf="actionPerformed" class="status-message">
287
- <val-text
288
- [props]="{ content: successText$ | async, color: 'success', size: 'small', bold: false }"
289
- ></val-text>
290
- </div>
291
- </div>
292
- </div>
293
-
294
- <!-- Demo of synchronous global content access -->
295
- <div class="section">
296
- <h3>Synchronous Access Example</h3>
297
- <p>
298
- OK button text (sync): <strong>{{ getOkTextSync() }}</strong>
299
- </p>
300
- <p>
301
- Cancel button text (sync): <strong>{{ getCancelTextSync() }}</strong>
302
- </p>
303
- </div>
304
- </ion-card-content>
305
- </ion-card>
306
- `, isInline: true, styles: [".language-section{display:flex;align-items:center;gap:8px;margin-bottom:16px}.divider{height:1px;background-color:var(--ion-color-light);margin:16px 0}.section{margin-bottom:16px}.section h3{margin:0 0 12px;color:var(--ion-color-primary)}.button-grid{display:grid;grid-template-columns:repeat(auto-fit,minmax(120px,1fr));gap:8px}.state-messages{display:flex;flex-direction:column;gap:8px}.state-item{padding:8px;border-radius:4px;background-color:var(--ion-color-light-shade)}.state-item.loading{background-color:var(--ion-color-medium-tint)}.state-item.success{background-color:var(--ion-color-success-tint)}.state-item.error{background-color:var(--ion-color-danger-tint)}.state-item.warning{background-color:var(--ion-color-warning-tint)}.state-item.info{background-color:var(--ion-color-primary-tint)}.example-form{margin-top:12px}.confirmation-examples{padding:12px;background-color:var(--ion-color-light-tint);border-radius:8px}.mixed-content{padding:12px;background-color:var(--ion-color-light-shade);border-radius:8px}.action-buttons{display:flex;gap:8px;margin:12px 0}.status-message{padding:8px;background-color:var(--ion-color-success-tint);border-radius:4px;margin-top:8px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i2.NgForm, selector: "form:not([ngNoForm]):not([formGroup]),ng-form,[ngForm]", inputs: ["ngFormOptions"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { 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: IonItem, selector: "ion-item", inputs: ["button", "color", "detail", "detailIcon", "disabled", "download", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "component", type: IonInput, selector: "ion-input", inputs: ["accept", "autocapitalize", "autocomplete", "autocorrect", "autofocus", "clearInput", "clearOnEdit", "color", "counter", "counterFormatter", "debounce", "disabled", "enterkeyhint", "errorText", "fill", "helperText", "inputmode", "label", "labelPlacement", "max", "maxlength", "min", "minlength", "mode", "multiple", "name", "pattern", "placeholder", "readonly", "required", "shape", "size", "spellcheck", "step", "type", "value"] }, { kind: "component", type: IonTextarea, selector: "ion-textarea", inputs: ["autoGrow", "autocapitalize", "autofocus", "clearOnEdit", "color", "cols", "counter", "counterFormatter", "debounce", "disabled", "enterkeyhint", "errorText", "fill", "helperText", "inputmode", "label", "labelPlacement", "maxlength", "minlength", "mode", "name", "placeholder", "readonly", "required", "rows", "shape", "spellcheck", "value", "wrap"] }, { kind: "component", type: TextComponent, selector: "val-text", inputs: ["props"] }] }); }
307
- }
308
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: GlobalContentExampleComponent, decorators: [{
309
- type: Component,
310
- args: [{ selector: 'val-global-content-example', standalone: true, imports: [
311
- CommonModule,
312
- FormsModule,
313
- IonCard,
314
- IonCardContent,
315
- IonCardHeader,
316
- IonCardTitle,
317
- IonButton,
318
- IonItem,
319
- IonLabel,
320
- IonInput,
321
- IonTextarea,
322
- TextComponent,
323
- ], template: `
324
- <ion-card>
325
- <ion-card-header>
326
- <ion-card-title>
327
- <val-text [props]="titleProps"></val-text>
328
- </ion-card-title>
329
- </ion-card-header>
330
-
331
- <ion-card-content>
332
- <!-- Language switcher -->
333
- <div class="language-section">
334
- <val-text [props]="languageLabel"></val-text>
335
- <ion-button fill="outline" size="small" (click)="setLanguage('es')"> Español </ion-button>
336
- <ion-button fill="outline" size="small" (click)="setLanguage('en')"> English </ion-button>
337
- </div>
338
-
339
- <div class="divider"></div>
340
-
341
- <!-- Global buttons demonstration -->
342
- <div class="section">
343
- <h3>Global Buttons</h3>
344
- <div class="button-grid">
345
- <ion-button color="primary">{{ saveButton$ | async }}</ion-button>
346
- <ion-button color="secondary">{{ editButton$ | async }}</ion-button>
347
- <ion-button color="danger">{{ deleteButton$ | async }}</ion-button>
348
- <ion-button fill="outline">{{ cancelButton$ | async }}</ion-button>
349
- <ion-button fill="outline">{{ backButton$ | async }}</ion-button>
350
- <ion-button fill="outline">{{ nextButton$ | async }}</ion-button>
351
- <ion-button color="success">{{ okButton$ | async }}</ion-button>
352
- <ion-button fill="clear">{{ closeButton$ | async }}</ion-button>
353
- </div>
354
- </div>
355
-
356
- <div class="divider"></div>
357
-
358
- <!-- Global actions demonstration -->
359
- <div class="section">
360
- <h3>Global Actions</h3>
361
- <div class="button-grid">
362
- <ion-button size="small">{{ addButton$ | async }}</ion-button>
363
- <ion-button size="small">{{ removeButton$ | async }}</ion-button>
364
- <ion-button size="small">{{ searchButton$ | async }}</ion-button>
365
- <ion-button size="small">{{ filterButton$ | async }}</ion-button>
366
- <ion-button size="small">{{ sortButton$ | async }}</ion-button>
367
- <ion-button size="small">{{ refreshButton$ | async }}</ion-button>
368
- </div>
369
- </div>
370
-
371
- <div class="divider"></div>
372
-
373
- <!-- Global states demonstration -->
374
- <div class="section">
375
- <h3>Global States</h3>
376
- <div class="state-messages">
377
- <div class="state-item loading">
378
- <val-text
379
- [props]="{ content: loadingText$ | async, color: 'medium', size: 'small', bold: false }"
380
- ></val-text>
381
- </div>
382
- <div class="state-item success">
383
- <val-text
384
- [props]="{ content: successText$ | async, color: 'success', size: 'small', bold: false }"
385
- ></val-text>
386
- </div>
387
- <div class="state-item error">
388
- <val-text
389
- [props]="{ content: errorText$ | async, color: 'danger', size: 'small', bold: false }"
390
- ></val-text>
391
- </div>
392
- <div class="state-item warning">
393
- <val-text
394
- [props]="{ content: warningText$ | async, color: 'warning', size: 'small', bold: false }"
395
- ></val-text>
396
- </div>
397
- <div class="state-item info">
398
- <val-text
399
- [props]="{ content: infoText$ | async, color: 'primary', size: 'small', bold: false }"
400
- ></val-text>
401
- </div>
402
- <div class="state-item no-data">
403
- <val-text
404
- [props]="{ content: noDataText$ | async, color: 'medium', size: 'small', bold: false }"
405
- ></val-text>
406
- </div>
407
- </div>
408
- </div>
409
-
410
- <div class="divider"></div>
411
-
412
- <!-- Form with global placeholders and labels -->
413
- <div class="section">
414
- <h3>Form with Global Content</h3>
415
- <form class="example-form">
416
- <ion-item>
417
- <ion-label position="stacked">Name ({{ requiredText$ | async }})</ion-label>
418
- <ion-input [(ngModel)]="formData.name" [placeholder]="'Enter your name' + '...'" name="name"> </ion-input>
419
- </ion-item>
420
-
421
- <ion-item>
422
- <ion-label position="stacked">Email ({{ optionalText$ | async }})</ion-label>
423
- <ion-input [(ngModel)]="formData.email" [placeholder]="'Enter your email' + '...'" name="email">
424
- </ion-input>
425
- </ion-item>
426
-
427
- <ion-item>
428
- <ion-label position="stacked">Message</ion-label>
429
- <ion-textarea [(ngModel)]="formData.message" [placeholder]="searchPlaceholder$ | async" name="message">
430
- </ion-textarea>
431
- </ion-item>
432
- </form>
433
- </div>
434
-
435
- <div class="divider"></div>
436
-
437
- <!-- Global confirmations with interpolation -->
438
- <div class="section">
439
- <h3>Confirmations with Interpolation</h3>
440
- <div class="confirmation-examples">
441
- <p><strong>Delete Confirmation:</strong></p>
442
- <val-text
443
- [props]="{ content: deleteConfirmation$ | async, color: 'danger', size: 'medium', bold: false }"
444
- ></val-text>
445
-
446
- <br /><br />
447
- <p><strong>General Confirmation:</strong></p>
448
- <val-text
449
- [props]="{ content: areYouSureText$ | async, color: 'warning', size: 'medium', bold: false }"
450
- ></val-text>
451
-
452
- <br /><br />
453
- <p><strong>Unsaved Changes:</strong></p>
454
- <val-text
455
- [props]="{ content: unsavedChangesText$ | async, color: 'medium', size: 'medium', bold: false }"
456
- ></val-text>
457
- </div>
458
- </div>
459
-
460
- <div class="divider"></div>
461
-
462
- <!-- Mixed content: Global + Component specific -->
463
- <div class="section">
464
- <h3>Mixed Content Example</h3>
465
- <div class="mixed-content">
466
- <!-- Component-specific content -->
467
- <val-text [props]="componentDescriptionProps"></val-text>
468
-
469
- <br /><br />
470
-
471
- <!-- Action buttons mixing global and component content -->
472
- <div class="action-buttons">
473
- <ion-button color="primary" (click)="performAction()">
474
- {{ saveButton$ | async }}
475
- </ion-button>
476
- <ion-button fill="outline" (click)="cancelAction()">
477
- {{ cancelButton$ | async }}
478
- </ion-button>
479
- </div>
480
-
481
- <!-- Status message using global content -->
482
- <div *ngIf="actionPerformed" class="status-message">
483
- <val-text
484
- [props]="{ content: successText$ | async, color: 'success', size: 'small', bold: false }"
485
- ></val-text>
486
- </div>
487
- </div>
488
- </div>
489
-
490
- <!-- Demo of synchronous global content access -->
491
- <div class="section">
492
- <h3>Synchronous Access Example</h3>
493
- <p>
494
- OK button text (sync): <strong>{{ getOkTextSync() }}</strong>
495
- </p>
496
- <p>
497
- Cancel button text (sync): <strong>{{ getCancelTextSync() }}</strong>
498
- </p>
499
- </div>
500
- </ion-card-content>
501
- </ion-card>
502
- `, styles: [".language-section{display:flex;align-items:center;gap:8px;margin-bottom:16px}.divider{height:1px;background-color:var(--ion-color-light);margin:16px 0}.section{margin-bottom:16px}.section h3{margin:0 0 12px;color:var(--ion-color-primary)}.button-grid{display:grid;grid-template-columns:repeat(auto-fit,minmax(120px,1fr));gap:8px}.state-messages{display:flex;flex-direction:column;gap:8px}.state-item{padding:8px;border-radius:4px;background-color:var(--ion-color-light-shade)}.state-item.loading{background-color:var(--ion-color-medium-tint)}.state-item.success{background-color:var(--ion-color-success-tint)}.state-item.error{background-color:var(--ion-color-danger-tint)}.state-item.warning{background-color:var(--ion-color-warning-tint)}.state-item.info{background-color:var(--ion-color-primary-tint)}.example-form{margin-top:12px}.confirmation-examples{padding:12px;background-color:var(--ion-color-light-tint);border-radius:8px}.mixed-content{padding:12px;background-color:var(--ion-color-light-shade);border-radius:8px}.action-buttons{display:flex;gap:8px;margin:12px 0}.status-message{padding:8px;background-color:var(--ion-color-success-tint);border-radius:4px;margin-top:8px}\n"] }]
503
- }] });
504
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"global-content-example.component.js","sourceRoot":"","sources":["../../../../../../projects/valtech-components/src/lib/components/_examples/global-content-example.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, inject } from '@angular/core';\nimport { FormsModule } from '@angular/forms';\nimport {\n  IonButton,\n  IonCard,\n  IonCardContent,\n  IonCardHeader,\n  IonCardTitle,\n  IonInput,\n  IonItem,\n  IonLabel,\n  IonTextarea,\n} from '@ionic/angular/standalone';\nimport { ContentService } from '../../services/content.service';\nimport { TextComponent } from '../atoms/text/text.component';\nimport { TextMetadata } from '../atoms/text/types';\n\n/**\n * Comprehensive example demonstrating global content usage.\n *\n * This component shows:\n * 1. Global button texts\n * 2. Global state messages\n * 3. Global confirmations with interpolation\n * 4. Mixing global and component-specific content\n * 5. Language switching\n */\n@Component({\n  selector: 'val-global-content-example',\n  standalone: true,\n  imports: [\n    CommonModule,\n    FormsModule,\n    IonCard,\n    IonCardContent,\n    IonCardHeader,\n    IonCardTitle,\n    IonButton,\n    IonItem,\n    IonLabel,\n    IonInput,\n    IonTextarea,\n    TextComponent,\n  ],\n  template: `\n    <ion-card>\n      <ion-card-header>\n        <ion-card-title>\n          <val-text [props]=\"titleProps\"></val-text>\n        </ion-card-title>\n      </ion-card-header>\n\n      <ion-card-content>\n        <!-- Language switcher -->\n        <div class=\"language-section\">\n          <val-text [props]=\"languageLabel\"></val-text>\n          <ion-button fill=\"outline\" size=\"small\" (click)=\"setLanguage('es')\"> Español </ion-button>\n          <ion-button fill=\"outline\" size=\"small\" (click)=\"setLanguage('en')\"> English </ion-button>\n        </div>\n\n        <div class=\"divider\"></div>\n\n        <!-- Global buttons demonstration -->\n        <div class=\"section\">\n          <h3>Global Buttons</h3>\n          <div class=\"button-grid\">\n            <ion-button color=\"primary\">{{ saveButton$ | async }}</ion-button>\n            <ion-button color=\"secondary\">{{ editButton$ | async }}</ion-button>\n            <ion-button color=\"danger\">{{ deleteButton$ | async }}</ion-button>\n            <ion-button fill=\"outline\">{{ cancelButton$ | async }}</ion-button>\n            <ion-button fill=\"outline\">{{ backButton$ | async }}</ion-button>\n            <ion-button fill=\"outline\">{{ nextButton$ | async }}</ion-button>\n            <ion-button color=\"success\">{{ okButton$ | async }}</ion-button>\n            <ion-button fill=\"clear\">{{ closeButton$ | async }}</ion-button>\n          </div>\n        </div>\n\n        <div class=\"divider\"></div>\n\n        <!-- Global actions demonstration -->\n        <div class=\"section\">\n          <h3>Global Actions</h3>\n          <div class=\"button-grid\">\n            <ion-button size=\"small\">{{ addButton$ | async }}</ion-button>\n            <ion-button size=\"small\">{{ removeButton$ | async }}</ion-button>\n            <ion-button size=\"small\">{{ searchButton$ | async }}</ion-button>\n            <ion-button size=\"small\">{{ filterButton$ | async }}</ion-button>\n            <ion-button size=\"small\">{{ sortButton$ | async }}</ion-button>\n            <ion-button size=\"small\">{{ refreshButton$ | async }}</ion-button>\n          </div>\n        </div>\n\n        <div class=\"divider\"></div>\n\n        <!-- Global states demonstration -->\n        <div class=\"section\">\n          <h3>Global States</h3>\n          <div class=\"state-messages\">\n            <div class=\"state-item loading\">\n              <val-text\n                [props]=\"{ content: loadingText$ | async, color: 'medium', size: 'small', bold: false }\"\n              ></val-text>\n            </div>\n            <div class=\"state-item success\">\n              <val-text\n                [props]=\"{ content: successText$ | async, color: 'success', size: 'small', bold: false }\"\n              ></val-text>\n            </div>\n            <div class=\"state-item error\">\n              <val-text\n                [props]=\"{ content: errorText$ | async, color: 'danger', size: 'small', bold: false }\"\n              ></val-text>\n            </div>\n            <div class=\"state-item warning\">\n              <val-text\n                [props]=\"{ content: warningText$ | async, color: 'warning', size: 'small', bold: false }\"\n              ></val-text>\n            </div>\n            <div class=\"state-item info\">\n              <val-text\n                [props]=\"{ content: infoText$ | async, color: 'primary', size: 'small', bold: false }\"\n              ></val-text>\n            </div>\n            <div class=\"state-item no-data\">\n              <val-text\n                [props]=\"{ content: noDataText$ | async, color: 'medium', size: 'small', bold: false }\"\n              ></val-text>\n            </div>\n          </div>\n        </div>\n\n        <div class=\"divider\"></div>\n\n        <!-- Form with global placeholders and labels -->\n        <div class=\"section\">\n          <h3>Form with Global Content</h3>\n          <form class=\"example-form\">\n            <ion-item>\n              <ion-label position=\"stacked\">Name ({{ requiredText$ | async }})</ion-label>\n              <ion-input [(ngModel)]=\"formData.name\" [placeholder]=\"'Enter your name' + '...'\" name=\"name\"> </ion-input>\n            </ion-item>\n\n            <ion-item>\n              <ion-label position=\"stacked\">Email ({{ optionalText$ | async }})</ion-label>\n              <ion-input [(ngModel)]=\"formData.email\" [placeholder]=\"'Enter your email' + '...'\" name=\"email\">\n              </ion-input>\n            </ion-item>\n\n            <ion-item>\n              <ion-label position=\"stacked\">Message</ion-label>\n              <ion-textarea [(ngModel)]=\"formData.message\" [placeholder]=\"searchPlaceholder$ | async\" name=\"message\">\n              </ion-textarea>\n            </ion-item>\n          </form>\n        </div>\n\n        <div class=\"divider\"></div>\n\n        <!-- Global confirmations with interpolation -->\n        <div class=\"section\">\n          <h3>Confirmations with Interpolation</h3>\n          <div class=\"confirmation-examples\">\n            <p><strong>Delete Confirmation:</strong></p>\n            <val-text\n              [props]=\"{ content: deleteConfirmation$ | async, color: 'danger', size: 'medium', bold: false }\"\n            ></val-text>\n\n            <br /><br />\n            <p><strong>General Confirmation:</strong></p>\n            <val-text\n              [props]=\"{ content: areYouSureText$ | async, color: 'warning', size: 'medium', bold: false }\"\n            ></val-text>\n\n            <br /><br />\n            <p><strong>Unsaved Changes:</strong></p>\n            <val-text\n              [props]=\"{ content: unsavedChangesText$ | async, color: 'medium', size: 'medium', bold: false }\"\n            ></val-text>\n          </div>\n        </div>\n\n        <div class=\"divider\"></div>\n\n        <!-- Mixed content: Global + Component specific -->\n        <div class=\"section\">\n          <h3>Mixed Content Example</h3>\n          <div class=\"mixed-content\">\n            <!-- Component-specific content -->\n            <val-text [props]=\"componentDescriptionProps\"></val-text>\n\n            <br /><br />\n\n            <!-- Action buttons mixing global and component content -->\n            <div class=\"action-buttons\">\n              <ion-button color=\"primary\" (click)=\"performAction()\">\n                {{ saveButton$ | async }}\n              </ion-button>\n              <ion-button fill=\"outline\" (click)=\"cancelAction()\">\n                {{ cancelButton$ | async }}\n              </ion-button>\n            </div>\n\n            <!-- Status message using global content -->\n            <div *ngIf=\"actionPerformed\" class=\"status-message\">\n              <val-text\n                [props]=\"{ content: successText$ | async, color: 'success', size: 'small', bold: false }\"\n              ></val-text>\n            </div>\n          </div>\n        </div>\n\n        <!-- Demo of synchronous global content access -->\n        <div class=\"section\">\n          <h3>Synchronous Access Example</h3>\n          <p>\n            OK button text (sync): <strong>{{ getOkTextSync() }}</strong>\n          </p>\n          <p>\n            Cancel button text (sync): <strong>{{ getCancelTextSync() }}</strong>\n          </p>\n        </div>\n      </ion-card-content>\n    </ion-card>\n  `,\n  styles: [\n    `\n      .language-section {\n        display: flex;\n        align-items: center;\n        gap: 8px;\n        margin-bottom: 16px;\n      }\n\n      .divider {\n        height: 1px;\n        background-color: var(--ion-color-light);\n        margin: 16px 0;\n      }\n\n      .section {\n        margin-bottom: 16px;\n      }\n\n      .section h3 {\n        margin: 0 0 12px 0;\n        color: var(--ion-color-primary);\n      }\n\n      .button-grid {\n        display: grid;\n        grid-template-columns: repeat(auto-fit, minmax(120px, 1fr));\n        gap: 8px;\n      }\n\n      .state-messages {\n        display: flex;\n        flex-direction: column;\n        gap: 8px;\n      }\n\n      .state-item {\n        padding: 8px;\n        border-radius: 4px;\n        background-color: var(--ion-color-light-shade);\n      }\n\n      .state-item.loading {\n        background-color: var(--ion-color-medium-tint);\n      }\n\n      .state-item.success {\n        background-color: var(--ion-color-success-tint);\n      }\n\n      .state-item.error {\n        background-color: var(--ion-color-danger-tint);\n      }\n\n      .state-item.warning {\n        background-color: var(--ion-color-warning-tint);\n      }\n\n      .state-item.info {\n        background-color: var(--ion-color-primary-tint);\n      }\n\n      .example-form {\n        margin-top: 12px;\n      }\n\n      .confirmation-examples {\n        padding: 12px;\n        background-color: var(--ion-color-light-tint);\n        border-radius: 8px;\n      }\n\n      .mixed-content {\n        padding: 12px;\n        background-color: var(--ion-color-light-shade);\n        border-radius: 8px;\n      }\n\n      .action-buttons {\n        display: flex;\n        gap: 8px;\n        margin: 12px 0;\n      }\n\n      .status-message {\n        padding: 8px;\n        background-color: var(--ion-color-success-tint);\n        border-radius: 4px;\n        margin-top: 8px;\n      }\n    `,\n  ],\n})\nexport class GlobalContentExampleComponent {\n  private content = inject(ContentService);\n  private componentContent = this.content.forComponent('GlobalContentExample');\n\n  // Form data\n  formData = {\n    name: '',\n    email: '',\n    message: '',\n  };\n\n  // Action state\n  actionPerformed = false;\n\n  // Component-specific content properties\n  titleProps: TextMetadata = {\n    contentKey: 'title',\n    contentClass: 'GlobalContentExample',\n    contentFallback: 'Global Content Example',\n    color: 'primary',\n    size: 'xlarge',\n    bold: true,\n  };\n\n  languageLabel: TextMetadata = {\n    contentKey: 'languageLabel',\n    contentClass: 'GlobalContentExample',\n    contentFallback: 'Language:',\n    color: 'medium',\n    size: 'medium',\n    bold: true,\n  };\n\n  componentDescriptionProps: TextMetadata = {\n    contentKey: 'description',\n    contentClass: 'GlobalContentExample',\n    contentFallback: 'This example demonstrates global and component-specific content.',\n    color: 'dark',\n    size: 'medium',\n    bold: false,\n  };\n\n  // Global button content (reactive)\n  saveButton$ = this.content.fromContent({ key: 'save' });\n  editButton$ = this.content.fromContent({ key: 'edit' });\n  deleteButton$ = this.content.fromContent({ key: 'delete' });\n  cancelButton$ = this.content.fromContent({ key: 'cancel' });\n  backButton$ = this.content.fromContent({ key: 'back' });\n  nextButton$ = this.content.fromContent({ key: 'next' });\n  okButton$ = this.content.fromContent({ key: 'ok' });\n  closeButton$ = this.content.fromContent({ key: 'close' });\n\n  // Global action content\n  addButton$ = this.content.fromContent({ key: 'add' });\n  removeButton$ = this.content.fromContent({ key: 'remove' });\n  searchButton$ = this.content.fromContent({ key: 'search' });\n  filterButton$ = this.content.fromContent({ key: 'filter' });\n  sortButton$ = this.content.fromContent({ key: 'sort' });\n  refreshButton$ = this.content.fromContent({ key: 'refresh' });\n\n  // Global state content\n  loadingText$ = this.content.fromContent({ key: 'loading' });\n  successText$ = this.content.fromContent({ key: 'success' });\n  errorText$ = this.content.fromContent({ key: 'error' });\n  warningText$ = this.content.fromContent({ key: 'warning' });\n  infoText$ = this.content.fromContent({ key: 'info' });\n  noDataText$ = this.content.fromContent({ key: 'noData' });\n\n  // Global form content\n  requiredText$ = this.content.fromContent({ key: 'required' });\n  optionalText$ = this.content.fromContent({ key: 'optional' });\n  searchPlaceholder$ = this.content.fromContent({ key: 'searchPlaceholder' });\n\n  // Global confirmation content with interpolation\n  deleteConfirmation$ = this.content.fromContentWithInterpolation({\n    key: 'deleteConfirmation',\n    interpolation: { itemName: 'este registro' },\n  });\n\n  areYouSureText$ = this.content.fromContent({ key: 'areYouSure' });\n  unsavedChangesText$ = this.content.fromContent({ key: 'unsavedChanges' });\n\n  /**\n   * Set language\n   */\n  setLanguage(lang: 'es' | 'en') {\n    this.content.setLang(lang as any);\n  }\n\n  /**\n   * Perform an action to demonstrate status messages\n   */\n  performAction() {\n    this.actionPerformed = true;\n    setTimeout(() => {\n      this.actionPerformed = false;\n    }, 3000);\n  }\n\n  /**\n   * Cancel action\n   */\n  cancelAction() {\n    this.formData = { name: '', email: '', message: '' };\n    this.actionPerformed = false;\n  }\n\n  /**\n   * Example of synchronous global content access\n   */\n  getOkTextSync(): string {\n    return this.content.getGlobalText('ok');\n  }\n\n  /**\n   * Example of synchronous global content access using getText\n   */\n  getCancelTextSync(): string {\n    return this.content.getText('cancel');\n  }\n}\n"]}
@@ -1,43 +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
- // Example content for a sample component
9
- const exampleComponentContent = {
10
- es: {
11
- title: 'Ejemplo de Contenido Reactivo',
12
- subtitle: 'Este es un subtítulo que cambia automáticamente',
13
- description: 'Esta descripción se actualiza cuando cambias el idioma sin necesidad de recargar la página.',
14
- greeting: 'Hola {name}, tienes {count} mensajes nuevos',
15
- welcomeMessage: 'Bienvenido a la biblioteca de componentes Valtech',
16
- dynamicText: 'Este texto se actualiza automáticamente cuando cambias el idioma',
17
- buttonText: 'Cambiar Idioma',
18
- loadingText: 'Cargando...',
19
- errorMessage: 'Error: No se pudo cargar el contenido',
20
- successMessage: 'Contenido cargado exitosamente',
21
- confirmDialog: '¿Estás seguro de que quieres continuar?',
22
- cancelButton: 'Cancelar',
23
- acceptButton: 'Aceptar',
24
- },
25
- en: {
26
- title: 'Reactive Content Example',
27
- subtitle: 'This is a subtitle that changes automatically',
28
- description: 'This description updates when you change the language without needing to reload the page.',
29
- greeting: 'Hello {name}, you have {count} new messages',
30
- welcomeMessage: 'Welcome to the Valtech component library',
31
- dynamicText: 'This text updates automatically when you change the language',
32
- buttonText: 'Change Language',
33
- loadingText: 'Loading...',
34
- errorMessage: 'Error: Could not load content',
35
- successMessage: 'Content loaded successfully',
36
- confirmDialog: 'Are you sure you want to continue?',
37
- cancelButton: 'Cancel',
38
- acceptButton: 'Accept',
39
- },
40
- };
41
- // Export the content as a TextContent instance
42
- export default new TextContent(exampleComponentContent);
43
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVhY3RpdmUtY29udGVudC1leGFtcGxlLWNvbnRlbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy92YWx0ZWNoLWNvbXBvbmVudHMvc3JjL2xpYi9jb21wb25lbnRzL19leGFtcGxlcy9yZWFjdGl2ZS1jb250ZW50LWV4YW1wbGUtY29udGVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7R0FLRztBQUVILE9BQU8sRUFBb0IsV0FBVyxFQUFFLE1BQU0sb0NBQW9DLENBQUM7QUFFbkYseUNBQXlDO0FBQ3pDLE1BQU0sdUJBQXVCLEdBQXFCO0lBQ2hELEVBQUUsRUFBRTtRQUNGLEtBQUssRUFBRSwrQkFBK0I7UUFDdEMsUUFBUSxFQUFFLGlEQUFpRDtRQUMzRCxXQUFXLEVBQ1QsNkZBQTZGO1FBQy9GLFFBQVEsRUFBRSw2Q0FBNkM7UUFDdkQsY0FBYyxFQUFFLG1EQUFtRDtRQUNuRSxXQUFXLEVBQUUsa0VBQWtFO1FBQy9FLFVBQVUsRUFBRSxnQkFBZ0I7UUFDNUIsV0FBVyxFQUFFLGFBQWE7UUFDMUIsWUFBWSxFQUFFLHVDQUF1QztRQUNyRCxjQUFjLEVBQUUsZ0NBQWdDO1FBQ2hELGFBQWEsRUFBRSx5Q0FBeUM7UUFDeEQsWUFBWSxFQUFFLFVBQVU7UUFDeEIsWUFBWSxFQUFFLFNBQVM7S0FDeEI7SUFDRCxFQUFFLEVBQUU7UUFDRixLQUFLLEVBQUUsMEJBQTBCO1FBQ2pDLFFBQVEsRUFBRSwrQ0FBK0M7UUFDekQsV0FBVyxFQUNULDJGQUEyRjtRQUM3RixRQUFRLEVBQUUsNkNBQTZDO1FBQ3ZELGNBQWMsRUFBRSwwQ0FBMEM7UUFDMUQsV0FBVyxFQUFFLDhEQUE4RDtRQUMzRSxVQUFVLEVBQUUsaUJBQWlCO1FBQzdCLFdBQVcsRUFBRSxZQUFZO1FBQ3pCLFlBQVksRUFBRSwrQkFBK0I7UUFDN0MsY0FBYyxFQUFFLDZCQUE2QjtRQUM3QyxhQUFhLEVBQUUsb0NBQW9DO1FBQ25ELFlBQVksRUFBRSxRQUFRO1FBQ3RCLFlBQVksRUFBRSxRQUFRO0tBQ3ZCO0NBQ0YsQ0FBQztBQUVGLCtDQUErQztBQUMvQyxlQUFlLElBQUksV0FBVyxDQUFDLHVCQUF1QixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEV4YW1wbGUgY29udGVudCBjb25maWd1cmF0aW9uIGZvciBkZW1vbnN0cmF0aW5nIHRoZSByZWFjdGl2ZSBjb250ZW50IHN5c3RlbS5cbiAqXG4gKiBUaGlzIGZpbGUgc2hvd3MgaG93IHRvIHN0cnVjdHVyZSBjb250ZW50IGZvciBjb21wb25lbnRzIHRoYXQgdXNlIHRoZSByZWFjdGl2ZVxuICogbGFuZ3VhZ2Ugc3lzdGVtIHdpdGggdGhlIG5ldyBmcm9tQ29udGVudCBwYXR0ZXJuLlxuICovXG5cbmltcG9ydCB7IExhbmd1YWdlc0NvbnRlbnQsIFRleHRDb250ZW50IH0gZnJvbSAnLi4vLi4vc2VydmljZXMvbGFuZy1wcm92aWRlci90eXBlcyc7XG5cbi8vIEV4YW1wbGUgY29udGVudCBmb3IgYSBzYW1wbGUgY29tcG9uZW50XG5jb25zdCBleGFtcGxlQ29tcG9uZW50Q29udGVudDogTGFuZ3VhZ2VzQ29udGVudCA9IHtcbiAgZXM6IHtcbiAgICB0aXRsZTogJ0VqZW1wbG8gZGUgQ29udGVuaWRvIFJlYWN0aXZvJyxcbiAgICBzdWJ0aXRsZTogJ0VzdGUgZXMgdW4gc3VidMOtdHVsbyBxdWUgY2FtYmlhIGF1dG9tw6F0aWNhbWVudGUnLFxuICAgIGRlc2NyaXB0aW9uOlxuICAgICAgJ0VzdGEgZGVzY3JpcGNpw7NuIHNlIGFjdHVhbGl6YSBjdWFuZG8gY2FtYmlhcyBlbCBpZGlvbWEgc2luIG5lY2VzaWRhZCBkZSByZWNhcmdhciBsYSBww6FnaW5hLicsXG4gICAgZ3JlZXRpbmc6ICdIb2xhIHtuYW1lfSwgdGllbmVzIHtjb3VudH0gbWVuc2FqZXMgbnVldm9zJyxcbiAgICB3ZWxjb21lTWVzc2FnZTogJ0JpZW52ZW5pZG8gYSBsYSBiaWJsaW90ZWNhIGRlIGNvbXBvbmVudGVzIFZhbHRlY2gnLFxuICAgIGR5bmFtaWNUZXh0OiAnRXN0ZSB0ZXh0byBzZSBhY3R1YWxpemEgYXV0b23DoXRpY2FtZW50ZSBjdWFuZG8gY2FtYmlhcyBlbCBpZGlvbWEnLFxuICAgIGJ1dHRvblRleHQ6ICdDYW1iaWFyIElkaW9tYScsXG4gICAgbG9hZGluZ1RleHQ6ICdDYXJnYW5kby4uLicsXG4gICAgZXJyb3JNZXNzYWdlOiAnRXJyb3I6IE5vIHNlIHB1ZG8gY2FyZ2FyIGVsIGNvbnRlbmlkbycsXG4gICAgc3VjY2Vzc01lc3NhZ2U6ICdDb250ZW5pZG8gY2FyZ2FkbyBleGl0b3NhbWVudGUnLFxuICAgIGNvbmZpcm1EaWFsb2c6ICfCv0VzdMOhcyBzZWd1cm8gZGUgcXVlIHF1aWVyZXMgY29udGludWFyPycsXG4gICAgY2FuY2VsQnV0dG9uOiAnQ2FuY2VsYXInLFxuICAgIGFjY2VwdEJ1dHRvbjogJ0FjZXB0YXInLFxuICB9LFxuICBlbjoge1xuICAgIHRpdGxlOiAnUmVhY3RpdmUgQ29udGVudCBFeGFtcGxlJyxcbiAgICBzdWJ0aXRsZTogJ1RoaXMgaXMgYSBzdWJ0aXRsZSB0aGF0IGNoYW5nZXMgYXV0b21hdGljYWxseScsXG4gICAgZGVzY3JpcHRpb246XG4gICAgICAnVGhpcyBkZXNjcmlwdGlvbiB1cGRhdGVzIHdoZW4geW91IGNoYW5nZSB0aGUgbGFuZ3VhZ2Ugd2l0aG91dCBuZWVkaW5nIHRvIHJlbG9hZCB0aGUgcGFnZS4nLFxuICAgIGdyZWV0aW5nOiAnSGVsbG8ge25hbWV9LCB5b3UgaGF2ZSB7Y291bnR9IG5ldyBtZXNzYWdlcycsXG4gICAgd2VsY29tZU1lc3NhZ2U6ICdXZWxjb21lIHRvIHRoZSBWYWx0ZWNoIGNvbXBvbmVudCBsaWJyYXJ5JyxcbiAgICBkeW5hbWljVGV4dDogJ1RoaXMgdGV4dCB1cGRhdGVzIGF1dG9tYXRpY2FsbHkgd2hlbiB5b3UgY2hhbmdlIHRoZSBsYW5ndWFnZScsXG4gICAgYnV0dG9uVGV4dDogJ0NoYW5nZSBMYW5ndWFnZScsXG4gICAgbG9hZGluZ1RleHQ6ICdMb2FkaW5nLi4uJyxcbiAgICBlcnJvck1lc3NhZ2U6ICdFcnJvcjogQ291bGQgbm90IGxvYWQgY29udGVudCcsXG4gICAgc3VjY2Vzc01lc3NhZ2U6ICdDb250ZW50IGxvYWRlZCBzdWNjZXNzZnVsbHknLFxuICAgIGNvbmZpcm1EaWFsb2c6ICdBcmUgeW91IHN1cmUgeW91IHdhbnQgdG8gY29udGludWU/JyxcbiAgICBjYW5jZWxCdXR0b246ICdDYW5jZWwnLFxuICAgIGFjY2VwdEJ1dHRvbjogJ0FjY2VwdCcsXG4gIH0sXG59O1xuXG4vLyBFeHBvcnQgdGhlIGNvbnRlbnQgYXMgYSBUZXh0Q29udGVudCBpbnN0YW5jZVxuZXhwb3J0IGRlZmF1bHQgbmV3IFRleHRDb250ZW50KGV4YW1wbGVDb21wb25lbnRDb250ZW50KTtcblxuLyoqXG4gKiBUeXBlLXNhZmUgY29udGVudCBrZXlzIGZvciB0aGUgZXhhbXBsZSBjb21wb25lbnQuXG4gKiBVc2UgdGhpcyBpbnRlcmZhY2UgdG8gZW5zdXJlIHR5cGUgc2FmZXR5IHdoZW4gYWNjZXNzaW5nIGNvbnRlbnQga2V5cy5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBFeGFtcGxlQ29tcG9uZW50Q29udGVudEtleXMge1xuICB0aXRsZTogc3RyaW5nO1xuICBzdWJ0aXRsZTogc3RyaW5nO1xuICBkZXNjcmlwdGlvbjogc3RyaW5nO1xuICBncmVldGluZzogc3RyaW5nO1xuICB3ZWxjb21lTWVzc2FnZTogc3RyaW5nO1xuICBkeW5hbWljVGV4dDogc3RyaW5nO1xuICBidXR0b25UZXh0OiBzdHJpbmc7XG4gIGxvYWRpbmdUZXh0OiBzdHJpbmc7XG4gIGVycm9yTWVzc2FnZTogc3RyaW5nO1xuICBzdWNjZXNzTWVzc2FnZTogc3RyaW5nO1xuICBjb25maXJtRGlhbG9nOiBzdHJpbmc7XG4gIGNhbmNlbEJ1dHRvbjogc3RyaW5nO1xuICBhY2NlcHRCdXR0b246IHN0cmluZztcbn1cblxuLyoqXG4gKiBVdGlsaXR5IHR5cGUgZm9yIGdldHRpbmcgY29udGVudCBrZXlzIHdpdGggdHlwZSBzYWZldHkuXG4gKi9cbmV4cG9ydCB0eXBlIEV4YW1wbGVDb250ZW50S2V5ID0ga2V5b2YgRXhhbXBsZUNvbXBvbmVudENvbnRlbnRLZXlzO1xuIl19