valtech-components 2.0.286 → 2.0.288
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 +54 -14
- package/esm2022/lib/components/atoms/text/text.component.mjs +46 -10
- package/esm2022/lib/components/atoms/text/types.mjs +1 -1
- package/esm2022/lib/examples/custom-content-demo.component.mjs +291 -0
- package/esm2022/lib/examples/link-processing-example.component.mjs +139 -0
- package/esm2022/lib/services/lang-provider/content.mjs +1 -13
- package/esm2022/lib/services/link-processor.service.mjs +147 -0
- package/esm2022/lib/shared/pipes/process-links.pipe.mjs +69 -0
- package/esm2022/public-api.mjs +5 -4
- package/fesm2022/valtech-components.mjs +384 -925
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/atoms/text/text.component.d.ts +5 -1
- package/lib/components/atoms/text/types.d.ts +5 -0
- package/lib/examples/link-processing-example.component.d.ts +22 -0
- package/lib/services/link-processor.service.d.ts +92 -0
- package/lib/shared/pipes/process-links.pipe.d.ts +55 -0
- package/package.json +1 -1
- package/public-api.d.ts +4 -3
- package/esm2022/lib/components/_examples/custom-content-demo.component.mjs +0 -291
- package/esm2022/lib/components/_examples/global-content-example-content.mjs +0 -23
- package/esm2022/lib/components/_examples/global-content-example.component.mjs +0 -504
- package/esm2022/lib/components/_examples/reactive-content-example-content.mjs +0 -43
- package/esm2022/lib/components/_examples/reactive-content-example.component.mjs +0 -347
- package/esm2022/lib/services/lang-provider/components/theme-settings.mjs +0 -15
- package/lib/components/_examples/global-content-example-content.d.ts +0 -9
- package/lib/components/_examples/global-content-example.component.d.ts +0 -73
- package/lib/components/_examples/reactive-content-example-content.d.ts +0 -32
- package/lib/components/_examples/reactive-content-example.component.d.ts +0 -47
- package/lib/services/lang-provider/components/theme-settings.d.ts +0 -3
- /package/lib/{components/_examples → examples}/custom-content-demo.component.d.ts +0 -0
package/README.md
CHANGED
|
@@ -75,8 +75,8 @@ The library includes a powerful reactive internationalization system with **Cont
|
|
|
75
75
|
- **Buttons**: `ok`, `cancel`, `save`, `delete`, `edit`, `close`, `back`, `next`, `previous`
|
|
76
76
|
- **Actions**: `add`, `remove`, `search`, `filter`, `sort`, `refresh`
|
|
77
77
|
- **States**: `loading`, `error`, `success`, `warning`, `info`, `noData`
|
|
78
|
-
- **Forms**: `
|
|
79
|
-
- **Confirmations**: `
|
|
78
|
+
- **Forms**: `searchPlaceholder`
|
|
79
|
+
- **Confirmations**: `deleteConfirmation`, `unsavedChanges`
|
|
80
80
|
|
|
81
81
|
### Basic Setup
|
|
82
82
|
|
|
@@ -169,29 +169,68 @@ export class UserComponent {
|
|
|
169
169
|
content = inject(ContentService);
|
|
170
170
|
componentContent = this.content.forComponent('UserComponent');
|
|
171
171
|
|
|
172
|
-
greeting$ = this.componentContent.
|
|
173
|
-
|
|
174
|
-
|
|
172
|
+
greeting$ = this.componentContent.get('personalizedGreeting', {
|
|
173
|
+
interpolation: {
|
|
174
|
+
name: 'John',
|
|
175
|
+
count: 5,
|
|
176
|
+
}
|
|
175
177
|
});
|
|
176
178
|
// Results in: "Hello John, you have 5 messages"
|
|
177
179
|
}
|
|
178
180
|
```
|
|
179
181
|
|
|
180
|
-
|
|
182
|
+
## 🔗 Automatic Link Processing
|
|
181
183
|
|
|
182
|
-
|
|
184
|
+
The library now includes automatic link detection and processing capabilities. Convert URLs and internal routes in text content into clickable links automatically:
|
|
183
185
|
|
|
184
|
-
|
|
186
|
+
```typescript
|
|
187
|
+
// Basic usage
|
|
188
|
+
<val-text [props]="{
|
|
189
|
+
content: 'Visit https://angular.io or go to /dashboard',
|
|
190
|
+
processLinks: true,
|
|
191
|
+
linkConfig: {
|
|
192
|
+
openExternalInNewTab: true,
|
|
193
|
+
openInternalInNewTab: false
|
|
194
|
+
}
|
|
195
|
+
}"></val-text>
|
|
196
|
+
|
|
197
|
+
// Advanced configuration
|
|
198
|
+
linkConfig: LinkProcessorConfig = {
|
|
199
|
+
openExternalInNewTab: true,
|
|
200
|
+
linkClass: 'custom-link',
|
|
201
|
+
externalLinkClass: 'external',
|
|
202
|
+
internalLinkClass: 'internal'
|
|
203
|
+
};
|
|
204
|
+
```
|
|
205
|
+
|
|
206
|
+
### Features
|
|
207
|
+
|
|
208
|
+
- **Automatic Detection**: Identifies external URLs (http/https) and internal routes (starting with /)
|
|
209
|
+
- **Configurable Behavior**: Control whether links open in new tabs
|
|
210
|
+
- **Custom Styling**: Apply custom CSS classes to links
|
|
211
|
+
- **Secure Processing**: Uses Angular's DomSanitizer for safe HTML generation
|
|
212
|
+
- **Standalone Pipe**: Use `ProcessLinksPipe` independently in any template
|
|
185
213
|
|
|
186
|
-
|
|
187
|
-
- **[Global Content Guide](./GLOBAL_CONTENT_GUIDE.md)** - How to use reusable global content for buttons, states, and confirmations
|
|
188
|
-
- **[Reactive Content Guide](./REACTIVE_CONTENT_GUIDE.md)** - Advanced patterns for reactive content management
|
|
214
|
+
### Using the Service Directly
|
|
189
215
|
|
|
190
|
-
|
|
216
|
+
```typescript
|
|
217
|
+
import { LinkProcessorService } from 'valtech-components';
|
|
218
|
+
|
|
219
|
+
constructor(private linkProcessor: LinkProcessorService) {}
|
|
191
220
|
|
|
192
|
-
|
|
221
|
+
processText(text: string) {
|
|
222
|
+
// Convert links automatically
|
|
223
|
+
const processed = this.linkProcessor.processLinks(text);
|
|
224
|
+
|
|
225
|
+
// Check if text contains links
|
|
226
|
+
const hasLinks = this.linkProcessor.hasLinks(text);
|
|
227
|
+
|
|
228
|
+
// Extract all links with their types
|
|
229
|
+
const links = this.linkProcessor.extractLinks(text);
|
|
230
|
+
}
|
|
231
|
+
```
|
|
193
232
|
|
|
194
|
-
|
|
233
|
+
See the **[Link Processing Guide](./LINK_PROCESSING_GUIDE.md)** for complete documentation and examples.
|
|
195
234
|
|
|
196
235
|
## Project Structure
|
|
197
236
|
|
|
@@ -210,6 +249,7 @@ The library includes example components that demonstrate best practices:
|
|
|
210
249
|
|
|
211
250
|
- `ReactiveContentExampleComponent` - Demonstrates the reactive content system with component-specific content
|
|
212
251
|
- `GlobalContentExampleComponent` - Comprehensive demonstration of global and mixed content usage
|
|
252
|
+
- `LinkProcessingExampleComponent` - Shows automatic link detection and processing in different scenarios
|
|
213
253
|
- Check the `_examples/` directory for more implementation examples
|
|
214
254
|
|
|
215
255
|
## License
|
|
@@ -2,8 +2,10 @@ import { AsyncPipe } from '@angular/common';
|
|
|
2
2
|
import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
|
|
3
3
|
import { IonText } from '@ionic/angular/standalone';
|
|
4
4
|
import { of, Subscription } from 'rxjs';
|
|
5
|
+
import { ProcessLinksPipe } from '../../../shared/pipes/process-links.pipe';
|
|
5
6
|
import * as i0 from "@angular/core";
|
|
6
7
|
import * as i1 from "../../../services/content.service";
|
|
8
|
+
import * as i2 from "../../../services/link-processor.service";
|
|
7
9
|
/**
|
|
8
10
|
* val-text
|
|
9
11
|
*
|
|
@@ -44,6 +46,23 @@ import * as i1 from "../../../services/content.service";
|
|
|
44
46
|
* }"></val-text>
|
|
45
47
|
* ```
|
|
46
48
|
*
|
|
49
|
+
* @example With automatic link processing:
|
|
50
|
+
* ```html
|
|
51
|
+
* <val-text [props]="{
|
|
52
|
+
* content: 'Visit https://example.com or go to /profile for more info',
|
|
53
|
+
* processLinks: true,
|
|
54
|
+
* linkConfig: {
|
|
55
|
+
* openExternalInNewTab: true,
|
|
56
|
+
* openInternalInNewTab: false,
|
|
57
|
+
* linkClass: 'custom-link',
|
|
58
|
+
* externalLinkClass: 'external',
|
|
59
|
+
* internalLinkClass: 'internal'
|
|
60
|
+
* },
|
|
61
|
+
* color: 'primary',
|
|
62
|
+
* size: 'medium'
|
|
63
|
+
* }"></val-text>
|
|
64
|
+
* ```
|
|
65
|
+
*
|
|
47
66
|
* @example Using ContentService helper:
|
|
48
67
|
* ```typescript
|
|
49
68
|
* // In component
|
|
@@ -63,8 +82,9 @@ import * as i1 from "../../../services/content.service";
|
|
|
63
82
|
* @input props: TextMetadata - Configuration for the text (content, styling, and reactive content options)
|
|
64
83
|
*/
|
|
65
84
|
export class TextComponent {
|
|
66
|
-
constructor(contentService) {
|
|
85
|
+
constructor(contentService, linkProcessor) {
|
|
67
86
|
this.contentService = contentService;
|
|
87
|
+
this.linkProcessor = linkProcessor;
|
|
68
88
|
this.subscription = new Subscription();
|
|
69
89
|
}
|
|
70
90
|
ngOnInit() {
|
|
@@ -108,21 +128,37 @@ export class TextComponent {
|
|
|
108
128
|
this.displayContent$ = of(this.props.contentFallback || '');
|
|
109
129
|
}
|
|
110
130
|
}
|
|
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 }); }
|
|
112
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "
|
|
131
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TextComponent, deps: [{ token: i1.ContentService }, { token: i2.LinkProcessorService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
132
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: TextComponent, isStandalone: true, selector: "val-text", inputs: { props: "props" }, ngImport: i0, template: `
|
|
113
133
|
<ion-text [color]="props.color">
|
|
114
|
-
|
|
134
|
+
@if (props.processLinks) {
|
|
135
|
+
<p
|
|
136
|
+
[class]="props.size"
|
|
137
|
+
[class.bold]="props.bold"
|
|
138
|
+
[innerHTML]="displayContent$ | async | processLinks: props.linkConfig"
|
|
139
|
+
></p>
|
|
140
|
+
} @else {
|
|
141
|
+
<p [class]="props.size" [class.bold]="props.bold">{{ displayContent$ | async }}</p>
|
|
142
|
+
}
|
|
115
143
|
</ion-text>
|
|
116
|
-
`, isInline: true, styles: ["
|
|
144
|
+
`, isInline: true, styles: ["@charset \"UTF-8\";: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}}:host ::ng-deep .processed-link{color:var(--ion-color-primary, #3880ff);text-decoration:underline;text-decoration-thickness:1px;text-underline-offset:2px;transition:color .3s ease}:host ::ng-deep .processed-link:hover{color:var(--ion-color-primary-shade, #3171e0);text-decoration-thickness:2px}:host ::ng-deep .processed-link:active{color:var(--ion-color-primary-tint, #4c8dff)}:host ::ng-deep .external-link:after{content:\" \\2197\";font-size:.8em;opacity:.7}:host ::ng-deep .internal-link{color:var(--ion-color-secondary, #0cd1e8)}:host ::ng-deep .internal-link:hover{color:var(--ion-color-secondary-shade, #0bb8cc)}\n"], dependencies: [{ kind: "component", type: IonText, selector: "ion-text", inputs: ["color", "mode"] }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "pipe", type: ProcessLinksPipe, name: "processLinks" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
117
145
|
}
|
|
118
146
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TextComponent, decorators: [{
|
|
119
147
|
type: Component,
|
|
120
|
-
args: [{ selector: 'val-text', standalone: true, imports: [IonText, AsyncPipe], template: `
|
|
148
|
+
args: [{ selector: 'val-text', standalone: true, imports: [IonText, AsyncPipe, ProcessLinksPipe], template: `
|
|
121
149
|
<ion-text [color]="props.color">
|
|
122
|
-
|
|
150
|
+
@if (props.processLinks) {
|
|
151
|
+
<p
|
|
152
|
+
[class]="props.size"
|
|
153
|
+
[class.bold]="props.bold"
|
|
154
|
+
[innerHTML]="displayContent$ | async | processLinks: props.linkConfig"
|
|
155
|
+
></p>
|
|
156
|
+
} @else {
|
|
157
|
+
<p [class]="props.size" [class.bold]="props.bold">{{ displayContent$ | async }}</p>
|
|
158
|
+
}
|
|
123
159
|
</ion-text>
|
|
124
|
-
`, changeDetection: ChangeDetectionStrategy.OnPush, styles: ["
|
|
125
|
-
}], ctorParameters: () => [{ type: i1.ContentService }], propDecorators: { props: [{
|
|
160
|
+
`, changeDetection: ChangeDetectionStrategy.OnPush, styles: ["@charset \"UTF-8\";: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}}:host ::ng-deep .processed-link{color:var(--ion-color-primary, #3880ff);text-decoration:underline;text-decoration-thickness:1px;text-underline-offset:2px;transition:color .3s ease}:host ::ng-deep .processed-link:hover{color:var(--ion-color-primary-shade, #3171e0);text-decoration-thickness:2px}:host ::ng-deep .processed-link:active{color:var(--ion-color-primary-tint, #4c8dff)}:host ::ng-deep .external-link:after{content:\" \\2197\";font-size:.8em;opacity:.7}:host ::ng-deep .internal-link{color:var(--ion-color-secondary, #0cd1e8)}:host ::ng-deep .internal-link:hover{color:var(--ion-color-secondary-shade, #0bb8cc)}\n"] }]
|
|
161
|
+
}], ctorParameters: () => [{ type: i1.ContentService }, { type: i2.LinkProcessorService }], propDecorators: { props: [{
|
|
126
162
|
type: Input
|
|
127
163
|
}] } });
|
|
128
164
|
/**
|
|
@@ -159,4 +195,4 @@ export function createTextProps(contentConfig, styleConfig = {}) {
|
|
|
159
195
|
bold: styleConfig.bold || false,
|
|
160
196
|
};
|
|
161
197
|
}
|
|
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"]}
|
|
198
|
+
//# 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 { LinkProcessorService } from '../../../services/link-processor.service';\nimport { ProcessLinksPipe } from '../../../shared/pipes/process-links.pipe';\nimport { TextContentConfig, TextMetadata } from './types';\n\n@Component({\n  selector: 'val-text',\n  standalone: true,\n  imports: [IonText, AsyncPipe, ProcessLinksPipe],\n  template: `\n    <ion-text [color]=\"props.color\">\n      @if (props.processLinks) {\n        <p\n          [class]=\"props.size\"\n          [class.bold]=\"props.bold\"\n          [innerHTML]=\"displayContent$ | async | processLinks: props.linkConfig\"\n        ></p>\n      } @else {\n        <p [class]=\"props.size\" [class.bold]=\"props.bold\">{{ displayContent$ | async }}</p>\n      }\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 With automatic link processing:\n * ```html\n * <val-text [props]=\"{\n *   content: 'Visit https://example.com or go to /profile for more info',\n *   processLinks: true,\n *   linkConfig: {\n *     openExternalInNewTab: true,\n *     openInternalInNewTab: false,\n *     linkClass: 'custom-link',\n *     externalLinkClass: 'external',\n *     internalLinkClass: 'internal'\n *   },\n *   color: 'primary',\n *   size: 'medium'\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   * @property processLinks - Whether to automatically process and convert links in text (default: false)\n   * @property linkConfig - Configuration for link processing (colors, target behavior, etc.)\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(\n    private contentService: ContentService,\n    private linkProcessor: LinkProcessorService\n  ) {}\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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy92YWx0ZWNoLWNvbXBvbmVudHMvc3JjL2xpYi9jb21wb25lbnRzL2F0b21zL3RleHQvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbG9yIH0gZnJvbSAnQGlvbmljL2NvcmUnO1xuaW1wb3J0IHsgTGlua1Byb2Nlc3NvckNvbmZpZyB9IGZyb20gJy4uLy4uLy4uL3NlcnZpY2VzL2xpbmstcHJvY2Vzc29yLnNlcnZpY2UnO1xuXG4vKipcbiAqIFByb3BzIGZvciB2YWwtdGV4dCBjb21wb25lbnQuXG4gKlxuICogVGhlIGNvbXBvbmVudCBzdXBwb3J0cyBib3RoIHN0YXRpYyBjb250ZW50IGFuZCByZWFjdGl2ZSBjb250ZW50IGZyb20gdGhlIGxhbmd1YWdlIHNlcnZpY2UuXG4gKiBVc2UgZWl0aGVyIGBjb250ZW50YCBmb3Igc3RhdGljIHRleHQgb3IgYGNvbnRlbnRLZXlgICsgYGNvbnRlbnRDbGFzc2AgZm9yIHJlYWN0aXZlIHRleHQuXG4gKlxuICogQHByb3BlcnR5IGNvbnRlbnQgLSBTdGF0aWMgdGV4dCB0byBkaXNwbGF5ICh0YWtlcyBwcmVjZWRlbmNlIG92ZXIgY29udGVudEtleSlcbiAqIEBwcm9wZXJ0eSBjb250ZW50S2V5IC0gS2V5IGZvciBkeW5hbWljIGNvbnRlbnQgZnJvbSBsYW5ndWFnZSBzZXJ2aWNlXG4gKiBAcHJvcGVydHkgY29udGVudENsYXNzIC0gQ29tcG9uZW50IGNsYXNzIG5hbWUgZm9yIGNvbnRlbnQgbG9va3VwIChyZXF1aXJlZCB3aXRoIGNvbnRlbnRLZXkpXG4gKiBAcHJvcGVydHkgY29udGVudEZhbGxiYWNrIC0gRmFsbGJhY2sgdGV4dCBpZiBjb250ZW50S2V5IGlzIG5vdCBmb3VuZFxuICogQHByb3BlcnR5IGNvbnRlbnRJbnRlcnBvbGF0aW9uIC0gVmFsdWVzIHRvIGludGVycG9sYXRlIGludG8gdGhlIGNvbnRlbnQgc3RyaW5nXG4gKiBAcHJvcGVydHkgY29sb3IgLSBUaGUgdGV4dCBjb2xvciAoSW9uaWMgY29sb3Igc3RyaW5nKVxuICogQHByb3BlcnR5IHNpemUgLSBUaGUgdGV4dCBzaXplICgnc21hbGwnIHwgJ21lZGl1bScgfCAnbGFyZ2UnIHwgJ3hsYXJnZScpXG4gKiBAcHJvcGVydHkgYm9sZCAtIFdoZXRoZXIgdGhlIHRleHQgaXMgYm9sZFxuICogQHByb3BlcnR5IHByb2Nlc3NMaW5rcyAtIFdoZXRoZXIgdG8gYXV0b21hdGljYWxseSBwcm9jZXNzIGFuZCBjb252ZXJ0IGxpbmtzIGluIHRoZSB0ZXh0IChkZWZhdWx0OiBmYWxzZSlcbiAqIEBwcm9wZXJ0eSBsaW5rQ29uZmlnIC0gQ29uZmlndXJhdGlvbiBmb3IgbGluayBwcm9jZXNzaW5nIChjb2xvcnMsIHRhcmdldCBiZWhhdmlvciwgZXRjLilcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBUZXh0TWV0YWRhdGEge1xuICBzaXplOiAnc21hbGwnIHwgJ21lZGl1bScgfCAnbGFyZ2UnIHwgJ3hsYXJnZSc7XG4gIGNvbG9yOiBDb2xvcjtcbiAgY29udGVudD86IHN0cmluZztcbiAgYm9sZDogYm9vbGVhbjtcblxuICAvLyBSZWFjdGl2ZSBjb250ZW50IHByb3BlcnRpZXNcbiAgY29udGVudEtleT86IHN0cmluZztcbiAgY29udGVudENsYXNzPzogc3RyaW5nO1xuICBjb250ZW50RmFsbGJhY2s/OiBzdHJpbmc7XG4gIGNvbnRlbnRJbnRlcnBvbGF0aW9uPzogUmVjb3JkPHN0cmluZywgc3RyaW5nIHwgbnVtYmVyPjtcblxuICAvLyBMaW5rIHByb2Nlc3NpbmcgcHJvcGVydGllc1xuICBwcm9jZXNzTGlua3M/OiBib29sZWFuO1xuICBsaW5rQ29uZmlnPzogTGlua1Byb2Nlc3NvckNvbmZpZztcbn1cblxuLyoqXG4gKiBDb25maWd1cmF0aW9uIGZvciByZWFjdGl2ZSBjb250ZW50IGluIHZhbC10ZXh0IGNvbXBvbmVudC5cbiAqIFVzZSB0aGlzIGludGVyZmFjZSB3aGVuIHlvdSBvbmx5IG5lZWQgdG8gc3BlY2lmeSBjb250ZW50LXJlbGF0ZWQgcHJvcGVydGllcy5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBUZXh0Q29udGVudENvbmZpZyB7XG4gIGNvbnRlbnRLZXk6IHN0cmluZztcbiAgY29udGVudENsYXNzOiBzdHJpbmc7XG4gIGNvbnRlbnRGYWxsYmFjaz86IHN0cmluZztcbiAgY29udGVudEludGVycG9sYXRpb24/OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmcgfCBudW1iZXI+O1xufVxuIl19
|
|
@@ -0,0 +1,291 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { Component, inject } from '@angular/core';
|
|
3
|
+
import { IonButton, IonCard, IonCardContent, IonCardHeader, IonCardTitle, IonItem, IonLabel, } from '@ionic/angular/standalone';
|
|
4
|
+
import { ContentService } from '../services/content.service';
|
|
5
|
+
import { LangOption } from '../services/lang-provider/types';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "@angular/common";
|
|
8
|
+
export class CustomContentDemoComponent {
|
|
9
|
+
constructor() {
|
|
10
|
+
this.content = inject(ContentService);
|
|
11
|
+
this.currentLang = 'es';
|
|
12
|
+
// Contenido global predefinido (sin className)
|
|
13
|
+
this.saveText$ = this.content.fromContent({ key: 'save' });
|
|
14
|
+
this.cancelText$ = this.content.fromContent({ key: 'cancel' });
|
|
15
|
+
this.deleteText$ = this.content.fromContent({ key: 'delete' });
|
|
16
|
+
this.loadingText$ = this.content.fromContent({ key: 'loading' });
|
|
17
|
+
// Tu contenido global personalizado (sin className)
|
|
18
|
+
this.dashboardText$ = this.content.fromContent({ key: 'dashboard' });
|
|
19
|
+
this.profileText$ = this.content.fromContent({ key: 'profile' });
|
|
20
|
+
this.settingsText$ = this.content.fromContent({ key: 'settings' });
|
|
21
|
+
this.welcomeText$ = this.content.fromContent({
|
|
22
|
+
key: 'welcome',
|
|
23
|
+
interpolation: { appName: 'Mi Aplicación' },
|
|
24
|
+
});
|
|
25
|
+
// Contenido específico del componente Language (con className)
|
|
26
|
+
this.spanishText$ = this.content.fromContent({
|
|
27
|
+
className: 'Language',
|
|
28
|
+
key: 'spanish',
|
|
29
|
+
});
|
|
30
|
+
this.englishText$ = this.content.fromContent({
|
|
31
|
+
className: 'Language',
|
|
32
|
+
key: 'english',
|
|
33
|
+
});
|
|
34
|
+
this.descriptionText$ = this.content.fromContent({
|
|
35
|
+
className: 'Language',
|
|
36
|
+
key: 'description',
|
|
37
|
+
});
|
|
38
|
+
// Para debug - texto síncrono
|
|
39
|
+
this.saveTextSync = '';
|
|
40
|
+
this.dashboardTextSync = '';
|
|
41
|
+
this.languageDescSync = '';
|
|
42
|
+
}
|
|
43
|
+
ngOnInit() {
|
|
44
|
+
// Obtener idioma actual
|
|
45
|
+
this.content.currentLang$.subscribe(lang => {
|
|
46
|
+
this.currentLang = lang;
|
|
47
|
+
});
|
|
48
|
+
// Obtener textos síncronos para debug
|
|
49
|
+
this.updateSyncTexts();
|
|
50
|
+
// 🔍 DIAGNÓSTICO: Verificar configuración
|
|
51
|
+
this.diagnoseConfiguration();
|
|
52
|
+
}
|
|
53
|
+
switchLanguage() {
|
|
54
|
+
const newLang = this.currentLang === 'es' ? LangOption.EN : LangOption.ES;
|
|
55
|
+
this.content.setLang(newLang);
|
|
56
|
+
// Actualizar textos síncronos después del cambio
|
|
57
|
+
setTimeout(() => {
|
|
58
|
+
this.updateSyncTexts();
|
|
59
|
+
}, 100);
|
|
60
|
+
}
|
|
61
|
+
updateSyncTexts() {
|
|
62
|
+
// Contenido global predefinido
|
|
63
|
+
this.saveTextSync = this.content.getText('save');
|
|
64
|
+
// Tu contenido global personalizado
|
|
65
|
+
this.dashboardTextSync = this.content.getText('dashboard');
|
|
66
|
+
// Contenido específico del componente - SOLUCIÓN TEMPORAL
|
|
67
|
+
// En lugar de usar getText, acceder directamente
|
|
68
|
+
try {
|
|
69
|
+
const langService = this.content.langService;
|
|
70
|
+
const languageContent = langService.content['Language'];
|
|
71
|
+
const currentLang = langService.selectedLang?.value || 'es';
|
|
72
|
+
const classContent = languageContent?.Content?.[currentLang];
|
|
73
|
+
this.languageDescSync = classContent?.['description'] || 'No encontrado';
|
|
74
|
+
}
|
|
75
|
+
catch (error) {
|
|
76
|
+
this.languageDescSync = `Error: ${error}`;
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
diagnoseConfiguration() {
|
|
80
|
+
console.log('=== DIAGNÓSTICO DE CONFIGURACIÓN ===');
|
|
81
|
+
// Verificar acceso directo al servicio
|
|
82
|
+
console.log('ContentService available:', !!this.content);
|
|
83
|
+
console.log('Current language:', this.content.currentLang);
|
|
84
|
+
// 🔍 Diagnóstico detallado del contenido Language
|
|
85
|
+
console.log('\n--- DIAGNÓSTICO DETALLADO LANGUAGE ---');
|
|
86
|
+
try {
|
|
87
|
+
// Acceder directamente al LangService
|
|
88
|
+
const langService = this.content.langService;
|
|
89
|
+
console.log('LangService available:', !!langService);
|
|
90
|
+
// Verificar si existe el contenido Language
|
|
91
|
+
const languageContent = langService.content['Language'];
|
|
92
|
+
console.log('Language content exists:', !!languageContent);
|
|
93
|
+
if (languageContent) {
|
|
94
|
+
console.log('Language content structure:', languageContent);
|
|
95
|
+
console.log('Language content.Content:', languageContent.Content);
|
|
96
|
+
// Verificar contenido en español
|
|
97
|
+
const esContent = languageContent.Content?.es;
|
|
98
|
+
console.log('ES content:', esContent);
|
|
99
|
+
console.log('ES description:', esContent?.description);
|
|
100
|
+
// Verificar contenido en inglés
|
|
101
|
+
const enContent = languageContent.Content?.en;
|
|
102
|
+
console.log('EN content:', enContent);
|
|
103
|
+
console.log('EN description:', enContent?.description);
|
|
104
|
+
}
|
|
105
|
+
// Verificar current lang
|
|
106
|
+
const currentLang = langService.selectedLang?.value;
|
|
107
|
+
console.log('Current selected language:', currentLang);
|
|
108
|
+
}
|
|
109
|
+
catch (error) {
|
|
110
|
+
console.log('❌ Error accessing LangService:', error);
|
|
111
|
+
}
|
|
112
|
+
// Intentar acceso síncrono a contenido global predefinido
|
|
113
|
+
try {
|
|
114
|
+
const saveText = this.content.getText('save');
|
|
115
|
+
console.log('✅ Global predefinido (save):', saveText);
|
|
116
|
+
}
|
|
117
|
+
catch (error) {
|
|
118
|
+
console.log('❌ Error global predefinido (save):', error);
|
|
119
|
+
}
|
|
120
|
+
// Intentar acceso síncrono a contenido global personalizado
|
|
121
|
+
try {
|
|
122
|
+
const dashboardText = this.content.getText('dashboard');
|
|
123
|
+
console.log('✅ Global personalizado (dashboard):', dashboardText);
|
|
124
|
+
}
|
|
125
|
+
catch (error) {
|
|
126
|
+
console.log('❌ Error global personalizado (dashboard):', error);
|
|
127
|
+
}
|
|
128
|
+
// Intentar acceso síncrono a contenido de componente
|
|
129
|
+
try {
|
|
130
|
+
const spanishText = this.content.getText('Language', 'spanish');
|
|
131
|
+
console.log('✅ Componente Language (spanish):', spanishText);
|
|
132
|
+
// 🔍 Debug paso a paso para spanish
|
|
133
|
+
const langService = this.content.langService;
|
|
134
|
+
const languageContent = langService.content['Language'];
|
|
135
|
+
const currentLang = langService.selectedLang?.value;
|
|
136
|
+
const classContent = languageContent?.Content[currentLang];
|
|
137
|
+
console.log('🔍 Debug spanish - currentLang:', currentLang);
|
|
138
|
+
console.log('🔍 Debug spanish - classContent:', classContent);
|
|
139
|
+
console.log('🔍 Debug spanish - classContent["spanish"]:', classContent?.['spanish']);
|
|
140
|
+
}
|
|
141
|
+
catch (error) {
|
|
142
|
+
console.log('❌ Error componente Language (spanish):', error);
|
|
143
|
+
}
|
|
144
|
+
// Intentar acceso síncrono a contenido de componente
|
|
145
|
+
try {
|
|
146
|
+
const descriptionText = this.content.getText('Language', 'description');
|
|
147
|
+
console.log('✅ Componente Language (description):', descriptionText);
|
|
148
|
+
// 🔍 Debug paso a paso para description
|
|
149
|
+
const langService = this.content.langService;
|
|
150
|
+
const languageContent = langService.content['Language'];
|
|
151
|
+
const currentLang = langService.selectedLang?.value;
|
|
152
|
+
const classContent = languageContent?.Content[currentLang];
|
|
153
|
+
console.log('🔍 Debug description - currentLang:', currentLang);
|
|
154
|
+
console.log('🔍 Debug description - classContent:', classContent);
|
|
155
|
+
console.log('🔍 Debug description - classContent["description"]:', classContent?.['description']);
|
|
156
|
+
}
|
|
157
|
+
catch (error) {
|
|
158
|
+
console.log('❌ Error componente Language (description):', error);
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: CustomContentDemoComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
162
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: CustomContentDemoComponent, isStandalone: true, selector: "app-custom-content-demo", ngImport: i0, template: `
|
|
163
|
+
<ion-card>
|
|
164
|
+
<ion-card-header>
|
|
165
|
+
<ion-card-title>Demo de Contenido Personalizado</ion-card-title>
|
|
166
|
+
</ion-card-header>
|
|
167
|
+
|
|
168
|
+
<ion-card-content>
|
|
169
|
+
<!-- Contenido global predefinido -->
|
|
170
|
+
<div style="margin-bottom: 20px;">
|
|
171
|
+
<h3>Contenido Global Predefinido:</h3>
|
|
172
|
+
<ion-button>{{ saveText$ | async }}</ion-button>
|
|
173
|
+
<ion-button color="medium">{{ cancelText$ | async }}</ion-button>
|
|
174
|
+
<ion-button color="danger">{{ deleteText$ | async }}</ion-button>
|
|
175
|
+
<p><strong>Estado:</strong> {{ loadingText$ | async }}</p>
|
|
176
|
+
</div>
|
|
177
|
+
|
|
178
|
+
<!-- Tu contenido global personalizado -->
|
|
179
|
+
<div style="margin-bottom: 20px;">
|
|
180
|
+
<h3>Tu Contenido Global Personalizado:</h3>
|
|
181
|
+
<p><strong>Sección:</strong> {{ dashboardText$ | async }}</p>
|
|
182
|
+
<p><strong>Usuario:</strong> {{ profileText$ | async }}</p>
|
|
183
|
+
<p><strong>Config:</strong> {{ settingsText$ | async }}</p>
|
|
184
|
+
<p>{{ welcomeText$ | async }}</p>
|
|
185
|
+
</div>
|
|
186
|
+
|
|
187
|
+
<!-- Contenido específico del componente Language -->
|
|
188
|
+
<div style="margin-bottom: 20px;">
|
|
189
|
+
<h3>Contenido del Componente Language:</h3>
|
|
190
|
+
<ion-item>
|
|
191
|
+
<ion-label> <strong>Español:</strong> {{ spanishText$ | async }} </ion-label>
|
|
192
|
+
</ion-item>
|
|
193
|
+
<ion-item>
|
|
194
|
+
<ion-label> <strong>Inglés:</strong> {{ englishText$ | async }} </ion-label>
|
|
195
|
+
</ion-item>
|
|
196
|
+
<p>
|
|
197
|
+
<em>{{ descriptionText$ | async }}</em>
|
|
198
|
+
</p>
|
|
199
|
+
</div>
|
|
200
|
+
|
|
201
|
+
<!-- Botón para cambiar idioma -->
|
|
202
|
+
<div>
|
|
203
|
+
<h3>Control de Idioma:</h3>
|
|
204
|
+
<ion-button (click)="switchLanguage()" color="secondary">
|
|
205
|
+
Cambiar a {{ currentLang === 'es' ? 'English' : 'Español' }}
|
|
206
|
+
</ion-button>
|
|
207
|
+
<p>
|
|
208
|
+
<small>Idioma actual: {{ currentLang }}</small>
|
|
209
|
+
</p>
|
|
210
|
+
</div>
|
|
211
|
+
|
|
212
|
+
<!-- Debug info -->
|
|
213
|
+
<div style="margin-top: 20px; padding: 10px; background: #f5f5f5; border-radius: 4px;">
|
|
214
|
+
<h4>Debug Info:</h4>
|
|
215
|
+
<p><strong>Save (sync):</strong> {{ saveTextSync }}</p>
|
|
216
|
+
<p><strong>Dashboard (sync):</strong> {{ dashboardTextSync }}</p>
|
|
217
|
+
<p><strong>Language Description (sync):</strong> {{ languageDescSync }}</p>
|
|
218
|
+
</div>
|
|
219
|
+
</ion-card-content>
|
|
220
|
+
</ion-card>
|
|
221
|
+
`, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { 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"] }] }); }
|
|
222
|
+
}
|
|
223
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: CustomContentDemoComponent, decorators: [{
|
|
224
|
+
type: Component,
|
|
225
|
+
args: [{
|
|
226
|
+
selector: 'app-custom-content-demo',
|
|
227
|
+
standalone: true,
|
|
228
|
+
imports: [CommonModule, IonCard, IonCardContent, IonCardHeader, IonCardTitle, IonButton, IonItem, IonLabel],
|
|
229
|
+
template: `
|
|
230
|
+
<ion-card>
|
|
231
|
+
<ion-card-header>
|
|
232
|
+
<ion-card-title>Demo de Contenido Personalizado</ion-card-title>
|
|
233
|
+
</ion-card-header>
|
|
234
|
+
|
|
235
|
+
<ion-card-content>
|
|
236
|
+
<!-- Contenido global predefinido -->
|
|
237
|
+
<div style="margin-bottom: 20px;">
|
|
238
|
+
<h3>Contenido Global Predefinido:</h3>
|
|
239
|
+
<ion-button>{{ saveText$ | async }}</ion-button>
|
|
240
|
+
<ion-button color="medium">{{ cancelText$ | async }}</ion-button>
|
|
241
|
+
<ion-button color="danger">{{ deleteText$ | async }}</ion-button>
|
|
242
|
+
<p><strong>Estado:</strong> {{ loadingText$ | async }}</p>
|
|
243
|
+
</div>
|
|
244
|
+
|
|
245
|
+
<!-- Tu contenido global personalizado -->
|
|
246
|
+
<div style="margin-bottom: 20px;">
|
|
247
|
+
<h3>Tu Contenido Global Personalizado:</h3>
|
|
248
|
+
<p><strong>Sección:</strong> {{ dashboardText$ | async }}</p>
|
|
249
|
+
<p><strong>Usuario:</strong> {{ profileText$ | async }}</p>
|
|
250
|
+
<p><strong>Config:</strong> {{ settingsText$ | async }}</p>
|
|
251
|
+
<p>{{ welcomeText$ | async }}</p>
|
|
252
|
+
</div>
|
|
253
|
+
|
|
254
|
+
<!-- Contenido específico del componente Language -->
|
|
255
|
+
<div style="margin-bottom: 20px;">
|
|
256
|
+
<h3>Contenido del Componente Language:</h3>
|
|
257
|
+
<ion-item>
|
|
258
|
+
<ion-label> <strong>Español:</strong> {{ spanishText$ | async }} </ion-label>
|
|
259
|
+
</ion-item>
|
|
260
|
+
<ion-item>
|
|
261
|
+
<ion-label> <strong>Inglés:</strong> {{ englishText$ | async }} </ion-label>
|
|
262
|
+
</ion-item>
|
|
263
|
+
<p>
|
|
264
|
+
<em>{{ descriptionText$ | async }}</em>
|
|
265
|
+
</p>
|
|
266
|
+
</div>
|
|
267
|
+
|
|
268
|
+
<!-- Botón para cambiar idioma -->
|
|
269
|
+
<div>
|
|
270
|
+
<h3>Control de Idioma:</h3>
|
|
271
|
+
<ion-button (click)="switchLanguage()" color="secondary">
|
|
272
|
+
Cambiar a {{ currentLang === 'es' ? 'English' : 'Español' }}
|
|
273
|
+
</ion-button>
|
|
274
|
+
<p>
|
|
275
|
+
<small>Idioma actual: {{ currentLang }}</small>
|
|
276
|
+
</p>
|
|
277
|
+
</div>
|
|
278
|
+
|
|
279
|
+
<!-- Debug info -->
|
|
280
|
+
<div style="margin-top: 20px; padding: 10px; background: #f5f5f5; border-radius: 4px;">
|
|
281
|
+
<h4>Debug Info:</h4>
|
|
282
|
+
<p><strong>Save (sync):</strong> {{ saveTextSync }}</p>
|
|
283
|
+
<p><strong>Dashboard (sync):</strong> {{ dashboardTextSync }}</p>
|
|
284
|
+
<p><strong>Language Description (sync):</strong> {{ languageDescSync }}</p>
|
|
285
|
+
</div>
|
|
286
|
+
</ion-card-content>
|
|
287
|
+
</ion-card>
|
|
288
|
+
`,
|
|
289
|
+
}]
|
|
290
|
+
}] });
|
|
291
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-content-demo.component.js","sourceRoot":"","sources":["../../../../../projects/valtech-components/src/lib/examples/custom-content-demo.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, inject, OnInit } from '@angular/core';\nimport {\n  IonButton,\n  IonCard,\n  IonCardContent,\n  IonCardHeader,\n  IonCardTitle,\n  IonItem,\n  IonLabel,\n} from '@ionic/angular/standalone';\nimport { ContentService } from '../services/content.service';\nimport { LangOption } from '../services/lang-provider/types';\n\n@Component({\n  selector: 'app-custom-content-demo',\n  standalone: true,\n  imports: [CommonModule, IonCard, IonCardContent, IonCardHeader, IonCardTitle, IonButton, IonItem, IonLabel],\n  template: `\n    <ion-card>\n      <ion-card-header>\n        <ion-card-title>Demo de Contenido Personalizado</ion-card-title>\n      </ion-card-header>\n\n      <ion-card-content>\n        <!-- Contenido global predefinido -->\n        <div style=\"margin-bottom: 20px;\">\n          <h3>Contenido Global Predefinido:</h3>\n          <ion-button>{{ saveText$ | async }}</ion-button>\n          <ion-button color=\"medium\">{{ cancelText$ | async }}</ion-button>\n          <ion-button color=\"danger\">{{ deleteText$ | async }}</ion-button>\n          <p><strong>Estado:</strong> {{ loadingText$ | async }}</p>\n        </div>\n\n        <!-- Tu contenido global personalizado -->\n        <div style=\"margin-bottom: 20px;\">\n          <h3>Tu Contenido Global Personalizado:</h3>\n          <p><strong>Sección:</strong> {{ dashboardText$ | async }}</p>\n          <p><strong>Usuario:</strong> {{ profileText$ | async }}</p>\n          <p><strong>Config:</strong> {{ settingsText$ | async }}</p>\n          <p>{{ welcomeText$ | async }}</p>\n        </div>\n\n        <!-- Contenido específico del componente Language -->\n        <div style=\"margin-bottom: 20px;\">\n          <h3>Contenido del Componente Language:</h3>\n          <ion-item>\n            <ion-label> <strong>Español:</strong> {{ spanishText$ | async }} </ion-label>\n          </ion-item>\n          <ion-item>\n            <ion-label> <strong>Inglés:</strong> {{ englishText$ | async }} </ion-label>\n          </ion-item>\n          <p>\n            <em>{{ descriptionText$ | async }}</em>\n          </p>\n        </div>\n\n        <!-- Botón para cambiar idioma -->\n        <div>\n          <h3>Control de Idioma:</h3>\n          <ion-button (click)=\"switchLanguage()\" color=\"secondary\">\n            Cambiar a {{ currentLang === 'es' ? 'English' : 'Español' }}\n          </ion-button>\n          <p>\n            <small>Idioma actual: {{ currentLang }}</small>\n          </p>\n        </div>\n\n        <!-- Debug info -->\n        <div style=\"margin-top: 20px; padding: 10px; background: #f5f5f5; border-radius: 4px;\">\n          <h4>Debug Info:</h4>\n          <p><strong>Save (sync):</strong> {{ saveTextSync }}</p>\n          <p><strong>Dashboard (sync):</strong> {{ dashboardTextSync }}</p>\n          <p><strong>Language Description (sync):</strong> {{ languageDescSync }}</p>\n        </div>\n      </ion-card-content>\n    </ion-card>\n  `,\n})\nexport class CustomContentDemoComponent implements OnInit {\n  private content = inject(ContentService);\n\n  currentLang = 'es';\n\n  // Contenido global predefinido (sin className)\n  saveText$ = this.content.fromContent({ key: 'save' });\n  cancelText$ = this.content.fromContent({ key: 'cancel' });\n  deleteText$ = this.content.fromContent({ key: 'delete' });\n  loadingText$ = this.content.fromContent({ key: 'loading' });\n\n  // Tu contenido global personalizado (sin className)\n  dashboardText$ = this.content.fromContent({ key: 'dashboard' });\n  profileText$ = this.content.fromContent({ key: 'profile' });\n  settingsText$ = this.content.fromContent({ key: 'settings' });\n  welcomeText$ = this.content.fromContent({\n    key: 'welcome',\n    interpolation: { appName: 'Mi Aplicación' },\n  });\n\n  // Contenido específico del componente Language (con className)\n  spanishText$ = this.content.fromContent({\n    className: 'Language',\n    key: 'spanish',\n  });\n  englishText$ = this.content.fromContent({\n    className: 'Language',\n    key: 'english',\n  });\n  descriptionText$ = this.content.fromContent({\n    className: 'Language',\n    key: 'description',\n  });\n\n  // Para debug - texto síncrono\n  saveTextSync = '';\n  dashboardTextSync = '';\n  languageDescSync = '';\n\n  ngOnInit() {\n    // Obtener idioma actual\n    this.content.currentLang$.subscribe(lang => {\n      this.currentLang = lang;\n    });\n\n    // Obtener textos síncronos para debug\n    this.updateSyncTexts();\n\n    // 🔍 DIAGNÓSTICO: Verificar configuración\n    this.diagnoseConfiguration();\n  }\n\n  switchLanguage() {\n    const newLang = this.currentLang === 'es' ? LangOption.EN : LangOption.ES;\n    this.content.setLang(newLang);\n\n    // Actualizar textos síncronos después del cambio\n    setTimeout(() => {\n      this.updateSyncTexts();\n    }, 100);\n  }\n\n  private updateSyncTexts() {\n    // Contenido global predefinido\n    this.saveTextSync = this.content.getText('save');\n\n    // Tu contenido global personalizado\n    this.dashboardTextSync = this.content.getText('dashboard');\n\n    // Contenido específico del componente - SOLUCIÓN TEMPORAL\n    // En lugar de usar getText, acceder directamente\n    try {\n      const langService = (this.content as any).langService;\n      const languageContent = langService.content['Language'];\n      const currentLang = langService.selectedLang?.value || 'es';\n      const classContent = languageContent?.Content?.[currentLang];\n      this.languageDescSync = classContent?.['description'] || 'No encontrado';\n    } catch (error) {\n      this.languageDescSync = `Error: ${error}`;\n    }\n  }\n\n  private diagnoseConfiguration() {\n    console.log('=== DIAGNÓSTICO DE CONFIGURACIÓN ===');\n\n    // Verificar acceso directo al servicio\n    console.log('ContentService available:', !!this.content);\n    console.log('Current language:', this.content.currentLang);\n\n    // 🔍 Diagnóstico detallado del contenido Language\n    console.log('\\n--- DIAGNÓSTICO DETALLADO LANGUAGE ---');\n    try {\n      // Acceder directamente al LangService\n      const langService = (this.content as any).langService;\n      console.log('LangService available:', !!langService);\n\n      // Verificar si existe el contenido Language\n      const languageContent = langService.content['Language'];\n      console.log('Language content exists:', !!languageContent);\n\n      if (languageContent) {\n        console.log('Language content structure:', languageContent);\n        console.log('Language content.Content:', languageContent.Content);\n\n        // Verificar contenido en español\n        const esContent = languageContent.Content?.es;\n        console.log('ES content:', esContent);\n        console.log('ES description:', esContent?.description);\n\n        // Verificar contenido en inglés\n        const enContent = languageContent.Content?.en;\n        console.log('EN content:', enContent);\n        console.log('EN description:', enContent?.description);\n      }\n\n      // Verificar current lang\n      const currentLang = langService.selectedLang?.value;\n      console.log('Current selected language:', currentLang);\n    } catch (error) {\n      console.log('❌ Error accessing LangService:', error);\n    }\n\n    // Intentar acceso síncrono a contenido global predefinido\n    try {\n      const saveText = this.content.getText('save');\n      console.log('✅ Global predefinido (save):', saveText);\n    } catch (error) {\n      console.log('❌ Error global predefinido (save):', error);\n    }\n\n    // Intentar acceso síncrono a contenido global personalizado\n    try {\n      const dashboardText = this.content.getText('dashboard');\n      console.log('✅ Global personalizado (dashboard):', dashboardText);\n    } catch (error) {\n      console.log('❌ Error global personalizado (dashboard):', error);\n    }\n\n    // Intentar acceso síncrono a contenido de componente\n    try {\n      const spanishText = this.content.getText('Language', 'spanish');\n      console.log('✅ Componente Language (spanish):', spanishText);\n\n      // 🔍 Debug paso a paso para spanish\n      const langService = (this.content as any).langService;\n      const languageContent = langService.content['Language'];\n      const currentLang = langService.selectedLang?.value;\n      const classContent = languageContent?.Content[currentLang];\n      console.log('🔍 Debug spanish - currentLang:', currentLang);\n      console.log('🔍 Debug spanish - classContent:', classContent);\n      console.log('🔍 Debug spanish - classContent[\"spanish\"]:', classContent?.['spanish']);\n    } catch (error) {\n      console.log('❌ Error componente Language (spanish):', error);\n    }\n\n    // Intentar acceso síncrono a contenido de componente\n    try {\n      const descriptionText = this.content.getText('Language', 'description');\n      console.log('✅ Componente Language (description):', descriptionText);\n\n      // 🔍 Debug paso a paso para description\n      const langService = (this.content as any).langService;\n      const languageContent = langService.content['Language'];\n      const currentLang = langService.selectedLang?.value;\n      const classContent = languageContent?.Content[currentLang];\n      console.log('🔍 Debug description - currentLang:', currentLang);\n      console.log('🔍 Debug description - classContent:', classContent);\n      console.log('🔍 Debug description - classContent[\"description\"]:', classContent?.['description']);\n    } catch (error) {\n      console.log('❌ Error componente Language (description):', error);\n    }\n  }\n}\n"]}
|