@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.
@@ -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
- for (const errorKey of this.schema.errorPriority) {
1729
- if (this.control.hasError(errorKey)) {
1730
- const rule = this.schema.rules.find(r => r.name === (errorKey === 'required' ? 'required' : errorKey));
1731
- if (rule)
1732
- return this.getReadableError(rule.errorKey || '');
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 });