@leanix/components 0.2.250 → 0.2.253

Sign up to get free protection for your applications and to get access to all the features.
package/index.d.ts CHANGED
@@ -3,8 +3,6 @@ export * from './lib/core-ui/core-ui.constants';
3
3
  export * from './lib/core-ui/annotations/required';
4
4
  export * from './lib/core-ui/pipes/br.pipe';
5
5
  export * from './lib/core-ui/pipes/custom-date.pipe';
6
- export * from './lib/core-ui/pipes/linkify/linkify.pipe';
7
- export * from './lib/core-ui/pipes/linkify/unlinkify.pipe';
8
6
  export * from './lib/core-ui/pipes/lx-is-uuid.pipe';
9
7
  export * from './lib/core-ui/pipes/lx-time-ago.pipe';
10
8
  export * from './lib/core-ui/pipes/lx-translate.pipe';
@@ -38,6 +36,8 @@ export * from './lib/core-ui/tooltip/tooltip-position.interface';
38
36
  export * from './lib/core-ui/tooltip/tooltip.component';
39
37
  export * from './lib/core-ui/tooltip/tooltip.directive';
40
38
  export * from './lib/core-ui/tooltip/tooltip.module';
39
+ export * from './lib/core-ui/linkify/linkify.pipe';
40
+ export * from './lib/core-ui/linkify/unlinkify.pipe';
41
41
  export * from './lib/core-ui/functions/core-css.helpers';
42
42
  export * from './lib/core-ui/services/resize-observer.service';
43
43
  export * from './lib/forms-ui/forms-ui.module';
