@hmcts/ccd-case-ui-toolkit 7.2.55-markdown-add-validation → 7.2.56

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.
@@ -52,15 +52,10 @@ export class FormValidatorsService {
52
52
  return validator;
53
53
  }
54
54
  static markDownPatternValidator() {
55
- // Matches: [text](url), ![alt](url), <img ...>, <a ...>...</a>
56
- const inlineMarkdownPattern = /(?:!?\[[^\]]{0,500}\]\([^)]{0,500}\)|<(?:img\b[^>]{0,500}>|a\b[^>]{0,500}>[\s\S]*?<\/a>))/i;
57
- // Matches: [text][id], ![alt][id], and the collapsed form [text][]
58
- const referenceBoxPattern = /(!)?\[((?:[^[\]\\]|\\.){0,500})\]\s*\[([^\]]{0,100})\]/;
59
- // Matches: autolinks such as <http://example.com>
60
- const autolinkPattern = /<(?:[A-Za-z][A-Za-z0-9+.-]*:[^ <>\n]*|[A-Za-z0-9.!#$%&'*+/=?^_`{|}~-]+@[A-Za-z0-9-]+(?:\.[A-Za-z0-9-]+)+)>/;
55
+ const pattern = /(\[[^\]]{0,500}\]\([^)]{0,500}\)|!\[[^\]]{0,500}\]\([^)]{0,500}\)|<img[^>]{0,500}>|<a[^>]{0,500}>.*?<\/a>)/;
61
56
  return (control) => {
62
57
  const value = control?.value?.toString().trim();
63
- return (value && (inlineMarkdownPattern.test(value) || referenceBoxPattern.test(value) || this.matchesReferenceUrlDef(value) || autolinkPattern.test(value) || this.hasMultiBracket(value))) ? { markDownPattern: {} } : null;
58
+ return (value && pattern.test(value)) ? { markDownPattern: {} } : null;
64
59
  };
65
60
  }
66
61
  // TODO: Strip this out as it's only here for the moment because
@@ -78,77 +73,10 @@ export class FormValidatorsService {
78
73
  }
79
74
  return control;
80
75
  }
81
- // Check for multi-bracket markdown links and validate destination URL
82
- static hasMultiBracket(value) {
83
- // Sonar-friendly detector: opening-run + text + first closing ']'
84
- const openingTextClosePattern = /\[{1,10}[^[\]\n]{1,60}\]/;
85
- // Can add an additional RegEx for additional URL validation rules if needed here
86
- let scanIndex = 0;
87
- const totalLength = value.length;
88
- while (scanIndex < totalLength) {
89
- const seg = this.findOpeningTextClose(value, scanIndex, openingTextClosePattern);
90
- if (!seg) {
91
- return false; // no candidate -> no match
92
- }
93
- const runs = this.extendClosingRunAndRequireParen(value, seg.absStart, seg.afterFirstClose);
94
- if (runs && runs.openingRunCount === runs.closingRunCount) {
95
- // If there were additional validation rules, they would be applied here
96
- return true;
97
- }
98
- // Advance to avoid stalling on overlaps
99
- scanIndex = seg.absStart + 1;
100
- }
101
- return false;
102
- }
103
- // Find opening '[' run, text, and first closing ']'
104
- static findOpeningTextClose(source, fromIndex, pattern) {
105
- const slice = source.slice(fromIndex);
106
- const match = pattern.exec(slice);
107
- if (!match) {
108
- return null;
109
- }
110
- const absStart = fromIndex + match.index;
111
- const afterFirstClose = absStart + match[0].length; // index just after the first ']'
112
- return { absStart, afterFirstClose };
113
- }
114
- // Count opening '[' run, extend the ']' run, and require '(' right after the full ']' run
115
- static extendClosingRunAndRequireParen(source, absStart, afterFirstClose) {
116
- const n = source.length;
117
- // Count opening '[' run (e.g., '[[[')
118
- let openingRunCount = 0;
119
- for (let i = absStart; i < n && source[i] === '['; i++) {
120
- openingRunCount++;
121
- }
122
- // Extend closing ']' run forward from the first one
123
- let closingRunCount = 1;
124
- let afterClosingRun = afterFirstClose;
125
- while (afterClosingRun < n && source[afterClosingRun] === ']') {
126
- closingRunCount++;
127
- afterClosingRun++;
128
- }
129
- return { openingRunCount, closingRunCount, afterOpenParen: afterClosingRun + 1 };
130
- }
131
- static isValidReferenceUrlTitleTail(tail) {
132
- const possibleTitle = tail.trim();
133
- // Accept exactly one of: "title", 'title', (title) — bounded and single-line.
134
- if (!possibleTitle || /^"[^"\r\n]{0,300}"$/.test(possibleTitle) || /^'[^'\r\n]{0,300}'$/.test(possibleTitle) || /^\([^)\r\n]{0,300}\)$/.test(possibleTitle)) {
135
- return true;
136
- }
137
- return false;
138
- }
139
- static matchesReferenceUrlDef(line) {
140
- // Single-line, pragmatic CommonMark-style reference definition e.g. [text]: http://example.com
141
- const baseReferenceUrlPattern = /^[ \t]{0,3}\[([^\]]{1,100})\]:[ \t]*<?([^\s>]{1,2048})>?[ \t]*([^ \t\r\n].*)?$/m;
142
- const mainRegEx = baseReferenceUrlPattern.exec(line);
143
- if (!mainRegEx)
144
- return false;
145
- const tail = mainRegEx[3] ?? "";
146
- return this.isValidReferenceUrlTitleTail(tail);
147
- }
148
76
  static ɵfac = function FormValidatorsService_Factory(__ngFactoryType__) { return new (__ngFactoryType__ || FormValidatorsService)(); };
