@schenkerjon/ng-govbr-tw 0.2.1 → 0.3.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.
@@ -0,0 +1,270 @@
1
+ import { Component, Input, Optional, Self, ViewEncapsulation, } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "@angular/forms";
4
+ export class BrInputCnpjComponent {
5
+ static { this.nextId = 0; }
6
+ static { this.defaultErrorMessages = {
7
+ required: 'Campo obrigatório',
8
+ cnpjInvalid: 'CNPJ inválido',
9
+ cnpjIncomplete: 'CNPJ incompleto',
10
+ }; }
11
+ constructor(ngControl) {
12
+ this.ngControl = ngControl;
13
+ this.placeholder = '00.000.000/0000-00';
14
+ this.required = false;
15
+ this.density = 'medium';
16
+ /** Mensagens customizadas para erros */
17
+ this.errorMessages = {};
18
+ this.displayValue = '';
19
+ this.isDisabled = false;
20
+ this.rawValue = '';
21
+ this.inputId = `br-input-cnpj-${BrInputCnpjComponent.nextId++}`;
22
+ this.onChange = () => { };
23
+ this.onTouched = () => { };
24
+ this.onValidatorChange = () => { };
25
+ if (this.ngControl) {
26
+ this.ngControl.valueAccessor = this;
27
+ }
28
+ }
29
+ get displayError() {
30
+ if (this.error)
31
+ return this.error;
32
+ if (!this.ngControl?.control)
33
+ return undefined;
34
+ const control = this.ngControl.control;
35
+ if (!control.errors || (!control.touched && !control.dirty))
36
+ return undefined;
37
+ return this.getFirstErrorMessage(control.errors);
38
+ }
39
+ getFirstErrorMessage(errors) {
40
+ for (const key of Object.keys(errors)) {
41
+ if (this.errorMessages[key])
42
+ return this.errorMessages[key];
43
+ const defaultMsg = BrInputCnpjComponent.defaultErrorMessages[key];
44
+ if (defaultMsg)
45
+ return defaultMsg;
46
+ }
47
+ return 'Campo inválido';
48
+ }
49
+ get inputClasses() {
50
+ const densityHeights = {
51
+ small: 'h-[var(--govbr-input-small)] text-govbr-sm',
52
+ medium: 'h-[var(--govbr-input-medium)] text-govbr-base',
53
+ large: 'h-[var(--govbr-input-large)] text-govbr-lg',
54
+ };
55
+ const base = `w-full px-4 pr-10 font-rawline govbr-input ${densityHeights[this.density]}`;
56
+ const statusCls = this.displayError ? 'govbr-input--danger' : '';
57
+ const disabledCls = this.isDisabled
58
+ ? 'bg-govbr-gray-2 text-govbr-gray-40 cursor-not-allowed'
59
+ : 'bg-white text-govbr-gray-80';
60
+ return [base, statusCls, disabledCls].filter(Boolean).join(' ');
61
+ }
62
+ onInput(event) {
63
+ const input = event.target;
64
+ const digits = input.value.replace(/\D/g, '').slice(0, 14);
65
+ this.rawValue = digits;
66
+ this.displayValue = this.formatCnpj(digits);
67
+ this.onChange(digits);
68
+ setTimeout(() => {
69
+ input.value = this.displayValue;
70
+ });
71
+ }
72
+ onBlur() {
73
+ this.onTouched();
74
+ }
75
+ formatCnpj(digits) {
76
+ if (!digits)
77
+ return '';
78
+ let formatted = digits;
79
+ // 00.000.000/0000-00
80
+ if (digits.length > 2) {
81
+ formatted = digits.slice(0, 2) + '.' + digits.slice(2);
82
+ }
83
+ if (digits.length > 5) {
84
+ formatted = formatted.slice(0, 6) + '.' + formatted.slice(6);
85
+ }
86
+ if (digits.length > 8) {
87
+ formatted = formatted.slice(0, 10) + '/' + formatted.slice(10);
88
+ }
89
+ if (digits.length > 12) {
90
+ formatted = formatted.slice(0, 15) + '-' + formatted.slice(15);
91
+ }
92
+ return formatted;
93
+ }
94
+ // ControlValueAccessor
95
+ writeValue(value) {
96
+ const digits = (value || '').replace(/\D/g, '').slice(0, 14);
97
+ this.rawValue = digits;
98
+ this.displayValue = this.formatCnpj(digits);
99
+ }
100
+ registerOnChange(fn) {
101
+ this.onChange = fn;
102
+ }
103
+ registerOnTouched(fn) {
104
+ this.onTouched = fn;
105
+ }
106
+ setDisabledState(isDisabled) {
107
+ this.isDisabled = isDisabled;
108
+ }
109
+ // Validator
110
+ validate(control) {
111
+ const value = control.value;
112
+ if (!value) {
113
+ return this.required ? { required: true } : null;
114
+ }
115
+ const digits = value.replace(/\D/g, '');
116
+ if (digits.length < 14) {
117
+ return { cnpjIncomplete: true };
118
+ }
119
+ if (!this.isValidCnpj(digits)) {
120
+ return { cnpjInvalid: true };
121
+ }
122
+ return null;
123
+ }
124
+ registerOnValidatorChange(fn) {
125
+ this.onValidatorChange = fn;
126
+ }
127
+ isValidCnpj(cnpj) {
128
+ const digits = cnpj.replace(/\D/g, '');
129
+ if (digits.length !== 14)
130
+ return false;
131
+ // Verifica se todos os dígitos são iguais
132
+ if (/^(\d)\1+$/.test(digits))
133
+ return false;
134
+ // Calcula primeiro dígito verificador
135
+ const weights1 = [5, 4, 3, 2, 9, 8, 7, 6, 5, 4, 3, 2];
136
+ let sum = 0;
137
+ for (let i = 0; i < 12; i++) {
138
+ sum += parseInt(digits[i], 10) * weights1[i];
139
+ }
140
+ let remainder = sum % 11;
141
+ const digit1 = remainder < 2 ? 0 : 11 - remainder;
142
+ if (digit1 !== parseInt(digits[12], 10))
143
+ return false;
144
+ // Calcula segundo dígito verificador
145
+ const weights2 = [6, 5, 4, 3, 2, 9, 8, 7, 6, 5, 4, 3, 2];
146
+ sum = 0;
147
+ for (let i = 0; i < 13; i++) {
148
+ sum += parseInt(digits[i], 10) * weights2[i];
149
+ }
150
+ remainder = sum % 11;
151
+ const digit2 = remainder < 2 ? 0 : 11 - remainder;
152
+ if (digit2 !== parseInt(digits[13], 10))
153
+ return false;
154
+ return true;
155
+ }
156
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BrInputCnpjComponent, deps: [{ token: i1.NgControl, optional: true, self: true }], target: i0.ɵɵFactoryTarget.Component }); }
157
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: BrInputCnpjComponent, isStandalone: true, selector: "br-input-cnpj", inputs: { label: "label", placeholder: "placeholder", required: "required", helperText: "helperText", error: "error", density: "density", errorMessages: "errorMessages" }, ngImport: i0, template: `
158
+ <div class="mb-4 font-rawline">
159
+ @if (label) {
160
+ <label
161
+ [for]="inputId"
162
+ class="block text-govbr-sm font-semibold text-govbr-gray-80 mb-1"
163
+ >
164
+ {{ label }}
165
+ @if (required) {
166
+ <span class="text-govbr-danger ml-0.5">*</span>
167
+ }
168
+ </label>
169
+ }
170
+
171
+ @if (helperText && !displayError) {
172
+ <span class="block text-govbr-xs text-govbr-gray-60 mb-1">
173
+ {{ helperText }}
174
+ </span>
175
+ }
176
+
177
+ <div class="relative">
178
+ <input
179
+ [id]="inputId"
180
+ type="text"
181
+ inputmode="numeric"
182
+ [placeholder]="placeholder"
183
+ [disabled]="isDisabled"
184
+ [class]="inputClasses"
185
+ [value]="displayValue"
186
+ (input)="onInput($event)"
187
+ (blur)="onBlur()"
188
+ maxlength="18"
189
+ />
190
+ <i class="fa-solid fa-building absolute right-3 top-1/2 -translate-y-1/2 text-govbr-gray-40 pointer-events-none"></i>
191
+ </div>
192
+
193
+ @if (displayError) {
194
+ <span class="flex items-center mt-1 text-govbr-sm text-govbr-danger">
195
+ <i class="fa-solid fa-times-circle mr-1"></i>{{ displayError }}
196
+ </span>
197
+ }
198
+ </div>
199
+ `, isInline: true, encapsulation: i0.ViewEncapsulation.None }); }
200
+ }
201
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BrInputCnpjComponent, decorators: [{
202
+ type: Component,
203
+ args: [{
204
+ selector: 'br-input-cnpj',
205
+ standalone: true,
206
+ encapsulation: ViewEncapsulation.None,
207
+ template: `
208
+ <div class="mb-4 font-rawline">
209
+ @if (label) {
210
+ <label
211
+ [for]="inputId"
212
+ class="block text-govbr-sm font-semibold text-govbr-gray-80 mb-1"
213
+ >
214
+ {{ label }}
215
+ @if (required) {
216
+ <span class="text-govbr-danger ml-0.5">*</span>
217
+ }
218
+ </label>
219
+ }
220
+
221
+ @if (helperText && !displayError) {
222
+ <span class="block text-govbr-xs text-govbr-gray-60 mb-1">
223
+ {{ helperText }}
224
+ </span>
225
+ }
226
+
227
+ <div class="relative">
228
+ <input
229
+ [id]="inputId"
230
+ type="text"
231
+ inputmode="numeric"
232
+ [placeholder]="placeholder"
233
+ [disabled]="isDisabled"
234
+ [class]="inputClasses"
235
+ [value]="displayValue"
236
+ (input)="onInput($event)"
237
+ (blur)="onBlur()"
238
+ maxlength="18"
239
+ />
240
+ <i class="fa-solid fa-building absolute right-3 top-1/2 -translate-y-1/2 text-govbr-gray-40 pointer-events-none"></i>
241
+ </div>
242
+
243
+ @if (displayError) {
244
+ <span class="flex items-center mt-1 text-govbr-sm text-govbr-danger">
245
+ <i class="fa-solid fa-times-circle mr-1"></i>{{ displayError }}
246
+ </span>
247
+ }
248
+ </div>
249
+ `,
250
+ }]
251
+ }], ctorParameters: () => [{ type: i1.NgControl, decorators: [{
252
+ type: Self
253
+ }, {
254
+ type: Optional
255
+ }] }], propDecorators: { label: [{
256
+ type: Input
257
+ }], placeholder: [{
258
+ type: Input
259
+ }], required: [{
260
+ type: Input
261
+ }], helperText: [{
262
+ type: Input
263
+ }], error: [{
264
+ type: Input
265
+ }], density: [{
266
+ type: Input
267
+ }], errorMessages: [{
268
+ type: Input
269
+ }] } });
270
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"br-input-cnpj.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-govbr-tw/src/lib/components/input-cnpj/br-input-cnpj.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  Input,\r\n  Optional,\r\n  Self,\r\n  ViewEncapsulation,\r\n} from '@angular/core';\r\n\r\nimport { ControlValueAccessor, NgControl, ValidationErrors, Validator, AbstractControl } from '@angular/forms';\r\n\r\n@Component({\r\n  selector: 'br-input-cnpj',\r\n  standalone: true,\r\n  encapsulation: ViewEncapsulation.None,\r\n  template: `\r\n    <div class=\"mb-4 font-rawline\">\r\n      @if (label) {\r\n        <label\r\n          [for]=\"inputId\"\r\n          class=\"block text-govbr-sm font-semibold text-govbr-gray-80 mb-1\"\r\n        >\r\n          {{ label }}\r\n          @if (required) {\r\n            <span class=\"text-govbr-danger ml-0.5\">*</span>\r\n          }\r\n        </label>\r\n      }\r\n\r\n      @if (helperText && !displayError) {\r\n        <span class=\"block text-govbr-xs text-govbr-gray-60 mb-1\">\r\n          {{ helperText }}\r\n        </span>\r\n      }\r\n\r\n      <div class=\"relative\">\r\n        <input\r\n          [id]=\"inputId\"\r\n          type=\"text\"\r\n          inputmode=\"numeric\"\r\n          [placeholder]=\"placeholder\"\r\n          [disabled]=\"isDisabled\"\r\n          [class]=\"inputClasses\"\r\n          [value]=\"displayValue\"\r\n          (input)=\"onInput($event)\"\r\n          (blur)=\"onBlur()\"\r\n          maxlength=\"18\"\r\n        />\r\n        <i class=\"fa-solid fa-building absolute right-3 top-1/2 -translate-y-1/2 text-govbr-gray-40 pointer-events-none\"></i>\r\n      </div>\r\n\r\n      @if (displayError) {\r\n        <span class=\"flex items-center mt-1 text-govbr-sm text-govbr-danger\">\r\n          <i class=\"fa-solid fa-times-circle mr-1\"></i>{{ displayError }}\r\n        </span>\r\n      }\r\n    </div>\r\n  `,\r\n})\r\nexport class BrInputCnpjComponent implements ControlValueAccessor, Validator {\r\n  @Input() label?: string;\r\n  @Input() placeholder = '00.000.000/0000-00';\r\n  @Input() required = false;\r\n  @Input() helperText?: string;\r\n  @Input() error?: string;\r\n  @Input() density: 'small' | 'medium' | 'large' = 'medium';\r\n\r\n  /** Mensagens customizadas para erros */\r\n  @Input() errorMessages: Record<string, string> = {};\r\n\r\n  displayValue = '';\r\n  isDisabled = false;\r\n  private rawValue = '';\r\n\r\n  private static nextId = 0;\r\n  inputId = `br-input-cnpj-${BrInputCnpjComponent.nextId++}`;\r\n\r\n  onChange: (value: string) => void = () => {};\r\n  onTouched: () => void = () => {};\r\n  onValidatorChange: () => void = () => {};\r\n\r\n  private static readonly defaultErrorMessages: Record<string, string> = {\r\n    required: 'Campo obrigatório',\r\n    cnpjInvalid: 'CNPJ inválido',\r\n    cnpjIncomplete: 'CNPJ incompleto',\r\n  };\r\n\r\n  constructor(@Self() @Optional() private ngControl: NgControl) {\r\n    if (this.ngControl) {\r\n      this.ngControl.valueAccessor = this;\r\n    }\r\n  }\r\n\r\n  get displayError(): string | undefined {\r\n    if (this.error) return this.error;\r\n\r\n    if (!this.ngControl?.control) return undefined;\r\n\r\n    const control = this.ngControl.control;\r\n    if (!control.errors || (!control.touched && !control.dirty)) return undefined;\r\n\r\n    return this.getFirstErrorMessage(control.errors);\r\n  }\r\n\r\n  private getFirstErrorMessage(errors: ValidationErrors): string {\r\n    for (const key of Object.keys(errors)) {\r\n      if (this.errorMessages[key]) return this.errorMessages[key];\r\n      const defaultMsg = BrInputCnpjComponent.defaultErrorMessages[key];\r\n      if (defaultMsg) return defaultMsg;\r\n    }\r\n    return 'Campo inválido';\r\n  }\r\n\r\n  get inputClasses(): string {\r\n    const densityHeights = {\r\n      small: 'h-[var(--govbr-input-small)] text-govbr-sm',\r\n      medium: 'h-[var(--govbr-input-medium)] text-govbr-base',\r\n      large: 'h-[var(--govbr-input-large)] text-govbr-lg',\r\n    } as const;\r\n\r\n    const base = `w-full px-4 pr-10 font-rawline govbr-input ${densityHeights[this.density]}`;\r\n    const statusCls = this.displayError ? 'govbr-input--danger' : '';\r\n    const disabledCls = this.isDisabled\r\n      ? 'bg-govbr-gray-2 text-govbr-gray-40 cursor-not-allowed'\r\n      : 'bg-white text-govbr-gray-80';\r\n\r\n    return [base, statusCls, disabledCls].filter(Boolean).join(' ');\r\n  }\r\n\r\n  onInput(event: Event): void {\r\n    const input = event.target as HTMLInputElement;\r\n    const digits = input.value.replace(/\\D/g, '').slice(0, 14);\r\n    this.rawValue = digits;\r\n    this.displayValue = this.formatCnpj(digits);\r\n    this.onChange(digits);\r\n\r\n    setTimeout(() => {\r\n      input.value = this.displayValue;\r\n    });\r\n  }\r\n\r\n  onBlur(): void {\r\n    this.onTouched();\r\n  }\r\n\r\n  private formatCnpj(digits: string): string {\r\n    if (!digits) return '';\r\n    let formatted = digits;\r\n    // 00.000.000/0000-00\r\n    if (digits.length > 2) {\r\n      formatted = digits.slice(0, 2) + '.' + digits.slice(2);\r\n    }\r\n    if (digits.length > 5) {\r\n      formatted = formatted.slice(0, 6) + '.' + formatted.slice(6);\r\n    }\r\n    if (digits.length > 8) {\r\n      formatted = formatted.slice(0, 10) + '/' + formatted.slice(10);\r\n    }\r\n    if (digits.length > 12) {\r\n      formatted = formatted.slice(0, 15) + '-' + formatted.slice(15);\r\n    }\r\n    return formatted;\r\n  }\r\n\r\n  // ControlValueAccessor\r\n  writeValue(value: string): void {\r\n    const digits = (value || '').replace(/\\D/g, '').slice(0, 14);\r\n    this.rawValue = digits;\r\n    this.displayValue = this.formatCnpj(digits);\r\n  }\r\n\r\n  registerOnChange(fn: (value: string) => void): void {\r\n    this.onChange = fn;\r\n  }\r\n\r\n  registerOnTouched(fn: () => void): void {\r\n    this.onTouched = fn;\r\n  }\r\n\r\n  setDisabledState(isDisabled: boolean): void {\r\n    this.isDisabled = isDisabled;\r\n  }\r\n\r\n  // Validator\r\n  validate(control: AbstractControl): ValidationErrors | null {\r\n    const value = control.value as string;\r\n    if (!value) {\r\n      return this.required ? { required: true } : null;\r\n    }\r\n\r\n    const digits = value.replace(/\\D/g, '');\r\n    if (digits.length < 14) {\r\n      return { cnpjIncomplete: true };\r\n    }\r\n\r\n    if (!this.isValidCnpj(digits)) {\r\n      return { cnpjInvalid: true };\r\n    }\r\n\r\n    return null;\r\n  }\r\n\r\n  registerOnValidatorChange(fn: () => void): void {\r\n    this.onValidatorChange = fn;\r\n  }\r\n\r\n  private isValidCnpj(cnpj: string): boolean {\r\n    const digits = cnpj.replace(/\\D/g, '');\r\n\r\n    if (digits.length !== 14) return false;\r\n\r\n    // Verifica se todos os dígitos são iguais\r\n    if (/^(\\d)\\1+$/.test(digits)) return false;\r\n\r\n    // Calcula primeiro dígito verificador\r\n    const weights1 = [5, 4, 3, 2, 9, 8, 7, 6, 5, 4, 3, 2];\r\n    let sum = 0;\r\n    for (let i = 0; i < 12; i++) {\r\n      sum += parseInt(digits[i], 10) * weights1[i];\r\n    }\r\n    let remainder = sum % 11;\r\n    const digit1 = remainder < 2 ? 0 : 11 - remainder;\r\n    if (digit1 !== parseInt(digits[12], 10)) return false;\r\n\r\n    // Calcula segundo dígito verificador\r\n    const weights2 = [6, 5, 4, 3, 2, 9, 8, 7, 6, 5, 4, 3, 2];\r\n    sum = 0;\r\n    for (let i = 0; i < 13; i++) {\r\n      sum += parseInt(digits[i], 10) * weights2[i];\r\n    }\r\n    remainder = sum % 11;\r\n    const digit2 = remainder < 2 ? 0 : 11 - remainder;\r\n    if (digit2 !== parseInt(digits[13], 10)) return false;\r\n\r\n    return true;\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,268 @@
1
+ import { Component, Input, Optional, Self, ViewEncapsulation, } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "@angular/forms";
4
+ export class BrInputCpfComponent {
5
+ static { this.nextId = 0; }
6
+ static { this.defaultErrorMessages = {
7
+ required: 'Campo obrigatório',
8
+ cpfInvalid: 'CPF inválido',
9
+ cpfIncomplete: 'CPF incompleto',
10
+ }; }
11
+ constructor(ngControl) {
12
+ this.ngControl = ngControl;
13
+ this.placeholder = '000.000.000-00';
14
+ this.required = false;
15
+ this.density = 'medium';
16
+ /** Mensagens customizadas para erros */
17
+ this.errorMessages = {};
18
+ this.displayValue = '';
19
+ this.isDisabled = false;
20
+ this.rawValue = '';
21
+ this.inputId = `br-input-cpf-${BrInputCpfComponent.nextId++}`;
22
+ this.onChange = () => { };
23
+ this.onTouched = () => { };
24
+ this.onValidatorChange = () => { };
25
+ if (this.ngControl) {
26
+ this.ngControl.valueAccessor = this;
27
+ }
28
+ }
29
+ get displayError() {
30
+ if (this.error)
31
+ return this.error;
32
+ if (!this.ngControl?.control)
33
+ return undefined;
34
+ const control = this.ngControl.control;
35
+ if (!control.errors || (!control.touched && !control.dirty))
36
+ return undefined;
37
+ return this.getFirstErrorMessage(control.errors);
38
+ }
39
+ getFirstErrorMessage(errors) {
40
+ for (const key of Object.keys(errors)) {
41
+ if (this.errorMessages[key])
42
+ return this.errorMessages[key];
43
+ const defaultMsg = BrInputCpfComponent.defaultErrorMessages[key];
44
+ if (defaultMsg)
45
+ return defaultMsg;
46
+ }
47
+ return 'Campo inválido';
48
+ }
49
+ get inputClasses() {
50
+ const densityHeights = {
51
+ small: 'h-[var(--govbr-input-small)] text-govbr-sm',
52
+ medium: 'h-[var(--govbr-input-medium)] text-govbr-base',
53
+ large: 'h-[var(--govbr-input-large)] text-govbr-lg',
54
+ };
55
+ const base = `w-full px-4 pr-10 font-rawline govbr-input ${densityHeights[this.density]}`;
56
+ const statusCls = this.displayError ? 'govbr-input--danger' : '';
57
+ const disabledCls = this.isDisabled
58
+ ? 'bg-govbr-gray-2 text-govbr-gray-40 cursor-not-allowed'
59
+ : 'bg-white text-govbr-gray-80';
60
+ return [base, statusCls, disabledCls].filter(Boolean).join(' ');
61
+ }
62
+ onInput(event) {
63
+ const input = event.target;
64
+ const digits = input.value.replace(/\D/g, '').slice(0, 11);
65
+ this.rawValue = digits;
66
+ this.displayValue = this.formatCpf(digits);
67
+ this.onChange(digits);
68
+ // Reposition cursor
69
+ setTimeout(() => {
70
+ input.value = this.displayValue;
71
+ });
72
+ }
73
+ onBlur() {
74
+ this.onTouched();
75
+ }
76
+ formatCpf(digits) {
77
+ if (!digits)
78
+ return '';
79
+ let formatted = digits;
80
+ if (digits.length > 3) {
81
+ formatted = digits.slice(0, 3) + '.' + digits.slice(3);
82
+ }
83
+ if (digits.length > 6) {
84
+ formatted = formatted.slice(0, 7) + '.' + formatted.slice(7);
85
+ }
86
+ if (digits.length > 9) {
87
+ formatted = formatted.slice(0, 11) + '-' + formatted.slice(11);
88
+ }
89
+ return formatted;
90
+ }
91
+ // ControlValueAccessor
92
+ writeValue(value) {
93
+ const digits = (value || '').replace(/\D/g, '').slice(0, 11);
94
+ this.rawValue = digits;
95
+ this.displayValue = this.formatCpf(digits);
96
+ }
97
+ registerOnChange(fn) {
98
+ this.onChange = fn;
99
+ }
100
+ registerOnTouched(fn) {
101
+ this.onTouched = fn;
102
+ }
103
+ setDisabledState(isDisabled) {
104
+ this.isDisabled = isDisabled;
105
+ }
106
+ // Validator
107
+ validate(control) {
108
+ const value = control.value;
109
+ if (!value) {
110
+ return this.required ? { required: true } : null;
111
+ }
112
+ const digits = value.replace(/\D/g, '');
113
+ if (digits.length < 11) {
114
+ return { cpfIncomplete: true };
115
+ }
116
+ if (!this.isValidCpf(digits)) {
117
+ return { cpfInvalid: true };
118
+ }
119
+ return null;
120
+ }
121
+ registerOnValidatorChange(fn) {
122
+ this.onValidatorChange = fn;
123
+ }
124
+ isValidCpf(cpf) {
125
+ // Remove caracteres não numéricos
126
+ const digits = cpf.replace(/\D/g, '');
127
+ if (digits.length !== 11)
128
+ return false;
129
+ // Verifica se todos os dígitos são iguais
130
+ if (/^(\d)\1+$/.test(digits))
131
+ return false;
132
+ // Calcula primeiro dígito verificador
133
+ let sum = 0;
134
+ for (let i = 0; i < 9; i++) {
135
+ sum += parseInt(digits[i], 10) * (10 - i);
136
+ }
137
+ let remainder = (sum * 10) % 11;
138
+ if (remainder === 10 || remainder === 11)
139
+ remainder = 0;
140
+ if (remainder !== parseInt(digits[9], 10))
141
+ return false;
142
+ // Calcula segundo dígito verificador
143
+ sum = 0;
144
+ for (let i = 0; i < 10; i++) {
145
+ sum += parseInt(digits[i], 10) * (11 - i);
146
+ }
147
+ remainder = (sum * 10) % 11;
148
+ if (remainder === 10 || remainder === 11)
149
+ remainder = 0;
150
+ if (remainder !== parseInt(digits[10], 10))
151
+ return false;
152
+ return true;
153
+ }
154
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BrInputCpfComponent, deps: [{ token: i1.NgControl, optional: true, self: true }], target: i0.ɵɵFactoryTarget.Component }); }
155
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: BrInputCpfComponent, isStandalone: true, selector: "br-input-cpf", inputs: { label: "label", placeholder: "placeholder", required: "required", helperText: "helperText", error: "error", density: "density", errorMessages: "errorMessages" }, ngImport: i0, template: `
156
+ <div class="mb-4 font-rawline">
157
+ @if (label) {
158
+ <label
159
+ [for]="inputId"
160
+ class="block text-govbr-sm font-semibold text-govbr-gray-80 mb-1"
161
+ >
162
+ {{ label }}
163
+ @if (required) {
164
+ <span class="text-govbr-danger ml-0.5">*</span>
165
+ }
166
+ </label>
167
+ }
168
+
169
+ @if (helperText && !displayError) {
170
+ <span class="block text-govbr-xs text-govbr-gray-60 mb-1">
171
+ {{ helperText }}
172
+ </span>
173
+ }
174
+
175
+ <div class="relative">
176
+ <input
177
+ [id]="inputId"
178
+ type="text"
179
+ inputmode="numeric"
180
+ [placeholder]="placeholder"
181
+ [disabled]="isDisabled"
182
+ [class]="inputClasses"
183
+ [value]="displayValue"
184
+ (input)="onInput($event)"
185
+ (blur)="onBlur()"
186
+ maxlength="14"
187
+ />
188
+ <i class="fa-solid fa-id-card absolute right-3 top-1/2 -translate-y-1/2 text-govbr-gray-40 pointer-events-none"></i>
189
+ </div>
190
+
191
+ @if (displayError) {
192
+ <span class="flex items-center mt-1 text-govbr-sm text-govbr-danger">
193
+ <i class="fa-solid fa-times-circle mr-1"></i>{{ displayError }}
194
+ </span>
195
+ }
196
+ </div>
197
+ `, isInline: true, encapsulation: i0.ViewEncapsulation.None }); }
198
+ }
199
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BrInputCpfComponent, decorators: [{
200
+ type: Component,
201
+ args: [{
202
+ selector: 'br-input-cpf',
203
+ standalone: true,
204
+ encapsulation: ViewEncapsulation.None,
205
+ template: `
206
+ <div class="mb-4 font-rawline">
207
+ @if (label) {
208
+ <label
209
+ [for]="inputId"
210
+ class="block text-govbr-sm font-semibold text-govbr-gray-80 mb-1"
211
+ >
212
+ {{ label }}
213
+ @if (required) {
214
+ <span class="text-govbr-danger ml-0.5">*</span>
215
+ }
216
+ </label>
217
+ }
218
+
219
+ @if (helperText && !displayError) {
220
+ <span class="block text-govbr-xs text-govbr-gray-60 mb-1">
221
+ {{ helperText }}
222
+ </span>
223
+ }
224
+
225
+ <div class="relative">
226
+ <input
227
+ [id]="inputId"
228
+ type="text"
229
+ inputmode="numeric"
230
+ [placeholder]="placeholder"
231
+ [disabled]="isDisabled"
232
+ [class]="inputClasses"
233
+ [value]="displayValue"
234
+ (input)="onInput($event)"
235
+ (blur)="onBlur()"
236
+ maxlength="14"
237
+ />
238
+ <i class="fa-solid fa-id-card absolute right-3 top-1/2 -translate-y-1/2 text-govbr-gray-40 pointer-events-none"></i>
239
+ </div>
240
+
241
+ @if (displayError) {
242
+ <span class="flex items-center mt-1 text-govbr-sm text-govbr-danger">
243
+ <i class="fa-solid fa-times-circle mr-1"></i>{{ displayError }}
244
+ </span>
245
+ }
246
+ </div>
247
+ `,
248
+ }]
249
+ }], ctorParameters: () => [{ type: i1.NgControl, decorators: [{
250
+ type: Self
251
+ }, {
252
+ type: Optional
253
+ }] }], propDecorators: { label: [{
254
+ type: Input
255
+ }], placeholder: [{
256
+ type: Input
257
+ }], required: [{
258
+ type: Input
259
+ }], helperText: [{
260
+ type: Input
261
+ }], error: [{
262
+ type: Input
263
+ }], density: [{
264
+ type: Input
265
+ }], errorMessages: [{
266
+ type: Input
267
+ }] } });
268
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"br-input-cpf.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-govbr-tw/src/lib/components/input-cpf/br-input-cpf.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  Component,\r\n  Input,\r\n  Optional,\r\n  Self,\r\n  ViewEncapsulation,\r\n} from '@angular/core';\r\n\r\nimport { ControlValueAccessor, NgControl, ValidationErrors, Validator, AbstractControl } from '@angular/forms';\r\n\r\n@Component({\r\n  selector: 'br-input-cpf',\r\n  standalone: true,\r\n  encapsulation: ViewEncapsulation.None,\r\n  template: `\r\n    <div class=\"mb-4 font-rawline\">\r\n      @if (label) {\r\n        <label\r\n          [for]=\"inputId\"\r\n          class=\"block text-govbr-sm font-semibold text-govbr-gray-80 mb-1\"\r\n        >\r\n          {{ label }}\r\n          @if (required) {\r\n            <span class=\"text-govbr-danger ml-0.5\">*</span>\r\n          }\r\n        </label>\r\n      }\r\n\r\n      @if (helperText && !displayError) {\r\n        <span class=\"block text-govbr-xs text-govbr-gray-60 mb-1\">\r\n          {{ helperText }}\r\n        </span>\r\n      }\r\n\r\n      <div class=\"relative\">\r\n        <input\r\n          [id]=\"inputId\"\r\n          type=\"text\"\r\n          inputmode=\"numeric\"\r\n          [placeholder]=\"placeholder\"\r\n          [disabled]=\"isDisabled\"\r\n          [class]=\"inputClasses\"\r\n          [value]=\"displayValue\"\r\n          (input)=\"onInput($event)\"\r\n          (blur)=\"onBlur()\"\r\n          maxlength=\"14\"\r\n        />\r\n        <i class=\"fa-solid fa-id-card absolute right-3 top-1/2 -translate-y-1/2 text-govbr-gray-40 pointer-events-none\"></i>\r\n      </div>\r\n\r\n      @if (displayError) {\r\n        <span class=\"flex items-center mt-1 text-govbr-sm text-govbr-danger\">\r\n          <i class=\"fa-solid fa-times-circle mr-1\"></i>{{ displayError }}\r\n        </span>\r\n      }\r\n    </div>\r\n  `,\r\n})\r\nexport class BrInputCpfComponent implements ControlValueAccessor, Validator {\r\n  @Input() label?: string;\r\n  @Input() placeholder = '000.000.000-00';\r\n  @Input() required = false;\r\n  @Input() helperText?: string;\r\n  @Input() error?: string;\r\n  @Input() density: 'small' | 'medium' | 'large' = 'medium';\r\n\r\n  /** Mensagens customizadas para erros */\r\n  @Input() errorMessages: Record<string, string> = {};\r\n\r\n  displayValue = '';\r\n  isDisabled = false;\r\n  private rawValue = '';\r\n\r\n  private static nextId = 0;\r\n  inputId = `br-input-cpf-${BrInputCpfComponent.nextId++}`;\r\n\r\n  onChange: (value: string) => void = () => {};\r\n  onTouched: () => void = () => {};\r\n  onValidatorChange: () => void = () => {};\r\n\r\n  private static readonly defaultErrorMessages: Record<string, string> = {\r\n    required: 'Campo obrigatório',\r\n    cpfInvalid: 'CPF inválido',\r\n    cpfIncomplete: 'CPF incompleto',\r\n  };\r\n\r\n  constructor(@Self() @Optional() private ngControl: NgControl) {\r\n    if (this.ngControl) {\r\n      this.ngControl.valueAccessor = this;\r\n    }\r\n  }\r\n\r\n  get displayError(): string | undefined {\r\n    if (this.error) return this.error;\r\n\r\n    if (!this.ngControl?.control) return undefined;\r\n\r\n    const control = this.ngControl.control;\r\n    if (!control.errors || (!control.touched && !control.dirty)) return undefined;\r\n\r\n    return this.getFirstErrorMessage(control.errors);\r\n  }\r\n\r\n  private getFirstErrorMessage(errors: ValidationErrors): string {\r\n    for (const key of Object.keys(errors)) {\r\n      if (this.errorMessages[key]) return this.errorMessages[key];\r\n      const defaultMsg = BrInputCpfComponent.defaultErrorMessages[key];\r\n      if (defaultMsg) return defaultMsg;\r\n    }\r\n    return 'Campo inválido';\r\n  }\r\n\r\n  get inputClasses(): string {\r\n    const densityHeights = {\r\n      small: 'h-[var(--govbr-input-small)] text-govbr-sm',\r\n      medium: 'h-[var(--govbr-input-medium)] text-govbr-base',\r\n      large: 'h-[var(--govbr-input-large)] text-govbr-lg',\r\n    } as const;\r\n\r\n    const base = `w-full px-4 pr-10 font-rawline govbr-input ${densityHeights[this.density]}`;\r\n    const statusCls = this.displayError ? 'govbr-input--danger' : '';\r\n    const disabledCls = this.isDisabled\r\n      ? 'bg-govbr-gray-2 text-govbr-gray-40 cursor-not-allowed'\r\n      : 'bg-white text-govbr-gray-80';\r\n\r\n    return [base, statusCls, disabledCls].filter(Boolean).join(' ');\r\n  }\r\n\r\n  onInput(event: Event): void {\r\n    const input = event.target as HTMLInputElement;\r\n    const digits = input.value.replace(/\\D/g, '').slice(0, 11);\r\n    this.rawValue = digits;\r\n    this.displayValue = this.formatCpf(digits);\r\n    this.onChange(digits);\r\n\r\n    // Reposition cursor\r\n    setTimeout(() => {\r\n      input.value = this.displayValue;\r\n    });\r\n  }\r\n\r\n  onBlur(): void {\r\n    this.onTouched();\r\n  }\r\n\r\n  private formatCpf(digits: string): string {\r\n    if (!digits) return '';\r\n    let formatted = digits;\r\n    if (digits.length > 3) {\r\n      formatted = digits.slice(0, 3) + '.' + digits.slice(3);\r\n    }\r\n    if (digits.length > 6) {\r\n      formatted = formatted.slice(0, 7) + '.' + formatted.slice(7);\r\n    }\r\n    if (digits.length > 9) {\r\n      formatted = formatted.slice(0, 11) + '-' + formatted.slice(11);\r\n    }\r\n    return formatted;\r\n  }\r\n\r\n  // ControlValueAccessor\r\n  writeValue(value: string): void {\r\n    const digits = (value || '').replace(/\\D/g, '').slice(0, 11);\r\n    this.rawValue = digits;\r\n    this.displayValue = this.formatCpf(digits);\r\n  }\r\n\r\n  registerOnChange(fn: (value: string) => void): void {\r\n    this.onChange = fn;\r\n  }\r\n\r\n  registerOnTouched(fn: () => void): void {\r\n    this.onTouched = fn;\r\n  }\r\n\r\n  setDisabledState(isDisabled: boolean): void {\r\n    this.isDisabled = isDisabled;\r\n  }\r\n\r\n  // Validator\r\n  validate(control: AbstractControl): ValidationErrors | null {\r\n    const value = control.value as string;\r\n    if (!value) {\r\n      return this.required ? { required: true } : null;\r\n    }\r\n\r\n    const digits = value.replace(/\\D/g, '');\r\n    if (digits.length < 11) {\r\n      return { cpfIncomplete: true };\r\n    }\r\n\r\n    if (!this.isValidCpf(digits)) {\r\n      return { cpfInvalid: true };\r\n    }\r\n\r\n    return null;\r\n  }\r\n\r\n  registerOnValidatorChange(fn: () => void): void {\r\n    this.onValidatorChange = fn;\r\n  }\r\n\r\n  private isValidCpf(cpf: string): boolean {\r\n    // Remove caracteres não numéricos\r\n    const digits = cpf.replace(/\\D/g, '');\r\n\r\n    if (digits.length !== 11) return false;\r\n\r\n    // Verifica se todos os dígitos são iguais\r\n    if (/^(\\d)\\1+$/.test(digits)) return false;\r\n\r\n    // Calcula primeiro dígito verificador\r\n    let sum = 0;\r\n    for (let i = 0; i < 9; i++) {\r\n      sum += parseInt(digits[i], 10) * (10 - i);\r\n    }\r\n    let remainder = (sum * 10) % 11;\r\n    if (remainder === 10 || remainder === 11) remainder = 0;\r\n    if (remainder !== parseInt(digits[9], 10)) return false;\r\n\r\n    // Calcula segundo dígito verificador\r\n    sum = 0;\r\n    for (let i = 0; i < 10; i++) {\r\n      sum += parseInt(digits[i], 10) * (11 - i);\r\n    }\r\n    remainder = (sum * 10) % 11;\r\n    if (remainder === 10 || remainder === 11) remainder = 0;\r\n    if (remainder !== parseInt(digits[10], 10)) return false;\r\n\r\n    return true;\r\n  }\r\n}\r\n"]}