valtech-components 2.0.288 → 2.0.290
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 +20 -20
- package/esm2022/lib/components/atoms/text/text.component.mjs +3 -3
- package/esm2022/lib/examples/link-processing-example.component.mjs +73 -1
- package/esm2022/lib/services/link-processor.service.mjs +94 -43
- package/fesm2022/valtech-components.mjs +167 -44
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/examples/link-processing-example.component.d.ts +3 -0
- package/lib/services/link-processor.service.d.ts +17 -11
- package/package.json +1 -1
- package/src/lib/components/styles/overrides.scss +34 -0
package/README.md
CHANGED
|
@@ -184,7 +184,7 @@ export class UserComponent {
|
|
|
184
184
|
The library now includes automatic link detection and processing capabilities. Convert URLs and internal routes in text content into clickable links automatically:
|
|
185
185
|
|
|
186
186
|
```typescript
|
|
187
|
-
// Basic usage
|
|
187
|
+
// Basic usage with direct URLs
|
|
188
188
|
<val-text [props]="{
|
|
189
189
|
content: 'Visit https://angular.io or go to /dashboard',
|
|
190
190
|
processLinks: true,
|
|
@@ -194,18 +194,27 @@ The library now includes automatic link detection and processing capabilities. C
|
|
|
194
194
|
}
|
|
195
195
|
}"></val-text>
|
|
196
196
|
|
|
197
|
-
//
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
197
|
+
// Markdown-style links [text](url)
|
|
198
|
+
<val-text [props]="{
|
|
199
|
+
content: 'Check [the documentation](https://angular.io/docs) for details',
|
|
200
|
+
processLinks: true,
|
|
201
|
+
linkConfig: {
|
|
202
|
+
processMarkdownLinks: true,
|
|
203
|
+
openExternalInNewTab: true
|
|
204
|
+
}
|
|
205
|
+
}"></val-text>
|
|
206
|
+
|
|
207
|
+
// Mixed formats in the same text
|
|
208
|
+
<val-text [props]="{
|
|
209
|
+
content: 'Read [the guide](https://angular.io/guide), check https://github.com/angular, or go to /examples',
|
|
210
|
+
processLinks: true
|
|
211
|
+
}"></val-text>
|
|
204
212
|
```
|
|
205
213
|
|
|
206
214
|
### Features
|
|
207
215
|
|
|
208
|
-
- **Automatic Detection**: Identifies external URLs (http/https)
|
|
216
|
+
- **Automatic Detection**: Identifies external URLs (http/https), internal routes (starting with /), and Markdown-style links `[text](url)`
|
|
217
|
+
- **Improved Regex**: Excludes punctuation marks (`,`, `;`, `.`, `!`, `?`, `()`) from URLs to prevent broken links
|
|
209
218
|
- **Configurable Behavior**: Control whether links open in new tabs
|
|
210
219
|
- **Custom Styling**: Apply custom CSS classes to links
|
|
211
220
|
- **Secure Processing**: Uses Angular's DomSanitizer for safe HTML generation
|
|
@@ -225,13 +234,12 @@ processText(text: string) {
|
|
|
225
234
|
// Check if text contains links
|
|
226
235
|
const hasLinks = this.linkProcessor.hasLinks(text);
|
|
227
236
|
|
|
228
|
-
// Extract all links with their types
|
|
237
|
+
// Extract all links with their types and text
|
|
229
238
|
const links = this.linkProcessor.extractLinks(text);
|
|
239
|
+
// Returns: [{ url: '...', type: 'external|internal', text: '...' }]
|
|
230
240
|
}
|
|
231
241
|
```
|
|
232
242
|
|
|
233
|
-
See the **[Link Processing Guide](./LINK_PROCESSING_GUIDE.md)** for complete documentation and examples.
|
|
234
|
-
|
|
235
243
|
## Project Structure
|
|
236
244
|
|
|
237
245
|
- `src/lib/components/atoms/` – Basic UI elements (e.g., button, icon, text)
|
|
@@ -243,14 +251,6 @@ See the **[Link Processing Guide](./LINK_PROCESSING_GUIDE.md)** for complete doc
|
|
|
243
251
|
- `src/lib/services/` – Utility and helper services (e.g., theme, navigation, i18n)
|
|
244
252
|
- `src/lib/shared/` – Shared constants and utility functions
|
|
245
253
|
|
|
246
|
-
## 📚 Examples
|
|
247
|
-
|
|
248
|
-
The library includes example components that demonstrate best practices:
|
|
249
|
-
|
|
250
|
-
- `ReactiveContentExampleComponent` - Demonstrates the reactive content system with component-specific content
|
|
251
|
-
- `GlobalContentExampleComponent` - Comprehensive demonstration of global and mixed content usage
|
|
252
|
-
- `LinkProcessingExampleComponent` - Shows automatic link detection and processing in different scenarios
|
|
253
|
-
- Check the `_examples/` directory for more implementation examples
|
|
254
254
|
|
|
255
255
|
## License
|
|
256
256
|
|
|
@@ -141,7 +141,7 @@ export class TextComponent {
|
|
|
141
141
|
<p [class]="props.size" [class.bold]="props.bold">{{ displayContent$ | async }}</p>
|
|
142
142
|
}
|
|
143
143
|
</ion-text>
|
|
144
|
-
`, isInline: true, styles: ["
|
|
144
|
+
`, isInline: true, styles: [":root{--ion-color-primary: #7026df;--ion-color-primary-rgb: 112, 38, 223;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #6321c4;--ion-color-primary-tint: #7e3ce2;--ion-color-secondary: #e2ccff;--ion-color-secondary-rgb: 226, 204, 255;--ion-color-secondary-contrast: #000000;--ion-color-secondary-contrast-rgb: 0, 0, 0;--ion-color-secondary-shade: #c7b4e0;--ion-color-secondary-tint: #e5d1ff;--ion-color-texti: #354c69;--ion-color-texti-rgb: 53, 76, 105;--ion-color-texti-contrast: #ffffff;--ion-color-texti-contrast-rgb: 255, 255, 255;--ion-color-texti-shade: #2f435c;--ion-color-texti-tint: #495e78;--ion-color-darki: #090f1b;--ion-color-darki-rgb: 9, 15, 27;--ion-color-darki-contrast: #ffffff;--ion-color-darki-contrast-rgb: 255, 255, 255;--ion-color-darki-shade: #080d18;--ion-color-darki-tint: #222732;--ion-color-medium: #9e9e9e;--ion-color-medium-rgb: 158, 158, 158;--ion-color-medium-contrast: #000000;--ion-color-medium-contrast-rgb: 0, 0, 0;--ion-color-medium-shade: #8b8b8b;--ion-color-medium-tint: #a8a8a8}@media (prefers-color-scheme: dark){:root{--ion-color-texti: #8fc1ff;--ion-color-texti-rgb: 143, 193, 255;--ion-color-texti-contrast: #000000;--ion-color-texti-contrast-rgb: 0, 0, 0;--ion-color-texti-shade: #7eaae0;--ion-color-texti-tint: #9ac7ff;--ion-color-darki: #ffffff;--ion-color-darki-rgb: 255, 255, 255;--ion-color-darki-contrast: #000000;--ion-color-darki-contrast-rgb: 0, 0, 0;--ion-color-darki-shade: #e0e0e0;--ion-color-darki-tint: #ffffff;--ion-color-primary: #8f49f8;--ion-color-primary-rgb: 143,73,248;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255,255,255;--ion-color-primary-shade: #7e40da;--ion-color-primary-tint: #9a5bf9}}.ion-color-texti{--ion-color-base: var(--ion-color-texti);--ion-color-base-rgb: var(--ion-color-texti-rgb);--ion-color-contrast: var(--ion-color-texti-contrast);--ion-color-contrast-rgb: var(--ion-color-texti-contrast-rgb);--ion-color-shade: var(--ion-color-texti-shade);--ion-color-tint: var(--ion-color-texti-tint)}.ion-color-darki{--ion-color-base: var(--ion-color-darki);--ion-color-base-rgb: var(--ion-color-darki-rgb);--ion-color-contrast: var(--ion-color-darki-contrast);--ion-color-contrast-rgb: var(--ion-color-darki-contrast-rgb);--ion-color-shade: var(--ion-color-darki-shade);--ion-color-tint: var(--ion-color-darki-tint)}.small{font-size:.75rem;line-height:1.25rem;font-weight:400}.small.bold{font-size:.75rem;line-height:1.25rem;font-weight:700}.medium{font-size:.875rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.medium{font-size:1rem;line-height:1.5rem}}.medium.bold{font-size:.875rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.medium.bold{font-size:1rem;line-height:1.5rem}}.large{font-size:1rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.large{font-size:1.125rem;line-height:1.5rem}}.large.bold{font-size:1rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.large.bold{font-size:1.125rem;line-height:1.5rem}}.xlarge{font-size:1.125rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.xlarge{font-size:1.5rem;line-height:2rem}}.xlarge.bold{font-size:1.125rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.xlarge.bold{font-size:1.5rem;line-height:2rem}}\n"], dependencies: [{ kind: "component", type: IonText, selector: "ion-text", inputs: ["color", "mode"] }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "pipe", type: ProcessLinksPipe, name: "processLinks" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
145
145
|
}
|
|
146
146
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TextComponent, decorators: [{
|
|
147
147
|
type: Component,
|
|
@@ -157,7 +157,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
157
157
|
<p [class]="props.size" [class.bold]="props.bold">{{ displayContent$ | async }}</p>
|
|
158
158
|
}
|
|
159
159
|
</ion-text>
|
|
160
|
-
`, changeDetection: ChangeDetectionStrategy.OnPush, styles: ["
|
|
160
|
+
`, changeDetection: ChangeDetectionStrategy.OnPush, styles: [":root{--ion-color-primary: #7026df;--ion-color-primary-rgb: 112, 38, 223;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #6321c4;--ion-color-primary-tint: #7e3ce2;--ion-color-secondary: #e2ccff;--ion-color-secondary-rgb: 226, 204, 255;--ion-color-secondary-contrast: #000000;--ion-color-secondary-contrast-rgb: 0, 0, 0;--ion-color-secondary-shade: #c7b4e0;--ion-color-secondary-tint: #e5d1ff;--ion-color-texti: #354c69;--ion-color-texti-rgb: 53, 76, 105;--ion-color-texti-contrast: #ffffff;--ion-color-texti-contrast-rgb: 255, 255, 255;--ion-color-texti-shade: #2f435c;--ion-color-texti-tint: #495e78;--ion-color-darki: #090f1b;--ion-color-darki-rgb: 9, 15, 27;--ion-color-darki-contrast: #ffffff;--ion-color-darki-contrast-rgb: 255, 255, 255;--ion-color-darki-shade: #080d18;--ion-color-darki-tint: #222732;--ion-color-medium: #9e9e9e;--ion-color-medium-rgb: 158, 158, 158;--ion-color-medium-contrast: #000000;--ion-color-medium-contrast-rgb: 0, 0, 0;--ion-color-medium-shade: #8b8b8b;--ion-color-medium-tint: #a8a8a8}@media (prefers-color-scheme: dark){:root{--ion-color-texti: #8fc1ff;--ion-color-texti-rgb: 143, 193, 255;--ion-color-texti-contrast: #000000;--ion-color-texti-contrast-rgb: 0, 0, 0;--ion-color-texti-shade: #7eaae0;--ion-color-texti-tint: #9ac7ff;--ion-color-darki: #ffffff;--ion-color-darki-rgb: 255, 255, 255;--ion-color-darki-contrast: #000000;--ion-color-darki-contrast-rgb: 0, 0, 0;--ion-color-darki-shade: #e0e0e0;--ion-color-darki-tint: #ffffff;--ion-color-primary: #8f49f8;--ion-color-primary-rgb: 143,73,248;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255,255,255;--ion-color-primary-shade: #7e40da;--ion-color-primary-tint: #9a5bf9}}.ion-color-texti{--ion-color-base: var(--ion-color-texti);--ion-color-base-rgb: var(--ion-color-texti-rgb);--ion-color-contrast: var(--ion-color-texti-contrast);--ion-color-contrast-rgb: var(--ion-color-texti-contrast-rgb);--ion-color-shade: var(--ion-color-texti-shade);--ion-color-tint: var(--ion-color-texti-tint)}.ion-color-darki{--ion-color-base: var(--ion-color-darki);--ion-color-base-rgb: var(--ion-color-darki-rgb);--ion-color-contrast: var(--ion-color-darki-contrast);--ion-color-contrast-rgb: var(--ion-color-darki-contrast-rgb);--ion-color-shade: var(--ion-color-darki-shade);--ion-color-tint: var(--ion-color-darki-tint)}.small{font-size:.75rem;line-height:1.25rem;font-weight:400}.small.bold{font-size:.75rem;line-height:1.25rem;font-weight:700}.medium{font-size:.875rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.medium{font-size:1rem;line-height:1.5rem}}.medium.bold{font-size:.875rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.medium.bold{font-size:1rem;line-height:1.5rem}}.large{font-size:1rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.large{font-size:1.125rem;line-height:1.5rem}}.large.bold{font-size:1rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.large.bold{font-size:1.125rem;line-height:1.5rem}}.xlarge{font-size:1.125rem;line-height:1.5rem;font-weight:400}@media (min-width: 768px){.xlarge{font-size:1.5rem;line-height:2rem}}.xlarge.bold{font-size:1.125rem;line-height:1.5rem;font-weight:700}@media (min-width: 768px){.xlarge.bold{font-size:1.5rem;line-height:2rem}}\n"] }]
|
|
161
161
|
}], ctorParameters: () => [{ type: i1.ContentService }, { type: i2.LinkProcessorService }], propDecorators: { props: [{
|
|
162
162
|
type: Input
|
|
163
163
|
}] } });
|
|
@@ -195,4 +195,4 @@ export function createTextProps(contentConfig, styleConfig = {}) {
|
|
|
195
195
|
bold: styleConfig.bold || false,
|
|
196
196
|
};
|
|
197
197
|
}
|
|
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"]}
|
|
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"]}
|
|
@@ -70,6 +70,48 @@ export class LinkProcessingExampleComponent {
|
|
|
70
70
|
internalLinkClass: 'internal-same-tab',
|
|
71
71
|
},
|
|
72
72
|
};
|
|
73
|
+
this.markdownLinksProps = {
|
|
74
|
+
content: 'Consulta [la documentación de Angular](https://angular.io/docs) y ve a [configuración del perfil](/profile/settings) para más opciones.',
|
|
75
|
+
size: 'medium',
|
|
76
|
+
color: 'dark',
|
|
77
|
+
bold: false,
|
|
78
|
+
processLinks: true,
|
|
79
|
+
linkConfig: {
|
|
80
|
+
openExternalInNewTab: true,
|
|
81
|
+
openInternalInNewTab: false,
|
|
82
|
+
processMarkdownLinks: true,
|
|
83
|
+
linkClass: 'markdown-link',
|
|
84
|
+
externalLinkClass: 'markdown-external',
|
|
85
|
+
internalLinkClass: 'markdown-internal',
|
|
86
|
+
},
|
|
87
|
+
};
|
|
88
|
+
this.mixedFormatsProps = {
|
|
89
|
+
content: 'Aquí hay [documentación oficial](https://angular.io/docs), un enlace directo https://github.com/angular/angular, y una ruta interna /dashboard/analytics. ¡Todos funcionan!',
|
|
90
|
+
size: 'medium',
|
|
91
|
+
color: 'dark',
|
|
92
|
+
bold: false,
|
|
93
|
+
processLinks: true,
|
|
94
|
+
linkConfig: {
|
|
95
|
+
openExternalInNewTab: true,
|
|
96
|
+
openInternalInNewTab: false,
|
|
97
|
+
processMarkdownLinks: true,
|
|
98
|
+
linkClass: 'mixed-link',
|
|
99
|
+
externalLinkClass: 'mixed-external',
|
|
100
|
+
internalLinkClass: 'mixed-internal',
|
|
101
|
+
},
|
|
102
|
+
};
|
|
103
|
+
this.punctuationTestProps = {
|
|
104
|
+
content: 'URLs con puntuación: https://ionicframework.com/docs, revisa https://angular.io! También https://github.com/angular? Y finalmente https://typescript.org. ¡Todos deben funcionar correctamente!',
|
|
105
|
+
size: 'medium',
|
|
106
|
+
color: 'dark',
|
|
107
|
+
bold: false,
|
|
108
|
+
processLinks: true,
|
|
109
|
+
linkConfig: {
|
|
110
|
+
openExternalInNewTab: true,
|
|
111
|
+
linkClass: 'punctuation-test',
|
|
112
|
+
externalLinkClass: 'external-punct',
|
|
113
|
+
},
|
|
114
|
+
};
|
|
73
115
|
}
|
|
74
116
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: LinkProcessingExampleComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
75
117
|
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: LinkProcessingExampleComponent, isStandalone: true, selector: "val-link-processing-example", ngImport: i0, template: `
|
|
@@ -100,6 +142,21 @@ export class LinkProcessingExampleComponent {
|
|
|
100
142
|
<h3>Enlaces sin abrir en nueva pestaña:</h3>
|
|
101
143
|
<val-text [props]="sameTabLinksProps"></val-text>
|
|
102
144
|
</div>
|
|
145
|
+
|
|
146
|
+
<div class="example-section">
|
|
147
|
+
<h3>Enlaces estilo Markdown [texto](url):</h3>
|
|
148
|
+
<val-text [props]="markdownLinksProps"></val-text>
|
|
149
|
+
</div>
|
|
150
|
+
|
|
151
|
+
<div class="example-section">
|
|
152
|
+
<h3>Mezcla de enlaces directos y Markdown:</h3>
|
|
153
|
+
<val-text [props]="mixedFormatsProps"></val-text>
|
|
154
|
+
</div>
|
|
155
|
+
|
|
156
|
+
<div class="example-section">
|
|
157
|
+
<h3>Corrección de puntuación en URLs:</h3>
|
|
158
|
+
<val-text [props]="punctuationTestProps"></val-text>
|
|
159
|
+
</div>
|
|
103
160
|
</div>
|
|
104
161
|
`, isInline: true, styles: [".link-examples{padding:20px;max-width:800px}.example-section{margin-bottom:24px;padding:16px;border:1px solid var(--ion-color-light, #f4f5f8);border-radius:8px;background:var(--ion-color-light-tint, #f5f6f9)}h2{color:var(--ion-color-primary, #3880ff);margin-bottom:20px}h3{color:var(--ion-color-dark, #222428);margin-bottom:10px;font-size:16px}\n"], dependencies: [{ kind: "component", type: TextComponent, selector: "val-text", inputs: ["props"] }] }); }
|
|
105
162
|
}
|
|
@@ -133,7 +190,22 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
133
190
|
<h3>Enlaces sin abrir en nueva pestaña:</h3>
|
|
134
191
|
<val-text [props]="sameTabLinksProps"></val-text>
|
|
135
192
|
</div>
|
|
193
|
+
|
|
194
|
+
<div class="example-section">
|
|
195
|
+
<h3>Enlaces estilo Markdown [texto](url):</h3>
|
|
196
|
+
<val-text [props]="markdownLinksProps"></val-text>
|
|
197
|
+
</div>
|
|
198
|
+
|
|
199
|
+
<div class="example-section">
|
|
200
|
+
<h3>Mezcla de enlaces directos y Markdown:</h3>
|
|
201
|
+
<val-text [props]="mixedFormatsProps"></val-text>
|
|
202
|
+
</div>
|
|
203
|
+
|
|
204
|
+
<div class="example-section">
|
|
205
|
+
<h3>Corrección de puntuación en URLs:</h3>
|
|
206
|
+
<val-text [props]="punctuationTestProps"></val-text>
|
|
207
|
+
</div>
|
|
136
208
|
</div>
|
|
137
209
|
`, styles: [".link-examples{padding:20px;max-width:800px}.example-section{margin-bottom:24px;padding:16px;border:1px solid var(--ion-color-light, #f4f5f8);border-radius:8px;background:var(--ion-color-light-tint, #f5f6f9)}h2{color:var(--ion-color-primary, #3880ff);margin-bottom:20px}h3{color:var(--ion-color-dark, #222428);margin-bottom:10px;font-size:16px}\n"] }]
|
|
138
210
|
}] });
|
|
139
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGluay1wcm9jZXNzaW5nLWV4YW1wbGUuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvdmFsdGVjaC1jb21wb25lbnRzL3NyYy9saWIvZXhhbXBsZXMvbGluay1wcm9jZXNzaW5nLWV4YW1wbGUuY29tcG9uZW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDMUMsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLHlDQUF5QyxDQUFDOztBQUV4RTs7Ozs7Ozs7OztHQVVHO0FBK0RILE1BQU0sT0FBTyw4QkFBOEI7SUE5RDNDO1FBK0RFLG1CQUFjLEdBQWlCO1lBQzdCLE9BQU8sRUFBRSx1RkFBdUY7WUFDaEcsSUFBSSxFQUFFLFFBQVE7WUFDZCxLQUFLLEVBQUUsTUFBTTtZQUNiLElBQUksRUFBRSxLQUFLO1lBQ1gsWUFBWSxFQUFFLEtBQUs7U0FDcEIsQ0FBQztRQUVGLG9CQUFlLEdBQWlCO1lBQzlCLE9BQU8sRUFBRSx5RkFBeUY7WUFDbEcsSUFBSSxFQUFFLFFBQVE7WUFDZCxLQUFLLEVBQUUsTUFBTTtZQUNiLElBQUksRUFBRSxLQUFLO1lBQ1gsWUFBWSxFQUFFLElBQUk7U0FDbkIsQ0FBQztRQUVGLHFCQUFnQixHQUFpQjtZQUMvQixPQUFPLEVBQUUsZ0ZBQWdGO1lBQ3pGLElBQUksRUFBRSxRQUFRO1lBQ2QsS0FBSyxFQUFFLE1BQU07WUFDYixJQUFJLEVBQUUsS0FBSztZQUNYLFlBQVksRUFBRSxJQUFJO1lBQ2xCLFVBQVUsRUFBRTtnQkFDVixvQkFBb0IsRUFBRSxJQUFJO2dCQUMxQixvQkFBb0IsRUFBRSxLQUFLO2dCQUMzQixTQUFTLEVBQUUsbUJBQW1CO2dCQUM5QixpQkFBaUIsRUFBRSxpQkFBaUI7Z0JBQ3BDLGlCQUFpQixFQUFFLGlCQUFpQjthQUNyQztTQUNGLENBQUM7UUFFRixvQkFBZSxHQUFpQjtZQUM5QixPQUFPLEVBQ0wseUxBQXlMO1lBQzNMLElBQUksRUFBRSxRQUFRO1lBQ2QsS0FBSyxFQUFFLE1BQU07WUFDYixJQUFJLEVBQUUsS0FBSztZQUNYLFlBQVksRUFBRSxJQUFJO1lBQ2xCLFVBQVUsRUFBRTtnQkFDVixvQkFBb0IsRUFBRSxJQUFJO2dCQUMxQixvQkFBb0IsRUFBRSxLQUFLO2dCQUMzQixTQUFTLEVBQUUsZ0JBQWdCO2dCQUMzQixpQkFBaUIsRUFBRSxlQUFlO2dCQUNsQyxpQkFBaUIsRUFBRSxlQUFlO2FBQ25DO1NBQ0YsQ0FBQztRQUVGLHNCQUFpQixHQUFpQjtZQUNoQyxPQUFPLEVBQUUsc0VBQXNFO1lBQy9FLElBQUksRUFBRSxRQUFRO1lBQ2QsS0FBSyxFQUFFLE1BQU07WUFDYixJQUFJLEVBQUUsS0FBSztZQUNYLFlBQVksRUFBRSxJQUFJO1lBQ2xCLFVBQVUsRUFBRTtnQkFDVixvQkFBb0IsRUFBRSxLQUFLO2dCQUMzQixvQkFBb0IsRUFBRSxLQUFLO2dCQUMzQixTQUFTLEVBQUUsZUFBZTtnQkFDMUIsaUJBQWlCLEVBQUUsbUJBQW1CO2dCQUN0QyxpQkFBaUIsRUFBRSxtQkFBbUI7YUFDdkM7U0FDRixDQUFDO0tBQ0g7K0dBOURZLDhCQUE4QjttR0FBOUIsOEJBQThCLHVGQTFEL0I7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBNkJULG9hQTlCUyxhQUFhOzs0RkEyRFosOEJBQThCO2tCQTlEMUMsU0FBUzsrQkFDRSw2QkFBNkIsY0FDM0IsSUFBSSxXQUNQLENBQUMsYUFBYSxDQUFDLFlBQ2Q7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBNkJUIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBUZXh0Q29tcG9uZW50IH0gZnJvbSAnLi4vY29tcG9uZW50cy9hdG9tcy90ZXh0L3RleHQuY29tcG9uZW50JztcbmltcG9ydCB7IFRleHRNZXRhZGF0YSB9IGZyb20gJy4uL2NvbXBvbmVudHMvYXRvbXMvdGV4dC90eXBlcyc7XG4vKipcbiAqIExpbmtQcm9jZXNzaW5nRXhhbXBsZUNvbXBvbmVudCAtIENvbXBvbmVudGUgZGUgZWplbXBsbyBxdWUgZGVtdWVzdHJhIGVsIHByb2Nlc2FtaWVudG8gYXV0b23DoXRpY28gZGUgZW5sYWNlcy5cbiAqXG4gKiBFc3RlIGNvbXBvbmVudGUgbXVlc3RyYSBkaWZlcmVudGVzIGNhc29zIGRlIHVzbyBwYXJhIGVsIHByb2Nlc2FtaWVudG8gYXV0b23DoXRpY28gZGUgZW5sYWNlc1xuICogZW4gZWwgY29tcG9uZW50ZSB2YWwtdGV4dCwgaW5jbHV5ZW5kbyBlbmxhY2VzIGV4dGVybm9zLCBydXRhcyBpbnRlcm5hcyB5IGNvbmZpZ3VyYWNpb25lcyBwZXJzb25hbGl6YWRhcy5cbiAqXG4gKiBAZXhhbXBsZSBVc28gZW4gdGVtcGxhdGU6XG4gKiBgYGBodG1sXG4gKiA8dmFsLWxpbmstcHJvY2Vzc2luZy1leGFtcGxlPjwvdmFsLWxpbmstcHJvY2Vzc2luZy1leGFtcGxlPlxuICogYGBgXG4gKi9cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ3ZhbC1saW5rLXByb2Nlc3NpbmctZXhhbXBsZScsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtUZXh0Q29tcG9uZW50XSxcbiAgdGVtcGxhdGU6IGBcbiAgICA8ZGl2IGNsYXNzPVwibGluay1leGFtcGxlc1wiPlxuICAgICAgPGgyPkVqZW1wbG9zIGRlIFByb2Nlc2FtaWVudG8gZGUgRW5sYWNlczwvaDI+XG5cbiAgICAgIDxkaXYgY2xhc3M9XCJleGFtcGxlLXNlY3Rpb25cIj5cbiAgICAgICAgPGgzPlRleHRvIHNpbiBwcm9jZXNhbWllbnRvIGRlIGVubGFjZXM6PC9oMz5cbiAgICAgICAgPHZhbC10ZXh0IFtwcm9wc109XCJiYXNpY1RleHRQcm9wc1wiPjwvdmFsLXRleHQ+XG4gICAgICA8L2Rpdj5cblxuICAgICAgPGRpdiBjbGFzcz1cImV4YW1wbGUtc2VjdGlvblwiPlxuICAgICAgICA8aDM+VGV4dG8gY29uIHByb2Nlc2FtaWVudG8gYsOhc2ljbyBkZSBlbmxhY2VzOjwvaDM+XG4gICAgICAgIDx2YWwtdGV4dCBbcHJvcHNdPVwiYmFzaWNMaW5rc1Byb3BzXCI+PC92YWwtdGV4dD5cbiAgICAgIDwvZGl2PlxuXG4gICAgICA8ZGl2IGNsYXNzPVwiZXhhbXBsZS1zZWN0aW9uXCI+XG4gICAgICAgIDxoMz5FbmxhY2VzIGNvbiBjb25maWd1cmFjacOzbiBwZXJzb25hbGl6YWRhOjwvaDM+XG4gICAgICAgIDx2YWwtdGV4dCBbcHJvcHNdPVwiY3VzdG9tTGlua3NQcm9wc1wiPjwvdmFsLXRleHQ+XG4gICAgICA8L2Rpdj5cblxuICAgICAgPGRpdiBjbGFzcz1cImV4YW1wbGUtc2VjdGlvblwiPlxuICAgICAgICA8aDM+RW5sYWNlcyBpbnRlcm5vcyB5IGV4dGVybm9zIG1lemNsYWRvczo8L2gzPlxuICAgICAgICA8dmFsLXRleHQgW3Byb3BzXT1cIm1peGVkTGlua3NQcm9wc1wiPjwvdmFsLXRleHQ+XG4gICAgICA8L2Rpdj5cblxuICAgICAgPGRpdiBjbGFzcz1cImV4YW1wbGUtc2VjdGlvblwiPlxuICAgICAgICA8aDM+RW5sYWNlcyBzaW4gYWJyaXIgZW4gbnVldmEgcGVzdGHDsWE6PC9oMz5cbiAgICAgICAgPHZhbC10ZXh0IFtwcm9wc109XCJzYW1lVGFiTGlua3NQcm9wc1wiPjwvdmFsLXRleHQ+XG4gICAgICA8L2Rpdj5cbiAgICA8L2Rpdj5cbiAgYCxcbiAgc3R5bGVzOiBbXG4gICAgYFxuICAgICAgLmxpbmstZXhhbXBsZXMge1xuICAgICAgICBwYWRkaW5nOiAyMHB4O1xuICAgICAgICBtYXgtd2lkdGg6IDgwMHB4O1xuICAgICAgfVxuXG4gICAgICAuZXhhbXBsZS1zZWN0aW9uIHtcbiAgICAgICAgbWFyZ2luLWJvdHRvbTogMjRweDtcbiAgICAgICAgcGFkZGluZzogMTZweDtcbiAgICAgICAgYm9yZGVyOiAxcHggc29saWQgdmFyKC0taW9uLWNvbG9yLWxpZ2h0LCAjZjRmNWY4KTtcbiAgICAgICAgYm9yZGVyLXJhZGl1czogOHB4O1xuICAgICAgICBiYWNrZ3JvdW5kOiB2YXIoLS1pb24tY29sb3ItbGlnaHQtdGludCwgI2Y1ZjZmOSk7XG4gICAgICB9XG5cbiAgICAgIGgyIHtcbiAgICAgICAgY29sb3I6IHZhcigtLWlvbi1jb2xvci1wcmltYXJ5LCAjMzg4MGZmKTtcbiAgICAgICAgbWFyZ2luLWJvdHRvbTogMjBweDtcbiAgICAgIH1cblxuICAgICAgaDMge1xuICAgICAgICBjb2xvcjogdmFyKC0taW9uLWNvbG9yLWRhcmssICMyMjI0MjgpO1xuICAgICAgICBtYXJnaW4tYm90dG9tOiAxMHB4O1xuICAgICAgICBmb250LXNpemU6IDE2cHg7XG4gICAgICB9XG4gICAgYCxcbiAgXSxcbn0pXG5leHBvcnQgY2xhc3MgTGlua1Byb2Nlc3NpbmdFeGFtcGxlQ29tcG9uZW50IHtcbiAgYmFzaWNUZXh0UHJvcHM6IFRleHRNZXRhZGF0YSA9IHtcbiAgICBjb250ZW50OiAnRXN0ZSB0ZXh0byBjb250aWVuZSBodHRwczovL2FuZ3VsYXIuaW8geSAvZGFzaGJvYXJkIHBlcm8gbm8gc2UgcHJvY2VzYW4gY29tbyBlbmxhY2VzLicsXG4gICAgc2l6ZTogJ21lZGl1bScsXG4gICAgY29sb3I6ICdkYXJrJyxcbiAgICBib2xkOiBmYWxzZSxcbiAgICBwcm9jZXNzTGlua3M6IGZhbHNlLFxuICB9O1xuXG4gIGJhc2ljTGlua3NQcm9wczogVGV4dE1ldGFkYXRhID0ge1xuICAgIGNvbnRlbnQ6ICdWaXNpdGEgaHR0cHM6Ly9hbmd1bGFyLmlvIHBhcmEgZG9jdW1lbnRhY2nDs24gbyB2ZSBhIC9kYXNoYm9hcmQgcGFyYSBlbCBwYW5lbCBwcmluY2lwYWwuJyxcbiAgICBzaXplOiAnbWVkaXVtJyxcbiAgICBjb2xvcjogJ2RhcmsnLFxuICAgIGJvbGQ6IGZhbHNlLFxuICAgIHByb2Nlc3NMaW5rczogdHJ1ZSxcbiAgfTtcblxuICBjdXN0b21MaW5rc1Byb3BzOiBUZXh0TWV0YWRhdGEgPSB7XG4gICAgY29udGVudDogJ0VubGFjZXMgcGVyc29uYWxpemFkb3M6IGh0dHBzOi8vZ2l0aHViLmNvbS9hbmd1bGFyL2FuZ3VsYXIgeSAvcHJvZmlsZS9zZXR0aW5ncycsXG4gICAgc2l6ZTogJ21lZGl1bScsXG4gICAgY29sb3I6ICdkYXJrJyxcbiAgICBib2xkOiBmYWxzZSxcbiAgICBwcm9jZXNzTGlua3M6IHRydWUsXG4gICAgbGlua0NvbmZpZzoge1xuICAgICAgb3BlbkV4dGVybmFsSW5OZXdUYWI6IHRydWUsXG4gICAgICBvcGVuSW50ZXJuYWxJbk5ld1RhYjogZmFsc2UsXG4gICAgICBsaW5rQ2xhc3M6ICdjdXN0b20tbGluay1zdHlsZScsXG4gICAgICBleHRlcm5hbExpbmtDbGFzczogJ2V4dGVybmFsLWN1c3RvbScsXG4gICAgICBpbnRlcm5hbExpbmtDbGFzczogJ2ludGVybmFsLWN1c3RvbScsXG4gICAgfSxcbiAgfTtcblxuICBtaXhlZExpbmtzUHJvcHM6IFRleHRNZXRhZGF0YSA9IHtcbiAgICBjb250ZW50OlxuICAgICAgJ0NvbnN1bHRhIGxhIGRvY3VtZW50YWNpw7NuIGVuIGh0dHBzOi8vaW9uaWNmcmFtZXdvcmsuY29tL2RvY3MsIHJldmlzYSBlbCBjw7NkaWdvIGVuIGh0dHBzOi8vZ2l0aHViLmNvbS9pb25pYy10ZWFtL2lvbmljLWZyYW1ld29yaywgbyBuYXZlZ2EgYSAvY29tcG9uZW50cy9idXR0b25zIHBhcmEgZWplbXBsb3MgaW50ZXJub3MuJyxcbiAgICBzaXplOiAnbWVkaXVtJyxcbiAgICBjb2xvcjogJ2RhcmsnLFxuICAgIGJvbGQ6IGZhbHNlLFxuICAgIHByb2Nlc3NMaW5rczogdHJ1ZSxcbiAgICBsaW5rQ29uZmlnOiB7XG4gICAgICBvcGVuRXh0ZXJuYWxJbk5ld1RhYjogdHJ1ZSxcbiAgICAgIG9wZW5JbnRlcm5hbEluTmV3VGFiOiBmYWxzZSxcbiAgICAgIGxpbmtDbGFzczogJ3Byb2Nlc3NlZC1saW5rJyxcbiAgICAgIGV4dGVybmFsTGlua0NsYXNzOiAnZXh0ZXJuYWwtbGluaycsXG4gICAgICBpbnRlcm5hbExpbmtDbGFzczogJ2ludGVybmFsLWxpbmsnLFxuICAgIH0sXG4gIH07XG5cbiAgc2FtZVRhYkxpbmtzUHJvcHM6IFRleHRNZXRhZGF0YSA9IHtcbiAgICBjb250ZW50OiAnRXN0b3MgZW5sYWNlcyBubyBhYnJlbiBlbiBudWV2YSBwZXN0YcOxYTogaHR0cHM6Ly9leGFtcGxlLmNvbSB5IC9ob21lJyxcbiAgICBzaXplOiAnbWVkaXVtJyxcbiAgICBjb2xvcjogJ2RhcmsnLFxuICAgIGJvbGQ6IGZhbHNlLFxuICAgIHByb2Nlc3NMaW5rczogdHJ1ZSxcbiAgICBsaW5rQ29uZmlnOiB7XG4gICAgICBvcGVuRXh0ZXJuYWxJbk5ld1RhYjogZmFsc2UsXG4gICAgICBvcGVuSW50ZXJuYWxJbk5ld1RhYjogZmFsc2UsXG4gICAgICBsaW5rQ2xhc3M6ICdzYW1lLXRhYi1saW5rJyxcbiAgICAgIGV4dGVybmFsTGlua0NsYXNzOiAnZXh0ZXJuYWwtc2FtZS10YWInLFxuICAgICAgaW50ZXJuYWxMaW5rQ2xhc3M6ICdpbnRlcm5hbC1zYW1lLXRhYicsXG4gICAgfSxcbiAgfTtcbn1cbiJdfQ==
|
|
211
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"link-processing-example.component.js","sourceRoot":"","sources":["../../../../../projects/valtech-components/src/lib/examples/link-processing-example.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component } from '@angular/core';\nimport { TextComponent } from '../components/atoms/text/text.component';\nimport { TextMetadata } from '../components/atoms/text/types';\n/**\n * LinkProcessingExampleComponent - Componente de ejemplo que demuestra el procesamiento automático de enlaces.\n *\n * Este componente muestra diferentes casos de uso para el procesamiento automático de enlaces\n * en el componente val-text, incluyendo enlaces externos, rutas internas y configuraciones personalizadas.\n *\n * @example Uso en template:\n * ```html\n * <val-link-processing-example></val-link-processing-example>\n * ```\n */\n@Component({\n  selector: 'val-link-processing-example',\n  standalone: true,\n  imports: [TextComponent],\n  template: `\n    <div class=\"link-examples\">\n      <h2>Ejemplos de Procesamiento de Enlaces</h2>\n\n      <div class=\"example-section\">\n        <h3>Texto sin procesamiento de enlaces:</h3>\n        <val-text [props]=\"basicTextProps\"></val-text>\n      </div>\n\n      <div class=\"example-section\">\n        <h3>Texto con procesamiento básico de enlaces:</h3>\n        <val-text [props]=\"basicLinksProps\"></val-text>\n      </div>\n\n      <div class=\"example-section\">\n        <h3>Enlaces con configuración personalizada:</h3>\n        <val-text [props]=\"customLinksProps\"></val-text>\n      </div>\n\n      <div class=\"example-section\">\n        <h3>Enlaces internos y externos mezclados:</h3>\n        <val-text [props]=\"mixedLinksProps\"></val-text>\n      </div>\n\n      <div class=\"example-section\">\n        <h3>Enlaces sin abrir en nueva pestaña:</h3>\n        <val-text [props]=\"sameTabLinksProps\"></val-text>\n      </div>\n\n      <div class=\"example-section\">\n        <h3>Enlaces estilo Markdown [texto](url):</h3>\n        <val-text [props]=\"markdownLinksProps\"></val-text>\n      </div>\n\n      <div class=\"example-section\">\n        <h3>Mezcla de enlaces directos y Markdown:</h3>\n        <val-text [props]=\"mixedFormatsProps\"></val-text>\n      </div>\n\n      <div class=\"example-section\">\n        <h3>Corrección de puntuación en URLs:</h3>\n        <val-text [props]=\"punctuationTestProps\"></val-text>\n      </div>\n    </div>\n  `,\n  styles: [\n    `\n      .link-examples {\n        padding: 20px;\n        max-width: 800px;\n      }\n\n      .example-section {\n        margin-bottom: 24px;\n        padding: 16px;\n        border: 1px solid var(--ion-color-light, #f4f5f8);\n        border-radius: 8px;\n        background: var(--ion-color-light-tint, #f5f6f9);\n      }\n\n      h2 {\n        color: var(--ion-color-primary, #3880ff);\n        margin-bottom: 20px;\n      }\n\n      h3 {\n        color: var(--ion-color-dark, #222428);\n        margin-bottom: 10px;\n        font-size: 16px;\n      }\n    `,\n  ],\n})\nexport class LinkProcessingExampleComponent {\n  basicTextProps: TextMetadata = {\n    content: 'Este texto contiene https://angular.io y /dashboard pero no se procesan como enlaces.',\n    size: 'medium',\n    color: 'dark',\n    bold: false,\n    processLinks: false,\n  };\n\n  basicLinksProps: TextMetadata = {\n    content: 'Visita https://angular.io para documentación o ve a /dashboard para el panel principal.',\n    size: 'medium',\n    color: 'dark',\n    bold: false,\n    processLinks: true,\n  };\n\n  customLinksProps: TextMetadata = {\n    content: 'Enlaces personalizados: https://github.com/angular/angular y /profile/settings',\n    size: 'medium',\n    color: 'dark',\n    bold: false,\n    processLinks: true,\n    linkConfig: {\n      openExternalInNewTab: true,\n      openInternalInNewTab: false,\n      linkClass: 'custom-link-style',\n      externalLinkClass: 'external-custom',\n      internalLinkClass: 'internal-custom',\n    },\n  };\n\n  mixedLinksProps: TextMetadata = {\n    content:\n      'Consulta la documentación en https://ionicframework.com/docs, revisa el código en https://github.com/ionic-team/ionic-framework, o navega a /components/buttons para ejemplos internos.',\n    size: 'medium',\n    color: 'dark',\n    bold: false,\n    processLinks: true,\n    linkConfig: {\n      openExternalInNewTab: true,\n      openInternalInNewTab: false,\n      linkClass: 'processed-link',\n      externalLinkClass: 'external-link',\n      internalLinkClass: 'internal-link',\n    },\n  };\n\n  sameTabLinksProps: TextMetadata = {\n    content: 'Estos enlaces no abren en nueva pestaña: https://example.com y /home',\n    size: 'medium',\n    color: 'dark',\n    bold: false,\n    processLinks: true,\n    linkConfig: {\n      openExternalInNewTab: false,\n      openInternalInNewTab: false,\n      linkClass: 'same-tab-link',\n      externalLinkClass: 'external-same-tab',\n      internalLinkClass: 'internal-same-tab',\n    },\n  };\n\n  markdownLinksProps: TextMetadata = {\n    content:\n      'Consulta [la documentación de Angular](https://angular.io/docs) y ve a [configuración del perfil](/profile/settings) para más opciones.',\n    size: 'medium',\n    color: 'dark',\n    bold: false,\n    processLinks: true,\n    linkConfig: {\n      openExternalInNewTab: true,\n      openInternalInNewTab: false,\n      processMarkdownLinks: true,\n      linkClass: 'markdown-link',\n      externalLinkClass: 'markdown-external',\n      internalLinkClass: 'markdown-internal',\n    },\n  };\n\n  mixedFormatsProps: TextMetadata = {\n    content:\n      'Aquí hay [documentación oficial](https://angular.io/docs), un enlace directo https://github.com/angular/angular, y una ruta interna /dashboard/analytics. ¡Todos funcionan!',\n    size: 'medium',\n    color: 'dark',\n    bold: false,\n    processLinks: true,\n    linkConfig: {\n      openExternalInNewTab: true,\n      openInternalInNewTab: false,\n      processMarkdownLinks: true,\n      linkClass: 'mixed-link',\n      externalLinkClass: 'mixed-external',\n      internalLinkClass: 'mixed-internal',\n    },\n  };\n\n  punctuationTestProps: TextMetadata = {\n    content:\n      'URLs con puntuación: https://ionicframework.com/docs, revisa https://angular.io! También https://github.com/angular? Y finalmente https://typescript.org. ¡Todos deben funcionar correctamente!',\n    size: 'medium',\n    color: 'dark',\n    bold: false,\n    processLinks: true,\n    linkConfig: {\n      openExternalInNewTab: true,\n      linkClass: 'punctuation-test',\n      externalLinkClass: 'external-punct',\n    },\n  };\n}\n"]}
|
|
@@ -4,15 +4,15 @@ import * as i1 from "@angular/platform-browser";
|
|
|
4
4
|
/**
|
|
5
5
|
* LinkProcessorService - Service for processing text content to convert URLs and internal routes into clickable links.
|
|
6
6
|
*
|
|
7
|
-
* This service automatically detects external URLs (http/https)
|
|
8
|
-
* and converts them into HTML anchor elements with appropriate attributes.
|
|
7
|
+
* This service automatically detects external URLs (http/https), internal routes (starting with /),
|
|
8
|
+
* and Markdown-style links [text](url) and converts them into HTML anchor elements with appropriate attributes.
|
|
9
9
|
*
|
|
10
10
|
* @example Basic usage:
|
|
11
11
|
* ```typescript
|
|
12
12
|
* constructor(private linkProcessor: LinkProcessorService) {}
|
|
13
13
|
*
|
|
14
14
|
* processText() {
|
|
15
|
-
* const text = 'Visit https://example.com
|
|
15
|
+
* const text = 'Visit https://example.com, go to /profile, or [check docs](https://docs.example.com)';
|
|
16
16
|
* const processed = this.linkProcessor.processLinks(text);
|
|
17
17
|
* // Returns SafeHtml with clickable links
|
|
18
18
|
* }
|
|
@@ -21,14 +21,16 @@ import * as i1 from "@angular/platform-browser";
|
|
|
21
21
|
export class LinkProcessorService {
|
|
22
22
|
constructor(sanitizer) {
|
|
23
23
|
this.sanitizer = sanitizer;
|
|
24
|
-
// Regex para detectar URLs completas (http/https)
|
|
25
|
-
this.urlRegex = /(https?:\/\/[^\s]
|
|
26
|
-
// Regex para detectar rutas internas (empiezan con / pero no son URLs completas)
|
|
27
|
-
this.internalRouteRegex = /(\s|^)(\/[^\s]
|
|
24
|
+
// Regex para detectar URLs completas (http/https) - permite caracteres válidos pero excluye puntuación al final
|
|
25
|
+
this.urlRegex = /(https?:\/\/[^\s]+?)(?=[.,;!?()\s]|$)/g;
|
|
26
|
+
// Regex para detectar rutas internas (empiezan con / pero no son URLs completas) - excluye puntuación al final
|
|
27
|
+
this.internalRouteRegex = /(\s|^)(\/[^\s]*?)(?=[.,;!?()\s]|$)/g;
|
|
28
|
+
// Regex para detectar enlaces estilo Markdown [texto](url)
|
|
29
|
+
this.markdownLinkRegex = /\[([^\]]+)\]\(([^)]+)\)/g;
|
|
28
30
|
}
|
|
29
31
|
/**
|
|
30
32
|
* Procesa texto para convertir enlaces en elementos <a> clickeables.
|
|
31
|
-
* Detecta automáticamente URLs externas
|
|
33
|
+
* Detecta automáticamente URLs externas, rutas internas y enlaces estilo Markdown.
|
|
32
34
|
*
|
|
33
35
|
* @param text - Texto a procesar
|
|
34
36
|
* @param config - Configuración del procesamiento
|
|
@@ -37,10 +39,11 @@ export class LinkProcessorService {
|
|
|
37
39
|
* @example
|
|
38
40
|
* ```typescript
|
|
39
41
|
* const result = this.linkProcessor.processLinks(
|
|
40
|
-
* 'Visit https://example.com
|
|
42
|
+
* 'Visit https://example.com, go to /profile, or [check docs](https://docs.example.com)',
|
|
41
43
|
* {
|
|
42
44
|
* openExternalInNewTab: true,
|
|
43
45
|
* openInternalInNewTab: false,
|
|
46
|
+
* processMarkdownLinks: true,
|
|
44
47
|
* linkClass: 'custom-link'
|
|
45
48
|
* }
|
|
46
49
|
* );
|
|
@@ -49,33 +52,61 @@ export class LinkProcessorService {
|
|
|
49
52
|
processLinks(text, config = {}) {
|
|
50
53
|
if (!text)
|
|
51
54
|
return '';
|
|
52
|
-
const { openExternalInNewTab = true, openInternalInNewTab = false, linkClass = 'processed-link', externalLinkClass = 'external-link', internalLinkClass = 'internal-link', } = config;
|
|
55
|
+
const { openExternalInNewTab = true, openInternalInNewTab = false, linkClass = 'processed-link', externalLinkClass = 'external-link', internalLinkClass = 'internal-link', processMarkdownLinks = true, } = config;
|
|
53
56
|
let hasLinks = false;
|
|
54
57
|
let processedText = text;
|
|
55
|
-
// Procesar
|
|
56
|
-
if (
|
|
57
|
-
|
|
58
|
-
this.
|
|
59
|
-
|
|
60
|
-
const
|
|
61
|
-
const
|
|
62
|
-
|
|
58
|
+
// 1. Procesar enlaces estilo Markdown [texto](url) primero
|
|
59
|
+
if (processMarkdownLinks) {
|
|
60
|
+
this.markdownLinkRegex.lastIndex = 0; // Reset regex
|
|
61
|
+
processedText = processedText.replace(this.markdownLinkRegex, (match, linkText, url) => {
|
|
62
|
+
hasLinks = true;
|
|
63
|
+
const isExternal = /^https?:\/\//.test(url);
|
|
64
|
+
const target = (isExternal ? openExternalInNewTab : openInternalInNewTab)
|
|
65
|
+
? isExternal
|
|
66
|
+
? ' target="_blank" rel="noopener noreferrer"'
|
|
67
|
+
: ' target="_blank"'
|
|
68
|
+
: '';
|
|
69
|
+
const typeClass = isExternal ? externalLinkClass : internalLinkClass;
|
|
70
|
+
const classes = `${linkClass} ${typeClass}`.trim();
|
|
71
|
+
return `<a href="${url}"${target} class="${classes}">${linkText}</a>`;
|
|
63
72
|
});
|
|
64
73
|
}
|
|
65
|
-
// Procesar
|
|
66
|
-
|
|
74
|
+
// 2. Procesar URLs externas directas (solo si no están ya en un enlace HTML)
|
|
75
|
+
this.urlRegex.lastIndex = 0; // Reset regex
|
|
76
|
+
processedText = processedText.replace(this.urlRegex, (fullMatch, url) => {
|
|
77
|
+
// Verificar que no esté ya dentro de un enlace HTML existente
|
|
78
|
+
const urlPosition = processedText.indexOf(fullMatch);
|
|
79
|
+
const textBefore = processedText.substring(0, urlPosition);
|
|
80
|
+
// Buscar la última apertura y cierre de enlace antes de esta posición
|
|
81
|
+
const lastOpenTag = textBefore.lastIndexOf('<a ');
|
|
82
|
+
const lastCloseTag = textBefore.lastIndexOf('</a>');
|
|
83
|
+
// Si hay un tag <a abierto sin cerrar, no procesamos
|
|
84
|
+
if (lastOpenTag > lastCloseTag) {
|
|
85
|
+
return fullMatch; // Mantener original
|
|
86
|
+
}
|
|
67
87
|
hasLinks = true;
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
88
|
+
const target = openExternalInNewTab ? ' target="_blank" rel="noopener noreferrer"' : '';
|
|
89
|
+
const classes = `${linkClass} ${externalLinkClass}`.trim();
|
|
90
|
+
return `<a href="${url}"${target} class="${classes}">${url}</a>`;
|
|
91
|
+
});
|
|
92
|
+
// 3. Procesar rutas internas (solo si no están ya en un enlace HTML)
|
|
93
|
+
this.internalRouteRegex.lastIndex = 0; // Reset regex
|
|
94
|
+
processedText = processedText.replace(this.internalRouteRegex, (match, prefix, route) => {
|
|
95
|
+
// Verificar que no esté ya dentro de un enlace HTML existente
|
|
96
|
+
const matchPosition = processedText.indexOf(match);
|
|
97
|
+
const textBefore = processedText.substring(0, matchPosition);
|
|
98
|
+
// Buscar la última apertura y cierre de enlace antes de esta posición
|
|
99
|
+
const lastOpenTag = textBefore.lastIndexOf('<a ');
|
|
100
|
+
const lastCloseTag = textBefore.lastIndexOf('</a>');
|
|
101
|
+
// Si hay un tag <a abierto sin cerrar, no procesamos
|
|
102
|
+
if (lastOpenTag > lastCloseTag) {
|
|
103
|
+
return match; // Mantener original
|
|
104
|
+
}
|
|
105
|
+
hasLinks = true;
|
|
106
|
+
const target = openInternalInNewTab ? ' target="_blank"' : '';
|
|
107
|
+
const classes = `${linkClass} ${internalLinkClass}`.trim();
|
|
108
|
+
return `${prefix}<a href="${route}"${target} class="${classes}">${route}</a>`;
|
|
109
|
+
});
|
|
79
110
|
// Si hay enlaces, sanitizar el HTML
|
|
80
111
|
if (hasLinks) {
|
|
81
112
|
return this.sanitizer.bypassSecurityTrustHtml(processedText);
|
|
@@ -83,14 +114,14 @@ export class LinkProcessorService {
|
|
|
83
114
|
return text;
|
|
84
115
|
}
|
|
85
116
|
/**
|
|
86
|
-
* Detecta si un texto contiene enlaces (URLs o
|
|
117
|
+
* Detecta si un texto contiene enlaces (URLs, rutas internas o enlaces Markdown).
|
|
87
118
|
*
|
|
88
119
|
* @param text - Texto a analizar
|
|
89
120
|
* @returns true si contiene enlaces
|
|
90
121
|
*
|
|
91
122
|
* @example
|
|
92
123
|
* ```typescript
|
|
93
|
-
* const hasLinks = this.linkProcessor.hasLinks('Visit https://example.com');
|
|
124
|
+
* const hasLinks = this.linkProcessor.hasLinks('Visit https://example.com or [docs](https://docs.com)');
|
|
94
125
|
* // Returns: true
|
|
95
126
|
* ```
|
|
96
127
|
*/
|
|
@@ -100,20 +131,24 @@ export class LinkProcessorService {
|
|
|
100
131
|
// Reset regex indices
|
|
101
132
|
this.urlRegex.lastIndex = 0;
|
|
102
133
|
this.internalRouteRegex.lastIndex = 0;
|
|
103
|
-
|
|
134
|
+
this.markdownLinkRegex.lastIndex = 0;
|
|
135
|
+
return (this.urlRegex.test(text) ||
|
|
136
|
+
this.internalRouteRegex.test(text) ||
|
|
137
|
+
this.markdownLinkRegex.test(text));
|
|
104
138
|
}
|
|
105
139
|
/**
|
|
106
140
|
* Extrae todos los enlaces de un texto.
|
|
107
141
|
*
|
|
108
142
|
* @param text - Texto a analizar
|
|
109
|
-
* @returns Array de enlaces encontrados con su tipo
|
|
143
|
+
* @returns Array de enlaces encontrados con su tipo y texto (si es Markdown)
|
|
110
144
|
*
|
|
111
145
|
* @example
|
|
112
146
|
* ```typescript
|
|
113
|
-
* const links = this.linkProcessor.extractLinks('Visit https://example.com or
|
|
147
|
+
* const links = this.linkProcessor.extractLinks('Visit https://example.com, /profile, or [docs](https://docs.com)');
|
|
114
148
|
* // Returns: [
|
|
115
|
-
* // { url: 'https://example.com', type: 'external' },
|
|
116
|
-
* // { url: '/profile', type: 'internal' }
|
|
149
|
+
* // { url: 'https://example.com', type: 'external', text: 'https://example.com' },
|
|
150
|
+
* // { url: '/profile', type: 'internal', text: '/profile' },
|
|
151
|
+
* // { url: 'https://docs.com', type: 'external', text: 'docs' }
|
|
117
152
|
* // ]
|
|
118
153
|
* ```
|
|
119
154
|
*/
|
|
@@ -124,14 +159,30 @@ export class LinkProcessorService {
|
|
|
124
159
|
// Reset regex indices
|
|
125
160
|
this.urlRegex.lastIndex = 0;
|
|
126
161
|
this.internalRouteRegex.lastIndex = 0;
|
|
127
|
-
|
|
162
|
+
this.markdownLinkRegex.lastIndex = 0;
|
|
163
|
+
// Extraer enlaces Markdown primero
|
|
128
164
|
let match;
|
|
165
|
+
while ((match = this.markdownLinkRegex.exec(text)) !== null) {
|
|
166
|
+
const url = match[2];
|
|
167
|
+
const linkText = match[1];
|
|
168
|
+
const type = /^https?:\/\//.test(url) ? 'external' : 'internal';
|
|
169
|
+
links.push({ url, type, text: linkText });
|
|
170
|
+
}
|
|
171
|
+
// Extraer URLs externas directas
|
|
129
172
|
while ((match = this.urlRegex.exec(text)) !== null) {
|
|
130
|
-
|
|
173
|
+
const url = match[1];
|
|
174
|
+
// Verificar que no esté ya capturado como Markdown link
|
|
175
|
+
if (!links.some(link => link.url === url)) {
|
|
176
|
+
links.push({ url, type: 'external', text: url });
|
|
177
|
+
}
|
|
131
178
|
}
|
|
132
|
-
// Extraer rutas internas
|
|
179
|
+
// Extraer rutas internas directas
|
|
133
180
|
while ((match = this.internalRouteRegex.exec(text)) !== null) {
|
|
134
|
-
|
|
181
|
+
const url = match[2];
|
|
182
|
+
// Verificar que no esté ya capturado como Markdown link
|
|
183
|
+
if (!links.some(link => link.url === url)) {
|
|
184
|
+
links.push({ url, type: 'internal', text: url });
|
|
185
|
+
}
|
|
135
186
|
}
|
|
136
187
|
return links;
|
|
137
188
|
}
|
|
@@ -144,4 +195,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
144
195
|
providedIn: 'root',
|
|
145
196
|
}]
|
|
146
197
|
}], ctorParameters: () => [{ type: i1.DomSanitizer }] });
|
|
147
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"link-processor.service.js","sourceRoot":"","sources":["../../../../../projects/valtech-components/src/lib/services/link-processor.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { DomSanitizer, SafeHtml } from '@angular/platform-browser';\n\nexport interface LinkProcessorConfig {\n  /** Whether to open external links in new tab (default: true) */\n  openExternalInNewTab?: boolean;\n  /** Whether to open internal links in new tab (default: false) */\n  openInternalInNewTab?: boolean;\n  /** Custom CSS classes for links */\n  linkClass?: string;\n  /** Custom CSS classes for external links */\n  externalLinkClass?: string;\n  /** Custom CSS classes for internal links */\n  internalLinkClass?: string;\n}\n\n/**\n * LinkProcessorService - Service for processing text content to convert URLs and internal routes into clickable links.\n *\n * This service automatically detects external URLs (http/https) and internal routes (starting with /)\n * and converts them into HTML anchor elements with appropriate attributes.\n *\n * @example Basic usage:\n * ```typescript\n * constructor(private linkProcessor: LinkProcessorService) {}\n *\n * processText() {\n *   const text = 'Visit https://example.com or go to /profile';\n *   const processed = this.linkProcessor.processLinks(text);\n *   // Returns SafeHtml with clickable links\n * }\n * ```\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class LinkProcessorService {\n  // Regex para detectar URLs completas (http/https)\n  private readonly urlRegex = /(https?:\\/\\/[^\\s]+)/g;\n\n  // Regex para detectar rutas internas (empiezan con / pero no son URLs completas)\n  private readonly internalRouteRegex = /(\\s|^)(\\/[^\\s]*)/g;\n\n  constructor(private sanitizer: DomSanitizer) {}\n\n  /**\n   * Procesa texto para convertir enlaces en elementos <a> clickeables.\n   * Detecta automáticamente URLs externas e internas y las convierte en enlaces.\n   *\n   * @param text - Texto a procesar\n   * @param config - Configuración del procesamiento\n   * @returns SafeHtml con enlaces procesados o string original\n   *\n   * @example\n   * ```typescript\n   * const result = this.linkProcessor.processLinks(\n   *   'Visit https://example.com or /profile',\n   *   {\n   *     openExternalInNewTab: true,\n   *     openInternalInNewTab: false,\n   *     linkClass: 'custom-link'\n   *   }\n   * );\n   * ```\n   */\n  processLinks(text: string, config: LinkProcessorConfig = {}): SafeHtml | string {\n    if (!text) return '';\n\n    const {\n      openExternalInNewTab = true,\n      openInternalInNewTab = false,\n      linkClass = 'processed-link',\n      externalLinkClass = 'external-link',\n      internalLinkClass = 'internal-link',\n    } = config;\n\n    let hasLinks = false;\n    let processedText = text;\n\n    // Procesar URLs externas primero\n    if (this.urlRegex.test(text)) {\n      hasLinks = true;\n      this.urlRegex.lastIndex = 0; // Reset regex\n\n      processedText = processedText.replace(this.urlRegex, url => {\n        const target = openExternalInNewTab ? ' target=\"_blank\" rel=\"noopener noreferrer\"' : '';\n        const classes = `${linkClass} ${externalLinkClass}`.trim();\n        return `<a href=\"${url}\"${target} class=\"${classes}\">${url}</a>`;\n      });\n    }\n\n    // Procesar rutas internas después\n    if (this.internalRouteRegex.test(processedText)) {\n      hasLinks = true;\n      this.internalRouteRegex.lastIndex = 0; // Reset regex\n\n      processedText = processedText.replace(this.internalRouteRegex, (match, prefix, route) => {\n        // Solo procesar si no está ya dentro de un enlace\n        if (processedText.indexOf(`href=\"${route}\"`) === -1) {\n          const target = openInternalInNewTab ? ' target=\"_blank\"' : '';\n          const classes = `${linkClass} ${internalLinkClass}`.trim();\n          return `${prefix}<a href=\"${route}\"${target} class=\"${classes}\">${route}</a>`;\n        }\n        return match;\n      });\n    }\n\n    // Si hay enlaces, sanitizar el HTML\n    if (hasLinks) {\n      return this.sanitizer.bypassSecurityTrustHtml(processedText);\n    }\n\n    return text;\n  }\n\n  /**\n   * Detecta si un texto contiene enlaces (URLs o rutas internas).\n   *\n   * @param text - Texto a analizar\n   * @returns true si contiene enlaces\n   *\n   * @example\n   * ```typescript\n   * const hasLinks = this.linkProcessor.hasLinks('Visit https://example.com');\n   * // Returns: true\n   * ```\n   */\n  hasLinks(text: string): boolean {\n    if (!text) return false;\n\n    // Reset regex indices\n    this.urlRegex.lastIndex = 0;\n    this.internalRouteRegex.lastIndex = 0;\n\n    return this.urlRegex.test(text) || this.internalRouteRegex.test(text);\n  }\n\n  /**\n   * Extrae todos los enlaces de un texto.\n   *\n   * @param text - Texto a analizar\n   * @returns Array de enlaces encontrados con su tipo\n   *\n   * @example\n   * ```typescript\n   * const links = this.linkProcessor.extractLinks('Visit https://example.com or /profile');\n   * // Returns: [\n   * //   { url: 'https://example.com', type: 'external' },\n   * //   { url: '/profile', type: 'internal' }\n   * // ]\n   * ```\n   */\n  extractLinks(text: string): Array<{ url: string; type: 'external' | 'internal' }> {\n    if (!text) return [];\n\n    const links: Array<{ url: string; type: 'external' | 'internal' }> = [];\n\n    // Reset regex indices\n    this.urlRegex.lastIndex = 0;\n    this.internalRouteRegex.lastIndex = 0;\n\n    // Extraer URLs externas\n    let match;\n    while ((match = this.urlRegex.exec(text)) !== null) {\n      links.push({ url: match[1], type: 'external' });\n    }\n\n    // Extraer rutas internas\n    while ((match = this.internalRouteRegex.exec(text)) !== null) {\n      links.push({ url: match[2], type: 'internal' });\n    }\n\n    return links;\n  }\n}\n"]}
|
|
198
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"link-processor.service.js","sourceRoot":"","sources":["../../../../../projects/valtech-components/src/lib/services/link-processor.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { DomSanitizer, SafeHtml } from '@angular/platform-browser';\n\nexport interface LinkProcessorConfig {\n  /** Whether to open external links in new tab (default: true) */\n  openExternalInNewTab?: boolean;\n  /** Whether to open internal links in new tab (default: false) */\n  openInternalInNewTab?: boolean;\n  /** Custom CSS classes for links */\n  linkClass?: string;\n  /** Custom CSS classes for external links */\n  externalLinkClass?: string;\n  /** Custom CSS classes for internal links */\n  internalLinkClass?: string;\n  /** Whether to process Markdown-style links [text](url) (default: true) */\n  processMarkdownLinks?: boolean;\n}\n\n/**\n * LinkProcessorService - Service for processing text content to convert URLs and internal routes into clickable links.\n *\n * This service automatically detects external URLs (http/https), internal routes (starting with /),\n * and Markdown-style links [text](url) and converts them into HTML anchor elements with appropriate attributes.\n *\n * @example Basic usage:\n * ```typescript\n * constructor(private linkProcessor: LinkProcessorService) {}\n *\n * processText() {\n *   const text = 'Visit https://example.com, go to /profile, or [check docs](https://docs.example.com)';\n *   const processed = this.linkProcessor.processLinks(text);\n *   // Returns SafeHtml with clickable links\n * }\n * ```\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class LinkProcessorService {\n  // Regex para detectar URLs completas (http/https) - permite caracteres válidos pero excluye puntuación al final\n  private readonly urlRegex = /(https?:\\/\\/[^\\s]+?)(?=[.,;!?()\\s]|$)/g;\n\n  // Regex para detectar rutas internas (empiezan con / pero no son URLs completas) - excluye puntuación al final\n  private readonly internalRouteRegex = /(\\s|^)(\\/[^\\s]*?)(?=[.,;!?()\\s]|$)/g;\n\n  // Regex para detectar enlaces estilo Markdown [texto](url)\n  private readonly markdownLinkRegex = /\\[([^\\]]+)\\]\\(([^)]+)\\)/g;\n\n  constructor(private sanitizer: DomSanitizer) {}\n\n  /**\n   * Procesa texto para convertir enlaces en elementos <a> clickeables.\n   * Detecta automáticamente URLs externas, rutas internas y enlaces estilo Markdown.\n   *\n   * @param text - Texto a procesar\n   * @param config - Configuración del procesamiento\n   * @returns SafeHtml con enlaces procesados o string original\n   *\n   * @example\n   * ```typescript\n   * const result = this.linkProcessor.processLinks(\n   *   'Visit https://example.com, go to /profile, or [check docs](https://docs.example.com)',\n   *   {\n   *     openExternalInNewTab: true,\n   *     openInternalInNewTab: false,\n   *     processMarkdownLinks: true,\n   *     linkClass: 'custom-link'\n   *   }\n   * );\n   * ```\n   */\n  processLinks(text: string, config: LinkProcessorConfig = {}): SafeHtml | string {\n    if (!text) return '';\n\n    const {\n      openExternalInNewTab = true,\n      openInternalInNewTab = false,\n      linkClass = 'processed-link',\n      externalLinkClass = 'external-link',\n      internalLinkClass = 'internal-link',\n      processMarkdownLinks = true,\n    } = config;\n\n    let hasLinks = false;\n    let processedText = text;\n\n    // 1. Procesar enlaces estilo Markdown [texto](url) primero\n    if (processMarkdownLinks) {\n      this.markdownLinkRegex.lastIndex = 0; // Reset regex\n\n      processedText = processedText.replace(this.markdownLinkRegex, (match, linkText, url) => {\n        hasLinks = true;\n        const isExternal = /^https?:\\/\\//.test(url);\n        const target = (isExternal ? openExternalInNewTab : openInternalInNewTab)\n          ? isExternal\n            ? ' target=\"_blank\" rel=\"noopener noreferrer\"'\n            : ' target=\"_blank\"'\n          : '';\n        const typeClass = isExternal ? externalLinkClass : internalLinkClass;\n        const classes = `${linkClass} ${typeClass}`.trim();\n        return `<a href=\"${url}\"${target} class=\"${classes}\">${linkText}</a>`;\n      });\n    }\n\n    // 2. Procesar URLs externas directas (solo si no están ya en un enlace HTML)\n    this.urlRegex.lastIndex = 0; // Reset regex\n\n    processedText = processedText.replace(this.urlRegex, (fullMatch, url) => {\n      // Verificar que no esté ya dentro de un enlace HTML existente\n      const urlPosition = processedText.indexOf(fullMatch);\n      const textBefore = processedText.substring(0, urlPosition);\n\n      // Buscar la última apertura y cierre de enlace antes de esta posición\n      const lastOpenTag = textBefore.lastIndexOf('<a ');\n      const lastCloseTag = textBefore.lastIndexOf('</a>');\n\n      // Si hay un tag <a abierto sin cerrar, no procesamos\n      if (lastOpenTag > lastCloseTag) {\n        return fullMatch; // Mantener original\n      }\n\n      hasLinks = true;\n      const target = openExternalInNewTab ? ' target=\"_blank\" rel=\"noopener noreferrer\"' : '';\n      const classes = `${linkClass} ${externalLinkClass}`.trim();\n      return `<a href=\"${url}\"${target} class=\"${classes}\">${url}</a>`;\n    });\n\n    // 3. Procesar rutas internas (solo si no están ya en un enlace HTML)\n    this.internalRouteRegex.lastIndex = 0; // Reset regex\n\n    processedText = processedText.replace(this.internalRouteRegex, (match, prefix, route) => {\n      // Verificar que no esté ya dentro de un enlace HTML existente\n      const matchPosition = processedText.indexOf(match);\n      const textBefore = processedText.substring(0, matchPosition);\n\n      // Buscar la última apertura y cierre de enlace antes de esta posición\n      const lastOpenTag = textBefore.lastIndexOf('<a ');\n      const lastCloseTag = textBefore.lastIndexOf('</a>');\n\n      // Si hay un tag <a abierto sin cerrar, no procesamos\n      if (lastOpenTag > lastCloseTag) {\n        return match; // Mantener original\n      }\n\n      hasLinks = true;\n      const target = openInternalInNewTab ? ' target=\"_blank\"' : '';\n      const classes = `${linkClass} ${internalLinkClass}`.trim();\n      return `${prefix}<a href=\"${route}\"${target} class=\"${classes}\">${route}</a>`;\n    });\n\n    // Si hay enlaces, sanitizar el HTML\n    if (hasLinks) {\n      return this.sanitizer.bypassSecurityTrustHtml(processedText);\n    }\n\n    return text;\n  }\n\n  /**\n   * Detecta si un texto contiene enlaces (URLs, rutas internas o enlaces Markdown).\n   *\n   * @param text - Texto a analizar\n   * @returns true si contiene enlaces\n   *\n   * @example\n   * ```typescript\n   * const hasLinks = this.linkProcessor.hasLinks('Visit https://example.com or [docs](https://docs.com)');\n   * // Returns: true\n   * ```\n   */\n  hasLinks(text: string): boolean {\n    if (!text) return false;\n\n    // Reset regex indices\n    this.urlRegex.lastIndex = 0;\n    this.internalRouteRegex.lastIndex = 0;\n    this.markdownLinkRegex.lastIndex = 0;\n\n    return (\n      this.urlRegex.test(text) ||\n      this.internalRouteRegex.test(text) ||\n      this.markdownLinkRegex.test(text)\n    );\n  }\n\n  /**\n   * Extrae todos los enlaces de un texto.\n   *\n   * @param text - Texto a analizar\n   * @returns Array de enlaces encontrados con su tipo y texto (si es Markdown)\n   *\n   * @example\n   * ```typescript\n   * const links = this.linkProcessor.extractLinks('Visit https://example.com, /profile, or [docs](https://docs.com)');\n   * // Returns: [\n   * //   { url: 'https://example.com', type: 'external', text: 'https://example.com' },\n   * //   { url: '/profile', type: 'internal', text: '/profile' },\n   * //   { url: 'https://docs.com', type: 'external', text: 'docs' }\n   * // ]\n   * ```\n   */\n  extractLinks(text: string): Array<{ url: string; type: 'external' | 'internal'; text: string }> {\n    if (!text) return [];\n\n    const links: Array<{ url: string; type: 'external' | 'internal'; text: string }> = [];\n\n    // Reset regex indices\n    this.urlRegex.lastIndex = 0;\n    this.internalRouteRegex.lastIndex = 0;\n    this.markdownLinkRegex.lastIndex = 0;\n\n    // Extraer enlaces Markdown primero\n    let match;\n    while ((match = this.markdownLinkRegex.exec(text)) !== null) {\n      const url = match[2];\n      const linkText = match[1];\n      const type = /^https?:\\/\\//.test(url) ? 'external' : 'internal';\n      links.push({ url, type, text: linkText });\n    }\n\n    // Extraer URLs externas directas\n    while ((match = this.urlRegex.exec(text)) !== null) {\n      const url = match[1];\n      // Verificar que no esté ya capturado como Markdown link\n      if (!links.some(link => link.url === url)) {\n        links.push({ url, type: 'external', text: url });\n      }\n    }\n\n    // Extraer rutas internas directas\n    while ((match = this.internalRouteRegex.exec(text)) !== null) {\n      const url = match[2];\n      // Verificar que no esté ya capturado como Markdown link\n      if (!links.some(link => link.url === url)) {\n        links.push({ url, type: 'internal', text: url });\n      }\n    }\n\n    return links;\n  }\n}\n"]}
|