@@ -17,6 +17,11 @@ export declare class EllipsisComponent implements OnInit, OnDestroy {
17
17
  private translateService;
18
18
  static DEFAULT_RESIZE_DEBOUNCE_MS: number;
19
19
  content: string;
20
+ /**
21
+ * Only set this to false if the content is not a user provided string
22
+ * or if you sanitize the provided content yourself.
23
+ */
24
+ escapeHtmlInContent: boolean;
20
25
  private contentSpanEl$;
21
26
  contentSpanEl: ElementRef<HTMLSpanElement>;
22
27
  private showMoreButtonEl$;
@@ -24,6 +29,7 @@ export declare class EllipsisComponent implements OnInit, OnDestroy {
24
29
  isShowingMore$: BehaviorSubject<boolean>;
25
30
  showButton$: Observable<boolean>;
26
31
  showMoreButtonLabel$: Observable<string>;
32
+ sanitizedContent$: Observable<string>;
27
33
  private content$;
28
34
  private destroyed$;
29
35
  constructor(debounceMsAfterResize: number, cdRef: ChangeDetectorRef, hostEl: ElementRef, resizeObserverService: ResizeObserverService, translateService: TranslateService);
@@ -33,5 +39,5 @@ export declare class EllipsisComponent implements OnInit, OnDestroy {
33
39
  private detectChangesWhenObservableEmits;
34
40
  private isContentOverflowing;
35
41
  static ɵfac: i0.ɵɵFactoryDeclaration<EllipsisComponent, never>;
36
- static ɵcmp: i0.ɵɵComponentDeclaration<EllipsisComponent, "lx-ellipsis", never, { "content": "content"; }, {}, never, never>;
42
+ static ɵcmp: i0.ɵɵComponentDeclaration<EllipsisComponent, "lx-ellipsis", never, { "content": "content"; "escapeHtmlInContent": "escapeHtmlInContent"; }, {}, never, never>;
37
43
  }
@@ -15,8 +15,8 @@ import * as i13 from "./pipes/highlight-range.pipe";
15
15
  import * as i14 from "./pipes/highlight-term.pipe";
16
16
  import * as i15 from "./directives/html.directive";
17
17
  import * as i16 from "./components/icon-scale/icon-scale.component";
18
- import * as i17 from "./pipes/linkify/linkify.pipe";
19
- import * as i18 from "./pipes/linkify/unlinkify.pipe";
18
+ import * as i17 from "./linkify/linkify.pipe";
19
+ import * as i18 from "./linkify/unlinkify.pipe";
20
20
  import * as i19 from "./pipes/lx-time-ago.pipe";
21
21
  import * as i20 from "./pipes/lx-translate.pipe";
22
22
  import * as i21 from "./pipes/markdown.pipe";
@@ -6,6 +6,8 @@ import * as i0 from "@angular/core";
6
6
  * - markdown link syntax
7
7
  * ... into clickable anchor elements.
8
8
  *
9
+ * The characters "<" and ">" are escaped with their HTML entities &lt; and &gt;.
10
+ *
9
11
  * You have an user interface where you don't want clickable links but also
10
12
  * don't want users to see the "ugly" markdown link syntax?
11
13
  * -> Use the 'lxUnlikify' pipe to replace markdown link syntax with just the link name
@@ -43,6 +45,15 @@ export declare class LxLinkifyPipe implements PipeTransform {
43
45
  private turnMarkdownStyleLinksIntoAnchorTags;
44
46
  private wrapRawHttpLinksWithAnchorTags;
45
47
  private getAllRegexMatches;
48
+ /**
49
+ * We assume that lxLinkify is exclusively used on user provided strings.
50
+ * This is why we want to escape any other HTML tags that are already present in the string.
51
+ * The logic implemented here has been used with no issues for three years in our Fact Sheet comments. See https://github.com/gregjacobs/Autolinker.js/pull/313
52
+ *
53
+ * When using lxLinkify in conjunction with other pipes that add HTML, make sure to use lxLinkify first,
54
+ * so that it doesn't escape the HTML of any previous pipes.
55
+ */
56
+ private escapeHtmlInUserProvidedString;
46
57
  static ɵfac: i0.ɵɵFactoryDeclaration<LxLinkifyPipe, never>;
47
58
  static ɵpipe: i0.ɵɵPipeDeclaration<LxLinkifyPipe, "lxLinkify">;
48
59
  }
@@ -23,6 +23,7 @@ export declare class BasicDropdownComponent extends KeyboardSelectDirective impl
23
23
  createNewOptionSelected: EventEmitter<void>;
24
24
  optionTemplateRef: TemplateRef<any>;
25
25
  createNewOptionTemplateRef: TemplateRef<any>;
26
+ descriptionTemplateRef: TemplateRef<any>;
26
27
  selectOption(option: any): void;
27
28
  get isNewItem(): boolean;
28
29
  trackByProp(prop?: string): (index: number, pill: any) => any;
@@ -30,5 +31,5 @@ export declare class BasicDropdownComponent extends KeyboardSelectDirective impl
30
31
  onNewItemSelected(): void;
31
32
  onCreateNewOptionSelected(): void;
32
33
  static ɵfac: i0.ɵɵFactoryDeclaration<BasicDropdownComponent, never>;
33
- static ɵcmp: i0.ɵɵComponentDeclaration<BasicDropdownComponent, "lx-basic-dropdown", never, { "options": "options"; "initiallySelectedIndex": "initiallySelectedIndex"; "labelKey": "labelKey"; "itemKey": "itemKey"; "placeholder": "placeholder"; "loading": "loading"; "newOptionLabel": "newOptionLabel"; "padding": "padding"; "showCreateNewOption": "showCreateNewOption"; "disabledOptions": "disabledOptions"; }, { "onItemSelected": "onItemSelected"; "triggerRequestForMoreEntries": "triggerRequestForMoreEntries"; "newOptionLabelSelected": "newOptionLabelSelected"; "createNewOptionSelected": "createNewOptionSelected"; }, ["optionTemplateRef", "createNewOptionTemplateRef"], never>;
34
+ static ɵcmp: i0.ɵɵComponentDeclaration<BasicDropdownComponent, "lx-basic-dropdown", never, { "options": "options"; "initiallySelectedIndex": "initiallySelectedIndex"; "labelKey": "labelKey"; "itemKey": "itemKey"; "placeholder": "placeholder"; "loading": "loading"; "newOptionLabel": "newOptionLabel"; "padding": "padding"; "showCreateNewOption": "showCreateNewOption"; "disabledOptions": "disabledOptions"; }, { "onItemSelected": "onItemSelected"; "triggerRequestForMoreEntries": "triggerRequestForMoreEntries"; "newOptionLabelSelected": "newOptionLabelSelected"; "createNewOptionSelected": "createNewOptionSelected"; }, ["optionTemplateRef", "createNewOptionTemplateRef", "descriptionTemplateRef"], never>;
34
35
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@leanix/components",
3
- "version": "0.2.250",
3
+ "version": "0.2.253",
4
4
  "license": "Apache-2.0",
5
5
  "author": "LeanIX GmbH",
6
6
  "repository": {
@@ -1,130 +0,0 @@
1
- import { Pipe } from '@angular/core';
2
- import * as i0 from "@angular/core";
3
- /**
4
- * This pipe transforms...
5
- * - "raw" http(s) links
6
- * - markdown link syntax
7
- * ... into clickable anchor elements.
8
- *
9
- * You have an user interface where you don't want clickable links but also
10
- * don't want users to see the "ugly" markdown link syntax?
11
- * -> Use the 'lxUnlikify' pipe to replace markdown link syntax with just the link name
12
- */
13
- export class LxLinkifyPipe {
14
- transform(text) {
15
- if (text && typeof text === 'string') {
16
- const textWithRawLinks = this.wrapRawHttpLinksWithAnchorTags(text);
17
- const textWithRawAndNamedLinks = this.turnMarkdownStyleLinksIntoAnchorTags(textWithRawLinks);
18
- return textWithRawAndNamedLinks;
19
- }
20
- else {
21
- return text;
22
- }
23
- }
24
- turnMarkdownStyleLinksIntoAnchorTags(text) {
25
- let textWithRawAndNamedLinks = text;
26
- const namedLinkMatches = this.getAllRegexMatches(LxLinkifyPipe.NAMED_LINK_REGEX, text);
27
- namedLinkMatches.forEach((namedLinkMatch) => {
28
- const [source, name, url] = namedLinkMatch;
29
- const urlIsValid = url && !/javascript\:/i.test(url);
30
- if (source && name && urlIsValid) {
31
- textWithRawAndNamedLinks = textWithRawAndNamedLinks.replace(source, `<a href="${url}" target="_blank" rel="noopener noreferrer">${name}</a>`);
32
- }
33
- });
34
- return textWithRawAndNamedLinks;
35
- }
36
- wrapRawHttpLinksWithAnchorTags(text) {
37
- let textWithRawLinks = text;
38
- /**
39
- * Keeping track of this index prevents infinite loops
40
- * where a previously processed link starts with the same characters
41
- * as a second link.
42
- * e.g. https://angular.io/docs followed by https://angular.io
43
- */
44
- let nextIndexToStartReplacingFrom = 0;
45
- const rawLinkMatches = this.getAllRegexMatches(LxLinkifyPipe.HTTP_LINK_REGEX, text);
46
- rawLinkMatches.forEach((rawLinkMatch) => {
47
- const [url] = rawLinkMatch;
48
- const wrapUrlInAnchor = (sanitizedUrlMatch) => {
49
- const firstPart = textWithRawLinks.substring(0, nextIndexToStartReplacingFrom);
50
- const anchorTagHtml = `<a href="${sanitizedUrlMatch}" target="_blank" rel="noopener noreferrer">${sanitizedUrlMatch}</a>`;
51
- const secondPart = textWithRawLinks.substring(nextIndexToStartReplacingFrom).replace(sanitizedUrlMatch, anchorTagHtml);
52
- textWithRawLinks = firstPart + secondPart;
53
- nextIndexToStartReplacingFrom = rawLinkMatch.index + anchorTagHtml.length;
54
- };
55
- if (url) {
56
- /*
57
- * TODO: get rid of all this code once Safari supports negative lookbehinds in regular expressions
58
- * The following is RegExp that handles the same stuff as the JS code below:
59
- *
60
- * /(?:(?:(?<!\]\())(?:https|http):\/\/)(?:[^\s/$.?#][^\s]*(?<![\.)]))/gi;
61
- *
62
- * Demo on regex101: https://regex101.com/r/7Vl9bg/1
63
- *
64
- * Check lookbehind support here: https://caniuse.com/?search=lookbehind
65
- */
66
- const lastUrlCharacterIndex = rawLinkMatch.index + url.length - 1;
67
- const textUsedToPerformMatching = rawLinkMatch.input;
68
- const lastCharacterInUrl = textUsedToPerformMatching[lastUrlCharacterIndex];
69
- const twoCharactersInFrontOfTheLink = rawLinkMatch.index > 3
70
- ? `${textUsedToPerformMatching[rawLinkMatch.index - 2]}${textUsedToPerformMatching[rawLinkMatch.index - 1]}`
71
- : '';
72
- const isMarkdownSyntaxLink = twoCharactersInFrontOfTheLink === '](';
73
- if (!isMarkdownSyntaxLink && lastCharacterInUrl === '.') {
74
- const characterAfterUrl = textUsedToPerformMatching[lastUrlCharacterIndex + 1];
75
- if (!characterAfterUrl || characterAfterUrl === ' ' || characterAfterUrl === '\n') {
76
- const urlWithoutDotAtTheEnd = url.slice(0, -1);
77
- wrapUrlInAnchor(urlWithoutDotAtTheEnd);
78
- }
79
- }
80
- else if (!isMarkdownSyntaxLink) {
81
- wrapUrlInAnchor(url);
82
- }
83
- }
84
- });
85
- return textWithRawLinks;
86
- }
87
- getAllRegexMatches(regex, input) {
88
- let match;
89
- const matches = [];
90
- while ((match = regex.exec(input)) !== null) {
91
- matches.push(match);
92
- }
93
- return matches;
94
- }
95
- }
96
- /**
97
- * This is not the "one URL regex to rule them all", but a more realistic one which should work
98
- * for any URLs that our customers include in text fields on a Fact Sheet.
99
- *
100
- * Regex rules explained in plain text:
101
- *
102
- * (?:(?:https?):\/\/) -> Links must start with "https://" or "http://"
103
- *
104
- * (?:[^\s/$.?#][^\s]*(?<![\.)])) LET'S SPLIT THIS ONE UP
105
- *
106
- * [^\s/$.?#][^\s]* -> Match any legal URL character until the next whitespace
107
- * (?<![\.)] -> A negative lookahead to prevent matching a dot or parenthesis following a URL
108
- *
109
- * Link to regex101: https://regex101.com/r/d3KtfH/1 (NOTE: please update this link when changing the regex)
110
- */
111
- LxLinkifyPipe.HTTP_LINK_REGEX = /(?:(?:https?):\/\/)(?:[^\s/$.?#][^\s]*)/gi;
112
- /**
113
- * This will match the markdown link syntax: [link name](url)
114
- * Regex rules explained in plain text:
115
- *
116
- * (?:\[([^\]]*)\]) -> Match any characters inside square brackets
117
- * \(([^\s\/$.?#][^\s]*)\) -> Notice that this is the same regex as the HTTP_LINK_REGEX above,
118
- * but without the requirement for the http protocol.
119
- * This allows for links without including the protocol or also "mailto:hello@world.de" links
120
- *
121
- * Link to regex101: https://regex101.com/r/5UMUH8/1
122
- */
123
- LxLinkifyPipe.NAMED_LINK_REGEX = /(?:\[([^\]]*)\])\(([^\s\/$.?#][^\s]*)\)/gi;
124
- LxLinkifyPipe.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.2", ngImport: i0, type: LxLinkifyPipe, deps: [], target: i0.ɵɵFactoryTarget.Pipe });
125
- LxLinkifyPipe.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "12.0.0", version: "13.3.2", ngImport: i0, type: LxLinkifyPipe, name: "lxLinkify" });
126
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.2", ngImport: i0, type: LxLinkifyPipe, decorators: [{
127
- type: Pipe,
128
- args: [{ name: 'lxLinkify' }]
129
- }] });
130
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"linkify.pipe.js","sourceRoot":"","sources":["../../../../../../../../libs/components/src/lib/core-ui/pipes/linkify/linkify.pipe.ts"],"names":[],"mappings":"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","sourcesContent":["import { Pipe, PipeTransform } from '@angular/core';\n\n/**\n * This pipe transforms...\n * - \"raw\" http(s) links\n * - markdown link syntax\n * ... into clickable anchor elements.\n *\n * You have an user interface where you don't want clickable links but also\n * don't want users to see the \"ugly\" markdown link syntax?\n * -> Use the 'lxUnlikify' pipe to replace markdown link syntax with just the link name\n */\n@Pipe({ name: 'lxLinkify' })\nexport class LxLinkifyPipe implements PipeTransform {\n  /**\n   * This is not the \"one URL regex to rule them all\", but a more realistic one which should work\n   * for any URLs that our customers include in text fields on a Fact Sheet.\n   *\n   * Regex rules explained in plain text:\n   *\n   * (?:(?:https?):\\/\\/)       ->    Links must start with \"https://\" or \"http://\"\n   *\n   * (?:[^\\s/$.?#][^\\s]*(?<![\\.)])) LET'S SPLIT THIS ONE UP\n   *\n   * [^\\s/$.?#][^\\s]*   ->    Match any legal URL character until the next whitespace\n   * (?<![\\.)]          ->    A negative lookahead to prevent matching a dot or parenthesis following a URL\n   *\n   * Link to regex101: https://regex101.com/r/d3KtfH/1 (NOTE: please update this link when changing the regex)\n   */\n  public static readonly HTTP_LINK_REGEX = /(?:(?:https?):\\/\\/)(?:[^\\s/$.?#][^\\s]*)/gi;\n\n  /**\n   * This will match the markdown link syntax: [link name](url)\n   * Regex rules explained in plain text:\n   *\n   * (?:\\[([^\\]]*)\\])            ->     Match any characters inside square brackets\n   * \\(([^\\s\\/$.?#][^\\s]*)\\)     ->     Notice that this is the same regex as the HTTP_LINK_REGEX above,\n   *                                    but without the requirement for the http protocol.\n   *                                    This allows for links without including the protocol or also \"mailto:hello@world.de\" links\n   *\n   * Link to regex101: https://regex101.com/r/5UMUH8/1\n   */\n  public static readonly NAMED_LINK_REGEX = /(?:\\[([^\\]]*)\\])\\(([^\\s\\/$.?#][^\\s]*)\\)/gi;\n\n  transform(text?: string | null): string | undefined | null {\n    if (text && typeof text === 'string') {\n      const textWithRawLinks = this.wrapRawHttpLinksWithAnchorTags(text);\n      const textWithRawAndNamedLinks = this.turnMarkdownStyleLinksIntoAnchorTags(textWithRawLinks);\n      return textWithRawAndNamedLinks;\n    } else {\n      return text;\n    }\n  }\n\n  private turnMarkdownStyleLinksIntoAnchorTags(text: string) {\n    let textWithRawAndNamedLinks = text;\n    const namedLinkMatches = this.getAllRegexMatches(LxLinkifyPipe.NAMED_LINK_REGEX, text);\n    namedLinkMatches.forEach((namedLinkMatch) => {\n      const [source, name, url] = namedLinkMatch;\n      const urlIsValid = url && !/javascript\\:/i.test(url);\n      if (source && name && urlIsValid) {\n        textWithRawAndNamedLinks = textWithRawAndNamedLinks.replace(\n          source,\n          `<a href=\"${url}\" target=\"_blank\" rel=\"noopener noreferrer\">${name}</a>`\n        );\n      }\n    });\n    return textWithRawAndNamedLinks;\n  }\n\n  private wrapRawHttpLinksWithAnchorTags(text: string) {\n    let textWithRawLinks = text;\n    /**\n     * Keeping track of this index prevents infinite loops\n     * where a previously processed link starts with the same characters\n     * as a second link.\n     * e.g. https://angular.io/docs followed by https://angular.io\n     */\n    let nextIndexToStartReplacingFrom = 0;\n    const rawLinkMatches = this.getAllRegexMatches(LxLinkifyPipe.HTTP_LINK_REGEX, text);\n    rawLinkMatches.forEach((rawLinkMatch) => {\n      const [url] = rawLinkMatch;\n      const wrapUrlInAnchor = (sanitizedUrlMatch: string) => {\n        const firstPart = textWithRawLinks.substring(0, nextIndexToStartReplacingFrom);\n        const anchorTagHtml = `<a href=\"${sanitizedUrlMatch}\" target=\"_blank\" rel=\"noopener noreferrer\">${sanitizedUrlMatch}</a>`;\n        const secondPart = textWithRawLinks.substring(nextIndexToStartReplacingFrom).replace(sanitizedUrlMatch!, anchorTagHtml);\n        textWithRawLinks = firstPart + secondPart;\n        nextIndexToStartReplacingFrom = rawLinkMatch.index + anchorTagHtml.length;\n      };\n      if (url) {\n        /*\n         * TODO: get rid of all this code once Safari supports negative lookbehinds in regular expressions\n         * The following is RegExp that handles the same stuff as the JS code below:\n         *\n         * /(?:(?:(?<!\\]\\())(?:https|http):\\/\\/)(?:[^\\s/$.?#][^\\s]*(?<![\\.)]))/gi;\n         *\n         * Demo on regex101: https://regex101.com/r/7Vl9bg/1\n         *\n         * Check lookbehind support here: https://caniuse.com/?search=lookbehind\n         */\n        const lastUrlCharacterIndex = rawLinkMatch.index + url.length - 1;\n        const textUsedToPerformMatching = rawLinkMatch.input;\n        const lastCharacterInUrl = textUsedToPerformMatching[lastUrlCharacterIndex];\n        const twoCharactersInFrontOfTheLink =\n          rawLinkMatch.index > 3\n            ? `${textUsedToPerformMatching[rawLinkMatch.index - 2]}${textUsedToPerformMatching[rawLinkMatch.index - 1]}`\n            : '';\n        const isMarkdownSyntaxLink = twoCharactersInFrontOfTheLink === '](';\n\n        if (!isMarkdownSyntaxLink && lastCharacterInUrl === '.') {\n          const characterAfterUrl = textUsedToPerformMatching[lastUrlCharacterIndex + 1];\n          if (!characterAfterUrl || characterAfterUrl === ' ' || characterAfterUrl === '\\n') {\n            const urlWithoutDotAtTheEnd = url.slice(0, -1);\n            wrapUrlInAnchor(urlWithoutDotAtTheEnd);\n          }\n        } else if (!isMarkdownSyntaxLink) {\n          wrapUrlInAnchor(url);\n        }\n      }\n    });\n    return textWithRawLinks;\n  }\n\n  private getAllRegexMatches(regex: RegExp, input: string) {\n    let match: RegExpExecArray | null;\n    const matches: RegExpExecArray[] = [];\n    while ((match = regex.exec(input)) !== null) {\n      matches.push(match);\n    }\n    return matches;\n  }\n}\n"]}
@@ -1,48 +0,0 @@
1
- import { Pipe } from '@angular/core';
2
- import { LxLinkifyPipe } from './linkify.pipe';
3
- import * as i0 from "@angular/core";
4
- /**
5
- * This pipe replaces markdown link syntax with just the link name (omits the link altogether).
6
- * Example: [Angular documentation](http://angular.io/docs) -> Angular documentation
7
- *
8
- * It can also be used for "bridging the gap" until your view is ready to use "lxLinkify"
9
- * and you just want to get rid of the "useless" markdown syntax fast.
10
- *
11
- * While there are views where we want markdown style links to be clickable anchor tags,
12
- * there are other views where this can degrade the UX.
13
- * Example: in the Fact Sheet list on the inventory list view, having an arbitrary number
14
- * of links on the page can slow down the keyboard navigation while navigating through the list.
15
- */
16
- export class LxUnlinkifyPipe {
17
- transform(text) {
18
- if (text && typeof text === 'string') {
19
- let textWhereMarkdownLinkSyntaxIsReplacedWithJustTheLinkName = text;
20
- const markdownLinkSyntaxMatches = this.getAllRegexMatches(LxLinkifyPipe.NAMED_LINK_REGEX, text);
21
- markdownLinkSyntaxMatches.forEach((markdownLinkSyntaxMatch) => {
22
- const [source, name] = markdownLinkSyntaxMatch;
23
- if (source && name) {
24
- textWhereMarkdownLinkSyntaxIsReplacedWithJustTheLinkName = textWhereMarkdownLinkSyntaxIsReplacedWithJustTheLinkName.replace(source, name);
25
- }
26
- });
27
- return textWhereMarkdownLinkSyntaxIsReplacedWithJustTheLinkName;
28
- }
29
- else {
30
- return text;
31
- }
32
- }
33
- getAllRegexMatches(regex, input) {
34
- let match;
35
- const matches = [];
36
- while ((match = regex.exec(input)) !== null) {
37
- matches.push(match);
38
- }
39
- return matches;
40
- }
41
- }
42
- LxUnlinkifyPipe.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.2", ngImport: i0, type: LxUnlinkifyPipe, deps: [], target: i0.ɵɵFactoryTarget.Pipe });
43
- LxUnlinkifyPipe.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "12.0.0", version: "13.3.2", ngImport: i0, type: LxUnlinkifyPipe, name: "lxUnlinkify" });
44
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.2", ngImport: i0, type: LxUnlinkifyPipe, decorators: [{
45
- type: Pipe,
46
- args: [{ name: 'lxUnlinkify' }]
47
- }] });
48
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidW5saW5raWZ5LnBpcGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9saWJzL2NvbXBvbmVudHMvc3JjL2xpYi9jb3JlLXVpL3BpcGVzL2xpbmtpZnkvdW5saW5raWZ5LnBpcGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLElBQUksRUFBaUIsTUFBTSxlQUFlLENBQUM7QUFDcEQsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLGdCQUFnQixDQUFDOztBQUUvQzs7Ozs7Ozs7Ozs7R0FXRztBQUVILE1BQU0sT0FBTyxlQUFlO0lBQzFCLFNBQVMsQ0FBQyxJQUFvQjtRQUM1QixJQUFJLElBQUksSUFBSSxPQUFPLElBQUksS0FBSyxRQUFRLEVBQUU7WUFDcEMsSUFBSSx3REFBd0QsR0FBRyxJQUFJLENBQUM7WUFDcEUsTUFBTSx5QkFBeUIsR0FBRyxJQUFJLENBQUMsa0JBQWtCLENBQUMsYUFBYSxDQUFDLGdCQUFnQixFQUFFLElBQUksQ0FBQyxDQUFDO1lBQ2hHLHlCQUF5QixDQUFDLE9BQU8sQ0FBQyxDQUFDLHVCQUF1QixFQUFFLEVBQUU7Z0JBQzVELE1BQU0sQ0FBQyxNQUFNLEVBQUUsSUFBSSxDQUFDLEdBQUcsdUJBQXVCLENBQUM7Z0JBQy9DLElBQUksTUFBTSxJQUFJLElBQUksRUFBRTtvQkFDbEIsd0RBQXdELEdBQUcsd0RBQXdELENBQUMsT0FBTyxDQUN6SCxNQUFNLEVBQ04sSUFBSSxDQUNMLENBQUM7aUJBQ0g7WUFDSCxDQUFDLENBQUMsQ0FBQztZQUVILE9BQU8sd0RBQXdELENBQUM7U0FDakU7YUFBTTtZQUNMLE9BQU8sSUFBSSxDQUFDO1NBQ2I7SUFDSCxDQUFDO0lBRU8sa0JBQWtCLENBQUMsS0FBYSxFQUFFLEtBQWE7UUFDckQsSUFBSSxLQUE2QixDQUFDO1FBQ2xDLE1BQU0sT0FBTyxHQUFzQixFQUFFLENBQUM7UUFDdEMsT0FBTyxDQUFDLEtBQUssR0FBRyxLQUFLLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDLEtBQUssSUFBSSxFQUFFO1lBQzNDLE9BQU8sQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7U0FDckI7UUFDRCxPQUFPLE9BQU8sQ0FBQztJQUNqQixDQUFDOzs0R0E1QlUsZUFBZTswR0FBZixlQUFlOzJGQUFmLGVBQWU7a0JBRDNCLElBQUk7bUJBQUMsRUFBRSxJQUFJLEVBQUUsYUFBYSxFQUFFIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUGlwZSwgUGlwZVRyYW5zZm9ybSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgTHhMaW5raWZ5UGlwZSB9IGZyb20gJy4vbGlua2lmeS5waXBlJztcblxuLyoqXG4gKiBUaGlzIHBpcGUgcmVwbGFjZXMgbWFya2Rvd24gbGluayBzeW50YXggd2l0aCBqdXN0IHRoZSBsaW5rIG5hbWUgKG9taXRzIHRoZSBsaW5rIGFsdG9nZXRoZXIpLlxuICogRXhhbXBsZTogW0FuZ3VsYXIgZG9jdW1lbnRhdGlvbl0oaHR0cDovL2FuZ3VsYXIuaW8vZG9jcykgLT4gQW5ndWxhciBkb2N1bWVudGF0aW9uXG4gKlxuICogSXQgY2FuIGFsc28gYmUgdXNlZCBmb3IgXCJicmlkZ2luZyB0aGUgZ2FwXCIgdW50aWwgeW91ciB2aWV3IGlzIHJlYWR5IHRvIHVzZSBcImx4TGlua2lmeVwiXG4gKiBhbmQgeW91IGp1c3Qgd2FudCB0byBnZXQgcmlkIG9mIHRoZSBcInVzZWxlc3NcIiBtYXJrZG93biBzeW50YXggZmFzdC5cbiAqXG4gKiBXaGlsZSB0aGVyZSBhcmUgdmlld3Mgd2hlcmUgd2Ugd2FudCBtYXJrZG93biBzdHlsZSBsaW5rcyB0byBiZSBjbGlja2FibGUgYW5jaG9yIHRhZ3MsXG4gKiB0aGVyZSBhcmUgb3RoZXIgdmlld3Mgd2hlcmUgdGhpcyBjYW4gZGVncmFkZSB0aGUgVVguXG4gKiBFeGFtcGxlOiBpbiB0aGUgRmFjdCBTaGVldCBsaXN0IG9uIHRoZSBpbnZlbnRvcnkgbGlzdCB2aWV3LCBoYXZpbmcgYW4gYXJiaXRyYXJ5IG51bWJlclxuICogb2YgbGlua3Mgb24gdGhlIHBhZ2UgY2FuIHNsb3cgZG93biB0aGUga2V5Ym9hcmQgbmF2aWdhdGlvbiB3aGlsZSBuYXZpZ2F0aW5nIHRocm91Z2ggdGhlIGxpc3QuXG4gKi9cbkBQaXBlKHsgbmFtZTogJ2x4VW5saW5raWZ5JyB9KVxuZXhwb3J0IGNsYXNzIEx4VW5saW5raWZ5UGlwZSBpbXBsZW1lbnRzIFBpcGVUcmFuc2Zvcm0ge1xuICB0cmFuc2Zvcm0odGV4dD86IHN0cmluZyB8IG51bGwpOiBzdHJpbmcgfCB1bmRlZmluZWQgfCBudWxsIHtcbiAgICBpZiAodGV4dCAmJiB0eXBlb2YgdGV4dCA9PT0gJ3N0cmluZycpIHtcbiAgICAgIGxldCB0ZXh0V2hlcmVNYXJrZG93bkxpbmtTeW50YXhJc1JlcGxhY2VkV2l0aEp1c3RUaGVMaW5rTmFtZSA9IHRleHQ7XG4gICAgICBjb25zdCBtYXJrZG93bkxpbmtTeW50YXhNYXRjaGVzID0gdGhpcy5nZXRBbGxSZWdleE1hdGNoZXMoTHhMaW5raWZ5UGlwZS5OQU1FRF9MSU5LX1JFR0VYLCB0ZXh0KTtcbiAgICAgIG1hcmtkb3duTGlua1N5bnRheE1hdGNoZXMuZm9yRWFjaCgobWFya2Rvd25MaW5rU3ludGF4TWF0Y2gpID0+IHtcbiAgICAgICAgY29uc3QgW3NvdXJjZSwgbmFtZV0gPSBtYXJrZG93bkxpbmtTeW50YXhNYXRjaDtcbiAgICAgICAgaWYgKHNvdXJjZSAmJiBuYW1lKSB7XG4gICAgICAgICAgdGV4dFdoZXJlTWFya2Rvd25MaW5rU3ludGF4SXNSZXBsYWNlZFdpdGhKdXN0VGhlTGlua05hbWUgPSB0ZXh0V2hlcmVNYXJrZG93bkxpbmtTeW50YXhJc1JlcGxhY2VkV2l0aEp1c3RUaGVMaW5rTmFtZS5yZXBsYWNlKFxuICAgICAgICAgICAgc291cmNlLFxuICAgICAgICAgICAgbmFtZVxuICAgICAgICAgICk7XG4gICAgICAgIH1cbiAgICAgIH0pO1xuXG4gICAgICByZXR1cm4gdGV4dFdoZXJlTWFya2Rvd25MaW5rU3ludGF4SXNSZXBsYWNlZFdpdGhKdXN0VGhlTGlua05hbWU7XG4gICAgfSBlbHNlIHtcbiAgICAgIHJldHVybiB0ZXh0O1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgZ2V0QWxsUmVnZXhNYXRjaGVzKHJlZ2V4OiBSZWdFeHAsIGlucHV0OiBzdHJpbmcpIHtcbiAgICBsZXQgbWF0Y2g6IFJlZ0V4cEV4ZWNBcnJheSB8IG51bGw7XG4gICAgY29uc3QgbWF0Y2hlczogUmVnRXhwRXhlY0FycmF5W10gPSBbXTtcbiAgICB3aGlsZSAoKG1hdGNoID0gcmVnZXguZXhlYyhpbnB1dCkpICE9PSBudWxsKSB7XG4gICAgICBtYXRjaGVzLnB1c2gobWF0Y2gpO1xuICAgIH1cbiAgICByZXR1cm4gbWF0Y2hlcztcbiAgfVxufVxuIl19