149
77
  static ɵprov = /*@__PURE__*/ i0.ɵɵdefineInjectable({ token: FormValidatorsService, factory: FormValidatorsService.ɵfac });
150
78
  }
151
79
  (() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(FormValidatorsService, [{
152
80
  type: Injectable
153
81
  }], null, null); })();
154
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-validators.service.js","sourceRoot":"","sources":["../../../../../../../projects/ccd-case-ui-toolkit/src/lib/shared/services/form/form-validators.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { AbstractControl, ValidationErrors, ValidatorFn, Validators } from '@angular/forms';\n\nimport { Constants } from '../../commons/constants';\nimport { CaseField } from '../../domain/definition/case-field.model';\nimport { FieldTypeEnum } from '../../domain/definition/field-type-enum.model';\n\n@Injectable()\nexport class FormValidatorsService {\n  private static readonly CUSTOM_VALIDATED_TYPES: FieldTypeEnum[] = [\n    'Date', 'MoneyGBP', 'Label', 'JudicialUser'\n  ];\n  private static readonly DEFAULT_INPUT_TEXT = 'text';\n  private static readonly DEFAULT_INPUT_TEXTAREA = 'textAreas';\n\n  public static addValidators(caseField: CaseField, control: AbstractControl): AbstractControl {\n    if (\n      caseField.display_context === Constants.MANDATORY &&\n      FormValidatorsService.CUSTOM_VALIDATED_TYPES.indexOf(caseField.field_type.type) === -1\n    ) {\n      const validators = [Validators.required];\n      if (caseField.field_type.type === 'Text') {\n        validators.push(this.markDownPatternValidator());\n        if (caseField.field_type.regular_expression) {\n          validators.push(Validators.pattern(caseField.field_type.regular_expression));\n        } else {\n          validators.push(this.emptyValidator());\n        }\n        if (caseField.field_type.min && (typeof caseField.field_type.min === 'number')) {\n          validators.push(Validators.minLength(caseField.field_type.min));\n        }\n        if (caseField.field_type.max && (typeof caseField.field_type.max === 'number')) {\n          validators.push(Validators.maxLength(caseField.field_type.max));\n        }\n      }\n\n      if (caseField.field_type.type === 'TextArea') {\n        validators.push(this.emptyValidator());\n        validators.push(this.markDownPatternValidator());\n      }\n\n      if (control.validator) {\n        validators.push(control.validator);\n      }\n      control.setValidators(validators);\n    } else if (caseField.display_context === 'OPTIONAL' && (caseField.field_type.type === 'Text' || caseField.field_type.type === 'TextArea')\n      || (caseField.display_context === 'COMPLEX' && caseField.field_type.type === 'Complex')) {\n      control.setValidators(this.markDownPatternValidator());\n    }\n\n    return control;\n  }\n\n  public static emptyValidator(): ValidatorFn {\n    const validator = (control: AbstractControl): ValidationErrors | null => {\n      if (control?.value?.toString().trim().length === 0) {\n        return { required: {} };\n      }\n      return null;\n    };\n    return validator;\n  }\n\n  public static markDownPatternValidator(): ValidatorFn {\n    // Matches: [text](url), ![alt](url), <img ...>, <a ...>...</a>\n    const inlineMarkdownPattern = /(?:!?\\[[^\\]]{0,500}\\]\\([^)]{0,500}\\)|<(?:img\\b[^>]{0,500}>|a\\b[^>]{0,500}>[\\s\\S]*?<\\/a>))/i;\n\n    // Matches: [text][id], ![alt][id], and the collapsed form [text][]\n    const referenceBoxPattern = /(!)?\\[((?:[^[\\]\\\\]|\\\\.){0,500})\\]\\s*\\[([^\\]]{0,100})\\]/;\n\n    // Matches: autolinks such as <http://example.com>\n    const autolinkPattern = /<(?:[A-Za-z][A-Za-z0-9+.-]*:[^ <>\\n]*|[A-Za-z0-9.!#$%&'*+/=?^_`{|}~-]+@[A-Za-z0-9-]+(?:\\.[A-Za-z0-9-]+)+)>/;\n\n    return (control: AbstractControl): ValidationErrors | null => {\n      const value = control?.value?.toString().trim();\n      return (value && (inlineMarkdownPattern.test(value) || referenceBoxPattern.test(value) || this.matchesReferenceUrlDef(value) || autolinkPattern.test(value) || this.hasMultiBracket(value as string))) ? { markDownPattern: {} } : null;\n    };\n  }\n\n  // TODO: Strip this out as it's only here for the moment because\n  // the service is being injected all over the place but it doesn't\n  // need to be as FormValidatorsService.addValidators is perfectly\n  // happy being static.\n  public addValidators(caseField: CaseField, control: AbstractControl): AbstractControl {\n    return FormValidatorsService.addValidators(caseField, control);\n  }\n\n  public addMarkDownValidators(formGroup: AbstractControl, controlPath: string): AbstractControl {\n    const control = formGroup.get(controlPath);\n    if (control) {\n      control.setValidators(FormValidatorsService.markDownPatternValidator());\n      control.updateValueAndValidity();\n    }\n    return control;\n  }\n\n  // Check for multi-bracket markdown links and validate destination URL\n  private static hasMultiBracket(value: string): boolean {\n\n    // Sonar-friendly detector: opening-run + text + first closing ']'\n    const openingTextClosePattern = /\\[{1,10}[^[\\]\\n]{1,60}\\]/;\n\n    // Can add an additional RegEx for additional URL validation rules if needed here\n\n    let scanIndex = 0;\n    const totalLength = value.length;\n\n    while (scanIndex < totalLength) {\n      const seg = this.findOpeningTextClose(value, scanIndex, openingTextClosePattern);\n      if (!seg) {\n        return false; // no candidate -> no match\n      }\n\n      const runs = this.extendClosingRunAndRequireParen(value, seg.absStart, seg.afterFirstClose);\n      if (runs && runs.openingRunCount === runs.closingRunCount) {\n        // If there were additional validation rules, they would be applied here\n        return true;\n      }\n\n      // Advance to avoid stalling on overlaps\n      scanIndex = seg.absStart + 1;\n    }\n    return false;\n  }\n\n  // Find opening '[' run, text, and first closing ']'\n  private static findOpeningTextClose(\n    source: string,\n    fromIndex: number,\n    pattern: RegExp\n  ): { absStart: number; afterFirstClose: number } | null {\n    const slice = source.slice(fromIndex);\n    const match = pattern.exec(slice);\n    if (!match) {\n      return null;\n    }\n    const absStart = fromIndex + match.index;\n    const afterFirstClose = absStart + match[0].length; // index just after the first ']'\n    return { absStart, afterFirstClose };\n  }\n\n  // Count opening '[' run, extend the ']' run, and require '(' right after the full ']' run\n  private static extendClosingRunAndRequireParen(\n    source: string,\n    absStart: number,\n    afterFirstClose: number\n  ): { openingRunCount: number; closingRunCount: number; afterOpenParen: number } | null {\n    const n = source.length;\n\n    // Count opening '[' run (e.g., '[[[')\n    let openingRunCount = 0;\n    for (let i = absStart; i < n && source[i] === '['; i++) {\n      openingRunCount++;\n    }\n\n    // Extend closing ']' run forward from the first one\n    let closingRunCount = 1;\n    let afterClosingRun = afterFirstClose;\n    while (afterClosingRun < n && source[afterClosingRun] === ']') {\n      closingRunCount++;\n      afterClosingRun++;\n    }\n\n    return { openingRunCount, closingRunCount, afterOpenParen: afterClosingRun + 1 };\n  }\n\n  private static isValidReferenceUrlTitleTail(tail: string): boolean {\n    const possibleTitle = tail.trim();\n    // Accept exactly one of: \"title\", 'title', (title) — bounded and single-line.\n    if (!possibleTitle || /^\"[^\"\\r\\n]{0,300}\"$/.test(possibleTitle) || /^'[^'\\r\\n]{0,300}'$/.test(possibleTitle) || /^\\([^)\\r\\n]{0,300}\\)$/.test(possibleTitle)) {\n      return true;\n    }\n    return false;\n  }\n\n\n  private static matchesReferenceUrlDef(line: string): boolean {\n    // Single-line, pragmatic CommonMark-style reference definition e.g. [text]: http://example.com\n    const baseReferenceUrlPattern = /^[ \\t]{0,3}\\[([^\\]]{1,100})\\]:[ \\t]*<?([^\\s>]{1,2048})>?[ \\t]*([^ \\t\\r\\n].*)?$/m;\n\n    const mainRegEx = baseReferenceUrlPattern.exec(line);\n    if (!mainRegEx) return false;\n    const tail = mainRegEx[3] ?? \"\";\n    return this.isValidReferenceUrlTitleTail(tail);\n  }\n}\n"]}
82
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-validators.service.js","sourceRoot":"","sources":["../../../../../../../projects/ccd-case-ui-toolkit/src/lib/shared/services/form/form-validators.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { AbstractControl, ValidationErrors, ValidatorFn, Validators } from '@angular/forms';\n\nimport { Constants } from '../../commons/constants';\nimport { CaseField } from '../../domain/definition/case-field.model';\nimport { FieldTypeEnum } from '../../domain/definition/field-type-enum.model';\n\n@Injectable()\nexport class FormValidatorsService {\n  private static readonly CUSTOM_VALIDATED_TYPES: FieldTypeEnum[] = [\n    'Date', 'MoneyGBP', 'Label', 'JudicialUser'\n  ];\n  private static readonly DEFAULT_INPUT_TEXT = 'text';\n  private static readonly DEFAULT_INPUT_TEXTAREA = 'textAreas';\n\n  public static addValidators(caseField: CaseField, control: AbstractControl): AbstractControl {\n    if (\n      caseField.display_context === Constants.MANDATORY &&\n      FormValidatorsService.CUSTOM_VALIDATED_TYPES.indexOf(caseField.field_type.type) === -1\n    ) {\n      const validators = [Validators.required];\n      if (caseField.field_type.type === 'Text') {\n        validators.push(this.markDownPatternValidator());\n        if (caseField.field_type.regular_expression) {\n          validators.push(Validators.pattern(caseField.field_type.regular_expression));\n        } else {\n          validators.push(this.emptyValidator());\n        }\n        if (caseField.field_type.min && (typeof caseField.field_type.min === 'number')) {\n          validators.push(Validators.minLength(caseField.field_type.min));\n        }\n        if (caseField.field_type.max && (typeof caseField.field_type.max === 'number')) {\n          validators.push(Validators.maxLength(caseField.field_type.max));\n        }\n      }\n\n      if (caseField.field_type.type === 'TextArea') {\n        validators.push(this.emptyValidator());\n        validators.push(this.markDownPatternValidator());\n      }\n\n      if (control.validator) {\n        validators.push(control.validator);\n      }\n      control.setValidators(validators);\n    } else if (caseField.display_context === 'OPTIONAL' && (caseField.field_type.type === 'Text' || caseField.field_type.type === 'TextArea')\n    || (caseField.display_context === 'COMPLEX' && caseField.field_type.type === 'Complex')) {\n      control.setValidators(this.markDownPatternValidator());\n    }\n\n    return control;\n  }\n\n  public static emptyValidator(): ValidatorFn {\n    const validator = (control: AbstractControl):ValidationErrors | null => {\n      if (control?.value?.toString().trim().length === 0) {\n        return { required: {} };\n      }\n      return null;\n    };\n    return validator;\n  }\n\n  public static markDownPatternValidator(): ValidatorFn {\n    const pattern = /(\\[[^\\]]{0,500}\\]\\([^)]{0,500}\\)|!\\[[^\\]]{0,500}\\]\\([^)]{0,500}\\)|<img[^>]{0,500}>|<a[^>]{0,500}>.*?<\\/a>)/;\n\n    return (control: AbstractControl): ValidationErrors | null => {\n      const value = control?.value?.toString().trim();\n      return (value && pattern.test(value)) ? { markDownPattern: {} } : null;\n    };\n  }\n\n  // TODO: Strip this out as it's only here for the moment because\n  // the service is being injected all over the place but it doesn't\n  // need to be as FormValidatorsService.addValidators is perfectly\n  // happy being static.\n  public addValidators(caseField: CaseField, control: AbstractControl): AbstractControl {\n    return FormValidatorsService.addValidators(caseField, control);\n  }\n\n  public addMarkDownValidators(formGroup: AbstractControl, controlPath: string): AbstractControl {\n    const control = formGroup.get(controlPath);\n    if (control) {\n      control.setValidators(FormValidatorsService.markDownPatternValidator());\n      control.updateValueAndValidity();\n    }\n    return control;\n  }\n}\n"]}
@@ -5093,15 +5093,10 @@ class FormValidatorsService {
5093
5093
  return validator;
5094
5094
  }
