@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.
- package/bignumber-input/lib/bignumber-input/bignumber-input.directive.d.ts +58 -25
- package/dynamic-form/dynamic-form/components/lookup/lookup.component.d.ts +8 -8
- package/esm2022/bignumber-input/lib/bignumber-input/bignumber-input.directive.mjs +140 -99
- package/esm2022/dynamic-form/dynamic-form/components/lookup/lookup.component.mjs +12 -12
- package/esm2022/dynamic-form/dynamic-form/form-field/fields/bignumber/bignumber-field.component.mjs +3 -3
- package/esm2022/dynamic-form/dynamic-form/form-field/fields/currency/currency-field.component.mjs +3 -3
- package/esm2022/dynamic-form/dynamic-form/form-field/fields/number/number-field.component.mjs +3 -3
- package/esm2022/inline-edit/lib/inline-edit/components/fields/inline-edit-number/inline-edit-number.component.mjs +1 -1
- package/esm2022/inline-edit/lib/inline-edit/fields/inline-edit-number-field.mjs +2 -3
- package/esm2022/locale/lib/locale/apply-localized-mask.mjs +19 -0
- package/esm2022/locale/lib/locale/pipes/localized-bignumber-impure.pipe.mjs +1 -1
- package/esm2022/locale/lib/locale/pipes/localized-bignumber.pipe.mjs +5 -4
- package/esm2022/locale/public-api.mjs +2 -1
- package/esm2022/number-input/lib/number-input/number-input.directive.mjs +12 -148
- package/esm2022/number-input/lib/number-input/number-input.module.mjs +5 -5
- package/esm2022/numeric-mask/lib/numeric-mask/numeric-mask.directive.mjs +32 -35
- package/esm2022/table/lib/table/table-column/table-columns.component.mjs +20 -4
- package/esm2022/table/public-api.mjs +2 -1
- package/fesm2022/seniorsistemas-angular-components-bignumber-input.mjs +139 -98
- package/fesm2022/seniorsistemas-angular-components-bignumber-input.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-dynamic-form.mjs +19 -19
- package/fesm2022/seniorsistemas-angular-components-dynamic-form.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-inline-edit.mjs +2 -3
- package/fesm2022/seniorsistemas-angular-components-inline-edit.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-locale.mjs +22 -4
- package/fesm2022/seniorsistemas-angular-components-locale.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-number-input.mjs +14 -151
- package/fesm2022/seniorsistemas-angular-components-number-input.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-numeric-mask.mjs +31 -34
- package/fesm2022/seniorsistemas-angular-components-numeric-mask.mjs.map +1 -1
- package/fesm2022/seniorsistemas-angular-components-table.mjs +21 -6
- package/fesm2022/seniorsistemas-angular-components-table.mjs.map +1 -1
- package/inline-edit/lib/inline-edit/fields/inline-edit-number-field.d.ts +2 -3
- package/locale/lib/locale/apply-localized-mask.d.ts +9 -0
- package/locale/lib/locale/pipes/localized-bignumber-impure.pipe.d.ts +2 -3
- package/locale/lib/locale/pipes/localized-bignumber.pipe.d.ts +3 -2
- package/locale/public-api.d.ts +1 -0
- package/number-input/lib/number-input/number-input.directive.d.ts +7 -47
- package/number-input/lib/number-input/number-input.module.d.ts +2 -2
- package/numeric-mask/lib/numeric-mask/numeric-mask.directive.d.ts +8 -9
- package/package.json +1 -1
- package/table/public-api.d.ts +1 -0
|
@@ -1,9 +1,9 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import {
|
|
3
|
-
import {
|
|
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
|
|
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
|
|
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
|
-
|
|
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
|
|
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
|
-
|
|
71
|
+
thousandSeparator: import("@angular/core").ModelSignal<string>;
|
|
49
72
|
/**
|
|
50
|
-
* @
|
|
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:
|
|
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
|
|
86
|
+
placeholder: import("@angular/core").ModelSignal<string | undefined>;
|
|
65
87
|
maxLength?: number;
|
|
66
88
|
/**
|
|
67
|
-
* @description Emitido quando uma
|
|
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:
|
|
71
|
-
private
|
|
72
|
-
|
|
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
|
-
*
|
|
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
|
-
|
|
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
|
|
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 {
|
|
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):
|
|
132
|
-
getMoneyMaskConfig(col: CurrencyFieldConfig):
|
|
133
|
-
getIntegerMaskConfig(col: BignumberFieldConfig | NumberFieldConfig):
|
|
134
|
-
getNumberMaskConfig(col: NumberFieldConfig | BignumberFieldConfig):
|
|
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,
|
|
1
|
+
import { Directive, forwardRef, input, model, output } from '@angular/core';
|
|
2
2
|
import { NG_VALUE_ACCESSOR } from '@angular/forms';
|
|
3
|
-
import {
|
|
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
|
|
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
|
|
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
|
-
|
|
72
|
+
thousandSeparator = model('.');
|
|
52
73
|
/**
|
|
53
|
-
* @
|
|
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 = '
|
|
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
|
|
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 =
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
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
|
-
|
|
94
|
-
|
|
95
|
-
|
|
108
|
+
if (changes['decimalSeparator'] && this.decimalSeparator()) {
|
|
109
|
+
this._decimalSeparatorExplicit = true;
|
|
110
|
+
}
|
|
111
|
+
if (changes['thousandSeparator'] && this.thousandSeparator()) {
|
|
112
|
+
this._thousandSeparatorExplicit = true;
|
|
96
113
|
}
|
|
97
|
-
|
|
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['
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
if (
|
|
104
|
-
this.
|
|
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(`\\${
|
|
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
|
-
*
|
|
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
|
-
|
|
123
|
-
this.
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
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
|
|
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 +=
|
|
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
|
-
|
|
146
|
-
this.
|
|
147
|
-
|
|
148
|
-
.
|
|
149
|
-
|
|
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: [
|
|
155
|
-
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "
|
|
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: () => [
|
|
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"]}
|