@fovestta2/web-angular 1.0.26 → 1.0.27
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/add-update-form/add-update-form.component.mjs +30 -1
- package/esm2022/lib/fv-dropdown/fv-dropdown.component.mjs +7 -2
- package/esm2022/lib/fv-email-field/fv-email-field.component.mjs +7 -2
- package/esm2022/lib/fv-entry-field/fv-entry-field.component.mjs +8 -2
- package/esm2022/lib/fv-name-code/fv-name-code.component.mjs +28 -14
- package/esm2022/lib/fv-number-field/fv-number-field.component.mjs +7 -2
- package/fesm2022/fovestta2-web-angular.mjs +80 -17
- package/fesm2022/fovestta2-web-angular.mjs.map +1 -1
- package/lib/add-update-form/add-update-form.component.d.ts +1 -1
- package/lib/fv-name-code/fv-name-code.component.d.ts +2 -1
- package/package.json +2 -2
|
@@ -53,8 +53,9 @@ export class FvNumberFieldComponent {
|
|
|
53
53
|
return;
|
|
54
54
|
const result = Validator.validate(value, this.schema);
|
|
55
55
|
this.errorMessage = result.errorKey;
|
|
56
|
+
const customMessage = result.message;
|
|
56
57
|
if (!result.isValid && result.errorKey) {
|
|
57
|
-
this.control.setErrors({ [result.errorKey]: true });
|
|
58
|
+
this.control.setErrors({ [result.errorKey]: true, message: customMessage });
|
|
58
59
|
}
|
|
59
60
|
else {
|
|
60
61
|
this.control.setErrors(null);
|
|
@@ -75,10 +76,14 @@ export class FvNumberFieldComponent {
|
|
|
75
76
|
getErrorMessage() {
|
|
76
77
|
if (!this.errorMessage)
|
|
77
78
|
return '';
|
|
79
|
+
if (this.control?.errors?.['message']) {
|
|
80
|
+
return this.control.errors['message'];
|
|
81
|
+
}
|
|
78
82
|
const errorMessages = {
|
|
79
83
|
ERR_REQUIRED: 'This field is required',
|
|
80
84
|
ERR_NUMERIC_INVALID: 'Please enter a valid number',
|
|
81
85
|
ERR_RANGE_INVALID: 'Number is out of range',
|
|
86
|
+
ERR_DUPLICATE: 'This value already exists',
|
|
82
87
|
};
|
|
83
88
|
return errorMessages[this.errorMessage] || this.errorMessage;
|
|
84
89
|
}
|
|
@@ -113,4 +118,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
113
118
|
}], focus: [{
|
|
114
119
|
type: Output
|
|
115
120
|
}] } });
|
|
116
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fv-number-field.component.js","sourceRoot":"","sources":["../../../../../projects/fv-controls/src/lib/fv-number-field/fv-number-field.component.ts","../../../../../projects/fv-controls/src/lib/fv-number-field/fv-number-field.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  EventEmitter,\r\n  Input,\r\n  OnDestroy,\r\n  OnInit,\r\n  Output,\r\n  OnChanges,\r\n  SimpleChanges,\r\n} from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { FormControl, ReactiveFormsModule } from '@angular/forms';\r\nimport { Subscription } from 'rxjs';\r\nimport { Validator, ValidationSchema } from '@fovestta2/validation-engine';\r\n\r\n@Component({\r\n  selector: 'fv-number-field',\r\n  standalone: true,\r\n  imports: [CommonModule, ReactiveFormsModule],\r\n  templateUrl: './fv-number-field.component.html',\r\n  styleUrl: './fv-number-field.component.css',\r\n})\r\nexport class FvNumberFieldComponent implements OnInit, OnDestroy, OnChanges {\r\n  @Input() label: string = '';\r\n  @Input() placeholder: string = '';\r\n  @Input() schema!: ValidationSchema;\r\n  @Input() control!: FormControl;\r\n  @Input() disabled: boolean = false;\r\n  @Input() readonly: boolean = false;\r\n  @Input() min?: any;\r\n  @Input() max?: any;\r\n  @Input() step: number = 1;\r\n\r\n  @Output() valueChange = new EventEmitter<number>();\r\n  @Output() blur = new EventEmitter<void>();\r\n  @Output() focus = new EventEmitter<void>();\r\n\r\n  errorMessage: string | null = null;\r\n  private subscription?: Subscription;\r\n\r\n  ngOnInit(): void {\r\n    if (!this.control) {\r\n      console.error('FvNumberField: control is required');\r\n      return;\r\n    }\r\n\r\n    if (!this.schema) {\r\n      console.warn('FvNumberField: schema is not provided');\r\n      return;\r\n    }\r\n\r\n    this.subscription = this.control.valueChanges.subscribe((value) => {\r\n      this.validateValue(value);\r\n      this.valueChange.emit(value);\r\n    });\r\n\r\n    this.validateValue(this.control.value);\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.subscription?.unsubscribe();\r\n  }\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (changes['disabled'] && this.control) {\r\n      if (this.disabled) {\r\n        this.control.disable({ emitEvent: false });\r\n      } else {\r\n        this.control.enable({ emitEvent: false });\r\n      }\r\n    }\r\n  }\r\n\r\n  private validateValue(value: any): void {\r\n    if (!this.schema) return;\r\n\r\n    const result = Validator.validate(value, this.schema);\r\n    this.errorMessage = result.errorKey;\r\n\r\n    if (!result.isValid && result.errorKey) {\r\n      this.control.setErrors({ [result.errorKey]: true });\r\n    } else {\r\n      this.control.setErrors(null);\r\n    }\r\n  }\r\n\r\n  onBlur(): void {\r\n    if (this.control && this.schema) {\r\n      this.validateValue(this.control.value);\r\n    }\r\n    this.blur.emit();\r\n  }\r\n\r\n  onFocus(): void {\r\n    this.focus.emit();\r\n  }\r\n\r\n  isRequired(): boolean {\r\n    return (\r\n      this.schema?.rules?.some(\r\n        (r) => r.name === 'required' && r.params?.['enabled']\r\n      ) || false\r\n    );\r\n  }\r\n\r\n  getErrorMessage(): string {\r\n    if (!this.errorMessage) return '';\r\n\r\n    const errorMessages: Record<string, string> = {\r\n      ERR_REQUIRED: 'This field is required',\r\n      ERR_NUMERIC_INVALID: 'Please enter a valid number',\r\n      ERR_RANGE_INVALID: 'Number is out of range',\r\n    };\r\n\r\n    return errorMessages[this.errorMessage] || this.errorMessage;\r\n  }\r\n}\r\n","<div class=\"fv-number-field-container\">\r\n  <label *ngIf=\"label\" class=\"fv-label\">\r\n    {{ label }}\r\n    <span *ngIf=\"isRequired()\" class=\"fv-required-asterisk\">*</span>\r\n  </label>\r\n\r\n  <input type=\"number\" [formControl]=\"control\" [placeholder]=\"placeholder\" [readonly]=\"readonly\" [min]=\"min\" [max]=\"max\"\r\n    [step]=\"step\" (blur)=\"onBlur()\" (focus)=\"onFocus()\" class=\"fv-input\"\r\n    [class.fv-input-error]=\"errorMessage && control?.touched\" [class.fv-input-disabled]=\"disabled\" />\r\n\r\n  <span *ngIf=\"errorMessage && control?.touched\" class=\"fv-error-message\">\r\n    {{ getErrorMessage() }}\r\n  </span>\r\n</div>"]}
|
|
121
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fv-number-field.component.js","sourceRoot":"","sources":["../../../../../projects/fv-controls/src/lib/fv-number-field/fv-number-field.component.ts","../../../../../projects/fv-controls/src/lib/fv-number-field/fv-number-field.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  EventEmitter,\r\n  Input,\r\n  OnDestroy,\r\n  OnInit,\r\n  Output,\r\n  OnChanges,\r\n  SimpleChanges,\r\n} from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { FormControl, ReactiveFormsModule } from '@angular/forms';\r\nimport { Subscription } from 'rxjs';\r\nimport { Validator, ValidationSchema } from '@fovestta2/validation-engine';\r\n\r\n@Component({\r\n  selector: 'fv-number-field',\r\n  standalone: true,\r\n  imports: [CommonModule, ReactiveFormsModule],\r\n  templateUrl: './fv-number-field.component.html',\r\n  styleUrl: './fv-number-field.component.css',\r\n})\r\nexport class FvNumberFieldComponent implements OnInit, OnDestroy, OnChanges {\r\n  @Input() label: string = '';\r\n  @Input() placeholder: string = '';\r\n  @Input() schema!: ValidationSchema;\r\n  @Input() control!: FormControl;\r\n  @Input() disabled: boolean = false;\r\n  @Input() readonly: boolean = false;\r\n  @Input() min?: any;\r\n  @Input() max?: any;\r\n  @Input() step: number = 1;\r\n\r\n  @Output() valueChange = new EventEmitter<number>();\r\n  @Output() blur = new EventEmitter<void>();\r\n  @Output() focus = new EventEmitter<void>();\r\n\r\n  errorMessage: string | null = null;\r\n  private subscription?: Subscription;\r\n\r\n  ngOnInit(): void {\r\n    if (!this.control) {\r\n      console.error('FvNumberField: control is required');\r\n      return;\r\n    }\r\n\r\n    if (!this.schema) {\r\n      console.warn('FvNumberField: schema is not provided');\r\n      return;\r\n    }\r\n\r\n    this.subscription = this.control.valueChanges.subscribe((value) => {\r\n      this.validateValue(value);\r\n      this.valueChange.emit(value);\r\n    });\r\n\r\n    this.validateValue(this.control.value);\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.subscription?.unsubscribe();\r\n  }\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    if (changes['disabled'] && this.control) {\r\n      if (this.disabled) {\r\n        this.control.disable({ emitEvent: false });\r\n      } else {\r\n        this.control.enable({ emitEvent: false });\r\n      }\r\n    }\r\n  }\r\n\r\n  private validateValue(value: any): void {\r\n    if (!this.schema) return;\r\n\r\n    const result = Validator.validate(value, this.schema);\r\n    this.errorMessage = result.errorKey;\r\n    const customMessage = result.message;\r\n\r\n    if (!result.isValid && result.errorKey) {\r\n      this.control.setErrors({ [result.errorKey]: true, message: customMessage });\r\n    } else {\r\n      this.control.setErrors(null);\r\n    }\r\n  }\r\n\r\n  onBlur(): void {\r\n    if (this.control && this.schema) {\r\n      this.validateValue(this.control.value);\r\n    }\r\n    this.blur.emit();\r\n  }\r\n\r\n  onFocus(): void {\r\n    this.focus.emit();\r\n  }\r\n\r\n  isRequired(): boolean {\r\n    return (\r\n      this.schema?.rules?.some(\r\n        (r) => r.name === 'required' && r.params?.['enabled']\r\n      ) || false\r\n    );\r\n  }\r\n\r\n  getErrorMessage(): string {\r\n    if (!this.errorMessage) return '';\r\n\r\n    if (this.control?.errors?.['message']) {\r\n      return this.control.errors['message'];\r\n    }\r\n\r\n    const errorMessages: Record<string, string> = {\r\n      ERR_REQUIRED: 'This field is required',\r\n      ERR_NUMERIC_INVALID: 'Please enter a valid number',\r\n      ERR_RANGE_INVALID: 'Number is out of range',\r\n      ERR_DUPLICATE: 'This value already exists',\r\n    };\r\n\r\n    return errorMessages[this.errorMessage] || this.errorMessage;\r\n  }\r\n}\r\n","<div class=\"fv-number-field-container\">\r\n  <label *ngIf=\"label\" class=\"fv-label\">\r\n    {{ label }}\r\n    <span *ngIf=\"isRequired()\" class=\"fv-required-asterisk\">*</span>\r\n  </label>\r\n\r\n  <input type=\"number\" [formControl]=\"control\" [placeholder]=\"placeholder\" [readonly]=\"readonly\" [min]=\"min\" [max]=\"max\"\r\n    [step]=\"step\" (blur)=\"onBlur()\" (focus)=\"onFocus()\" class=\"fv-input\"\r\n    [class.fv-input-error]=\"errorMessage && control?.touched\" [class.fv-input-disabled]=\"disabled\" />\r\n\r\n  <span *ngIf=\"errorMessage && control?.touched\" class=\"fv-error-message\">\r\n    {{ getErrorMessage() }}\r\n  </span>\r\n</div>"]}
|
|
@@ -59,8 +59,9 @@ class FvEntryFieldComponent {
|
|
|
59
59
|
return;
|
|
60
60
|
const result = Validator.validate(value, this.schema);
|
|
61
61
|
this.errorMessage = result.errorKey;
|
|
62
|
+
const customMessage = result.message;
|
|
62
63
|
if (!result.isValid && result.errorKey) {
|
|
63
|
-
this.control.setErrors({ [result.errorKey]: true });
|
|
64
|
+
this.control.setErrors({ [result.errorKey]: true, message: customMessage });
|
|
64
65
|
}
|
|
65
66
|
else {
|
|
66
67
|
this.control.setErrors(null);
|
|
@@ -101,12 +102,17 @@ class FvEntryFieldComponent {
|
|
|
101
102
|
getErrorMessage() {
|
|
102
103
|
if (!this.errorMessage)
|
|
103
104
|
return '';
|
|
105
|
+
// If there is a custom message in the control errors, use it
|
|
106
|
+
if (this.control?.errors?.['message']) {
|
|
107
|
+
return this.control.errors['message'];
|
|
108
|
+
}
|
|
104
109
|
// You can implement a translation service here
|
|
105
110
|
const errorMessages = {
|
|
106
111
|
ERR_REQUIRED: 'This field is required',
|
|
107
112
|
ERR_MIN_LENGTH: 'Value is too short',
|
|
108
113
|
ERR_MAX_LENGTH: 'Value is too long',
|
|
109
114
|
ERR_REGEX_MISMATCH: 'Invalid format',
|
|
115
|
+
ERR_DUPLICATE: 'This value already exists',
|
|
110
116
|
};
|
|
111
117
|
return errorMessages[this.errorMessage] || this.errorMessage;
|
|
112
118
|
}
|
|
@@ -473,8 +479,9 @@ class FvNumberFieldComponent {
|
|
|
473
479
|
return;
|
|
474
480
|
const result = Validator.validate(value, this.schema);
|
|
475
481
|
this.errorMessage = result.errorKey;
|
|
482
|
+
const customMessage = result.message;
|
|
476
483
|
if (!result.isValid && result.errorKey) {
|
|
477
|
-
this.control.setErrors({ [result.errorKey]: true });
|
|
484
|
+
this.control.setErrors({ [result.errorKey]: true, message: customMessage });
|
|
478
485
|
}
|
|
479
486
|
else {
|
|
480
487
|
this.control.setErrors(null);
|
|
@@ -495,10 +502,14 @@ class FvNumberFieldComponent {
|
|
|
495
502
|
getErrorMessage() {
|
|
496
503
|
if (!this.errorMessage)
|
|
497
504
|
return '';
|
|
505
|
+
if (this.control?.errors?.['message']) {
|
|
506
|
+
return this.control.errors['message'];
|
|
507
|
+
}
|
|
498
508
|
const errorMessages = {
|
|
499
509
|
ERR_REQUIRED: 'This field is required',
|
|
500
510
|
ERR_NUMERIC_INVALID: 'Please enter a valid number',
|
|
501
511
|
ERR_RANGE_INVALID: 'Number is out of range',
|
|
512
|
+
ERR_DUPLICATE: 'This value already exists',
|
|
502
513
|
};
|
|
503
514
|
return errorMessages[this.errorMessage] || this.errorMessage;
|
|
504
515
|
}
|
|
@@ -712,8 +723,9 @@ class FvDropdownComponent {
|
|
|
712
723
|
return;
|
|
713
724
|
const result = Validator.validate(value, this.schema);
|
|
714
725
|
this.errorMessage = result.errorKey;
|
|
726
|
+
const customMessage = result.message;
|
|
715
727
|
if (!result.isValid && result.errorKey) {
|
|
716
|
-
this.control.setErrors({ [result.errorKey]: true });
|
|
728
|
+
this.control.setErrors({ [result.errorKey]: true, message: customMessage });
|
|
717
729
|
}
|
|
718
730
|
else {
|
|
719
731
|
this.control.setErrors(null);
|
|
@@ -936,9 +948,13 @@ class FvDropdownComponent {
|
|
|
936
948
|
getErrorMessage() {
|
|
937
949
|
if (!this.errorMessage)
|
|
938
950
|
return '';
|
|
951
|
+
if (this.control?.errors?.['message']) {
|
|
952
|
+
return this.control.errors['message'];
|
|
953
|
+
}
|
|
939
954
|
const errorMessages = {
|
|
940
955
|
ERR_REQUIRED: 'This field is required',
|
|
941
956
|
ERR_INVALID_VALUE: 'Invalid selection',
|
|
957
|
+
ERR_DUPLICATE: 'This selection already exists',
|
|
942
958
|
};
|
|
943
959
|
return errorMessages[this.errorMessage] || this.errorMessage;
|
|
944
960
|
}
|
|
@@ -1633,6 +1649,7 @@ class FvNameCodeComponent {
|
|
|
1633
1649
|
if (this.control.disabled !== this.disabled) {
|
|
1634
1650
|
this.setDisabledState(this.control.disabled);
|
|
1635
1651
|
}
|
|
1652
|
+
this.validateValue(this.control.value);
|
|
1636
1653
|
}
|
|
1637
1654
|
}
|
|
1638
1655
|
filterOptions(term) {
|
|
@@ -1697,6 +1714,7 @@ class FvNameCodeComponent {
|
|
|
1697
1714
|
this.searchControl.setValue('', { emitEvent: false });
|
|
1698
1715
|
}
|
|
1699
1716
|
}
|
|
1717
|
+
this.validateValue(value);
|
|
1700
1718
|
}
|
|
1701
1719
|
registerOnChange(fn) {
|
|
1702
1720
|
this.onChange = fn;
|
|
@@ -1713,6 +1731,20 @@ class FvNameCodeComponent {
|
|
|
1713
1731
|
this.searchControl.enable({ emitEvent: false });
|
|
1714
1732
|
}
|
|
1715
1733
|
}
|
|
1734
|
+
errorMessage = null;
|
|
1735
|
+
validateValue(value) {
|
|
1736
|
+
if (!this.schema || !this.control)
|
|
1737
|
+
return;
|
|
1738
|
+
const result = Validator.validate(value, this.schema);
|
|
1739
|
+
this.errorMessage = result.errorKey;
|
|
1740
|
+
const customMessage = result.message;
|
|
1741
|
+
if (!result.isValid && result.errorKey) {
|
|
1742
|
+
this.control.setErrors({ [result.errorKey]: true, message: customMessage });
|
|
1743
|
+
}
|
|
1744
|
+
else {
|
|
1745
|
+
this.control.setErrors(null);
|
|
1746
|
+
}
|
|
1747
|
+
}
|
|
1716
1748
|
isInvalid() {
|
|
1717
1749
|
if (this.control) {
|
|
1718
1750
|
return this.control.invalid && (this.control.dirty || this.control.touched);
|
|
@@ -1725,24 +1757,21 @@ class FvNameCodeComponent {
|
|
|
1725
1757
|
getErrorMessage() {
|
|
1726
1758
|
if (!this.control || !this.control.errors || !this.schema)
|
|
1727
1759
|
return '';
|
|
1728
|
-
|
|
1729
|
-
|
|
1730
|
-
|
|
1731
|
-
|
|
1732
|
-
|
|
1733
|
-
|
|
1760
|
+
if (this.control.errors['message']) {
|
|
1761
|
+
return this.control.errors['message'];
|
|
1762
|
+
}
|
|
1763
|
+
if (this.errorMessage) {
|
|
1764
|
+
const map = {
|
|
1765
|
+
'ERR_REQUIRED': 'This field is required',
|
|
1766
|
+
'ERR_MIN_LENGTH': 'Value is too short',
|
|
1767
|
+
'ERR_DUPLICATE': 'This value already exists',
|
|
1768
|
+
};
|
|
1769
|
+
return map[this.errorMessage] || this.errorMessage;
|
|
1734
1770
|
}
|
|
1735
1771
|
if (this.control.hasError('required'))
|
|
1736
1772
|
return 'This field is required';
|
|
1737
1773
|
return 'Invalid value';
|
|
1738
1774
|
}
|
|
1739
|
-
getReadableError(key) {
|
|
1740
|
-
const map = {
|
|
1741
|
-
'ERR_REQUIRED': 'This field is required',
|
|
1742
|
-
'ERR_MIN_LENGTH': 'Value is too short',
|
|
1743
|
-
};
|
|
1744
|
-
return map[key] || 'Invalid value';
|
|
1745
|
-
}
|
|
1746
1775
|
onClickOutside(event) {
|
|
1747
1776
|
const clickedInside = event.target.closest('.fv-name-code-container');
|
|
1748
1777
|
if (!clickedInside) {
|
|
@@ -2502,8 +2531,9 @@ class FvEmailFieldComponent {
|
|
|
2502
2531
|
return;
|
|
2503
2532
|
const result = Validator.validate(value, this.schema);
|
|
2504
2533
|
this.errorMessage = result.errorKey;
|
|
2534
|
+
const customMessage = result.message;
|
|
2505
2535
|
if (!result.isValid && result.errorKey) {
|
|
2506
|
-
this.control.setErrors({ [result.errorKey]: true });
|
|
2536
|
+
this.control.setErrors({ [result.errorKey]: true, message: customMessage });
|
|
2507
2537
|
}
|
|
2508
2538
|
else {
|
|
2509
2539
|
this.control.setErrors(null);
|
|
@@ -2524,10 +2554,14 @@ class FvEmailFieldComponent {
|
|
|
2524
2554
|
getErrorMessage() {
|
|
2525
2555
|
if (!this.errorMessage)
|
|
2526
2556
|
return '';
|
|
2557
|
+
if (this.control?.errors?.['message']) {
|
|
2558
|
+
return this.control.errors['message'];
|
|
2559
|
+
}
|
|
2527
2560
|
const errorMessages = {
|
|
2528
2561
|
ERR_REQUIRED: 'Email is required',
|
|
2529
2562
|
ERR_EMAIL_INVALID: 'Invalid email format',
|
|
2530
2563
|
ERR_REGEX_MISMATCH: 'Invalid email format',
|
|
2564
|
+
ERR_DUPLICATE: 'Email already exists',
|
|
2531
2565
|
};
|
|
2532
2566
|
return errorMessages[this.errorMessage] || this.errorMessage;
|
|
2533
2567
|
}
|
|
@@ -3210,6 +3244,7 @@ class AddUpdateFormComponent {
|
|
|
3210
3244
|
name: 'required',
|
|
3211
3245
|
params: { enabled: true },
|
|
3212
3246
|
errorKey: 'ERR_REQUIRED',
|
|
3247
|
+
message: v.message
|
|
3213
3248
|
});
|
|
3214
3249
|
errorPriority.push('required');
|
|
3215
3250
|
break;
|
|
@@ -3220,6 +3255,7 @@ class AddUpdateFormComponent {
|
|
|
3220
3255
|
pattern: '^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\\.[a-zA-Z]{2,}$',
|
|
3221
3256
|
},
|
|
3222
3257
|
errorKey: 'ERR_REGEX_MISMATCH',
|
|
3258
|
+
message: v.message
|
|
3223
3259
|
});
|
|
3224
3260
|
errorPriority.push('regex');
|
|
3225
3261
|
break;
|
|
@@ -3228,6 +3264,7 @@ class AddUpdateFormComponent {
|
|
|
3228
3264
|
name: 'minLength',
|
|
3229
3265
|
params: { value: v.value },
|
|
3230
3266
|
errorKey: 'ERR_MIN_LENGTH',
|
|
3267
|
+
message: v.message
|
|
3231
3268
|
});
|
|
3232
3269
|
errorPriority.push('minLength');
|
|
3233
3270
|
break;
|
|
@@ -3236,6 +3273,7 @@ class AddUpdateFormComponent {
|
|
|
3236
3273
|
name: 'maxLength',
|
|
3237
3274
|
params: { value: v.value },
|
|
3238
3275
|
errorKey: 'ERR_MAX_LENGTH',
|
|
3276
|
+
message: v.message
|
|
3239
3277
|
});
|
|
3240
3278
|
errorPriority.push('maxLength');
|
|
3241
3279
|
break;
|
|
@@ -3244,6 +3282,7 @@ class AddUpdateFormComponent {
|
|
|
3244
3282
|
name: 'regex',
|
|
3245
3283
|
params: { pattern: v.value },
|
|
3246
3284
|
errorKey: 'ERR_REGEX_MISMATCH',
|
|
3285
|
+
message: v.message
|
|
3247
3286
|
});
|
|
3248
3287
|
errorPriority.push('regex');
|
|
3249
3288
|
break;
|
|
@@ -3266,6 +3305,7 @@ class AddUpdateFormComponent {
|
|
|
3266
3305
|
name: 'min',
|
|
3267
3306
|
params: { value: v.value },
|
|
3268
3307
|
errorKey: 'ERR_MIN_VALUE',
|
|
3308
|
+
message: v.message
|
|
3269
3309
|
});
|
|
3270
3310
|
errorPriority.push('min');
|
|
3271
3311
|
break;
|
|
@@ -3274,9 +3314,19 @@ class AddUpdateFormComponent {
|
|
|
3274
3314
|
name: 'max',
|
|
3275
3315
|
params: { value: v.value },
|
|
3276
3316
|
errorKey: 'ERR_MAX_VALUE',
|
|
3317
|
+
message: v.message
|
|
3277
3318
|
});
|
|
3278
3319
|
errorPriority.push('max');
|
|
3279
3320
|
break;
|
|
3321
|
+
case 'duplicate':
|
|
3322
|
+
rules.push({
|
|
3323
|
+
name: 'duplicate',
|
|
3324
|
+
params: { existingValues: v.value },
|
|
3325
|
+
errorKey: v.errorKey || 'ERR_DUPLICATE',
|
|
3326
|
+
message: v.message
|
|
3327
|
+
});
|
|
3328
|
+
errorPriority.push('duplicate');
|
|
3329
|
+
break;
|
|
3280
3330
|
}
|
|
3281
3331
|
});
|
|
3282
3332
|
}
|
|
@@ -3364,6 +3414,8 @@ class AddUpdateFormComponent {
|
|
|
3364
3414
|
return `Maximum value is ${errors['max'].max}`;
|
|
3365
3415
|
if (errors['pattern'])
|
|
3366
3416
|
return 'Invalid format';
|
|
3417
|
+
if (errors['duplicate'])
|
|
3418
|
+
return column?.validations?.find(v => v.type === 'duplicate')?.message || `${column?.label} already exists`;
|
|
3367
3419
|
return 'Invalid field';
|
|
3368
3420
|
}
|
|
3369
3421
|
isFieldDisabled(column, group) {
|
|
@@ -3639,6 +3691,15 @@ class AddUpdateFormComponent {
|
|
|
3639
3691
|
});
|
|
3640
3692
|
}
|
|
3641
3693
|
break;
|
|
3694
|
+
case 'duplicate':
|
|
3695
|
+
validators.push((control) => {
|
|
3696
|
+
const value = control.value;
|
|
3697
|
+
if (!value || !validation.value || !Array.isArray(validation.value))
|
|
3698
|
+
return null;
|
|
3699
|
+
const isDuplicate = validation.value.some((existing) => String(existing).toLowerCase().trim() === String(value).toLowerCase().trim());
|
|
3700
|
+
return isDuplicate ? { duplicate: true } : null;
|
|
3701
|
+
});
|
|
3702
|
+
break;
|
|
3642
3703
|
}
|
|
3643
3704
|
});
|
|
3644
3705
|
return validators;
|
|
@@ -3772,6 +3833,8 @@ class AddUpdateFormComponent {
|
|
|
3772
3833
|
return `Maximum value is ${field.errors['max'].max}`;
|
|
3773
3834
|
if (field.errors['pattern'])
|
|
3774
3835
|
return 'Invalid format';
|
|
3836
|
+
if (field.errors['duplicate'])
|
|
3837
|
+
return column?.validations?.find(v => v.type === 'duplicate')?.message || `${column?.label} already exists`;
|
|
3775
3838
|
return 'Invalid field';
|
|
3776
3839
|
}
|
|
3777
3840
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: AddUpdateFormComponent, deps: [{ token: i2.FormBuilder }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
|