@carefirst/library 2.0.29 → 2.0.30
Sign up to get free protection for your applications and to get access to all the features.
- package/esm2022/lib/components/form-input/form-input.component.mjs +13 -3
- package/esm2022/lib/utils/form-validators-utility.mjs +10 -1
- package/fesm2022/carefirst-library.mjs +129 -111
- package/fesm2022/carefirst-library.mjs.map +1 -1
- package/lib/components/form-input/form-input.component.d.ts +4 -2
- package/lib/utils/form-validators-utility.d.ts +1 -0
- package/package.json +1 -1
@@ -1,6 +1,7 @@
|
|
1
1
|
import { Component, Input, Output, EventEmitter } from '@angular/core';
|
2
2
|
import { FormControl } from '@angular/forms';
|
3
3
|
import { checkTruthAttribute, validateStringValue } from '../../utils/attribute.util';
|
4
|
+
import { CFPFormValidators } from '../../utils/form-validators-utility';
|
4
5
|
import * as i0 from "@angular/core";
|
5
6
|
import * as i1 from "@ionic/angular";
|
6
7
|
import * as i2 from "@angular/common";
|
@@ -15,8 +16,8 @@ const inputsC = {
|
|
15
16
|
};
|
16
17
|
export class FormInputComponent {
|
17
18
|
label;
|
18
|
-
min
|
19
|
-
max
|
19
|
+
min;
|
20
|
+
max;
|
20
21
|
labelPlacement;
|
21
22
|
placeholder;
|
22
23
|
inputmode;
|
@@ -37,6 +38,8 @@ export class FormInputComponent {
|
|
37
38
|
inputAutoCapitalize;
|
38
39
|
inputGreyBackground = false;
|
39
40
|
showPassword = false;
|
41
|
+
dateValidatorMin = null;
|
42
|
+
dateValidatorMax = null;
|
40
43
|
ngOnChanges(changes) {
|
41
44
|
this.inputLabelPlacement =
|
42
45
|
validateStringValue(changes, 'labelPlacement', inputsC.labelPlacement.slice(), this.inputLabelPlacement) || 'floating';
|
@@ -48,6 +51,13 @@ export class FormInputComponent {
|
|
48
51
|
this.inputAutoCapitalize =
|
49
52
|
validateStringValue(changes, 'autoCapitalize', inputsC.autoCapitalize.slice(), this.inputAutoCapitalize) || 'none';
|
50
53
|
this.inputGreyBackground = checkTruthAttribute(changes, 'greyBackground', this.inputGreyBackground);
|
54
|
+
if (this.inputType === 'date') {
|
55
|
+
if (changes['min']?.currentValue || changes['max']?.currentValue) {
|
56
|
+
this.dateValidatorMin = changes['min'] ? changes['min'].currentValue : this.dateValidatorMin;
|
57
|
+
this.dateValidatorMax = changes['max'] ? changes['max'].currentValue : this.dateValidatorMax;
|
58
|
+
this.control?.addValidators(CFPFormValidators.isValidDate(this.dateValidatorMin || null, this.dateValidatorMax || null));
|
59
|
+
}
|
60
|
+
}
|
51
61
|
}
|
52
62
|
togglePasswordShow() {
|
53
63
|
this.showPassword = !this.showPassword;
|
@@ -90,4 +100,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.3", ngImpor
|
|
90
100
|
}], valueChange: [{
|
91
101
|
type: Output
|
92
102
|
}] } });
|
93
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-input.component.js","sourceRoot":"","sources":["../../../../../../../projects/carefirst/library/src/lib/components/form-input/form-input.component.ts","../../../../../../../projects/carefirst/library/src/lib/components/form-input/form-input.component.html"],"names":[],"mappings":"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","sourcesContent":["/**\n * Carefirst Library Form Input\n *\n * @file          form-input.component\n * @description   Contains all the logic for generating a CareFirst Form Input Field\n * @author        Arno Jansen van Vuuren\n * @since         2023 - 12 - 14\n * @usage         <cf-form-input label=\"Label Name\" %labelPlacement% %inputMode% %noClearButton% %textCenter% %min% %max% %autoCapitalize% %type% %[(value)]=\"twoWayComs\"% %(valueChange)=\"currentValueEvent\"% %[control]%=\"formName.controls.controlName\"></cf-form-input>\n *                disable input by disabling the form control\n */\n\nimport { Component, Input, Output, type OnChanges, type SimpleChanges, EventEmitter } from '@angular/core';\nimport { FormControl } from '@angular/forms';\n//--- Utils\nimport { checkTruthAttribute, validateStringValue } from '../../utils/attribute.util';\n\n/**==============================================\n * @interface     inputsC\n * @description   Define available input values\n */\nconst inputsC = {\n  labelPlacement: ['fixed', 'floating', 'stacked', 'start', 'end'],\n  inputMode: ['text', 'email', 'numeric', 'tel', 'decimal', 'url', 'search'],\n  autoCapitalize: ['off', 'none', 'on', 'sentences', 'words', 'characters'],\n  type: ['date', 'datetime-local', 'email', 'month', 'number', 'password', 'search', 'tel', 'text', 'time', 'url', 'week'],\n} as const;\n\n@Component({\n  selector: 'cf-form-input',\n  templateUrl: './form-input.component.html',\n  styleUrl: './form-input.component.scss',\n})\nexport class FormInputComponent implements OnChanges {\n  @Input() label!: string;\n  @Input() min: number | string = 'none';\n  @Input() max: number | string = 'none';\n  @Input() labelPlacement?: (typeof inputsC.labelPlacement)[number];\n  @Input() placeholder?: string | undefined;\n  @Input() inputmode?: (typeof inputsC.inputMode)[number];\n  @Input() autoCapitalize?: (typeof inputsC.autoCapitalize)[number];\n  @Input() type?: (typeof inputsC.type)[number];\n  @Input() noClearButton?: boolean | string | undefined;\n  @Input() control?: FormControl | null;\n  @Input() textCenter?: boolean | string | undefined;\n  @Input() maxLength?: number | string;\n  @Input() greyBackground?: boolean | string | undefined;\n\n  //--- ngModel\n  @Input() value: string | undefined;\n  @Output() valueChange = new EventEmitter<typeof this.value>();\n\n  //--- Local variables\n  inputLabelPlacement: typeof this.labelPlacement;\n  localInputMode: typeof this.inputmode;\n  inputClear = false;\n  inputTextCenter = false;\n  inputType: typeof this.type;\n  inputAutoCapitalize: typeof this.autoCapitalize;\n  inputGreyBackground = false;\n  showPassword = false;\n\n  /**----------------------------------------------------------------\n   * @name          ngOnChanges\n   * @description   Detect changes to input values\n   * @param         {SimpleChanges} changes\n   */\n  ngOnChanges(changes: SimpleChanges): void {\n    //--- Label Placement\n    this.inputLabelPlacement =\n      validateStringValue<(typeof inputsC.labelPlacement)[number]>(\n        changes,\n        'labelPlacement',\n        inputsC.labelPlacement.slice(),\n        this.inputLabelPlacement\n      ) || 'floating';\n    //--- Input Type (keyboard)\n    this.localInputMode =\n      validateStringValue<(typeof inputsC.inputMode)[number]>(changes, 'inputmode', inputsC.inputMode.slice(), this.localInputMode) || 'text';\n    //--- Clear Button\n    this.inputClear = checkTruthAttribute(changes, 'noClearButton', this.inputClear);\n    //--- Text Center\n    this.inputTextCenter = checkTruthAttribute(changes, 'textCenter', this.inputTextCenter);\n    //--- Type\n    this.inputType = validateStringValue<(typeof inputsC.type)[number]>(changes, 'type', inputsC.type.slice(), this.inputType) || 'text';\n    //--- Auto Capitalize\n    this.inputAutoCapitalize =\n      validateStringValue<(typeof inputsC.autoCapitalize)[number]>(\n        changes,\n        'autoCapitalize',\n        inputsC.autoCapitalize.slice(),\n        this.inputAutoCapitalize\n      ) || 'none';\n    //--- Grey Background\n    this.inputGreyBackground = checkTruthAttribute(changes, 'greyBackground', this.inputGreyBackground);\n  }\n\n  /**----------------------------------------------------------------\n   * @name          togglePasswordShow\n   * @description   Toggle if password is showing or hidden\n   * @returns       {void}\n   */\n  togglePasswordShow(): void {\n    this.showPassword = !this.showPassword;\n    this.inputType = this.inputType === 'password' ? 'text' : 'password';\n  }\n}\n","<!-- ngModel -->\n<ion-input\n  #normalInput\n  *ngIf=\"!control\"\n  [ngClass]=\"{ 'text-center': inputTextCenter, 'grey-background': inputGreyBackground }\"\n  [label]=\"label\"\n  [labelPlacement]=\"inputLabelPlacement\"\n  [placeholder]=\"placeholder\"\n  fill=\"outline\"\n  [clearInput]=\"type === 'password' ? false : !inputClear\"\n  [autocapitalize]=\"autoCapitalize\"\n  mode=\"md\"\n  [inputmode]=\"localInputMode\"\n  [min]=\"min\"\n  [max]=\"max\"\n  [maxlength]=\"maxLength || null\"\n  [type]=\"inputType\"\n  (ionInput)=\"valueChange.emit($event.detail.value ?? undefined)\"\n  [value]=\"value\">\n  <cf-icon\n    slot=\"end\"\n    style=\"cursor: pointer\"\n    *ngIf=\"normalInput.value && type === 'password'\"\n    [icon]=\"showPassword ? 'hide-password' : 'show-password'\"\n    [height]=\"24\"\n    (click)=\"togglePasswordShow()\"></cf-icon\n></ion-input>\n<!-- Form Control -->\n<ion-input\n  #formControlInput\n  *ngIf=\"control\"\n  [ngClass]=\"{ 'text-center': inputTextCenter, 'grey-background': inputGreyBackground }\"\n  [label]=\"label\"\n  [labelPlacement]=\"inputLabelPlacement\"\n  [placeholder]=\"placeholder\"\n  fill=\"outline\"\n  [clearInput]=\"type === 'password' ? false : !inputClear\"\n  [autocapitalize]=\"autoCapitalize\"\n  mode=\"md\"\n  [inputmode]=\"localInputMode\"\n  [formControl]=\"control\"\n  [min]=\"min\"\n  [max]=\"max\"\n  [maxlength]=\"maxLength || null\"\n  [type]=\"inputType\"\n  (ionInput)=\"valueChange.emit($event.detail.value ?? undefined)\">\n  <cf-icon\n    slot=\"end\"\n    style=\"cursor: pointer\"\n    *ngIf=\"formControlInput.value && type === 'password'\"\n    [icon]=\"showPassword ? 'hide-password' : 'show-password'\"\n    [height]=\"24\"\n    (click)=\"togglePasswordShow()\"></cf-icon>\n</ion-input>\n<cf-form-validation *ngIf=\"control\" [control]=\"control\"></cf-form-validation>\n"]}
|
103
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-input.component.js","sourceRoot":"","sources":["../../../../../../../projects/carefirst/library/src/lib/components/form-input/form-input.component.ts","../../../../../../../projects/carefirst/library/src/lib/components/form-input/form-input.component.html"],"names":[],"mappings":"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","sourcesContent":["/**\n * Carefirst Library Form Input\n *\n * @file          form-input.component\n * @description   Contains all the logic for generating a CareFirst Form Input Field\n * @author        Arno Jansen van Vuuren\n * @since         2023 - 12 - 14\n * @usage         <cf-form-input label=\"Label Name\" %labelPlacement% %inputMode% %noClearButton% %textCenter% %min% %max% %autoCapitalize% %type% %[(value)]=\"twoWayComs\"% %(valueChange)=\"currentValueEvent\"% %[control]%=\"formName.controls.controlName\"></cf-form-input>\n *                disable input by disabling the form control\n */\n\nimport { Component, Input, Output, type OnChanges, type SimpleChanges, EventEmitter } from '@angular/core';\nimport { FormControl } from '@angular/forms';\n//--- Utils\nimport { checkTruthAttribute, validateStringValue } from '../../utils/attribute.util';\nimport { CFPFormValidators } from '../../utils/form-validators-utility';\n\n/**==============================================\n * @interface     inputsC\n * @description   Define available input values\n */\nconst inputsC = {\n  labelPlacement: ['fixed', 'floating', 'stacked', 'start', 'end'],\n  inputMode: ['text', 'email', 'numeric', 'tel', 'decimal', 'url', 'search'],\n  autoCapitalize: ['off', 'none', 'on', 'sentences', 'words', 'characters'],\n  type: ['date', 'datetime-local', 'email', 'month', 'number', 'password', 'search', 'tel', 'text', 'time', 'url', 'week'],\n} as const;\n\n@Component({\n  selector: 'cf-form-input',\n  templateUrl: './form-input.component.html',\n  styleUrl: './form-input.component.scss',\n})\nexport class FormInputComponent implements OnChanges {\n  @Input() label!: string;\n  @Input() min: string | number | undefined;\n  @Input() max: string | number | undefined;\n  @Input() labelPlacement?: (typeof inputsC.labelPlacement)[number];\n  @Input() placeholder?: string | undefined;\n  @Input() inputmode?: (typeof inputsC.inputMode)[number];\n  @Input() autoCapitalize?: (typeof inputsC.autoCapitalize)[number];\n  @Input() type?: (typeof inputsC.type)[number];\n  @Input() noClearButton?: boolean | string | undefined;\n  @Input() control?: FormControl | null;\n  @Input() textCenter?: boolean | string | undefined;\n  @Input() maxLength?: number | string;\n  @Input() greyBackground?: boolean | string | undefined;\n\n  //--- ngModel\n  @Input() value: string | undefined;\n  @Output() valueChange = new EventEmitter<typeof this.value>();\n\n  //--- Local variables\n  inputLabelPlacement: typeof this.labelPlacement;\n  localInputMode: typeof this.inputmode;\n  inputClear = false;\n  inputTextCenter = false;\n  inputType: typeof this.type;\n  inputAutoCapitalize: typeof this.autoCapitalize;\n  inputGreyBackground = false;\n  showPassword = false;\n  dateValidatorMin: string | null = null;\n  dateValidatorMax: string | null = null;\n\n  /**----------------------------------------------------------------\n   * @name          ngOnChanges\n   * @description   Detect changes to input values\n   * @param         {SimpleChanges} changes\n   */\n  ngOnChanges(changes: SimpleChanges): void {\n    //--- Label Placement\n    this.inputLabelPlacement =\n      validateStringValue<(typeof inputsC.labelPlacement)[number]>(\n        changes,\n        'labelPlacement',\n        inputsC.labelPlacement.slice(),\n        this.inputLabelPlacement\n      ) || 'floating';\n    //--- Input Type (keyboard)\n    this.localInputMode =\n      validateStringValue<(typeof inputsC.inputMode)[number]>(changes, 'inputmode', inputsC.inputMode.slice(), this.localInputMode) || 'text';\n    //--- Clear Button\n    this.inputClear = checkTruthAttribute(changes, 'noClearButton', this.inputClear);\n    //--- Text Center\n    this.inputTextCenter = checkTruthAttribute(changes, 'textCenter', this.inputTextCenter);\n    //--- Type\n    this.inputType = validateStringValue<(typeof inputsC.type)[number]>(changes, 'type', inputsC.type.slice(), this.inputType) || 'text';\n    //--- Auto Capitalize\n    this.inputAutoCapitalize =\n      validateStringValue<(typeof inputsC.autoCapitalize)[number]>(\n        changes,\n        'autoCapitalize',\n        inputsC.autoCapitalize.slice(),\n        this.inputAutoCapitalize\n      ) || 'none';\n    //--- Grey Background\n    this.inputGreyBackground = checkTruthAttribute(changes, 'greyBackground', this.inputGreyBackground);\n\n    //---<> adding multiple validators\n    //--- Automatically add isValidDate validator when input is a date type and has either a min or a max value\n    if (this.inputType === 'date') {\n      if (changes['min']?.currentValue || changes['max']?.currentValue) {\n        //--- Update min max input\n        this.dateValidatorMin = changes['min'] ? changes['min'].currentValue : this.dateValidatorMin;\n        this.dateValidatorMax = changes['max'] ? changes['max'].currentValue : this.dateValidatorMax;\n        //--- Add new validator\n        this.control?.addValidators(CFPFormValidators.isValidDate(this.dateValidatorMin || null, this.dateValidatorMax || null));\n      }\n    }\n  }\n\n  /**----------------------------------------------------------------\n   * @name          togglePasswordShow\n   * @description   Toggle if password is showing or hidden\n   * @returns       {void}\n   */\n  togglePasswordShow(): void {\n    this.showPassword = !this.showPassword;\n    this.inputType = this.inputType === 'password' ? 'text' : 'password';\n  }\n}\n","<!-- ngModel -->\n<ion-input\n  #normalInput\n  *ngIf=\"!control\"\n  [ngClass]=\"{ 'text-center': inputTextCenter, 'grey-background': inputGreyBackground }\"\n  [label]=\"label\"\n  [labelPlacement]=\"inputLabelPlacement\"\n  [placeholder]=\"placeholder\"\n  fill=\"outline\"\n  [clearInput]=\"type === 'password' ? false : !inputClear\"\n  [autocapitalize]=\"autoCapitalize\"\n  mode=\"md\"\n  [inputmode]=\"localInputMode\"\n  [min]=\"min\"\n  [max]=\"max\"\n  [maxlength]=\"maxLength || null\"\n  [type]=\"inputType\"\n  (ionInput)=\"valueChange.emit($event.detail.value ?? undefined)\"\n  [value]=\"value\">\n  <cf-icon\n    slot=\"end\"\n    style=\"cursor: pointer\"\n    *ngIf=\"normalInput.value && type === 'password'\"\n    [icon]=\"showPassword ? 'hide-password' : 'show-password'\"\n    [height]=\"24\"\n    (click)=\"togglePasswordShow()\"></cf-icon\n></ion-input>\n<!-- Form Control -->\n<ion-input\n  #formControlInput\n  *ngIf=\"control\"\n  [ngClass]=\"{ 'text-center': inputTextCenter, 'grey-background': inputGreyBackground }\"\n  [label]=\"label\"\n  [labelPlacement]=\"inputLabelPlacement\"\n  [placeholder]=\"placeholder\"\n  fill=\"outline\"\n  [clearInput]=\"type === 'password' ? false : !inputClear\"\n  [autocapitalize]=\"autoCapitalize\"\n  mode=\"md\"\n  [inputmode]=\"localInputMode\"\n  [formControl]=\"control\"\n  [min]=\"min\"\n  [max]=\"max\"\n  [maxlength]=\"maxLength || null\"\n  [type]=\"inputType\"\n  (ionInput)=\"valueChange.emit($event.detail.value ?? undefined)\">\n  <cf-icon\n    slot=\"end\"\n    style=\"cursor: pointer\"\n    *ngIf=\"formControlInput.value && type === 'password'\"\n    [icon]=\"showPassword ? 'hide-password' : 'show-password'\"\n    [height]=\"24\"\n    (click)=\"togglePasswordShow()\"></cf-icon>\n</ion-input>\n<cf-form-validation *ngIf=\"control\" [control]=\"control\"></cf-form-validation>\n"]}
|
@@ -109,5 +109,14 @@ export class CFPFormValidators {
|
|
109
109
|
return error ? { errorMessage: error } : null;
|
110
110
|
};
|
111
111
|
}
|
112
|
+
static isValidDate(minDate, maxDate) {
|
113
|
+
return (control) => {
|
114
|
+
if (!dayjs(new Date(control.value)).isValid())
|
115
|
+
return { errorMessage: 'The given date is not valid' };
|
116
|
+
if ((minDate && dayjs(new Date(control.value)).isBefore(minDate)) || (maxDate && dayjs(new Date(control.value)).isAfter(maxDate)))
|
117
|
+
return { errorMessage: 'The given date is not within the valid parameters' };
|
118
|
+
return null;
|
119
|
+
};
|
120
|
+
}
|
112
121
|
}
|
113
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-validators-utility.js","sourceRoot":"","sources":["../../../../../../projects/carefirst/library/src/lib/utils/form-validators-utility.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Form validators\n *\n * @file          form-validators\n * @description   Various form validators\n * @author        Jacques Coetzee\n * @since         2022 - 01 - 20\n * @usage         Display \"errorMessage\" key value as validator response message\n */\n\nimport { AbstractControl } from '@angular/forms';\nimport passwordValidator from 'password-validator';\nimport dayjs from 'dayjs';\nimport customParseFormat from 'dayjs/plugin/customParseFormat';\ndayjs.extend(customParseFormat);\n//--- Interfaces\nimport type { ValidationErrors, ValidatorFn } from '@angular/forms';\n\nexport class CFPFormValidators {\n  /**----------------------------------------------------------------\n   * @name          checkPSWRetype\n   * @description   Check if passwords match\n   * @param         {string} controlNameToCheckAgainst\n   * @returns       {ValidatorFn}\n   * @usage         FormValidators.compareTo('password')\n   */\n  static compareTo(controlNameToCheckAgainst: string): ValidatorFn {\n    return (control: AbstractControl): ValidationErrors | null => {\n      //--- Compare controls\n      const controlToCheckTo: AbstractControl | null = control?.parent?.get(controlNameToCheckAgainst) || null;\n\n      //--- Check primary control value\n      if (!controlToCheckTo?.value) return { errorMessage: `Password Required` };\n      //--- Check secondary control value\n      else if (!control.value) return { errorMessage: `Password confirmation required` };\n      //--- Compare controls\n      else if (control.value && control?.value !== controlToCheckTo?.value) return { errorMessage: `Passwords do not match` };\n\n      return null;\n    };\n  }\n\n  /**----------------------------------------------------------------\n   * @name          asPassword\n   * @description   Validate control as password\n   * @returns       {ValidatorFn}\n   * @usage         FormValidators.asPassword()\n   */\n  static asPassword(): ValidatorFn {\n    return (control: AbstractControl): ValidationErrors | null => {\n      const schema = new passwordValidator();\n\n      //--- Setup password requirements\n      schema\n        //--- Minimum length 8\n        .is()\n        .min(8, 'Minimum of 8 characters')\n        //--- Maximum length 50\n        .is()\n        .max(30, 'Maximum of 30 characters')\n        //--- Must have uppercase letters\n        .has()\n        .uppercase(undefined, 'At least one UPPERCASE character')\n        //--- Must have lowercase letters\n        .has()\n        .lowercase(undefined, 'At least one lowercase character')\n        //--- Must have at least 1 digits\n        .has()\n        .digits(undefined, 'At least one digit')\n        //--- Special characters\n        .has()\n        .symbols(undefined, 'At least one special character');\n\n      //--- Validate string\n      const inValid = schema.validate(control.value, { details: true }) as {\n        validation: string;\n        message: string;\n        arguments?: number;\n      }[];\n\n      const returnResponse: { [key: string]: unknown } = {};\n      inValid.forEach((error) => {\n        returnResponse['password-' + error.validation] = error.message;\n      });\n      return inValid.length ? returnResponse : null;\n    };\n  }\n\n  /**----------------------------------------------------------------\n   * @name          asEmail\n   * @description   Validate control as email\n   * @param         {true} canBeEmpty\n   * @returns       {ValidatorFn}\n   * @usage         FormValidators.asEmail(true?)\n   */\n  static asEmail(canBeEmpty?: true): ValidatorFn {\n    return (control: AbstractControl): ValidationErrors | null => {\n      let setErrors = null;\n      const emailRegEx = new RegExp('^[\\\\w-.]+@[\\\\w]+\\\\.[\\\\w]+');\n\n      //--- Check if email can be empty\n      if (canBeEmpty && (control.value === null || control.value === '')) {\n        return null;\n      }\n\n      //--- Test RegExp\n      if (!(control.value && emailRegEx.test(control.value) && typeof control.value === 'string' && !control.value.includes(' '))) {\n        setErrors = { errorMessage: `Not a valid email address` };\n      }\n\n      return setErrors;\n    };\n  }\n\n  /**----------------------------------------------------------------\n   * @name          isIn\n   * @description   Validate control value is part of a given list\n   * @param         {string | number[]} list\n   * @returns       {ValidatorFn}\n   * @usage         FormValidators.isIn()\n   */\n  static isIn(list: (string | number)[]): ValidatorFn {\n    return (control: AbstractControl): ValidationErrors | null => {\n      if (!list.includes(control.value)) return { errorMessage: 'Control value not part of eligible values' };\n      return null;\n    };\n  }\n\n  /**----------------------------------------------------------------\n   * @name          isSAIDNumber\n   * @description   Validate control value is a valid RSA ID Number\n   * @param         {true} canBeEmpty\n   * @returns       {ValidatorFn}\n   * @usage         FormValidators.isSAIDNumber(true?)\n   */\n  static isSAIDNumber(canBeEmpty?: true): ValidatorFn {\n    return (control: AbstractControl): ValidationErrors | null => {\n      const validateDate = (): boolean => {\n        if (typeof control.value !== 'string' || control.value.length < 6) return false;\n        const month = control.value.slice(2, 4);\n        const day = control.value.slice(4, 6);\n        if (+month < 0 && +month > 13) return false;\n        if (+day < 0 && +day > 31) return false;\n        if (!dayjs(control.value.slice(0, 6), 'YYMMDD', true).isValid()) return false;\n        return true;\n      };\n\n      const checksum = (): boolean => {\n        if (typeof control.value !== 'string' || control.value.length !== 13) return false;\n        let nSum = 0; //--- Total sum of all the values\n        let isSecond = false; //--- To double every other value\n        //--- Start from the end\n        for (let i = control.value.length - 1; i >= 0; i--) {\n          const val: string | undefined = control.value[i];\n          if (!val) return false;\n          //--- Get numerical value from ascii string\n          let d: number = val.charCodeAt(0) - '0'.charCodeAt(0);\n          //--- Double every other value\n          if (isSecond == true) d = d * 2;\n          //--- Sum the total of the doubled values if available\n          nSum += Math.floor(d / 10);\n          nSum += d % 10;\n          //--- Inverse second to determine every other index value\n          isSecond = !isSecond;\n        }\n        //--- Return the modulus 10 of the summation\n        return nSum % 10 == 0;\n      };\n\n      let error: string | null = null;\n      if (canBeEmpty && (!control.value || control.value?.length === 0)) return null;\n      //--- string\n      else if (typeof control.value !== 'string') error = 'Invalid RSA ID Number';\n      //--- Length = 13\n      else if (control.value.length !== 13) error = `Length required: ${13 - control.value.length}`;\n      //--- Validate date\n      else if (!validateDate()) error = `Invalid date of birth`;\n      //--- Citizenship\n      else if (!['0', '1'].includes(control.value.slice(10, 11))) error = `Invalid citizenship value`;\n      //--- Checksum\n      else if (!checksum()) error = `Invalid RSA ID number checksum`;\n\n      return error ? { errorMessage: error } : null;\n    };\n  }\n}\n"]}
|
122
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-validators-utility.js","sourceRoot":"","sources":["../../../../../../projects/carefirst/library/src/lib/utils/form-validators-utility.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Form validators\n *\n * @file          form-validators\n * @description   Various form validators\n * @author        Jacques Coetzee\n * @since         2022 - 01 - 20\n * @usage         Display \"errorMessage\" key value as validator response message\n */\n\nimport { AbstractControl } from '@angular/forms';\nimport passwordValidator from 'password-validator';\nimport dayjs from 'dayjs';\nimport customParseFormat from 'dayjs/plugin/customParseFormat';\ndayjs.extend(customParseFormat);\n//--- Interfaces\nimport type { ValidationErrors, ValidatorFn } from '@angular/forms';\n\nexport class CFPFormValidators {\n  /**----------------------------------------------------------------\n   * @name          checkPSWRetype\n   * @description   Check if passwords match\n   * @param         {string} controlNameToCheckAgainst\n   * @returns       {ValidatorFn}\n   * @usage         FormValidators.compareTo('password')\n   */\n  static compareTo(controlNameToCheckAgainst: string): ValidatorFn {\n    return (control: AbstractControl): ValidationErrors | null => {\n      //--- Compare controls\n      const controlToCheckTo: AbstractControl | null = control?.parent?.get(controlNameToCheckAgainst) || null;\n\n      //--- Check primary control value\n      if (!controlToCheckTo?.value) return { errorMessage: `Password Required` };\n      //--- Check secondary control value\n      else if (!control.value) return { errorMessage: `Password confirmation required` };\n      //--- Compare controls\n      else if (control.value && control?.value !== controlToCheckTo?.value) return { errorMessage: `Passwords do not match` };\n\n      return null;\n    };\n  }\n\n  /**----------------------------------------------------------------\n   * @name          asPassword\n   * @description   Validate control as password\n   * @returns       {ValidatorFn}\n   * @usage         FormValidators.asPassword()\n   */\n  static asPassword(): ValidatorFn {\n    return (control: AbstractControl): ValidationErrors | null => {\n      const schema = new passwordValidator();\n\n      //--- Setup password requirements\n      schema\n        //--- Minimum length 8\n        .is()\n        .min(8, 'Minimum of 8 characters')\n        //--- Maximum length 50\n        .is()\n        .max(30, 'Maximum of 30 characters')\n        //--- Must have uppercase letters\n        .has()\n        .uppercase(undefined, 'At least one UPPERCASE character')\n        //--- Must have lowercase letters\n        .has()\n        .lowercase(undefined, 'At least one lowercase character')\n        //--- Must have at least 1 digits\n        .has()\n        .digits(undefined, 'At least one digit')\n        //--- Special characters\n        .has()\n        .symbols(undefined, 'At least one special character');\n\n      //--- Validate string\n      const inValid = schema.validate(control.value, { details: true }) as {\n        validation: string;\n        message: string;\n        arguments?: number;\n      }[];\n\n      const returnResponse: { [key: string]: unknown } = {};\n      inValid.forEach((error) => {\n        returnResponse['password-' + error.validation] = error.message;\n      });\n      return inValid.length ? returnResponse : null;\n    };\n  }\n\n  /**----------------------------------------------------------------\n   * @name          asEmail\n   * @description   Validate control as email\n   * @param         {true} canBeEmpty\n   * @returns       {ValidatorFn}\n   * @usage         FormValidators.asEmail(true?)\n   */\n  static asEmail(canBeEmpty?: true): ValidatorFn {\n    return (control: AbstractControl): ValidationErrors | null => {\n      let setErrors = null;\n      const emailRegEx = new RegExp('^[\\\\w-.]+@[\\\\w]+\\\\.[\\\\w]+');\n\n      //--- Check if email can be empty\n      if (canBeEmpty && (control.value === null || control.value === '')) {\n        return null;\n      }\n\n      //--- Test RegExp\n      if (!(control.value && emailRegEx.test(control.value) && typeof control.value === 'string' && !control.value.includes(' '))) {\n        setErrors = { errorMessage: `Not a valid email address` };\n      }\n\n      return setErrors;\n    };\n  }\n\n  /**----------------------------------------------------------------\n   * @name          isIn\n   * @description   Validate control value is part of a given list\n   * @param         {(string | number)[]} list\n   * @returns       {ValidatorFn}\n   * @usage         FormValidators.isIn()\n   */\n  static isIn(list: (string | number)[]): ValidatorFn {\n    return (control: AbstractControl): ValidationErrors | null => {\n      if (!list.includes(control.value)) return { errorMessage: 'Control value not part of eligible values' };\n      return null;\n    };\n  }\n\n  /**----------------------------------------------------------------\n   * @name          isSAIDNumber\n   * @description   Validate control value is a valid RSA ID Number\n   * @param         {true} canBeEmpty\n   * @returns       {ValidatorFn}\n   * @usage         FormValidators.isSAIDNumber(true?)\n   */\n  static isSAIDNumber(canBeEmpty?: true): ValidatorFn {\n    return (control: AbstractControl): ValidationErrors | null => {\n      const validateDate = (): boolean => {\n        if (typeof control.value !== 'string' || control.value.length < 6) return false;\n        const month = control.value.slice(2, 4);\n        const day = control.value.slice(4, 6);\n        if (+month < 0 && +month > 13) return false;\n        if (+day < 0 && +day > 31) return false;\n        if (!dayjs(control.value.slice(0, 6), 'YYMMDD', true).isValid()) return false;\n        return true;\n      };\n\n      const checksum = (): boolean => {\n        if (typeof control.value !== 'string' || control.value.length !== 13) return false;\n        let nSum = 0; //--- Total sum of all the values\n        let isSecond = false; //--- To double every other value\n        //--- Start from the end\n        for (let i = control.value.length - 1; i >= 0; i--) {\n          const val: string | undefined = control.value[i];\n          if (!val) return false;\n          //--- Get numerical value from ascii string\n          let d: number = val.charCodeAt(0) - '0'.charCodeAt(0);\n          //--- Double every other value\n          if (isSecond == true) d = d * 2;\n          //--- Sum the total of the doubled values if available\n          nSum += Math.floor(d / 10);\n          nSum += d % 10;\n          //--- Inverse second to determine every other index value\n          isSecond = !isSecond;\n        }\n        //--- Return the modulus 10 of the summation\n        return nSum % 10 == 0;\n      };\n\n      let error: string | null = null;\n      if (canBeEmpty && (!control.value || control.value?.length === 0)) return null;\n      //--- string\n      else if (typeof control.value !== 'string') error = 'Invalid RSA ID Number';\n      //--- Length = 13\n      else if (control.value.length !== 13) error = `Length required: ${13 - control.value.length}`;\n      //--- Validate date\n      else if (!validateDate()) error = `Invalid date of birth`;\n      //--- Citizenship\n      else if (!['0', '1'].includes(control.value.slice(10, 11))) error = `Invalid citizenship value`;\n      //--- Checksum\n      else if (!checksum()) error = `Invalid RSA ID number checksum`;\n\n      return error ? { errorMessage: error } : null;\n    };\n  }\n\n  /**----------------------------------------------------------------\n   * @name          isValidDate\n   * @description   Validate control value is part of a given list\n   * @param         {string | null} minDate\n   * @param         {string | null} maxDate\n   * @returns       {ValidatorFn}\n   * @usage         FormValidators.isValidDate('1999-01-01', '2020-01-01')\n   */\n  static isValidDate(minDate: string | null, maxDate?: string | null): ValidatorFn {\n    return (control: AbstractControl): ValidationErrors | null => {\n      //--- Check to see if a invalid date is received like (31 Feb)\n      if (!dayjs(new Date(control.value)).isValid()) return { errorMessage: 'The given date is not valid' };\n      //--- Check date is not less than minDate and date doesn't exceed maxDate\n      if ((minDate && dayjs(new Date(control.value)).isBefore(minDate)) || (maxDate && dayjs(new Date(control.value)).isAfter(maxDate)))\n        return { errorMessage: 'The given date is not within the valid parameters' };\n      return null;\n    };\n  }\n}\n"]}
|
@@ -473,6 +473,124 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.3", ngImpor
|
|
473
473
|
type: Input
|
474
474
|
}] } });
|
475
475
|
|
476
|
+
dayjs.extend(customParseFormat);
|
477
|
+
class CFPFormValidators {
|
478
|
+
static compareTo(controlNameToCheckAgainst) {
|
479
|
+
return (control) => {
|
480
|
+
const controlToCheckTo = control?.parent?.get(controlNameToCheckAgainst) || null;
|
481
|
+
if (!controlToCheckTo?.value)
|
482
|
+
return { errorMessage: `Password Required` };
|
483
|
+
else if (!control.value)
|
484
|
+
return { errorMessage: `Password confirmation required` };
|
485
|
+
else if (control.value && control?.value !== controlToCheckTo?.value)
|
486
|
+
return { errorMessage: `Passwords do not match` };
|
487
|
+
return null;
|
488
|
+
};
|
489
|
+
}
|
490
|
+
static asPassword() {
|
491
|
+
return (control) => {
|
492
|
+
const schema = new passwordValidator();
|
493
|
+
schema
|
494
|
+
.is()
|
495
|
+
.min(8, 'Minimum of 8 characters')
|
496
|
+
.is()
|
497
|
+
.max(30, 'Maximum of 30 characters')
|
498
|
+
.has()
|
499
|
+
.uppercase(undefined, 'At least one UPPERCASE character')
|
500
|
+
.has()
|
501
|
+
.lowercase(undefined, 'At least one lowercase character')
|
502
|
+
.has()
|
503
|
+
.digits(undefined, 'At least one digit')
|
504
|
+
.has()
|
505
|
+
.symbols(undefined, 'At least one special character');
|
506
|
+
const inValid = schema.validate(control.value, { details: true });
|
507
|
+
const returnResponse = {};
|
508
|
+
inValid.forEach((error) => {
|
509
|
+
returnResponse['password-' + error.validation] = error.message;
|
510
|
+
});
|
511
|
+
return inValid.length ? returnResponse : null;
|
512
|
+
};
|
513
|
+
}
|
514
|
+
static asEmail(canBeEmpty) {
|
515
|
+
return (control) => {
|
516
|
+
let setErrors = null;
|
517
|
+
const emailRegEx = new RegExp('^[\\w-.]+@[\\w]+\\.[\\w]+');
|
518
|
+
if (canBeEmpty && (control.value === null || control.value === '')) {
|
519
|
+
return null;
|
520
|
+
}
|
521
|
+
if (!(control.value && emailRegEx.test(control.value) && typeof control.value === 'string' && !control.value.includes(' '))) {
|
522
|
+
setErrors = { errorMessage: `Not a valid email address` };
|
523
|
+
}
|
524
|
+
return setErrors;
|
525
|
+
};
|
526
|
+
}
|
527
|
+
static isIn(list) {
|
528
|
+
return (control) => {
|
529
|
+
if (!list.includes(control.value))
|
530
|
+
return { errorMessage: 'Control value not part of eligible values' };
|
531
|
+
return null;
|
532
|
+
};
|
533
|
+
}
|
534
|
+
static isSAIDNumber(canBeEmpty) {
|
535
|
+
return (control) => {
|
536
|
+
const validateDate = () => {
|
537
|
+
if (typeof control.value !== 'string' || control.value.length < 6)
|
538
|
+
return false;
|
539
|
+
const month = control.value.slice(2, 4);
|
540
|
+
const day = control.value.slice(4, 6);
|
541
|
+
if (+month < 0 && +month > 13)
|
542
|
+
return false;
|
543
|
+
if (+day < 0 && +day > 31)
|
544
|
+
return false;
|
545
|
+
if (!dayjs(control.value.slice(0, 6), 'YYMMDD', true).isValid())
|
546
|
+
return false;
|
547
|
+
return true;
|
548
|
+
};
|
549
|
+
const checksum = () => {
|
550
|
+
if (typeof control.value !== 'string' || control.value.length !== 13)
|
551
|
+
return false;
|
552
|
+
let nSum = 0;
|
553
|
+
let isSecond = false;
|
554
|
+
for (let i = control.value.length - 1; i >= 0; i--) {
|
555
|
+
const val = control.value[i];
|
556
|
+
if (!val)
|
557
|
+
return false;
|
558
|
+
let d = val.charCodeAt(0) - '0'.charCodeAt(0);
|
559
|
+
if (isSecond == true)
|
560
|
+
d = d * 2;
|
561
|
+
nSum += Math.floor(d / 10);
|
562
|
+
nSum += d % 10;
|
563
|
+
isSecond = !isSecond;
|
564
|
+
}
|
565
|
+
return nSum % 10 == 0;
|
566
|
+
};
|
567
|
+
let error = null;
|
568
|
+
if (canBeEmpty && (!control.value || control.value?.length === 0))
|
569
|
+
return null;
|
570
|
+
else if (typeof control.value !== 'string')
|
571
|
+
error = 'Invalid RSA ID Number';
|
572
|
+
else if (control.value.length !== 13)
|
573
|
+
error = `Length required: ${13 - control.value.length}`;
|
574
|
+
else if (!validateDate())
|
575
|
+
error = `Invalid date of birth`;
|
576
|
+
else if (!['0', '1'].includes(control.value.slice(10, 11)))
|
577
|
+
error = `Invalid citizenship value`;
|
578
|
+
else if (!checksum())
|
579
|
+
error = `Invalid RSA ID number checksum`;
|
580
|
+
return error ? { errorMessage: error } : null;
|
581
|
+
};
|
582
|
+
}
|
583
|
+
static isValidDate(minDate, maxDate) {
|
584
|
+
return (control) => {
|
585
|
+
if (!dayjs(new Date(control.value)).isValid())
|
586
|
+
return { errorMessage: 'The given date is not valid' };
|
587
|
+
if ((minDate && dayjs(new Date(control.value)).isBefore(minDate)) || (maxDate && dayjs(new Date(control.value)).isAfter(maxDate)))
|
588
|
+
return { errorMessage: 'The given date is not within the valid parameters' };
|
589
|
+
return null;
|
590
|
+
};
|
591
|
+
}
|
592
|
+
}
|
593
|
+
|
476
594
|
const inputsC = {
|
477
595
|
labelPlacement: ['fixed', 'floating', 'stacked', 'start', 'end'],
|
478
596
|
inputMode: ['text', 'email', 'numeric', 'tel', 'decimal', 'url', 'search'],
|
@@ -481,8 +599,8 @@ const inputsC = {
|
|
481
599
|
};
|
482
600
|
class FormInputComponent {
|
483
601
|
label;
|
484
|
-
min
|
485
|
-
max
|
602
|
+
min;
|
603
|
+
max;
|
486
604
|
labelPlacement;
|
487
605
|
placeholder;
|
488
606
|
inputmode;
|
@@ -503,6 +621,8 @@ class FormInputComponent {
|
|
503
621
|
inputAutoCapitalize;
|
504
622
|
inputGreyBackground = false;
|
505
623
|
showPassword = false;
|
624
|
+
dateValidatorMin = null;
|
625
|
+
dateValidatorMax = null;
|
506
626
|
ngOnChanges(changes) {
|
507
627
|
this.inputLabelPlacement =
|
508
628
|
validateStringValue(changes, 'labelPlacement', inputsC.labelPlacement.slice(), this.inputLabelPlacement) || 'floating';
|
@@ -514,6 +634,13 @@ class FormInputComponent {
|
|
514
634
|
this.inputAutoCapitalize =
|
515
635
|
validateStringValue(changes, 'autoCapitalize', inputsC.autoCapitalize.slice(), this.inputAutoCapitalize) || 'none';
|
516
636
|
this.inputGreyBackground = checkTruthAttribute(changes, 'greyBackground', this.inputGreyBackground);
|
637
|
+
if (this.inputType === 'date') {
|
638
|
+
if (changes['min']?.currentValue || changes['max']?.currentValue) {
|
639
|
+
this.dateValidatorMin = changes['min'] ? changes['min'].currentValue : this.dateValidatorMin;
|
640
|
+
this.dateValidatorMax = changes['max'] ? changes['max'].currentValue : this.dateValidatorMax;
|
641
|
+
this.control?.addValidators(CFPFormValidators.isValidDate(this.dateValidatorMin || null, this.dateValidatorMax || null));
|
642
|
+
}
|
643
|
+
}
|
517
644
|
}
|
518
645
|
togglePasswordShow() {
|
519
646
|
this.showPassword = !this.showPassword;
|
@@ -1007,115 +1134,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.3", ngImpor
|
|
1007
1134
|
}]
|
1008
1135
|
}] });
|
1009
1136
|
|
1010
|
-
dayjs.extend(customParseFormat);
|
1011
|
-
class CFPFormValidators {
|
1012
|
-
static compareTo(controlNameToCheckAgainst) {
|
1013
|
-
return (control) => {
|
1014
|
-
const controlToCheckTo = control?.parent?.get(controlNameToCheckAgainst) || null;
|
1015
|
-
if (!controlToCheckTo?.value)
|
1016
|
-
return { errorMessage: `Password Required` };
|
1017
|
-
else if (!control.value)
|
1018
|
-
return { errorMessage: `Password confirmation required` };
|
1019
|
-
else if (control.value && control?.value !== controlToCheckTo?.value)
|
1020
|
-
return { errorMessage: `Passwords do not match` };
|
1021
|
-
return null;
|
1022
|
-
};
|
1023
|
-
}
|
1024
|
-
static asPassword() {
|
1025
|
-
return (control) => {
|
1026
|
-
const schema = new passwordValidator();
|
1027
|
-
schema
|
1028
|
-
.is()
|
1029
|
-
.min(8, 'Minimum of 8 characters')
|
1030
|
-
.is()
|
1031
|
-
.max(30, 'Maximum of 30 characters')
|
1032
|
-
.has()
|
1033
|
-
.uppercase(undefined, 'At least one UPPERCASE character')
|
1034
|
-
.has()
|
1035
|
-
.lowercase(undefined, 'At least one lowercase character')
|
1036
|
-
.has()
|
1037
|
-
.digits(undefined, 'At least one digit')
|
1038
|
-
.has()
|
1039
|
-
.symbols(undefined, 'At least one special character');
|
1040
|
-
const inValid = schema.validate(control.value, { details: true });
|
1041
|
-
const returnResponse = {};
|
1042
|
-
inValid.forEach((error) => {
|
1043
|
-
returnResponse['password-' + error.validation] = error.message;
|
1044
|
-
});
|
1045
|
-
return inValid.length ? returnResponse : null;
|
1046
|
-
};
|
1047
|
-
}
|
1048
|
-
static asEmail(canBeEmpty) {
|
1049
|
-
return (control) => {
|
1050
|
-
let setErrors = null;
|
1051
|
-
const emailRegEx = new RegExp('^[\\w-.]+@[\\w]+\\.[\\w]+');
|
1052
|
-
if (canBeEmpty && (control.value === null || control.value === '')) {
|
1053
|
-
return null;
|
1054
|
-
}
|
1055
|
-
if (!(control.value && emailRegEx.test(control.value) && typeof control.value === 'string' && !control.value.includes(' '))) {
|
1056
|
-
setErrors = { errorMessage: `Not a valid email address` };
|
1057
|
-
}
|
1058
|
-
return setErrors;
|
1059
|
-
};
|
1060
|
-
}
|
1061
|
-
static isIn(list) {
|
1062
|
-
return (control) => {
|
1063
|
-
if (!list.includes(control.value))
|
1064
|
-
return { errorMessage: 'Control value not part of eligible values' };
|
1065
|
-
return null;
|
1066
|
-
};
|
1067
|
-
}
|
1068
|
-
static isSAIDNumber(canBeEmpty) {
|
1069
|
-
return (control) => {
|
1070
|
-
const validateDate = () => {
|
1071
|
-
if (typeof control.value !== 'string' || control.value.length < 6)
|
1072
|
-
return false;
|
1073
|
-
const month = control.value.slice(2, 4);
|
1074
|
-
const day = control.value.slice(4, 6);
|
1075
|
-
if (+month < 0 && +month > 13)
|
1076
|
-
return false;
|
1077
|
-
if (+day < 0 && +day > 31)
|
1078
|
-
return false;
|
1079
|
-
if (!dayjs(control.value.slice(0, 6), 'YYMMDD', true).isValid())
|
1080
|
-
return false;
|
1081
|
-
return true;
|
1082
|
-
};
|
1083
|
-
const checksum = () => {
|
1084
|
-
if (typeof control.value !== 'string' || control.value.length !== 13)
|
1085
|
-
return false;
|
1086
|
-
let nSum = 0;
|
1087
|
-
let isSecond = false;
|
1088
|
-
for (let i = control.value.length - 1; i >= 0; i--) {
|
1089
|
-
const val = control.value[i];
|
1090
|
-
if (!val)
|
1091
|
-
return false;
|
1092
|
-
let d = val.charCodeAt(0) - '0'.charCodeAt(0);
|
1093
|
-
if (isSecond == true)
|
1094
|
-
d = d * 2;
|
1095
|
-
nSum += Math.floor(d / 10);
|
1096
|
-
nSum += d % 10;
|
1097
|
-
isSecond = !isSecond;
|
1098
|
-
}
|
1099
|
-
return nSum % 10 == 0;
|
1100
|
-
};
|
1101
|
-
let error = null;
|
1102
|
-
if (canBeEmpty && (!control.value || control.value?.length === 0))
|
1103
|
-
return null;
|
1104
|
-
else if (typeof control.value !== 'string')
|
1105
|
-
error = 'Invalid RSA ID Number';
|
1106
|
-
else if (control.value.length !== 13)
|
1107
|
-
error = `Length required: ${13 - control.value.length}`;
|
1108
|
-
else if (!validateDate())
|
1109
|
-
error = `Invalid date of birth`;
|
1110
|
-
else if (!['0', '1'].includes(control.value.slice(10, 11)))
|
1111
|
-
error = `Invalid citizenship value`;
|
1112
|
-
else if (!checksum())
|
1113
|
-
error = `Invalid RSA ID number checksum`;
|
1114
|
-
return error ? { errorMessage: error } : null;
|
1115
|
-
};
|
1116
|
-
}
|
1117
|
-
}
|
1118
|
-
|
1119
1137
|
var alert_interface = /*#__PURE__*/Object.freeze({
|
1120
1138
|
__proto__: null
|
1121
1139
|
});
|