@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.
- package/esm2022/lib/components/input-cnpj/br-input-cnpj.component.mjs +270 -0
- package/esm2022/lib/components/input-cpf/br-input-cpf.component.mjs +268 -0
- package/esm2022/lib/components/toast/br-toast.component.mjs +138 -0
- package/esm2022/lib/components/toast/br-toast.service.mjs +52 -0
- package/esm2022/public-api.mjs +5 -1
- package/fesm2022/schenkerjon-ng-govbr-tw.mjs +719 -2
- package/fesm2022/schenkerjon-ng-govbr-tw.mjs.map +1 -1
- package/lib/components/input-cnpj/br-input-cnpj.component.d.ts +38 -0
- package/lib/components/input-cpf/br-input-cpf.component.d.ts +38 -0
- package/lib/components/toast/br-toast.component.d.ts +11 -0
- package/lib/components/toast/br-toast.service.d.ts +28 -0
- package/package.json +1 -1
- package/public-api.d.ts +5 -0
|
@@ -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":"AAAA,OAAO,EACL,SAAS,EACT,KAAK,EACL,QAAQ,EACR,IAAI,EACJ,iBAAiB,GAClB,MAAM,eAAe,CAAC;;;AAoDvB,MAAM,OAAO,mBAAmB;aAef,WAAM,GAAG,CAAC,AAAJ,CAAK;aAOF,yBAAoB,GAA2B;QACrE,QAAQ,EAAE,mBAAmB;QAC7B,UAAU,EAAE,cAAc;QAC1B,aAAa,EAAE,gBAAgB;KAChC,AAJ2C,CAI1C;IAEF,YAAwC,SAAoB;QAApB,cAAS,GAAT,SAAS,CAAW;QA1BnD,gBAAW,GAAG,gBAAgB,CAAC;QAC/B,aAAQ,GAAG,KAAK,CAAC;QAGjB,YAAO,GAAiC,QAAQ,CAAC;QAE1D,wCAAwC;QAC/B,kBAAa,GAA2B,EAAE,CAAC;QAEpD,iBAAY,GAAG,EAAE,CAAC;QAClB,eAAU,GAAG,KAAK,CAAC;QACX,aAAQ,GAAG,EAAE,CAAC;QAGtB,YAAO,GAAG,gBAAgB,mBAAmB,CAAC,MAAM,EAAE,EAAE,CAAC;QAEzD,aAAQ,GAA4B,GAAG,EAAE,GAAE,CAAC,CAAC;QAC7C,cAAS,GAAe,GAAG,EAAE,GAAE,CAAC,CAAC;QACjC,sBAAiB,GAAe,GAAG,EAAE,GAAE,CAAC,CAAC;QASvC,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACnB,IAAI,CAAC,SAAS,CAAC,aAAa,GAAG,IAAI,CAAC;QACtC,CAAC;IACH,CAAC;IAED,IAAI,YAAY;QACd,IAAI,IAAI,CAAC,KAAK;YAAE,OAAO,IAAI,CAAC,KAAK,CAAC;QAElC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,OAAO;YAAE,OAAO,SAAS,CAAC;QAE/C,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC;QACvC,IAAI,CAAC,OAAO,CAAC,MAAM,IAAI,CAAC,CAAC,OAAO,CAAC,OAAO,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC;YAAE,OAAO,SAAS,CAAC;QAE9E,OAAO,IAAI,CAAC,oBAAoB,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;IACnD,CAAC;IAEO,oBAAoB,CAAC,MAAwB;QACnD,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC;YACtC,IAAI,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC;gBAAE,OAAO,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC;YAC5D,MAAM,UAAU,GAAG,mBAAmB,CAAC,oBAAoB,CAAC,GAAG,CAAC,CAAC;YACjE,IAAI,UAAU;gBAAE,OAAO,UAAU,CAAC;QACpC,CAAC;QACD,OAAO,gBAAgB,CAAC;IAC1B,CAAC;IAED,IAAI,YAAY;QACd,MAAM,cAAc,GAAG;YACrB,KAAK,EAAE,4CAA4C;YACnD,MAAM,EAAE,+CAA+C;YACvD,KAAK,EAAE,4CAA4C;SAC3C,CAAC;QAEX,MAAM,IAAI,GAAG,8CAA8C,cAAc,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;QAC1F,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,EAAE,CAAC;QACjE,MAAM,WAAW,GAAG,IAAI,CAAC,UAAU;YACjC,CAAC,CAAC,uDAAuD;YACzD,CAAC,CAAC,6BAA6B,CAAC;QAElC,OAAO,CAAC,IAAI,EAAE,SAAS,EAAE,WAAW,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IAClE,CAAC;IAED,OAAO,CAAC,KAAY;QAClB,MAAM,KAAK,GAAG,KAAK,CAAC,MAA0B,CAAC;QAC/C,MAAM,MAAM,GAAG,KAAK,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QAC3D,IAAI,CAAC,QAAQ,GAAG,MAAM,CAAC;QACvB,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;QAC3C,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;QAEtB,oBAAoB;QACpB,UAAU,CAAC,GAAG,EAAE;YACd,KAAK,CAAC,KAAK,GAAG,IAAI,CAAC,YAAY,CAAC;QAClC,CAAC,CAAC,CAAC;IACL,CAAC;IAED,MAAM;QACJ,IAAI,CAAC,SAAS,EAAE,CAAC;IACnB,CAAC;IAEO,SAAS,CAAC,MAAc;QAC9B,IAAI,CAAC,MAAM;YAAE,OAAO,EAAE,CAAC;QACvB,IAAI,SAAS,GAAG,MAAM,CAAC;QACvB,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACtB,SAAS,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,GAAG,GAAG,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QACzD,CAAC;QACD,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACtB,SAAS,GAAG,SAAS,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,GAAG,GAAG,GAAG,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QAC/D,CAAC;QACD,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACtB,SAAS,GAAG,SAAS,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,GAAG,SAAS,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QACjE,CAAC;QACD,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,uBAAuB;IACvB,UAAU,CAAC,KAAa;QACtB,MAAM,MAAM,GAAG,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QAC7D,IAAI,CAAC,QAAQ,GAAG,MAAM,CAAC;QACvB,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;IAC7C,CAAC;IAED,gBAAgB,CAAC,EAA2B;QAC1C,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;IACrB,CAAC;IAED,iBAAiB,CAAC,EAAc;QAC9B,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;IACtB,CAAC;IAED,gBAAgB,CAAC,UAAmB;QAClC,IAAI,CAAC,UAAU,GAAG,UAAU,CAAC;IAC/B,CAAC;IAED,YAAY;IACZ,QAAQ,CAAC,OAAwB;QAC/B,MAAM,KAAK,GAAG,OAAO,CAAC,KAAe,CAAC;QACtC,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,OAAO,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC;QACnD,CAAC;QAED,MAAM,MAAM,GAAG,KAAK,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QACxC,IAAI,MAAM,CAAC,MAAM,GAAG,EAAE,EAAE,CAAC;YACvB,OAAO,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC;QACjC,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;YAC7B,OAAO,EAAE,UAAU,EAAE,IAAI,EAAE,CAAC;QAC9B,CAAC;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAED,yBAAyB,CAAC,EAAc;QACtC,IAAI,CAAC,iBAAiB,GAAG,EAAE,CAAC;IAC9B,CAAC;IAEO,UAAU,CAAC,GAAW;QAC5B,kCAAkC;QAClC,MAAM,MAAM,GAAG,GAAG,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEtC,IAAI,MAAM,CAAC,MAAM,KAAK,EAAE;YAAE,OAAO,KAAK,CAAC;QAEvC,0CAA0C;QAC1C,IAAI,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC;YAAE,OAAO,KAAK,CAAC;QAE3C,sCAAsC;QACtC,IAAI,GAAG,GAAG,CAAC,CAAC;QACZ,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;YAC3B,GAAG,IAAI,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,EAAE,GAAG,CAAC,CAAC,CAAC;QAC5C,CAAC;QACD,IAAI,SAAS,GAAG,CAAC,GAAG,GAAG,EAAE,CAAC,GAAG,EAAE,CAAC;QAChC,IAAI,SAAS,KAAK,EAAE,IAAI,SAAS,KAAK,EAAE;YAAE,SAAS,GAAG,CAAC,CAAC;QACxD,IAAI,SAAS,KAAK,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC;YAAE,OAAO,KAAK,CAAC;QAExD,qCAAqC;QACrC,GAAG,GAAG,CAAC,CAAC;QACR,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC;YAC5B,GAAG,IAAI,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,EAAE,GAAG,CAAC,CAAC,CAAC;QAC5C,CAAC;QACD,SAAS,GAAG,CAAC,GAAG,GAAG,EAAE,CAAC,GAAG,EAAE,CAAC;QAC5B,IAAI,SAAS,KAAK,EAAE,IAAI,SAAS,KAAK,EAAE;YAAE,SAAS,GAAG,CAAC,CAAC;QACxD,IAAI,SAAS,KAAK,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;YAAE,OAAO,KAAK,CAAC;QAEzD,OAAO,IAAI,CAAC;IACd,CAAC;+GA5KU,mBAAmB;mGAAnB,mBAAmB,oPA5CpB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA0CT;;4FAEU,mBAAmB;kBAhD/B,SAAS;mBAAC;oBACT,QAAQ,EAAE,cAAc;oBACxB,UAAU,EAAE,IAAI;oBAChB,aAAa,EAAE,iBAAiB,CAAC,IAAI;oBACrC,QAAQ,EAAE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA0CT;iBACF;;0BA6Bc,IAAI;;0BAAI,QAAQ;yCA3BpB,KAAK;sBAAb,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,UAAU;sBAAlB,KAAK;gBACG,KAAK;sBAAb,KAAK;gBACG,OAAO;sBAAf,KAAK;gBAGG,aAAa;sBAArB,KAAK","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"]}
|