5095
5095
  static markDownPatternValidator() {
5096
- // Matches: [text](url), ![alt](url), <img ...>, <a ...>...</a>
5097
- const inlineMarkdownPattern = /(?:!?\[[^\]]{0,500}\]\([^)]{0,500}\)|<(?:img\b[^>]{0,500}>|a\b[^>]{0,500}>[\s\S]*?<\/a>))/i;
5098
- // Matches: [text][id], ![alt][id], and the collapsed form [text][]
5099
- const referenceBoxPattern = /(!)?\[((?:[^[\]\\]|\\.){0,500})\]\s*\[([^\]]{0,100})\]/;
5100
- // Matches: autolinks such as <http://example.com>
5101
- const autolinkPattern = /<(?:[A-Za-z][A-Za-z0-9+.-]*:[^ <>\n]*|[A-Za-z0-9.!#$%&'*+/=?^_`{|}~-]+@[A-Za-z0-9-]+(?:\.[A-Za-z0-9-]+)+)>/;
5096
+ const pattern = /(\[[^\]]{0,500}\]\([^)]{0,500}\)|!\[[^\]]{0,500}\]\([^)]{0,500}\)|<img[^>]{0,500}>|<a[^>]{0,500}>.*?<\/a>)/;
5102
5097
  return (control) => {
5103
5098
  const value = control?.value?.toString().trim();
5104
- return (value && (inlineMarkdownPattern.test(value) || referenceBoxPattern.test(value) || this.matchesReferenceUrlDef(value) || autolinkPattern.test(value) || this.hasMultiBracket(value))) ? { markDownPattern: {} } : null;
5099
+ return (value && pattern.test(value)) ? { markDownPattern: {} } : null;
5105
5100
  };
