@seniorsistemas/angular-components 19.1.0 → 19.2.0

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.
Files changed (42) hide show
  1. package/bignumber-input/lib/bignumber-input/bignumber-input.directive.d.ts +58 -25
  2. package/dynamic-form/dynamic-form/components/lookup/lookup.component.d.ts +8 -8
  3. package/esm2022/bignumber-input/lib/bignumber-input/bignumber-input.directive.mjs +140 -99
  4. package/esm2022/dynamic-form/dynamic-form/components/lookup/lookup.component.mjs +12 -12
  5. package/esm2022/dynamic-form/dynamic-form/form-field/fields/bignumber/bignumber-field.component.mjs +3 -3
  6. package/esm2022/dynamic-form/dynamic-form/form-field/fields/currency/currency-field.component.mjs +3 -3
  7. package/esm2022/dynamic-form/dynamic-form/form-field/fields/number/number-field.component.mjs +3 -3
  8. package/esm2022/inline-edit/lib/inline-edit/components/fields/inline-edit-number/inline-edit-number.component.mjs +1 -1
  9. package/esm2022/inline-edit/lib/inline-edit/fields/inline-edit-number-field.mjs +2 -3
  10. package/esm2022/locale/lib/locale/apply-localized-mask.mjs +19 -0
  11. package/esm2022/locale/lib/locale/pipes/localized-bignumber-impure.pipe.mjs +1 -1
  12. package/esm2022/locale/lib/locale/pipes/localized-bignumber.pipe.mjs +5 -4
  13. package/esm2022/locale/public-api.mjs +2 -1
  14. package/esm2022/number-input/lib/number-input/number-input.directive.mjs +12 -148
  15. package/esm2022/number-input/lib/number-input/number-input.module.mjs +5 -5
  16. package/esm2022/numeric-mask/lib/numeric-mask/numeric-mask.directive.mjs +32 -35
  17. package/esm2022/table/lib/table/table-column/table-columns.component.mjs +20 -4
  18. package/esm2022/table/public-api.mjs +2 -1
  19. package/fesm2022/seniorsistemas-angular-components-bignumber-input.mjs +139 -98
  20. package/fesm2022/seniorsistemas-angular-components-bignumber-input.mjs.map +1 -1
  21. package/fesm2022/seniorsistemas-angular-components-dynamic-form.mjs +19 -19
  22. package/fesm2022/seniorsistemas-angular-components-dynamic-form.mjs.map +1 -1
  23. package/fesm2022/seniorsistemas-angular-components-inline-edit.mjs +2 -3
  24. package/fesm2022/seniorsistemas-angular-components-inline-edit.mjs.map +1 -1
  25. package/fesm2022/seniorsistemas-angular-components-locale.mjs +22 -4
  26. package/fesm2022/seniorsistemas-angular-components-locale.mjs.map +1 -1
  27. package/fesm2022/seniorsistemas-angular-components-number-input.mjs +14 -151
  28. package/fesm2022/seniorsistemas-angular-components-number-input.mjs.map +1 -1
  29. package/fesm2022/seniorsistemas-angular-components-numeric-mask.mjs +31 -34
  30. package/fesm2022/seniorsistemas-angular-components-numeric-mask.mjs.map +1 -1
  31. package/fesm2022/seniorsistemas-angular-components-table.mjs +21 -6
  32. package/fesm2022/seniorsistemas-angular-components-table.mjs.map +1 -1
  33. package/inline-edit/lib/inline-edit/fields/inline-edit-number-field.d.ts +2 -3
  34. package/locale/lib/locale/apply-localized-mask.d.ts +9 -0
  35. package/locale/lib/locale/pipes/localized-bignumber-impure.pipe.d.ts +2 -3
  36. package/locale/lib/locale/pipes/localized-bignumber.pipe.d.ts +3 -2
  37. package/locale/public-api.d.ts +1 -0
  38. package/number-input/lib/number-input/number-input.directive.d.ts +7 -47
  39. package/number-input/lib/number-input/number-input.module.d.ts +2 -2
  40. package/numeric-mask/lib/numeric-mask/numeric-mask.directive.d.ts +8 -9
  41. package/package.json +1 -1
  42. package/table/public-api.d.ts +1 -0
@@ -1,9 +1,9 @@
1
- import { ElementRef, EventEmitter, KeyValueDiffers, OnChanges, OnInit, SimpleChanges } from '@angular/core';
2
- import { CurrencyMaskDirective } from 'ng2-currency-mask';
3
- import { LocaleService } from '@seniorsistemas/angular-components/locale';
1
+ import { OnInit, OnChanges, SimpleChanges } from '@angular/core';
2
+ import { AbstractControl, ValidationErrors } from '@angular/forms';
3
+ import { NumericMaskDirective } from '@seniorsistemas/angular-components/numeric-mask';
4
4
  import * as i0 from "@angular/core";
5
5
  /**
6
- * @deprecated Should use 'AlignmentOptions' from @seniorsistemas/ng2-currency-mask instead
6
+ * @deprecated Este enum não tem mais efeito. O alinhamento do campo é sempre `'right'`.
7
7
  */
