@leanix/components 0.2.240 → 0.2.241

Sign up to get free protection for your applications and to get access to all the features.
@@ -13,72 +13,77 @@ import * as i0 from "@angular/core";
13
13
  export class LxLinkifyPipe {
14
14
  transform(text) {
15
15
  if (text && typeof text === 'string') {
16
- let textWithRawLinks = text;
17
- /**
18
- * Keeping track of this index prevents infinite loops
19
- * where a previously processed link starts with the same characters
20
- * as a second link.
21
- * e.g. https://angular.io/docs followed by https://angular.io
22
- */
23
- let nextIndexToStartReplacingFrom = 0;
24
- const rawLinkMatches = this.getAllRegexMatches(LxLinkifyPipe.HTTP_LINK_REGEX, text);
25
- rawLinkMatches.forEach((rawLinkMatch) => {
26
- const [url] = rawLinkMatch;
27
- const wrapUrlInAnchor = (sanitizedUrlMatch) => {
28
- const firstPart = textWithRawLinks.substring(0, nextIndexToStartReplacingFrom);
29
- const anchorTagHtml = `<a href="${sanitizedUrlMatch}" target="_blank" rel="noopener noreferrer">${sanitizedUrlMatch}</a>`;
30
- const secondPart = textWithRawLinks.substring(nextIndexToStartReplacingFrom).replace(sanitizedUrlMatch, anchorTagHtml);
31
- textWithRawLinks = firstPart + secondPart;
32
- nextIndexToStartReplacingFrom = rawLinkMatch.index + anchorTagHtml.length;
33
- };
34
- if (url) {
35
- /*
36
- * TODO: get rid of all this code once Safari supports negative lookbehinds in regular expressions
37
- * The following is RegExp that handles the same stuff as the JS code below:
38
- *
39
- * /(?:(?:(?<!\]\())(?:https|http):\/\/)(?:[^\s/$.?#][^\s]*(?<![\.)]))/gi;
40
- *
41
- * Demo on regex101: https://regex101.com/r/7Vl9bg/1
42
- *
43
- * Check lookbehind support here: https://caniuse.com/?search=lookbehind
44
- */
45
- const lastUrlCharacterIndex = rawLinkMatch.index + url.length - 1;
46
- const textUsedToPerformMatching = rawLinkMatch.input;
47
- const lastCharacterInUrl = textUsedToPerformMatching[lastUrlCharacterIndex];
48
- if (lastCharacterInUrl === '.') {
49
- const characterAfterUrl = textUsedToPerformMatching[lastUrlCharacterIndex + 1];
50
- if (!characterAfterUrl || characterAfterUrl === ' ' || characterAfterUrl === '\n') {
51
- const urlWithoutDotAtTheEnd = url.slice(0, -1);
52
- wrapUrlInAnchor(urlWithoutDotAtTheEnd);
53
- }
54
- }
55
- else if (rawLinkMatch.index > 3) {
56
- const twoCharactersInFrontOfTheLink = `${textUsedToPerformMatching[rawLinkMatch.index - 2]}${textUsedToPerformMatching[rawLinkMatch.index - 1]}`;
57
- if (twoCharactersInFrontOfTheLink && twoCharactersInFrontOfTheLink !== '](') {
58
- // only wrap url in anchor when it is not a named markdown link
59
- wrapUrlInAnchor(url);
60
- }
61
- }
62
- else {
63
- wrapUrlInAnchor(url);
64
- }
65
- }
66
- });
67
- let textWithRawAndNamedLinks = textWithRawLinks;
68
- const namedLinkMatches = this.getAllRegexMatches(LxLinkifyPipe.NAMED_LINK_REGEX, textWithRawLinks);
69
- namedLinkMatches.forEach((namedLinkMatch) => {
70
- const [source, name, url] = namedLinkMatch;
71
- const urlIsValid = url && !/javascript\:/i.test(url);
72
- if (source && name && urlIsValid) {
73
- textWithRawAndNamedLinks = textWithRawAndNamedLinks.replace(source, `<a href="${url}" target="_blank" rel="noopener noreferrer">${name}</a>`);
74
- }
75
- });
16
+ const textWithRawLinks = this.wrapRawHttpLinksWithAnchorTags(text);
17
+ const textWithRawAndNamedLinks = this.turnMarkdownStyleLinksIntoAnchorTags(textWithRawLinks);
76
18
  return textWithRawAndNamedLinks;
77
19
  }
78
20
  else {
79
21
  return text;
80
22
  }
81
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
+ }
82
87
  getAllRegexMatches(regex, input) {
83
88
  let match;
84
89
  const matches = [];
@@ -122,4 +127,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.2.5", ngImpor
122
127
  type: Pipe,
123
128
  args: [{ name: 'lxLinkify' }]
124
129
  }] });
