valtech-components 2.0.289 → 2.0.291
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/esm2022/lib/components/atoms/text/text.component.mjs +3 -3
- package/esm2022/lib/examples/comprehensive-link-test.component.mjs +208 -0
- package/esm2022/lib/examples/link-processing-example.component.mjs +47 -3
- package/esm2022/lib/services/link-processor.service.mjs +85 -36
- package/esm2022/public-api.mjs +2 -1
- package/fesm2022/valtech-components.mjs +338 -40
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/examples/comprehensive-link-test.component.d.ts +23 -0
- package/lib/examples/link-processing-example.component.d.ts +2 -0
- package/lib/services/link-processor.service.d.ts +6 -0
- package/package.json +1 -1
- package/public-api.d.ts +1 -0
- package/src/lib/components/styles/overrides.scss +34 -0
|
@@ -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":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AAC5C,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAE,KAAK,EAAqB,MAAM,eAAe,CAAC;AAC7F,OAAO,EAAE,OAAO,EAAE,MAAM,2BAA2B,CAAC;AACpD,OAAO,EAAc,EAAE,EAAE,YAAY,EAAE,MAAM,MAAM,CAAC;AAGpD,OAAO,EAAE,gBAAgB,EAAE,MAAM,0CAA0C,CAAC;;;;AAuB5E;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA0EG;AACH,MAAM,OAAO,aAAa;IAgCxB,YACU,cAA8B,EAC9B,aAAmC;QADnC,mBAAc,GAAd,cAAc,CAAgB;QAC9B,kBAAa,GAAb,aAAa,CAAsB;QAJrC,iBAAY,GAAG,IAAI,YAAY,EAAE,CAAC;IAKvC,CAAC;IAEJ,QAAQ;QACN,IAAI,CAAC,mBAAmB,EAAE,CAAC;IAC7B,CAAC;IAED,WAAW;QACT,IAAI,CAAC,YAAY,CAAC,WAAW,EAAE,CAAC;IAClC,CAAC;IAED;;;OAGG;IACK,mBAAmB;QACzB,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC;YACvB,kCAAkC;YAClC,IAAI,CAAC,eAAe,GAAG,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;QAChD,CAAC;aAAM,IAAI,IAAI,CAAC,KAAK,CAAC,UAAU,IAAI,IAAI,CAAC,KAAK,CAAC,YAAY,EAAE,CAAC;YAC5D,yCAAyC;YACzC,IAAI,IAAI,CAAC,KAAK,CAAC,oBAAoB,EAAE,CAAC;gBACpC,qBAAqB;gBACrB,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,cAAc,CAAC,4BAA4B,CAAC;oBACtE,SAAS,EAAE,IAAI,CAAC,KAAK,CAAC,YAAY;oBAClC,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,UAAU;oBAC1B,QAAQ,EAAE,IAAI,CAAC,KAAK,CAAC,eAAe;oBACpC,aAAa,EAAE,IAAI,CAAC,KAAK,CAAC,oBAAoB;iBAC/C,CAAC,CAAC;YACL,CAAC;iBAAM,CAAC;gBACN,0BAA0B;gBAC1B,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,cAAc,CAAC,WAAW,CAAC;oBACrD,SAAS,EAAE,IAAI,CAAC,KAAK,CAAC,YAAY;oBAClC,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,UAAU;oBAC1B,QAAQ,EAAE,IAAI,CAAC,KAAK,CAAC,eAAe;iBACrC,CAAC,CAAC;YACL,CAAC;QACH,CAAC;aAAM,CAAC;YACN,6DAA6D;YAC7D,OAAO,CAAC,IAAI,CACV,gJAAgJ,CACjJ,CAAC;YACF,IAAI,CAAC,eAAe,GAAG,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,eAAe,IAAI,EAAE,CAAC,CAAC;QAC9D,CAAC;IACH,CAAC;+GA9EU,aAAa;mGAAb,aAAa,gGA3Fd;;;;;;;;;;;;GAYT,k8HAbS,OAAO,2EAAE,SAAS,yCAAE,gBAAgB;;4FA4FnC,aAAa;kBA/FzB,SAAS;+BACE,UAAU,cACR,IAAI,WACP,CAAC,OAAO,EAAE,SAAS,EAAE,gBAAgB,CAAC,YACrC;;;;;;;;;;;;GAYT,mBAEgB,uBAAuB,CAAC,MAAM;sHAmG/C,KAAK;sBADJ,KAAK;;AA4DR;;;;;;;;;;;;;;;;;;;;;;GAsBG;AACH,MAAM,UAAU,eAAe,CAC7B,aAAgC,EAChC,cAAsE,EAAE;IAExE,OAAO;QACL,UAAU,EAAE,aAAa,CAAC,UAAU;QACpC,YAAY,EAAE,aAAa,CAAC,YAAY;QACxC,eAAe,EAAE,aAAa,CAAC,eAAe;QAC9C,oBAAoB,EAAE,aAAa,CAAC,oBAAoB;QACxD,KAAK,EAAE,WAAW,CAAC,KAAK,IAAI,MAAM;QAClC,IAAI,EAAE,WAAW,CAAC,IAAI,IAAI,QAAQ;QAClC,IAAI,EAAE,WAAW,CAAC,IAAI,IAAI,KAAK;KAChC,CAAC;AACJ,CAAC","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"]}
|
|
@@ -0,0 +1,208 @@
|
|
|
1
|
+
import { Component } from '@angular/core';
|
|
2
|
+
import { TextComponent } from '../components/atoms/text/text.component';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
/**
|
|
5
|
+
* ComprehensiveLinkTestComponent - Componente de prueba exhaustiva para el procesamiento de enlaces.
|
|
6
|
+
*
|
|
7
|
+
* Este componente demuestra todos los casos edge y escenarios complejos de procesamiento de enlaces,
|
|
8
|
+
* incluyendo puntuación, URLs complejas, y mezclas de formatos.
|
|
9
|
+
*
|
|
10
|
+
* @example Uso en template:
|
|
11
|
+
* ```html
|
|
12
|
+
* <val-comprehensive-link-test></val-comprehensive-link-test>
|
|
13
|
+
* ```
|
|
14
|
+
*/
|
|
15
|
+
export class ComprehensiveLinkTestComponent {
|
|
16
|
+
constructor() {
|
|
17
|
+
this.punctuationProps = {
|
|
18
|
+
content: 'Diferentes puntuaciones: https://angular.io, también https://github.com! ¿Conoces https://typescript.org? Final: https://rxjs.dev. Entre paréntesis (https://zone.js) y con comillas "https://ionic.io".',
|
|
19
|
+
size: 'medium',
|
|
20
|
+
color: 'dark',
|
|
21
|
+
bold: false,
|
|
22
|
+
processLinks: true,
|
|
23
|
+
linkConfig: {
|
|
24
|
+
openExternalInNewTab: true,
|
|
25
|
+
linkClass: 'test-punctuation',
|
|
26
|
+
externalLinkClass: 'test-external',
|
|
27
|
+
},
|
|
28
|
+
};
|
|
29
|
+
this.complexUrlProps = {
|
|
30
|
+
content: 'URLs complejas: https://api.github.com/repos/angular/angular/issues?state=open&sort=updated&per_page=50, búsqueda https://google.com/search?q=angular+ionic+components#results, y documentación https://angular.io/guide/getting-started#development-environment.',
|
|
31
|
+
size: 'medium',
|
|
32
|
+
color: 'dark',
|
|
33
|
+
bold: false,
|
|
34
|
+
processLinks: true,
|
|
35
|
+
linkConfig: {
|
|
36
|
+
openExternalInNewTab: true,
|
|
37
|
+
linkClass: 'test-complex',
|
|
38
|
+
externalLinkClass: 'test-external',
|
|
39
|
+
},
|
|
40
|
+
};
|
|
41
|
+
this.parenthesesProps = {
|
|
42
|
+
content: 'Paréntesis de contexto (ver https://docs.angular.io) vs URLs con paréntesis https://example.com/api/method(param) en el contenido. También funciona (https://ionic.io/docs).',
|
|
43
|
+
size: 'medium',
|
|
44
|
+
color: 'dark',
|
|
45
|
+
bold: false,
|
|
46
|
+
processLinks: true,
|
|
47
|
+
linkConfig: {
|
|
48
|
+
openExternalInNewTab: true,
|
|
49
|
+
linkClass: 'test-parentheses',
|
|
50
|
+
externalLinkClass: 'test-external',
|
|
51
|
+
},
|
|
52
|
+
};
|
|
53
|
+
this.mixedFormatsProps = {
|
|
54
|
+
content: 'Formatos mezclados: [Documentación oficial](https://angular.io/docs), enlace directo https://github.com/angular/angular, ruta interna /dashboard/settings, [guía de inicio](/getting-started), y API https://api.example.com/v1/users?active=true.',
|
|
55
|
+
size: 'medium',
|
|
56
|
+
color: 'dark',
|
|
57
|
+
bold: false,
|
|
58
|
+
processLinks: true,
|
|
59
|
+
linkConfig: {
|
|
60
|
+
openExternalInNewTab: true,
|
|
61
|
+
openInternalInNewTab: false,
|
|
62
|
+
processMarkdownLinks: true,
|
|
63
|
+
linkClass: 'test-mixed',
|
|
64
|
+
externalLinkClass: 'test-external',
|
|
65
|
+
internalLinkClass: 'test-internal',
|
|
66
|
+
},
|
|
67
|
+
};
|
|
68
|
+
this.edgeCasesProps = {
|
|
69
|
+
content: 'Casos extremos: "https://quoted-url.com", múltiple puntuación https://example.com?!!, URL al final de oración https://final-url.org. También consecutivos: https://first.com y https://second.com.',
|
|
70
|
+
size: 'medium',
|
|
71
|
+
color: 'dark',
|
|
72
|
+
bold: false,
|
|
73
|
+
processLinks: true,
|
|
74
|
+
linkConfig: {
|
|
75
|
+
openExternalInNewTab: true,
|
|
76
|
+
linkClass: 'test-edge',
|
|
77
|
+
externalLinkClass: 'test-external',
|
|
78
|
+
},
|
|
79
|
+
};
|
|
80
|
+
this.devUrlsProps = {
|
|
81
|
+
content: 'URLs de desarrollo: http://localhost:4200/dashboard, servidor local https://127.0.0.1:8080/api/status, desarrollo http://dev.example.com:3000/debug?verbose=true, y túnel https://abc123.ngrok.io/webhook.',
|
|
82
|
+
size: 'medium',
|
|
83
|
+
color: 'dark',
|
|
84
|
+
bold: false,
|
|
85
|
+
processLinks: true,
|
|
86
|
+
linkConfig: {
|
|
87
|
+
openExternalInNewTab: false, // Para desarrollo, puede ser útil no abrir en nueva pestaña
|
|
88
|
+
linkClass: 'test-dev',
|
|
89
|
+
externalLinkClass: 'test-dev-external',
|
|
90
|
+
},
|
|
91
|
+
};
|
|
92
|
+
}
|
|
93
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ComprehensiveLinkTestComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
94
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: ComprehensiveLinkTestComponent, isStandalone: true, selector: "val-comprehensive-link-test", ngImport: i0, template: `
|
|
95
|
+
<div class="comprehensive-test">
|
|
96
|
+
<h2>Prueba Exhaustiva de Procesamiento de Enlaces</h2>
|
|
97
|
+
|
|
98
|
+
<div class="test-section">
|
|
99
|
+
<h3>✅ Puntuación Final - SOLUCIONADO</h3>
|
|
100
|
+
<val-text [props]="punctuationProps"></val-text>
|
|
101
|
+
<p class="note">
|
|
102
|
+
<strong>Esperado:</strong> Los enlaces no incluyen puntuación final (.,;!?), pero la puntuación se preserva
|
|
103
|
+
como texto después del enlace.
|
|
104
|
+
</p>
|
|
105
|
+
</div>
|
|
106
|
+
|
|
107
|
+
<div class="test-section">
|
|
108
|
+
<h3>✅ URLs Complejas con Parámetros - SOLUCIONADO</h3>
|
|
109
|
+
<val-text [props]="complexUrlProps"></val-text>
|
|
110
|
+
<p class="note">
|
|
111
|
+
<strong>Esperado:</strong> URLs con query params, fragmentos y rutas complejas se preservan completamente.
|
|
112
|
+
</p>
|
|
113
|
+
</div>
|
|
114
|
+
|
|
115
|
+
<div class="test-section">
|
|
116
|
+
<h3>✅ Paréntesis Inteligentes - SOLUCIONADO</h3>
|
|
117
|
+
<val-text [props]="parenthesesProps"></val-text>
|
|
118
|
+
<p class="note">
|
|
119
|
+
<strong>Esperado:</strong> Paréntesis de contexto (texto) vs paréntesis de URL se manejan correctamente.
|
|
120
|
+
</p>
|
|
121
|
+
</div>
|
|
122
|
+
|
|
123
|
+
<div class="test-section">
|
|
124
|
+
<h3>✅ Mezcla de Formatos - SOLUCIONADO</h3>
|
|
125
|
+
<val-text [props]="mixedFormatsProps"></val-text>
|
|
126
|
+
<p class="note">
|
|
127
|
+
<strong>Esperado:</strong> Enlaces Markdown, URLs directas y rutas internas coexisten sin conflictos.
|
|
128
|
+
</p>
|
|
129
|
+
</div>
|
|
130
|
+
|
|
131
|
+
<div class="test-section">
|
|
132
|
+
<h3>✅ Casos Extremos - SOLUCIONADO</h3>
|
|
133
|
+
<val-text [props]="edgeCasesProps"></val-text>
|
|
134
|
+
<p class="note">
|
|
135
|
+
<strong>Esperado:</strong> Comillas, múltiple puntuación, y URLs al final de oraciones se procesan
|
|
136
|
+
correctamente.
|
|
137
|
+
</p>
|
|
138
|
+
</div>
|
|
139
|
+
|
|
140
|
+
<div class="test-section">
|
|
141
|
+
<h3>✅ URLs de Desarrollo - SOLUCIONADO</h3>
|
|
142
|
+
<val-text [props]="devUrlsProps"></val-text>
|
|
143
|
+
<p class="note">
|
|
144
|
+
<strong>Esperado:</strong> URLs de localhost, puertos, y rutas de desarrollo se detectan correctamente.
|
|
145
|
+
</p>
|
|
146
|
+
</div>
|
|
147
|
+
</div>
|
|
148
|
+
`, isInline: true, styles: [".comprehensive-test{padding:20px;max-width:1000px;font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,sans-serif}.test-section{margin-bottom:32px;padding:20px;border:2px solid var(--ion-color-success, #2dd36f);border-radius:12px;background:var(--ion-color-success-tint, #42d77d) 10}h2{color:var(--ion-color-primary, #3880ff);margin-bottom:24px;text-align:center}h3{color:var(--ion-color-success, #2dd36f);margin-bottom:16px;font-size:18px;display:flex;align-items:center;gap:8px}.note{margin-top:12px;padding:12px;background:var(--ion-color-light, #f4f5f8);border-radius:8px;font-size:14px;color:var(--ion-color-medium, #92949c);border-left:4px solid var(--ion-color-primary, #3880ff)}.note strong{color:var(--ion-color-dark, #222428)}\n"], dependencies: [{ kind: "component", type: TextComponent, selector: "val-text", inputs: ["props"] }] }); }
|
|
149
|
+
}
|
|
150
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ComprehensiveLinkTestComponent, decorators: [{
|
|
151
|
+
type: Component,
|
|
152
|
+
args: [{ selector: 'val-comprehensive-link-test', standalone: true, imports: [TextComponent], template: `
|
|
153
|
+
<div class="comprehensive-test">
|
|
154
|
+
<h2>Prueba Exhaustiva de Procesamiento de Enlaces</h2>
|
|
155
|
+
|
|
156
|
+
<div class="test-section">
|
|
157
|
+
<h3>✅ Puntuación Final - SOLUCIONADO</h3>
|
|
158
|
+
<val-text [props]="punctuationProps"></val-text>
|
|
159
|
+
<p class="note">
|
|
160
|
+
<strong>Esperado:</strong> Los enlaces no incluyen puntuación final (.,;!?), pero la puntuación se preserva
|
|
161
|
+
como texto después del enlace.
|
|
162
|
+
</p>
|
|
163
|
+
</div>
|
|
164
|
+
|
|
165
|
+
<div class="test-section">
|
|
166
|
+
<h3>✅ URLs Complejas con Parámetros - SOLUCIONADO</h3>
|
|
167
|
+
<val-text [props]="complexUrlProps"></val-text>
|
|
168
|
+
<p class="note">
|
|
169
|
+
<strong>Esperado:</strong> URLs con query params, fragmentos y rutas complejas se preservan completamente.
|
|
170
|
+
</p>
|
|
171
|
+
</div>
|
|
172
|
+
|
|
173
|
+
<div class="test-section">
|
|
174
|
+
<h3>✅ Paréntesis Inteligentes - SOLUCIONADO</h3>
|
|
175
|
+
<val-text [props]="parenthesesProps"></val-text>
|
|
176
|
+
<p class="note">
|
|
177
|
+
<strong>Esperado:</strong> Paréntesis de contexto (texto) vs paréntesis de URL se manejan correctamente.
|
|
178
|
+
</p>
|
|
179
|
+
</div>
|
|
180
|
+
|
|
181
|
+
<div class="test-section">
|
|
182
|
+
<h3>✅ Mezcla de Formatos - SOLUCIONADO</h3>
|
|
183
|
+
<val-text [props]="mixedFormatsProps"></val-text>
|
|
184
|
+
<p class="note">
|
|
185
|
+
<strong>Esperado:</strong> Enlaces Markdown, URLs directas y rutas internas coexisten sin conflictos.
|
|
186
|
+
</p>
|
|
187
|
+
</div>
|
|
188
|
+
|
|
189
|
+
<div class="test-section">
|
|
190
|
+
<h3>✅ Casos Extremos - SOLUCIONADO</h3>
|
|
191
|
+
<val-text [props]="edgeCasesProps"></val-text>
|
|
192
|
+
<p class="note">
|
|
193
|
+
<strong>Esperado:</strong> Comillas, múltiple puntuación, y URLs al final de oraciones se procesan
|
|
194
|
+
correctamente.
|
|
195
|
+
</p>
|
|
196
|
+
</div>
|
|
197
|
+
|
|
198
|
+
<div class="test-section">
|
|
199
|
+
<h3>✅ URLs de Desarrollo - SOLUCIONADO</h3>
|
|
200
|
+
<val-text [props]="devUrlsProps"></val-text>
|
|
201
|
+
<p class="note">
|
|
202
|
+
<strong>Esperado:</strong> URLs de localhost, puertos, y rutas de desarrollo se detectan correctamente.
|
|
203
|
+
</p>
|
|
204
|
+
</div>
|
|
205
|
+
</div>
|
|
206
|
+
`, styles: [".comprehensive-test{padding:20px;max-width:1000px;font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,sans-serif}.test-section{margin-bottom:32px;padding:20px;border:2px solid var(--ion-color-success, #2dd36f);border-radius:12px;background:var(--ion-color-success-tint, #42d77d) 10}h2{color:var(--ion-color-primary, #3880ff);margin-bottom:24px;text-align:center}h3{color:var(--ion-color-success, #2dd36f);margin-bottom:16px;font-size:18px;display:flex;align-items:center;gap:8px}.note{margin-top:12px;padding:12px;background:var(--ion-color-light, #f4f5f8);border-radius:8px;font-size:14px;color:var(--ion-color-medium, #92949c);border-left:4px solid var(--ion-color-primary, #3880ff)}.note strong{color:var(--ion-color-dark, #222428)}\n"] }]
|
|
207
|
+
}] });
|
|
208
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"comprehensive-link-test.component.js","sourceRoot":"","sources":["../../../../../projects/valtech-components/src/lib/examples/comprehensive-link-test.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAC1C,OAAO,EAAE,aAAa,EAAE,MAAM,yCAAyC,CAAC;;AAGxE;;;;;;;;;;GAUG;AA2GH,MAAM,OAAO,8BAA8B;IA1G3C;QA2GE,qBAAgB,GAAiB;YAC/B,OAAO,EACL,0MAA0M;YAC5M,IAAI,EAAE,QAAQ;YACd,KAAK,EAAE,MAAM;YACb,IAAI,EAAE,KAAK;YACX,YAAY,EAAE,IAAI;YAClB,UAAU,EAAE;gBACV,oBAAoB,EAAE,IAAI;gBAC1B,SAAS,EAAE,kBAAkB;gBAC7B,iBAAiB,EAAE,eAAe;aACnC;SACF,CAAC;QAEF,oBAAe,GAAiB;YAC9B,OAAO,EACL,mQAAmQ;YACrQ,IAAI,EAAE,QAAQ;YACd,KAAK,EAAE,MAAM;YACb,IAAI,EAAE,KAAK;YACX,YAAY,EAAE,IAAI;YAClB,UAAU,EAAE;gBACV,oBAAoB,EAAE,IAAI;gBAC1B,SAAS,EAAE,cAAc;gBACzB,iBAAiB,EAAE,eAAe;aACnC;SACF,CAAC;QAEF,qBAAgB,GAAiB;YAC/B,OAAO,EACL,8KAA8K;YAChL,IAAI,EAAE,QAAQ;YACd,KAAK,EAAE,MAAM;YACb,IAAI,EAAE,KAAK;YACX,YAAY,EAAE,IAAI;YAClB,UAAU,EAAE;gBACV,oBAAoB,EAAE,IAAI;gBAC1B,SAAS,EAAE,kBAAkB;gBAC7B,iBAAiB,EAAE,eAAe;aACnC;SACF,CAAC;QAEF,sBAAiB,GAAiB;YAChC,OAAO,EACL,oPAAoP;YACtP,IAAI,EAAE,QAAQ;YACd,KAAK,EAAE,MAAM;YACb,IAAI,EAAE,KAAK;YACX,YAAY,EAAE,IAAI;YAClB,UAAU,EAAE;gBACV,oBAAoB,EAAE,IAAI;gBAC1B,oBAAoB,EAAE,KAAK;gBAC3B,oBAAoB,EAAE,IAAI;gBAC1B,SAAS,EAAE,YAAY;gBACvB,iBAAiB,EAAE,eAAe;gBAClC,iBAAiB,EAAE,eAAe;aACnC;SACF,CAAC;QAEF,mBAAc,GAAiB;YAC7B,OAAO,EACL,oMAAoM;YACtM,IAAI,EAAE,QAAQ;YACd,KAAK,EAAE,MAAM;YACb,IAAI,EAAE,KAAK;YACX,YAAY,EAAE,IAAI;YAClB,UAAU,EAAE;gBACV,oBAAoB,EAAE,IAAI;gBAC1B,SAAS,EAAE,WAAW;gBACtB,iBAAiB,EAAE,eAAe;aACnC;SACF,CAAC;QAEF,iBAAY,GAAiB;YAC3B,OAAO,EACL,4MAA4M;YAC9M,IAAI,EAAE,QAAQ;YACd,KAAK,EAAE,MAAM;YACb,IAAI,EAAE,KAAK;YACX,YAAY,EAAE,IAAI;YAClB,UAAU,EAAE;gBACV,oBAAoB,EAAE,KAAK,EAAE,4DAA4D;gBACzF,SAAS,EAAE,UAAU;gBACrB,iBAAiB,EAAE,mBAAmB;aACvC;SACF,CAAC;KACH;+GAvFY,8BAA8B;mGAA9B,8BAA8B,uFAtG/B;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsDT,ozBAvDS,aAAa;;4FAuGZ,8BAA8B;kBA1G1C,SAAS;+BACE,6BAA6B,cAC3B,IAAI,WACP,CAAC,aAAa,CAAC,YACd;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsDT","sourcesContent":["import { Component } from '@angular/core';\nimport { TextComponent } from '../components/atoms/text/text.component';\nimport { TextMetadata } from '../components/atoms/text/types';\n\n/**\n * ComprehensiveLinkTestComponent - Componente de prueba exhaustiva para el procesamiento de enlaces.\n *\n * Este componente demuestra todos los casos edge y escenarios complejos de procesamiento de enlaces,\n * incluyendo puntuación, URLs complejas, y mezclas de formatos.\n *\n * @example Uso en template:\n * ```html\n * <val-comprehensive-link-test></val-comprehensive-link-test>\n * ```\n */\n@Component({\n  selector: 'val-comprehensive-link-test',\n  standalone: true,\n  imports: [TextComponent],\n  template: `\n    <div class=\"comprehensive-test\">\n      <h2>Prueba Exhaustiva de Procesamiento de Enlaces</h2>\n\n      <div class=\"test-section\">\n        <h3>✅ Puntuación Final - SOLUCIONADO</h3>\n        <val-text [props]=\"punctuationProps\"></val-text>\n        <p class=\"note\">\n          <strong>Esperado:</strong> Los enlaces no incluyen puntuación final (.,;!?), pero la puntuación se preserva\n          como texto después del enlace.\n        </p>\n      </div>\n\n      <div class=\"test-section\">\n        <h3>✅ URLs Complejas con Parámetros - SOLUCIONADO</h3>\n        <val-text [props]=\"complexUrlProps\"></val-text>\n        <p class=\"note\">\n          <strong>Esperado:</strong> URLs con query params, fragmentos y rutas complejas se preservan completamente.\n        </p>\n      </div>\n\n      <div class=\"test-section\">\n        <h3>✅ Paréntesis Inteligentes - SOLUCIONADO</h3>\n        <val-text [props]=\"parenthesesProps\"></val-text>\n        <p class=\"note\">\n          <strong>Esperado:</strong> Paréntesis de contexto (texto) vs paréntesis de URL se manejan correctamente.\n        </p>\n      </div>\n\n      <div class=\"test-section\">\n        <h3>✅ Mezcla de Formatos - SOLUCIONADO</h3>\n        <val-text [props]=\"mixedFormatsProps\"></val-text>\n        <p class=\"note\">\n          <strong>Esperado:</strong> Enlaces Markdown, URLs directas y rutas internas coexisten sin conflictos.\n        </p>\n      </div>\n\n      <div class=\"test-section\">\n        <h3>✅ Casos Extremos - SOLUCIONADO</h3>\n        <val-text [props]=\"edgeCasesProps\"></val-text>\n        <p class=\"note\">\n          <strong>Esperado:</strong> Comillas, múltiple puntuación, y URLs al final de oraciones se procesan\n          correctamente.\n        </p>\n      </div>\n\n      <div class=\"test-section\">\n        <h3>✅ URLs de Desarrollo - SOLUCIONADO</h3>\n        <val-text [props]=\"devUrlsProps\"></val-text>\n        <p class=\"note\">\n          <strong>Esperado:</strong> URLs de localhost, puertos, y rutas de desarrollo se detectan correctamente.\n        </p>\n      </div>\n    </div>\n  `,\n  styles: [\n    `\n      .comprehensive-test {\n        padding: 20px;\n        max-width: 1000px;\n        font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;\n      }\n\n      .test-section {\n        margin-bottom: 32px;\n        padding: 20px;\n        border: 2px solid var(--ion-color-success, #2dd36f);\n        border-radius: 12px;\n        background: var(--ion-color-success-tint, #42d77d) 10;\n      }\n\n      h2 {\n        color: var(--ion-color-primary, #3880ff);\n        margin-bottom: 24px;\n        text-align: center;\n      }\n\n      h3 {\n        color: var(--ion-color-success, #2dd36f);\n        margin-bottom: 16px;\n        font-size: 18px;\n        display: flex;\n        align-items: center;\n        gap: 8px;\n      }\n\n      .note {\n        margin-top: 12px;\n        padding: 12px;\n        background: var(--ion-color-light, #f4f5f8);\n        border-radius: 8px;\n        font-size: 14px;\n        color: var(--ion-color-medium, #92949c);\n        border-left: 4px solid var(--ion-color-primary, #3880ff);\n      }\n\n      .note strong {\n        color: var(--ion-color-dark, #222428);\n      }\n    `,\n  ],\n})\nexport class ComprehensiveLinkTestComponent {\n  punctuationProps: TextMetadata = {\n    content:\n      'Diferentes puntuaciones: https://angular.io, también https://github.com! ¿Conoces https://typescript.org? Final: https://rxjs.dev. Entre paréntesis (https://zone.js) y con comillas \"https://ionic.io\".',\n    size: 'medium',\n    color: 'dark',\n    bold: false,\n    processLinks: true,\n    linkConfig: {\n      openExternalInNewTab: true,\n      linkClass: 'test-punctuation',\n      externalLinkClass: 'test-external',\n    },\n  };\n\n  complexUrlProps: TextMetadata = {\n    content:\n      'URLs complejas: https://api.github.com/repos/angular/angular/issues?state=open&sort=updated&per_page=50, búsqueda https://google.com/search?q=angular+ionic+components#results, y documentación https://angular.io/guide/getting-started#development-environment.',\n    size: 'medium',\n    color: 'dark',\n    bold: false,\n    processLinks: true,\n    linkConfig: {\n      openExternalInNewTab: true,\n      linkClass: 'test-complex',\n      externalLinkClass: 'test-external',\n    },\n  };\n\n  parenthesesProps: TextMetadata = {\n    content:\n      'Paréntesis de contexto (ver https://docs.angular.io) vs URLs con paréntesis https://example.com/api/method(param) en el contenido. También funciona (https://ionic.io/docs).',\n    size: 'medium',\n    color: 'dark',\n    bold: false,\n    processLinks: true,\n    linkConfig: {\n      openExternalInNewTab: true,\n      linkClass: 'test-parentheses',\n      externalLinkClass: 'test-external',\n    },\n  };\n\n  mixedFormatsProps: TextMetadata = {\n    content:\n      'Formatos mezclados: [Documentación oficial](https://angular.io/docs), enlace directo https://github.com/angular/angular, ruta interna /dashboard/settings, [guía de inicio](/getting-started), y API https://api.example.com/v1/users?active=true.',\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: 'test-mixed',\n      externalLinkClass: 'test-external',\n      internalLinkClass: 'test-internal',\n    },\n  };\n\n  edgeCasesProps: TextMetadata = {\n    content:\n      'Casos extremos: \"https://quoted-url.com\", múltiple puntuación https://example.com?!!, URL al final de oración https://final-url.org. También consecutivos: https://first.com y https://second.com.',\n    size: 'medium',\n    color: 'dark',\n    bold: false,\n    processLinks: true,\n    linkConfig: {\n      openExternalInNewTab: true,\n      linkClass: 'test-edge',\n      externalLinkClass: 'test-external',\n    },\n  };\n\n  devUrlsProps: TextMetadata = {\n    content:\n      'URLs de desarrollo: http://localhost:4200/dashboard, servidor local https://127.0.0.1:8080/api/status, desarrollo http://dev.example.com:3000/debug?verbose=true, y túnel https://abc123.ngrok.io/webhook.',\n    size: 'medium',\n    color: 'dark',\n    bold: false,\n    processLinks: true,\n    linkConfig: {\n      openExternalInNewTab: false, // Para desarrollo, puede ser útil no abrir en nueva pestaña\n      linkClass: 'test-dev',\n      externalLinkClass: 'test-dev-external',\n    },\n  };\n}\n"]}
|
|
@@ -100,6 +100,30 @@ export class LinkProcessingExampleComponent {
|
|
|
100
100
|
internalLinkClass: 'mixed-internal',
|
|
101
101
|
},
|
|
102
102
|
};
|
|
103
|
+
this.punctuationTestProps = {
|
|
104
|
+
content: 'URLs con puntuación final: https://ionicframework.com/docs, también https://angular.io! Pregunta sobre https://github.com/angular? Y punto final: https://typescript.org. Paréntesis (https://rxjs.dev) y comillas "https://zone.js". ¡Todos funcionan!',
|
|
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
|
+
};
|
|
115
|
+
this.complexUrlsProps = {
|
|
116
|
+
content: 'URLs complejas: https://example.com/path?param=value&other=123#section, búsqueda en https://google.com/search?q=angular+components, y API https://api.github.com/repos/owner/repo/issues?state=open. Todos con parámetros y fragmentos.',
|
|
117
|
+
size: 'medium',
|
|
118
|
+
color: 'dark',
|
|
119
|
+
bold: false,
|
|
120
|
+
processLinks: true,
|
|
121
|
+
linkConfig: {
|
|
122
|
+
openExternalInNewTab: true,
|
|
123
|
+
linkClass: 'complex-url',
|
|
124
|
+
externalLinkClass: 'complex-external',
|
|
125
|
+
},
|
|
126
|
+
};
|
|
103
127
|
}
|
|
104
128
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: LinkProcessingExampleComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
105
129
|
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: `
|
|
@@ -140,8 +164,18 @@ export class LinkProcessingExampleComponent {
|
|
|
140
164
|
<h3>Mezcla de enlaces directos y Markdown:</h3>
|
|
141
165
|
<val-text [props]="mixedFormatsProps"></val-text>
|
|
142
166
|
</div>
|
|
167
|
+
|
|
168
|
+
<div class="example-section">
|
|
169
|
+
<h3>Corrección de puntuación en URLs:</h3>
|
|
170
|
+
<val-text [props]="punctuationTestProps"></val-text>
|
|
171
|
+
</div>
|
|
172
|
+
|
|
173
|
+
<div class="example-section">
|
|
174
|
+
<h3>URLs complejas con parámetros y fragmentos:</h3>
|
|
175
|
+
<val-text [props]="complexUrlsProps"></val-text>
|
|
176
|
+
</div>
|
|
143
177
|
</div>
|
|
144
|
-
`, isInline: true, styles: [".
|
|
178
|
+
`, isInline: true, styles: [".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"] }] }); }
|
|
145
179
|
}
|
|
146
180
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: LinkProcessingExampleComponent, decorators: [{
|
|
147
181
|
type: Component,
|
|
@@ -183,7 +217,17 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
183
217
|
<h3>Mezcla de enlaces directos y Markdown:</h3>
|
|
184
218
|
<val-text [props]="mixedFormatsProps"></val-text>
|
|
185
219
|
</div>
|
|
220
|
+
|
|
221
|
+
<div class="example-section">
|
|
222
|
+
<h3>Corrección de puntuación en URLs:</h3>
|
|
223
|
+
<val-text [props]="punctuationTestProps"></val-text>
|
|
224
|
+
</div>
|
|
225
|
+
|
|
226
|
+
<div class="example-section">
|
|
227
|
+
<h3>URLs complejas con parámetros y fragmentos:</h3>
|
|
228
|
+
<val-text [props]="complexUrlsProps"></val-text>
|
|
229
|
+
</div>
|
|
186
230
|
</div>
|
|
187
|
-
`, styles: [".
|
|
231
|
+
`, styles: [".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"] }]
|
|
188
232
|
}] });
|
|
189
|
-
//# 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    </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"]}
|
|
233
|
+
//# 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\n      <div class=\"example-section\">\n        <h3>URLs complejas con parámetros y fragmentos:</h3>\n        <val-text [props]=\"complexUrlsProps\"></val-text>\n      </div>\n    </div>\n  `,\n  styles: [\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 final: https://ionicframework.com/docs, también https://angular.io! Pregunta sobre https://github.com/angular? Y punto final: https://typescript.org. Paréntesis (https://rxjs.dev) y comillas \"https://zone.js\". ¡Todos funcionan!',\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  complexUrlsProps: TextMetadata = {\n    content:\n      'URLs complejas: https://example.com/path?param=value&other=123#section, búsqueda en https://google.com/search?q=angular+components, y API https://api.github.com/repos/owner/repo/issues?state=open. Todos con parámetros y fragmentos.',\n    size: 'medium',\n    color: 'dark',\n    bold: false,\n    processLinks: true,\n    linkConfig: {\n      openExternalInNewTab: true,\n      linkClass: 'complex-url',\n      externalLinkClass: 'complex-external',\n    },\n  };\n}\n"]}
|