8
8
  export declare enum BignumberAlignmentOption {
9
9
  RIGHT = "right",
@@ -22,66 +22,99 @@ export declare enum BignumberAlignmentOption {
22
22
  *
23
23
  * @category Inputs
24
24
  */
25
- export declare class BignumberInputDirective extends CurrencyMaskDirective implements OnInit, OnChanges {
26
- private readonly localeService;
25
+ export declare class BignumberInputDirective extends NumericMaskDirective implements OnInit, OnChanges {
27
26
  /**
28
- * @description Número total de dígitos significativos aceitos pelo campo.
27
+ * @description Número total de dígitos significativos aceitos pelo campo (parte inteira + parte decimal).
28
+ *
29
+ * Durante a digitação, o campo limita fisicamente o número de caracteres via `maxLength`.
30
+ * Ao colar um valor que ultrapasse o limite, o evento `pasteRejected` é emitido e a colagem é cancelada.
31
+ * Após sair do campo (blur) ou ao submeter o formulário, se o valor ultrapassar `precision`,
32
+ * o campo ficará inválido com o erro `exceedsPrecision: { precision, actual }`.
29
33
  *
30
34
  * @default 15
31
35
  */
32
- precision: number;
36
+ precision: import("@angular/core").InputSignal<number>;
33
37
  /**
38
+ * @deprecated Use `minDecimalPlaces` e `maxDecimalPlaces` ao invés deste.
34
39
  * @description Número de casas decimais exibidas e aceitas pelo campo.
40
+ * Quando informado, repassa o valor para `minDecimalPlaces` e `maxDecimalPlaces`.
41
+ *
42
+ * @default 0
43
+ */
44
+ scale: import("@angular/core").InputSignal<number>;
45
+ /**
46
+ * @description Número mínimo de casas decimais exibidas.
47
+ *
48
+ * @default 0
49
+ */
50
+ minDecimalPlaces: import("@angular/core").ModelSignal<number>;
51
+ /**
52
+ * @description Número máximo de casas decimais aceitas.
35
53
  *
36
54
  * @default 0
37
55
  */
38
- scale: number;
56
+ maxDecimalPlaces: import("@angular/core").ModelSignal<number>;
39
57
  /**
40
58
  * @description Separador decimal personalizado. Quando não informado,
41
59
  * utiliza o separador definido pelo locale ativo da aplicação.
42
60
  */
43
- decimalSeparator?: string;
61
+ decimalSeparator: import("@angular/core").ModelSignal<string>;
62
+ /**
63
+ * @deprecated Use `thousandSeparator` ao invés deste.
64
+ * @description Separador de milhar personalizado. Recomenda-se migrar para `thousandSeparator`.
65
+ */
66
+ thousandsSeparator: import("@angular/core").InputSignal<string | undefined>;
44
67
  /**
45
68
  * @description Separador de milhar personalizado. Quando não informado,
46
69
  * utiliza o separador definido pelo locale ativo da aplicação.
47
70
  */
48
- thousandsSeparator?: string;
71
+ thousandSeparator: import("@angular/core").ModelSignal<string>;
49
72
  /**
50
- * @description Alinhamento do conteúdo do campo. Aceita `'left'` ou `'right'`.
51
- *
52
- * @default 'left'
73
+ * @deprecated Este input não tem mais efeito e pode ser removido.
74
+ * O alinhamento do campo é sempre `'right'`, conforme o comportamento padrão da máscara numérica.
53
75
  */
54
- alignTo: 'left' | 'right';
76
+ alignTo: import("@angular/core").InputSignal<"right" | "left">;
55
77
  /**
56
78
  * @description Permite a entrada de valores negativos.
57
79
  *
58
80
  * @default true
59
81
  */
60
- allowNegative: boolean;
82
+ allowNegative: import("@angular/core").InputSignal<boolean>;
61
83
  /**
62
84
  * @description Texto de placeholder exibido quando o campo está vazio.
63
85
  */
64
- placeholder?: string;
86
+ placeholder: import("@angular/core").ModelSignal<string | undefined>;
65
87
  maxLength?: number;
66
88
  /**
67
- * @description Emitido quando uma colé (paste) é rejeitada pela validação do campo.
89
+ * @description Emitido quando uma colagem (paste) é rejeitada pela validação do campo.
68
90
  * Emite a string colada que foi recusada.
69
91
  */
70
- pasteRejected: EventEmitter<string>;
71
- private readonly regex;
72
- constructor(_elementRef: ElementRef, _keyValueDiffers: KeyValueDiffers, localeService: LocaleService);
92
+ pasteRejected: import("@angular/core").OutputEmitterRef<string>;
93
+ private _decimalSeparatorExplicit;
94
+ private _thousandSeparatorExplicit;
95
+ constructor();
73
96
  ngOnInit(): void;
74
97
  ngOnChanges(changes: SimpleChanges): void;
98
+ /**
99
+ * Overrides base paste handler to add precision validation.
100
+ * If the pasted value exceeds `precision` digits, the paste is rejected.
101
+ */
75
102
  onPaste(event: ClipboardEvent): void;
76
103
  /**
77
- * Update the options values according to the directive input values.
104
+ * Overrides locale-derived separators with explicitly set inputs, if provided.
105
+ * If no explicit separator is set, the locale-derived value is used.
106
+ */
107
+ protected updateSeparators(): void;
108
+ /**
109
+ * Overrides base validator to add precision validation.
110
+ * If the total number of significant digits exceeds `precision`, returns an `exceedsPrecision` error.
78
111
  */
79
- private updateVariables;
112
+ validate(control: AbstractControl): ValidationErrors | null;
80
113
  /**
81
114
  * Responsible to calculate the field maximum length considering the separators.
82
115
  */
83
116
  private calculateMaxLength;
84
- private onLocaleService;
117
+ private updatePlaceholder;
85
118
  static ɵfac: i0.ɵɵFactoryDeclaration<BignumberInputDirective, never>;
86
- static ɵdir: i0.ɵɵDirectiveDeclaration<BignumberInputDirective, "input[sBignumberInput]", never, { "precision": { "alias": "precision"; "required": false; }; "scale": { "alias": "scale"; "required": false; }; "decimalSeparator": { "alias": "decimalSeparator"; "required": false; }; "thousandsSeparator": { "alias": "thousandsSeparator"; "required": false; }; "alignTo": { "alias": "alignTo"; "required": false; }; "allowNegative": { "alias": "allowNegative"; "required": false; }; "placeholder": { "alias": "placeholder"; "required": false; }; }, { "pasteRejected": "pasteRejected"; }, never, never, false, never>;
119
+ static ɵdir: i0.ɵɵDirectiveDeclaration<BignumberInputDirective, "input[sBignumberInput]", never, { "precision": { "alias": "precision"; "required": false; "isSignal": true; }; "scale": { "alias": "scale"; "required": false; "isSignal": true; }; "minDecimalPlaces": { "alias": "minDecimalPlaces"; "required": false; "isSignal": true; }; "maxDecimalPlaces": { "alias": "maxDecimalPlaces"; "required": false; "isSignal": true; }; "decimalSeparator": { "alias": "decimalSeparator"; "required": false; "isSignal": true; }; "thousandsSeparator": { "alias": "thousandsSeparator"; "required": false; "isSignal": true; }; "thousandSeparator": { "alias": "thousandSeparator"; "required": false; "isSignal": true; }; "alignTo": { "alias": "alignTo"; "required": false; "isSignal": true; }; "allowNegative": { "alias": "allowNegative"; "required": false; "isSignal": true; }; "placeholder": { "alias": "placeholder"; "required": false; "isSignal": true; }; }, { "minDecimalPlaces": "minDecimalPlacesChange"; "maxDecimalPlaces": "maxDecimalPlacesChange"; "decimalSeparator": "decimalSeparatorChange"; "thousandSeparator": "thousandSeparatorChange"; "placeholder": "placeholderChange"; "pasteRejected": "pasteRejected"; }, never, never, false, never>;
87
120
  }
@@ -1,14 +1,12 @@
1
1
  import { ChangeDetectorRef, EventEmitter, OnChanges, OnDestroy, OnInit, SimpleChanges } from '@angular/core';
2
2
  import { ControlValueAccessor, FormGroup } from '@angular/forms';
3
3
  import { TranslateService } from '@ngx-translate/core';
4
- import { MaskConfig } from '@seniorsistemas/ng2-currency-mask';
4
+ import { LoadingStateIndicators } from '@seniorsistemas/angular-components/loading-state';
5
+ import { LocalizedBignumberOptions } from '@seniorsistemas/angular-components/locale';
5
6
  import { HotkeysService } from 'angular2-hotkeys';
6
7
  import { SortMeta } from 'primeng/api';
7
8
  import { AutoComplete, AutoCompleteCompleteEvent } from 'primeng/autocomplete';
8
9
  import { Table, TableLazyLoadEvent } from 'primeng/table';
9
- import { DefaultFilter } from './types/default-filter';
10
- import { LabelOptions } from './types/label-options';
11
- import { LoadingStateIndicators } from '@seniorsistemas/angular-components/loading-state';
12
10
  import { DynamicStructure } from '../../configurations/structure/structure';
13
11
  import { BignumberFieldConfig } from '../../form-field/configurations/fields/bignumber-field';
14
12
  import { CurrencyFieldConfig } from '../../form-field/configurations/fields/currency-field';
@@ -17,6 +15,8 @@ import { NumberFieldConfig } from '../../form-field/configurations/fields/number
17
15
  import { SelectFieldConfig } from '../../form-field/configurations/fields/select-field';
18
16
  import { TextFieldConfig } from '../../form-field/configurations/fields/text-field';
19
17
  import { LookupSearchRequest } from './models/lookup-api.model';
18
+ import { DefaultFilter } from './types/default-filter';
19
+ import { LabelOptions } from './types/label-options';
20
20
  import * as i0 from "@angular/core";
21
21
  export declare class LookupComponent implements ControlValueAccessor, OnChanges, OnInit, OnDestroy {
22
22
  private readonly hotkeysService;
@@ -128,10 +128,10 @@ export declare class LookupComponent implements ControlValueAccessor, OnChanges,
128
128
  getColWidth(label: string): string;
129
129
  getScale(scale: any): number;
130
130
  private isFunction;
131
- getDoubleMaskConfig(col: NumberFieldConfig | BignumberFieldConfig): MaskConfig;
132
- getMoneyMaskConfig(col: CurrencyFieldConfig): MaskConfig;
133
- getIntegerMaskConfig(col: BignumberFieldConfig | NumberFieldConfig): MaskConfig;
134
- getNumberMaskConfig(col: NumberFieldConfig | BignumberFieldConfig): MaskConfig;
131
+ getDoubleMaskConfig(col: NumberFieldConfig | BignumberFieldConfig): LocalizedBignumberOptions;
132
+ getMoneyMaskConfig(col: CurrencyFieldConfig): LocalizedBignumberOptions;
133
+ getIntegerMaskConfig(col: BignumberFieldConfig | NumberFieldConfig): LocalizedBignumberOptions;
134
+ getNumberMaskConfig(col: NumberFieldConfig | BignumberFieldConfig): LocalizedBignumberOptions;
135
135
  parseValueMaskValue(col: DynamicType, rowData: any): any;
136
136
  fieldHasMask(field: DynamicType): field is NumberFieldConfig | BignumberFieldConfig | TextFieldConfig;
137
137
  onTableRowDoubleClick(rowData: any): void;
@@ -1,11 +1,9 @@
1
- import { Directive, EventEmitter, forwardRef, HostBinding, HostListener, Input, Output, } from '@angular/core';
1
+ import { Directive, forwardRef, input, model, output } from '@angular/core';
2
2
  import { NG_VALUE_ACCESSOR } from '@angular/forms';
3
- import { CurrencyMaskDirective } from 'ng2-currency-mask';
4
- import { first } from 'rxjs/operators';
3
+ import { NumericMaskDirective } from '@seniorsistemas/angular-components/numeric-mask';
5
4
  import * as i0 from "@angular/core";
6
- import * as i1 from "@seniorsistemas/angular-components/locale";
7
5
  /**
8
- * @deprecated Should use 'AlignmentOptions' from @seniorsistemas/ng2-currency-mask instead
6
+ * @deprecated Este enum não tem mais efeito. O alinhamento do campo é sempre `'right'`.
9
7
  */
10
8
  export var BignumberAlignmentOption;
11
9
  (function (BignumberAlignmentOption) {
@@ -25,134 +23,198 @@ export var BignumberAlignmentOption;
25
23
  *
26
24
  * @category Inputs
27
25
  */
28
- export class BignumberInputDirective extends CurrencyMaskDirective {
29
- localeService;
26
+ export class BignumberInputDirective extends NumericMaskDirective {
30
27
  /**
31
- * @description Número total de dígitos significativos aceitos pelo campo.
28
+ * @description Número total de dígitos significativos aceitos pelo campo (parte inteira + parte decimal).
29
+ *
30
+ * Durante a digitação, o campo limita fisicamente o número de caracteres via `maxLength`.
31
+ * Ao colar um valor que ultrapasse o limite, o evento `pasteRejected` é emitido e a colagem é cancelada.
32
+ * Após sair do campo (blur) ou ao submeter o formulário, se o valor ultrapassar `precision`,
33
+ * o campo ficará inválido com o erro `exceedsPrecision: { precision, actual }`.
32
34
  *
33
35
  * @default 15
34
36
  */
35
- precision = 15;
37
+ precision = input(15);
36
38
  /**
39
+ * @deprecated Use `minDecimalPlaces` e `maxDecimalPlaces` ao invés deste.
37
40
  * @description Número de casas decimais exibidas e aceitas pelo campo.
41
+ * Quando informado, repassa o valor para `minDecimalPlaces` e `maxDecimalPlaces`.
38
42
  *
39
43
  * @default 0
40
44
  */
41
- scale = 0;
45
+ scale = input(0);
46
+ /**
47
+ * @description Número mínimo de casas decimais exibidas.
48
+ *
49
+ * @default 0
50
+ */
51
+ minDecimalPlaces = model(0);
52
+ /**
53
+ * @description Número máximo de casas decimais aceitas.
54
+ *
55
+ * @default 0
56
+ */
57
+ maxDecimalPlaces = model(0);
42
58
  /**
43
59
  * @description Separador decimal personalizado. Quando não informado,
44
60
  * utiliza o separador definido pelo locale ativo da aplicação.
45
61
  */
46
- decimalSeparator;
62
+ decimalSeparator = model(',');
63
+ /**
64
+ * @deprecated Use `thousandSeparator` ao invés deste.
65
+ * @description Separador de milhar personalizado. Recomenda-se migrar para `thousandSeparator`.
66
+ */
67
+ thousandsSeparator = input(undefined);
47
68
  /**
48
69
  * @description Separador de milhar personalizado. Quando não informado,
49
70
  * utiliza o separador definido pelo locale ativo da aplicação.
50
71
  */
51
- thousandsSeparator;
72
+ thousandSeparator = model('.');
52
73
  /**
53
- * @description Alinhamento do conteúdo do campo. Aceita `'left'` ou `'right'`.
54
- *
55
- * @default 'left'
74
+ * @deprecated Este input não tem mais efeito e pode ser removido.
75
+ * O alinhamento do campo é sempre `'right'`, conforme o comportamento padrão da máscara numérica.
56
76
  */
57
- alignTo = 'left';
77
+ alignTo = input('right');
58
78
  /**
59
79
  * @description Permite a entrada de valores negativos.
60
80
  *
61
81
  * @default true
62
82
  */
63
- allowNegative = true;
83
+ allowNegative = input(true);
64
84
  /**
65
85
  * @description Texto de placeholder exibido quando o campo está vazio.
66
86
  */
67
- placeholder;
87
+ placeholder = model(undefined);
68
88
  maxLength;
69
89
  /**
70
- * @description Emitido quando uma colé (paste) é rejeitada pela validação do campo.
90
+ * @description Emitido quando uma colagem (paste) é rejeitada pela validação do campo.
71
91
  * Emite a string colada que foi recusada.
72
92
  */
73
- pasteRejected = new EventEmitter();
74
- regex = /\d/;
75
- //TODO: Ajuste no ng2-currency-mask para utilizar esse Código (aplicado na tarefa: SDS-305)
76
- // private readonly _angularComponentsLocaleService: LocaleService;
77
- // constructor(injector: Injector) {
78
- // super(injector);
79
- // this._angularComponentsLocaleService = injector.get(LocaleService);
80
- // this.onLocaleService();
81
- // }
82
- //TODO: Remover esse constructor
83
- constructor(_elementRef, _keyValueDiffers, localeService) {
84
- super({ align: '', allowNegative: true, decimal: '', precision: 0, prefix: '', suffix: '', thousands: '' }, _elementRef, _keyValueDiffers);
85
- this.localeService = localeService;
86
- this.onLocaleService();
93
+ pasteRejected = output();
94
+ _decimalSeparatorExplicit = false;
95
+ _thousandSeparatorExplicit = false;
96
+ constructor() {
97
+ super();
87
98
  }
88
99
  ngOnInit() {
89
- this.updateVariables();
90
100
  super.ngOnInit();
101
+ this.updatePlaceholder();
91
102
  }
103
+ // ngOnChanges handles all signal writes reactively, avoiding the need for
104
+ // effect() with allowSignalWrites. The decimalSeparator/thousandSeparator inputs
105
+ // have truthy defaults, so ngOnChanges is necessary to distinguish "parent bound this"
106
+ // from "it's just the default value".
92
107
  ngOnChanges(changes) {
93
- const placeholderChange = changes['placeholder']?.currentValue;
94
- if (!placeholderChange && this.scale) {
95
- this.placeholder = `0${this.decimalSeparator}${''.padEnd(this.scale, '0')}`;
108
+ if (changes['decimalSeparator'] && this.decimalSeparator()) {
109
+ this._decimalSeparatorExplicit = true;
110
+ }
111
+ if (changes['thousandSeparator'] && this.thousandSeparator()) {
112
+ this._thousandSeparatorExplicit = true;
96
113
  }
97
- const configChanged = changes['scale'] ||
114
+ if (changes['thousandsSeparator']) {
115
+ const val = this.thousandsSeparator();
116
+ if (val !== undefined) {
117
+ this._thousandSeparatorExplicit = true;
118
+ this.thousandSeparator.set(val);
119
+ }
120
+ }
121
+ if (changes['scale']) {
122
+ this.minDecimalPlaces.set(this.scale());
123
+ this.maxDecimalPlaces.set(this.scale());
124
+ }
125
+ if (changes['scale'] ||
98
126
  changes['decimalSeparator'] ||
127
+ changes['thousandSeparator'] ||
99
128
  changes['thousandsSeparator'] ||
100
- changes['alignTo'] ||
101
- changes['precision'] ||
102
- changes['allowNegative'];
103
- if (configChanged) {
104
- this.updateVariables();
129
+ changes['precision']) {
130
+ this.calculateMaxLength();
131
+ }
132
+ if (changes['scale'] || changes['decimalSeparator']) {
133
+ this.updatePlaceholder();
105
134
  }
106
135
  }
136
+ /**
137
+ * Overrides base paste handler to add precision validation.
138
+ * If the pasted value exceeds `precision` digits, the paste is rejected.
139
+ */
107
140
  onPaste(event) {
108
141
  const pastedText = event.clipboardData?.getData('text') ?? '';
142
+ const effectiveThousandSep = this.thousandsSeparator() ?? this.thousandSeparator();
109
143
  const numericText = pastedText
110
- .replace(new RegExp(`\\${this.thousandsSeparator}`, 'g'), '')
111
- .replace(this.decimalSeparator, '.');
144
+ .replace(new RegExp(`\\${effectiveThousandSep}`, 'g'), '')
145
+ .replace(this.decimalSeparator(), '.');
112
146
  const [intPart, decimalPart = ''] = numericText.split('.');
113
- const totalDigits = intPart.replace(/\D/g, '').length + decimalPart.slice(0, this.scale).length;
114
- if (totalDigits > this.precision) {
147
+ const totalDigits = intPart.replace(/\D/g, '').length + decimalPart.slice(0, this.scale()).length;
148
+ if (totalDigits > this.precision()) {
115
149
  event.preventDefault();
116
150
  this.pasteRejected.emit(pastedText);
151
+ return;
117
152
  }
153
+ super.onPaste(event);
118
154
  }
119
155
  /**
120
- * Update the options values according to the directive input values.
156
+ * Overrides locale-derived separators with explicitly set inputs, if provided.
157
+ * If no explicit separator is set, the locale-derived value is used.
121
158
  */
122
- updateVariables() {
123
- this.options = {
124
- align: this.alignTo,
125
- decimal: this.decimalSeparator,
126
- thousands: this.thousandsSeparator,
127
- precision: this.scale || 0,
128
- prefix: '',
129
- suffix: '',
130
- allowNegative: this.allowNegative,
131
- };
159
+ updateSeparators() {
160
+ const explicitDecimal = this._decimalSeparatorExplicit ? this.decimalSeparator() : undefined;
161
+ const explicitThousand = this._thousandSeparatorExplicit
162
+ ? (this.thousandsSeparator() ?? this.thousandSeparator())
163
+ : undefined;
164
+ super.updateSeparators();
165
+ if (explicitDecimal !== undefined) {
166
+ this.decimalSeparator.set(explicitDecimal);
167
+ }
168
+ if (explicitThousand !== undefined) {
169
+ this.thousandSeparator.set(explicitThousand);
170
+ }
132
171
  this.calculateMaxLength();
172
+ this.updatePlaceholder();
173
+ }
174
+ /**
175
+ * Overrides base validator to add precision validation.
176
+ * If the total number of significant digits exceeds `precision`, returns an `exceedsPrecision` error.
177
+ */
178
+ validate(control) {
179
+ const baseErrors = super.validate(control);
180
+ if (!control.value || !this.precision()) {
181
+ return baseErrors ?? null;
182
+ }
183
+ const numericStr = String(control.value).replace('-', '');
184
+ const [intPart, decimalPart = ''] = numericStr.split('.');
185
+ const totalDigits = intPart.length + decimalPart.length;
186
+ if (totalDigits > this.precision()) {
187
+ return {
188
+ ...baseErrors,
189
+ exceedsPrecision: {
190
+ precision: this.precision(),
191
+ actual: totalDigits,
192
+ },
193
+ };
194
+ }
195
+ return baseErrors ?? null;
133
196
  }
134
197
  /**
135
198
  * Responsible to calculate the field maximum length considering the separators.
136
199
  */
137
200
  calculateMaxLength() {
138
- const decSepLength = this.decimalSeparator?.length ?? 0;
139
- const thoSepLength = this.thousandsSeparator?.length ?? 0;
201
+ const decSepLength = this.decimalSeparator()?.length ?? 0;
202
+ const effectiveThousandSep = this.thousandsSeparator() ?? this.thousandSeparator();
203
+ const thoSepLength = effectiveThousandSep?.length ?? 0;
140
204
  const DIGITS_GROUP_SIZE = 3;
141
- let maxLength = this.precision + (this.scale ? decSepLength : 0);
142
- maxLength += Math.ceil((this.precision - (this.scale ? this.scale : 0)) / DIGITS_GROUP_SIZE - 1) * thoSepLength;
205
+ let maxLength = this.precision() + (this.scale() ? decSepLength : 0);
206
+ maxLength +=
207
+ Math.ceil((this.precision() - (this.scale() ? this.scale() : 0)) / DIGITS_GROUP_SIZE - 1) * thoSepLength;
143
208
  this.maxLength = maxLength;
144
209
  }
145
- onLocaleService() {
146
- this.localeService
147
- .getLocale()
148
- .pipe(first())
149
- .subscribe(() => {
150
- this.decimalSeparator = this.decimalSeparator ?? this.localeService.getDecimalSeparator();
151
- this.thousandsSeparator = this.thousandsSeparator ?? this.localeService.getGroupingSeparator();
152
- });
210
+ updatePlaceholder() {
211
+ const scale = this.scale();
212
+ if (!this.placeholder() && scale) {
213
+ this.placeholder.set(`0${this.decimalSeparator()}${''.padEnd(scale, '0')}`);
214
+ }
153
215
  }
154
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: BignumberInputDirective, deps: [{ token: i0.ElementRef }, { token: i0.KeyValueDiffers }, { token: i1.LocaleService }], target: i0.ɵɵFactoryTarget.Directive });
155
- static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.14", type: BignumberInputDirective, selector: "input[sBignumberInput]", inputs: { precision: "precision", scale: "scale", decimalSeparator: "decimalSeparator", thousandsSeparator: "thousandsSeparator", alignTo: "alignTo", allowNegative: "allowNegative", placeholder: "placeholder" }, outputs: { pasteRejected: "pasteRejected" }, host: { listeners: { "paste": "onPaste($event)" }, properties: { "attr.placeholder": "this.placeholder", "attr.maxLength": "this.maxLength" } }, providers: [
216
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: BignumberInputDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
217
+ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.2.14", type: BignumberInputDirective, selector: "input[sBignumberInput]", inputs: { precision: { classPropertyName: "precision", publicName: "precision", isSignal: true, isRequired: false, transformFunction: null }, scale: { classPropertyName: "scale", publicName: "scale", isSignal: true, isRequired: false, transformFunction: null }, minDecimalPlaces: { classPropertyName: "minDecimalPlaces", publicName: "minDecimalPlaces", isSignal: true, isRequired: false, transformFunction: null }, maxDecimalPlaces: { classPropertyName: "maxDecimalPlaces", publicName: "maxDecimalPlaces", isSignal: true, isRequired: false, transformFunction: null }, decimalSeparator: { classPropertyName: "decimalSeparator", publicName: "decimalSeparator", isSignal: true, isRequired: false, transformFunction: null }, thousandsSeparator: { classPropertyName: "thousandsSeparator", publicName: "thousandsSeparator", isSignal: true, isRequired: false, transformFunction: null }, thousandSeparator: { classPropertyName: "thousandSeparator", publicName: "thousandSeparator", isSignal: true, isRequired: false, transformFunction: null }, alignTo: { classPropertyName: "alignTo", publicName: "alignTo", isSignal: true, isRequired: false, transformFunction: null }, allowNegative: { classPropertyName: "allowNegative", publicName: "allowNegative", isSignal: true, isRequired: false, transformFunction: null }, placeholder: { classPropertyName: "placeholder", publicName: "placeholder", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { minDecimalPlaces: "minDecimalPlacesChange", maxDecimalPlaces: "maxDecimalPlacesChange", decimalSeparator: "decimalSeparatorChange", thousandSeparator: "thousandSeparatorChange", placeholder: "placeholderChange", pasteRejected: "pasteRejected" }, host: { properties: { "attr.placeholder": "placeholder()", "attr.maxLength": "maxLength" } }, providers: [
156
218
  {
157
219
  provide: NG_VALUE_ACCESSOR,
158
220
  // NOSONAR - 'BignumberInputDirective' is deprecated.
@@ -165,6 +227,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
165
227
  type: Directive,
166
228
  args: [{
167
229
  selector: 'input[sBignumberInput]',
230
+ host: {
231
+ '[attr.placeholder]': 'placeholder()',
232
+ '[attr.maxLength]': 'maxLength',
233
+ },
168
234
  providers: [
169
235
  {
170
236
  provide: NG_VALUE_ACCESSOR,
@@ -174,30 +240,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
174
240
  },
175
241
  ],
176
242
  }]
177
- }], ctorParameters: () => [{ type: i0.ElementRef }, { type: i0.KeyValueDiffers }, { type: i1.LocaleService }], propDecorators: { precision: [{
178
- type: Input
179
- }], scale: [{
180
- type: Input
181
- }], decimalSeparator: [{
182
- type: Input
183
- }], thousandsSeparator: [{
184
- type: Input
185
- }], alignTo: [{
186
- type: Input
187
- }], allowNegative: [{
188
- type: Input
189
- }], placeholder: [{
190
- type: Input
191
- }, {
192
- type: HostBinding,
193
- args: ['attr.placeholder']
194
- }], maxLength: [{
195
- type: HostBinding,
196
- args: ['attr.maxLength']
197
- }], pasteRejected: [{
198
- type: Output
199
- }], onPaste: [{
200
- type: HostListener,
201
- args: ['paste', ['$event']]
202
- }] } });
203
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bignumber-input.directive.js","sourceRoot":"","sources":["../../../../../projects/angular-components/bignumber-input/src/lib/bignumber-input/bignumber-input.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    Directive,\n    ElementRef,\n    EventEmitter,\n    forwardRef,\n    HostBinding,\n    HostListener,\n    Input,\n    KeyValueDiffers,\n    OnChanges,\n    OnInit,\n    Output,\n    SimpleChanges,\n} from '@angular/core';\nimport { NG_VALUE_ACCESSOR } from '@angular/forms';\n\nimport { CurrencyMaskDirective } from 'ng2-currency-mask';\nimport { first } from 'rxjs/operators';\nimport { AlignmentOptions } from '@seniorsistemas/ng2-currency-mask';\nimport { LocaleService } from '@seniorsistemas/angular-components/locale';\n\n/**\n * @deprecated Should use 'AlignmentOptions' from @seniorsistemas/ng2-currency-mask instead\n */\nexport enum BignumberAlignmentOption {\n    RIGHT = 'right',\n    LEFT = 'left',\n}\n\n/**\n * @deprecated Utilize a diretiva `BigNumberInput` ao invés desta.\n * @description Diretiva de máscara para campos de entrada de números de grande precisão,\n * adaptada do `CurrencyMaskDirective`. Permite configurar precisão, escala,\n * separadores decimais e de milhar de acordo com o locale da aplicação.\n *\n * @example\n * ```html\n * <input sBignumberInput [precision]=\"15\" [scale]=\"2\" [allowNegative]=\"false\" />\n * ```\n *\n * @category Inputs\n */\n@Directive({\n    selector: 'input[sBignumberInput]',\n    providers: [\n        {\n            provide: NG_VALUE_ACCESSOR,\n            // NOSONAR - 'BignumberInputDirective' is deprecated.\n            useExisting: forwardRef(() => BignumberInputDirective),\n            multi: true,\n        },\n    ],\n})\nexport class BignumberInputDirective extends CurrencyMaskDirective implements OnInit, OnChanges {\n    /**\n     * @description Número total de dígitos significativos aceitos pelo campo.\n     *\n     * @default 15\n     */\n    @Input()\n    public precision = 15;\n\n    /**\n     * @description Número de casas decimais exibidas e aceitas pelo campo.\n     *\n     * @default 0\n     */\n    @Input()\n    public scale = 0;\n\n    /**\n     * @description Separador decimal personalizado. Quando não informado,\n     * utiliza o separador definido pelo locale ativo da aplicação.\n     */\n    @Input()\n    public decimalSeparator?: string;\n\n    /**\n     * @description Separador de milhar personalizado. Quando não informado,\n     * utiliza o separador definido pelo locale ativo da aplicação.\n     */\n    @Input()\n    public thousandsSeparator?: string;\n\n    /**\n     * @description Alinhamento do conteúdo do campo. Aceita `'left'` ou `'right'`.\n     *\n     * @default 'left'\n     */\n    @Input()\n    public alignTo: 'left' | 'right' = 'left';\n\n    /**\n     * @description Permite a entrada de valores negativos.\n     *\n     * @default true\n     */\n    @Input()\n    public allowNegative = true;\n\n    /**\n     * @description Texto de placeholder exibido quando o campo está vazio.\n     */\n    @Input()\n    @HostBinding('attr.placeholder')\n    public placeholder?: string;\n\n    @HostBinding('attr.maxLength')\n    public maxLength?: number;\n\n    /**\n     * @description Emitido quando uma colé (paste) é rejeitada pela validação do campo.\n     * Emite a string colada que foi recusada.\n     */\n    @Output()\n    public pasteRejected = new EventEmitter<string>();\n\n    private readonly regex = /\\d/;\n\n    //TODO: Ajuste no ng2-currency-mask para utilizar esse Código (aplicado na tarefa: SDS-305)\n    // private readonly _angularComponentsLocaleService: LocaleService;\n\n    // constructor(injector: Injector) {\n    // super(injector);\n\n    // this._angularComponentsLocaleService = injector.get(LocaleService);\n    // this.onLocaleService();\n    // }\n\n    //TODO:  Remover esse constructor\n    constructor(\n        _elementRef: ElementRef,\n        _keyValueDiffers: KeyValueDiffers,\n        private readonly localeService: LocaleService,\n    ) {\n        super(\n            { align: '', allowNegative: true, decimal: '', precision: 0, prefix: '', suffix: '', thousands: '' },\n            _elementRef,\n            _keyValueDiffers,\n        );\n\n        this.onLocaleService();\n    }\n\n    public override ngOnInit(): void {\n        this.updateVariables();\n\n        super.ngOnInit();\n    }\n\n    public ngOnChanges(changes: SimpleChanges) {\n        const placeholderChange = changes['placeholder']?.currentValue;\n        if (!placeholderChange && this.scale) {\n            this.placeholder = `0${this.decimalSeparator}${''.padEnd(this.scale, '0')}`;\n        }\n\n        const configChanged =\n            changes['scale'] ||\n            changes['decimalSeparator'] ||\n            changes['thousandsSeparator'] ||\n            changes['alignTo'] ||\n            changes['precision'] ||\n            changes['allowNegative'];\n\n        if (configChanged) {\n            this.updateVariables();\n        }\n    }\n\n    @HostListener('paste', ['$event'])\n    public onPaste(event: ClipboardEvent): void {\n        const pastedText = event.clipboardData?.getData('text') ?? '';\n        const numericText = pastedText\n            .replace(new RegExp(`\\\\${this.thousandsSeparator}`, 'g'), '')\n            .replace(this.decimalSeparator as string, '.');\n        const [intPart, decimalPart = ''] = numericText.split('.');\n        const totalDigits = intPart.replace(/\\D/g, '').length + decimalPart.slice(0, this.scale).length;\n\n        if (totalDigits > this.precision) {\n            event.preventDefault();\n            this.pasteRejected.emit(pastedText);\n        }\n    }\n\n    /**\n     * Update the options values according to the directive input values.\n     */\n    private updateVariables(): void {\n        this.options = {\n            align: this.alignTo,\n            decimal: this.decimalSeparator,\n            thousands: this.thousandsSeparator,\n            precision: this.scale || 0,\n            prefix: '',\n            suffix: '',\n            allowNegative: this.allowNegative,\n        };\n        this.calculateMaxLength();\n    }\n\n    /**\n     * Responsible to calculate the field maximum length considering the separators.\n     */\n    private calculateMaxLength(): void {\n        const decSepLength = this.decimalSeparator?.length ?? 0;\n        const thoSepLength = this.thousandsSeparator?.length ?? 0;\n\n        const DIGITS_GROUP_SIZE = 3;\n\n        let maxLength = this.precision + (this.scale ? decSepLength : 0);\n        maxLength += Math.ceil((this.precision - (this.scale ? this.scale : 0)) / DIGITS_GROUP_SIZE - 1) * thoSepLength;\n\n        this.maxLength = maxLength;\n    }\n\n    private onLocaleService(): void {\n        this.localeService\n            .getLocale()\n            .pipe(first())\n            .subscribe(() => {\n                this.decimalSeparator = this.decimalSeparator ?? this.localeService.getDecimalSeparator();\n                this.thousandsSeparator = this.thousandsSeparator ?? this.localeService.getGroupingSeparator();\n            });\n    }\n}\n\n"]}
243
+ }], ctorParameters: () => [] });
244
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bignumber-input.directive.js","sourceRoot":"","sources":["../../../../../projects/angular-components/bignumber-input/src/lib/bignumber-input/bignumber-input.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, forwardRef, input, model, OnInit, OnChanges, output, SimpleChanges } from '@angular/core';\nimport { NG_VALUE_ACCESSOR, AbstractControl, ValidationErrors } from '@angular/forms';\n\nimport { NumericMaskDirective } from '@seniorsistemas/angular-components/numeric-mask';\n\n/**\n * @deprecated Este enum não tem mais efeito. O alinhamento do campo é sempre `'right'`.\n */\nexport enum BignumberAlignmentOption {\n    RIGHT = 'right',\n    LEFT = 'left',\n}\n\n/**\n * @deprecated Utilize a diretiva `BigNumberInput` ao invés desta.\n * @description Diretiva de máscara para campos de entrada de números de grande precisão,\n * adaptada do `CurrencyMaskDirective`. Permite configurar precisão, escala,\n * separadores decimais e de milhar de acordo com o locale da aplicação.\n *\n * @example\n * ```html\n * <input sBignumberInput [precision]=\"15\" [scale]=\"2\" [allowNegative]=\"false\" />\n * ```\n *\n * @category Inputs\n */\n@Directive({\n    selector: 'input[sBignumberInput]',\n    host: {\n        '[attr.placeholder]': 'placeholder()',\n        '[attr.maxLength]': 'maxLength',\n    },\n    providers: [\n        {\n            provide: NG_VALUE_ACCESSOR,\n            // NOSONAR - 'BignumberInputDirective' is deprecated.\n            useExisting: forwardRef(() => BignumberInputDirective),\n            multi: true,\n        },\n    ],\n})\nexport class BignumberInputDirective extends NumericMaskDirective implements OnInit, OnChanges {\n    /**\n     * @description Número total de dígitos significativos aceitos pelo campo (parte inteira + parte decimal).\n     *\n     * Durante a digitação, o campo limita fisicamente o número de caracteres via `maxLength`.\n     * Ao colar um valor que ultrapasse o limite, o evento `pasteRejected` é emitido e a colagem é cancelada.\n     * Após sair do campo (blur) ou ao submeter o formulário, se o valor ultrapassar `precision`,\n     * o campo ficará inválido com o erro `exceedsPrecision: { precision, actual }`.\n     *\n     * @default 15\n     */\n    public precision = input(15);\n\n    /**\n     * @deprecated Use `minDecimalPlaces` e `maxDecimalPlaces` ao invés deste.\n     * @description Número de casas decimais exibidas e aceitas pelo campo.\n     * Quando informado, repassa o valor para `minDecimalPlaces` e `maxDecimalPlaces`.\n     *\n     * @default 0\n     */\n    public scale = input(0);\n\n    /**\n     * @description Número mínimo de casas decimais exibidas.\n     *\n     * @default 0\n     */\n    public override minDecimalPlaces = model(0);\n\n    /**\n     * @description Número máximo de casas decimais aceitas.\n     *\n     * @default 0\n     */\n    public override maxDecimalPlaces = model(0);\n\n    /**\n     * @description Separador decimal personalizado. Quando não informado,\n     * utiliza o separador definido pelo locale ativo da aplicação.\n     */\n    public override decimalSeparator = model(',');\n\n    /**\n     * @deprecated Use `thousandSeparator` ao invés deste.\n     * @description Separador de milhar personalizado. Recomenda-se migrar para `thousandSeparator`.\n     */\n    public thousandsSeparator = input<string | undefined>(undefined);\n\n    /**\n     * @description Separador de milhar personalizado. Quando não informado,\n     * utiliza o separador definido pelo locale ativo da aplicação.\n     */\n    public override thousandSeparator = model('.');\n\n    /**\n     * @deprecated Este input não tem mais efeito e pode ser removido.\n     * O alinhamento do campo é sempre `'right'`, conforme o comportamento padrão da máscara numérica.\n     */\n    public alignTo = input<'left' | 'right'>('right');\n\n    /**\n     * @description Permite a entrada de valores negativos.\n     *\n     * @default true\n     */\n    public override allowNegative = input(true);\n\n    /**\n     * @description Texto de placeholder exibido quando o campo está vazio.\n     */\n    public placeholder = model<string | undefined>(undefined);\n\n    public maxLength?: number;\n\n    /**\n     * @description Emitido quando uma colagem (paste) é rejeitada pela validação do campo.\n     * Emite a string colada que foi recusada.\n     */\n    public pasteRejected = output<string>();\n\n    private _decimalSeparatorExplicit = false;\n    private _thousandSeparatorExplicit = false;\n\n    constructor() {\n        super();\n    }\n\n    public override ngOnInit(): void {\n        super.ngOnInit();\n        this.updatePlaceholder();\n    }\n\n    // ngOnChanges handles all signal writes reactively, avoiding the need for\n    // effect() with allowSignalWrites. The decimalSeparator/thousandSeparator inputs\n    // have truthy defaults, so ngOnChanges is necessary to distinguish \"parent bound this\"\n    // from \"it's just the default value\".\n    public override ngOnChanges(changes: SimpleChanges): void {\n        if (changes['decimalSeparator'] && this.decimalSeparator()) {\n            this._decimalSeparatorExplicit = true;\n        }\n        if (changes['thousandSeparator'] && this.thousandSeparator()) {\n            this._thousandSeparatorExplicit = true;\n        }\n        if (changes['thousandsSeparator']) {\n            const val = this.thousandsSeparator();\n            if (val !== undefined) {\n                this._thousandSeparatorExplicit = true;\n                this.thousandSeparator.set(val);\n            }\n        }\n        if (changes['scale']) {\n            this.minDecimalPlaces.set(this.scale());\n            this.maxDecimalPlaces.set(this.scale());\n        }\n        if (\n            changes['scale'] ||\n            changes['decimalSeparator'] ||\n            changes['thousandSeparator'] ||\n            changes['thousandsSeparator'] ||\n            changes['precision']\n        ) {\n            this.calculateMaxLength();\n        }\n        if (changes['scale'] || changes['decimalSeparator']) {\n            this.updatePlaceholder();\n        }\n    }\n\n    /**\n     * Overrides base paste handler to add precision validation.\n     * If the pasted value exceeds `precision` digits, the paste is rejected.\n     */\n    public override onPaste(event: ClipboardEvent): void {\n        const pastedText = event.clipboardData?.getData('text') ?? '';\n        const effectiveThousandSep = this.thousandsSeparator() ?? this.thousandSeparator();\n        const numericText = pastedText\n            .replace(new RegExp(`\\\\${effectiveThousandSep}`, 'g'), '')\n            .replace(this.decimalSeparator(), '.');\n        const [intPart, decimalPart = ''] = numericText.split('.');\n        const totalDigits = intPart.replace(/\\D/g, '').length + decimalPart.slice(0, this.scale()).length;\n\n        if (totalDigits > this.precision()) {\n            event.preventDefault();\n            this.pasteRejected.emit(pastedText);\n            return;\n        }\n\n        super.onPaste(event);\n    }\n\n    /**\n     * Overrides locale-derived separators with explicitly set inputs, if provided.\n     * If no explicit separator is set, the locale-derived value is used.\n     */\n    protected override updateSeparators(): void {\n        const explicitDecimal = this._decimalSeparatorExplicit ? this.decimalSeparator() : undefined;\n        const explicitThousand = this._thousandSeparatorExplicit\n            ? (this.thousandsSeparator() ?? this.thousandSeparator())\n            : undefined;\n\n        super.updateSeparators();\n\n        if (explicitDecimal !== undefined) {\n            this.decimalSeparator.set(explicitDecimal);\n        }\n        if (explicitThousand !== undefined) {\n            this.thousandSeparator.set(explicitThousand);\n        }\n\n        this.calculateMaxLength();\n        this.updatePlaceholder();\n    }\n\n    /**\n     * Overrides base validator to add precision validation.\n     * If the total number of significant digits exceeds `precision`, returns an `exceedsPrecision` error.\n     */\n    public override validate(control: AbstractControl): ValidationErrors | null {\n        const baseErrors = super.validate(control);\n\n        if (!control.value || !this.precision()) {\n            return baseErrors ?? null;\n        }\n\n        const numericStr = String(control.value).replace('-', '');\n        const [intPart, decimalPart = ''] = numericStr.split('.');\n        const totalDigits = intPart.length + decimalPart.length;\n\n        if (totalDigits > this.precision()) {\n            return {\n                ...baseErrors,\n                exceedsPrecision: {\n                    precision: this.precision(),\n                    actual: totalDigits,\n                },\n            };\n        }\n\n        return baseErrors ?? null;\n    }\n\n    /**\n     * Responsible to calculate the field maximum length considering the separators.\n     */\n    private calculateMaxLength(): void {\n        const decSepLength = this.decimalSeparator()?.length ?? 0;\n        const effectiveThousandSep = this.thousandsSeparator() ?? this.thousandSeparator();\n        const thoSepLength = effectiveThousandSep?.length ?? 0;\n\n        const DIGITS_GROUP_SIZE = 3;\n\n        let maxLength = this.precision() + (this.scale() ? decSepLength : 0);\n        maxLength +=\n            Math.ceil((this.precision() - (this.scale() ? this.scale() : 0)) / DIGITS_GROUP_SIZE - 1) * thoSepLength;\n\n        this.maxLength = maxLength;\n    }\n\n    private updatePlaceholder(): void {\n        const scale = this.scale();\n        if (!this.placeholder() && scale) {\n            this.placeholder.set(`0${this.decimalSeparator()}${''.padEnd(scale, '0')}`);\n        }\n    }\n}\n\n"]}