125
- //# 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      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          if (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 (rawLinkMatch.index > 3) {\n            const twoCharactersInFrontOfTheLink = `${textUsedToPerformMatching[rawLinkMatch.index - 2]}${\n              textUsedToPerformMatching[rawLinkMatch.index - 1]\n            }`;\n            if (twoCharactersInFrontOfTheLink && twoCharactersInFrontOfTheLink !== '](') {\n              // only wrap url in anchor when it is not a named markdown link\n              wrapUrlInAnchor(url);\n            }\n          } else {\n            wrapUrlInAnchor(url);\n          }\n        }\n      });\n\n      let textWithRawAndNamedLinks = textWithRawLinks;\n      const namedLinkMatches = this.getAllRegexMatches(LxLinkifyPipe.NAMED_LINK_REGEX, textWithRawLinks);\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\n      return textWithRawAndNamedLinks;\n    } else {\n      return text;\n    }\n  }\n\n  private getAllRegexMatches(regex: RegExp, input: string) {\n    let match: RegExpExecArray | null;\n    const matches = [];\n    while ((match = regex.exec(input)) !== null) {\n      matches.push(match);\n    }\n    return matches;\n  }\n}\n"]}
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 = [];\n    while ((match = regex.exec(input)) !== null) {\n      matches.push(match);\n    }\n    return matches;\n  }\n}\n"]}
@@ -1167,72 +1167,77 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.2.5", ngImpor
1167
1167
  class LxLinkifyPipe {
1168
1168
  transform(text) {
1169
1169
  if (text && typeof text === 'string') {
1170
- let textWithRawLinks = text;
1171
- /**
1172
- * Keeping track of this index prevents infinite loops
1173
- * where a previously processed link starts with the same characters
1174
- * as a second link.
1175
- * e.g. https://angular.io/docs followed by https://angular.io
1176
- */
1177
- let nextIndexToStartReplacingFrom = 0;
1178
- const rawLinkMatches = this.getAllRegexMatches(LxLinkifyPipe.HTTP_LINK_REGEX, text);
1179
- rawLinkMatches.forEach((rawLinkMatch) => {
1180
- const [url] = rawLinkMatch;
1181
- const wrapUrlInAnchor = (sanitizedUrlMatch) => {
1182
- const firstPart = textWithRawLinks.substring(0, nextIndexToStartReplacingFrom);
1183
- const anchorTagHtml = `<a href="${sanitizedUrlMatch}" target="_blank" rel="noopener noreferrer">${sanitizedUrlMatch}</a>`;
1184
- const secondPart = textWithRawLinks.substring(nextIndexToStartReplacingFrom).replace(sanitizedUrlMatch, anchorTagHtml);
1185
- textWithRawLinks = firstPart + secondPart;
1186
- nextIndexToStartReplacingFrom = rawLinkMatch.index + anchorTagHtml.length;
1187
- };
1188
- if (url) {
1189
- /*
1190
- * TODO: get rid of all this code once Safari supports negative lookbehinds in regular expressions
1191
- * The following is RegExp that handles the same stuff as the JS code below:
1192
- *
1193
- * /(?:(?:(?<!\]\())(?:https|http):\/\/)(?:[^\s/$.?#][^\s]*(?<![\.)]))/gi;
1194
- *
1195
- * Demo on regex101: https://regex101.com/r/7Vl9bg/1
1196
- *
1197
- * Check lookbehind support here: https://caniuse.com/?search=lookbehind
1198
- */
1199
- const lastUrlCharacterIndex = rawLinkMatch.index + url.length - 1;
1200
- const textUsedToPerformMatching = rawLinkMatch.input;
1201
- const lastCharacterInUrl = textUsedToPerformMatching[lastUrlCharacterIndex];
1202
- if (lastCharacterInUrl === '.') {
1203
- const characterAfterUrl = textUsedToPerformMatching[lastUrlCharacterIndex + 1];
1204
- if (!characterAfterUrl || characterAfterUrl === ' ' || characterAfterUrl === '\n') {
1205
- const urlWithoutDotAtTheEnd = url.slice(0, -1);
1206
- wrapUrlInAnchor(urlWithoutDotAtTheEnd);
1207
- }
1208
- }
1209
- else if (rawLinkMatch.index > 3) {
1210
- const twoCharactersInFrontOfTheLink = `${textUsedToPerformMatching[rawLinkMatch.index - 2]}${textUsedToPerformMatching[rawLinkMatch.index - 1]}`;
1211
- if (twoCharactersInFrontOfTheLink && twoCharactersInFrontOfTheLink !== '](') {
1212
- // only wrap url in anchor when it is not a named markdown link
1213
- wrapUrlInAnchor(url);
1214
- }
1215
- }
1216
- else {
1217
- wrapUrlInAnchor(url);
1218
- }
1219
- }
1220
- });
1221
- let textWithRawAndNamedLinks = textWithRawLinks;
1222
- const namedLinkMatches = this.getAllRegexMatches(LxLinkifyPipe.NAMED_LINK_REGEX, textWithRawLinks);
1223
- namedLinkMatches.forEach((namedLinkMatch) => {
1224
- const [source, name, url] = namedLinkMatch;
1225
- const urlIsValid = url && !/javascript\:/i.test(url);
1226
- if (source && name && urlIsValid) {
1227
- textWithRawAndNamedLinks = textWithRawAndNamedLinks.replace(source, `<a href="${url}" target="_blank" rel="noopener noreferrer">${name}</a>`);
1228
- }
1229
- });
1170
+ const textWithRawLinks = this.wrapRawHttpLinksWithAnchorTags(text);
1171
+ const textWithRawAndNamedLinks = this.turnMarkdownStyleLinksIntoAnchorTags(textWithRawLinks);
1230
1172
  return textWithRawAndNamedLinks;
1231
1173
  }
1232
1174
  else {
1233
1175
  return text;
1234
1176
  }
1235
1177
  }
1178
+ turnMarkdownStyleLinksIntoAnchorTags(text) {
1179
+ let textWithRawAndNamedLinks = text;
1180
+ const namedLinkMatches = this.getAllRegexMatches(LxLinkifyPipe.NAMED_LINK_REGEX, text);
1181
+ namedLinkMatches.forEach((namedLinkMatch) => {
1182
+ const [source, name, url] = namedLinkMatch;
1183
+ const urlIsValid = url && !/javascript\:/i.test(url);
1184
+ if (source && name && urlIsValid) {
1185
+ textWithRawAndNamedLinks = textWithRawAndNamedLinks.replace(source, `<a href="${url}" target="_blank" rel="noopener noreferrer">${name}</a>`);
1186
+ }
1187
+ });
1188
+ return textWithRawAndNamedLinks;
1189
+ }
1190
+ wrapRawHttpLinksWithAnchorTags(text) {
1191
+ let textWithRawLinks = text;
1192
+ /**
1193
+ * Keeping track of this index prevents infinite loops
1194
+ * where a previously processed link starts with the same characters
1195
+ * as a second link.
1196
+ * e.g. https://angular.io/docs followed by https://angular.io
1197
+ */
1198
+ let nextIndexToStartReplacingFrom = 0;
1199
+ const rawLinkMatches = this.getAllRegexMatches(LxLinkifyPipe.HTTP_LINK_REGEX, text);
1200
+ rawLinkMatches.forEach((rawLinkMatch) => {
1201
+ const [url] = rawLinkMatch;
1202
+ const wrapUrlInAnchor = (sanitizedUrlMatch) => {
1203
+ const firstPart = textWithRawLinks.substring(0, nextIndexToStartReplacingFrom);
1204
+ const anchorTagHtml = `<a href="${sanitizedUrlMatch}" target="_blank" rel="noopener noreferrer">${sanitizedUrlMatch}</a>`;
1205
+ const secondPart = textWithRawLinks.substring(nextIndexToStartReplacingFrom).replace(sanitizedUrlMatch, anchorTagHtml);
1206
+ textWithRawLinks = firstPart + secondPart;
1207
+ nextIndexToStartReplacingFrom = rawLinkMatch.index + anchorTagHtml.length;
1208
+ };
1209
+ if (url) {
1210
+ /*
1211
+ * TODO: get rid of all this code once Safari supports negative lookbehinds in regular expressions
1212
+ * The following is RegExp that handles the same stuff as the JS code below:
1213
+ *
1214
+ * /(?:(?:(?<!\]\())(?:https|http):\/\/)(?:[^\s/$.?#][^\s]*(?<![\.)]))/gi;
1215
+ *
1216
+ * Demo on regex101: https://regex101.com/r/7Vl9bg/1
1217
+ *
1218
+ * Check lookbehind support here: https://caniuse.com/?search=lookbehind
1219
+ */
1220
+ const lastUrlCharacterIndex = rawLinkMatch.index + url.length - 1;
1221
+ const textUsedToPerformMatching = rawLinkMatch.input;
1222
+ const lastCharacterInUrl = textUsedToPerformMatching[lastUrlCharacterIndex];
1223
+ const twoCharactersInFrontOfTheLink = rawLinkMatch.index > 3
1224
+ ? `${textUsedToPerformMatching[rawLinkMatch.index - 2]}${textUsedToPerformMatching[rawLinkMatch.index - 1]}`
1225
+ : '';
1226
+ const isMarkdownSyntaxLink = twoCharactersInFrontOfTheLink === '](';
1227
+ if (!isMarkdownSyntaxLink && lastCharacterInUrl === '.') {
1228
+ const characterAfterUrl = textUsedToPerformMatching[lastUrlCharacterIndex + 1];
1229
+ if (!characterAfterUrl || characterAfterUrl === ' ' || characterAfterUrl === '\n') {
1230
+ const urlWithoutDotAtTheEnd = url.slice(0, -1);
1231
+ wrapUrlInAnchor(urlWithoutDotAtTheEnd);
1232
+ }
1233
+ }
1234
+ else if (!isMarkdownSyntaxLink) {
1235
+ wrapUrlInAnchor(url);
1236
+ }
1237
+ }
1238
+ });
1239
+ return textWithRawLinks;
1240
+ }
1236
1241
  getAllRegexMatches(regex, input) {
1237
1242
  let match;
1238
1243
  const matches = [];