5106
5101
  }
5107
5102
  // TODO: Strip this out as it's only here for the moment because
@@ -5119,73 +5114,6 @@ class FormValidatorsService {
5119
5114
  }
5120
5115
  return control;
5121
5116
  }
5122
- // Check for multi-bracket markdown links and validate destination URL
5123
- static hasMultiBracket(value) {
5124
- // Sonar-friendly detector: opening-run + text + first closing ']'
5125
- const openingTextClosePattern = /\[{1,10}[^[\]\n]{1,60}\]/;
5126
- // Can add an additional RegEx for additional URL validation rules if needed here
5127
- let scanIndex = 0;
5128
- const totalLength = value.length;
5129
- while (scanIndex < totalLength) {
5130
- const seg = this.findOpeningTextClose(value, scanIndex, openingTextClosePattern);
5131
- if (!seg) {
5132
- return false; // no candidate -> no match
5133
- }
5134
- const runs = this.extendClosingRunAndRequireParen(value, seg.absStart, seg.afterFirstClose);
5135
- if (runs && runs.openingRunCount === runs.closingRunCount) {
5136
- // If there were additional validation rules, they would be applied here
5137
- return true;
5138
- }
5139
- // Advance to avoid stalling on overlaps
5140
- scanIndex = seg.absStart + 1;
5141
- }
5142
- return false;
5143
- }
5144
- // Find opening '[' run, text, and first closing ']'
5145
- static findOpeningTextClose(source, fromIndex, pattern) {
5146
- const slice = source.slice(fromIndex);
5147
- const match = pattern.exec(slice);
5148
- if (!match) {
5149
- return null;
5150
- }
5151
- const absStart = fromIndex + match.index;
5152
- const afterFirstClose = absStart + match[0].length; // index just after the first ']'
5153
- return { absStart, afterFirstClose };
5154
- }
5155
- // Count opening '[' run, extend the ']' run, and require '(' right after the full ']' run
5156
- static extendClosingRunAndRequireParen(source, absStart, afterFirstClose) {
5157
- const n = source.length;
5158
- // Count opening '[' run (e.g., '[[[')
5159
- let openingRunCount = 0;
5160
- for (let i = absStart; i < n && source[i] === '['; i++) {
5161
- openingRunCount++;
5162
- }
5163
- // Extend closing ']' run forward from the first one
5164
- let closingRunCount = 1;
5165
- let afterClosingRun = afterFirstClose;
5166
- while (afterClosingRun < n && source[afterClosingRun] === ']') {
5167
- closingRunCount++;
5168
- afterClosingRun++;
5169
- }
5170
- return { openingRunCount, closingRunCount, afterOpenParen: afterClosingRun + 1 };
5171
- }
5172
- static isValidReferenceUrlTitleTail(tail) {
5173
- const possibleTitle = tail.trim();
5174
- // Accept exactly one of: "title", 'title', (title) — bounded and single-line.
5175
- if (!possibleTitle || /^"[^"\r\n]{0,300}"$/.test(possibleTitle) || /^'[^'\r\n]{0,300}'$/.test(possibleTitle) || /^\([^)\r\n]{0,300}\)$/.test(possibleTitle)) {
5176
- return true;
5177
- }
5178
- return false;
5179
- }
5180
- static matchesReferenceUrlDef(line) {
5181
- // Single-line, pragmatic CommonMark-style reference definition e.g. [text]: http://example.com
5182
- const baseReferenceUrlPattern = /^[ \t]{0,3}\[([^\]]{1,100})\]:[ \t]*<?([^\s>]{1,2048})>?[ \t]*([^ \t\r\n].*)?$/m;
5183
- const mainRegEx = baseReferenceUrlPattern.exec(line);
5184
- if (!mainRegEx)
5185
- return false;
5186
- const tail = mainRegEx[3] ?? "";
5187
- return this.isValidReferenceUrlTitleTail(tail);
5188
- }
5189
5117
  static ɵfac = function FormValidatorsService_Factory(__ngFactoryType__) { return new (__ngFactoryType__ || FormValidatorsService)(); };
5190
5118
  static ɵprov = /*@__PURE__*/ i0.ɵɵdefineInjectable({ token: FormValidatorsService, factory: FormValidatorsService.ɵfac });
5191
5119
  }