@schenkerjon/ng-govbr-tw 0.2.0 → 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.
@@ -1,9 +1,32 @@
1
- import { Component, EventEmitter, forwardRef, Input, Output, ViewEncapsulation, } from '@angular/core';
2
- import { NG_VALUE_ACCESSOR, FormsModule, } from '@angular/forms';
1
+ import { Component, EventEmitter, Input, Optional, Output, Self, ViewEncapsulation, } from '@angular/core';
2
+ import { FormsModule, } from '@angular/forms';
3
3
  import * as i0 from "@angular/core";
4
4
  import * as i1 from "@angular/forms";
5
5
  export class BrInputComponent {
6
- constructor() {
6
+ static { this.nextId = 0; }
7
+ static { this.defaultErrorMessages = {
8
+ required: 'Campo obrigatório',
9
+ email: 'Email inválido',
10
+ minlength: (err) => {
11
+ const e = err;
12
+ return `Mínimo de ${e.requiredLength} caracteres`;
13
+ },
14
+ maxlength: (err) => {
15
+ const e = err;
16
+ return `Máximo de ${e.requiredLength} caracteres`;
17
+ },
18
+ min: (err) => {
19
+ const e = err;
20
+ return `Valor mínimo: ${e.min}`;
21
+ },
22
+ max: (err) => {
23
+ const e = err;
24
+ return `Valor máximo: ${e.max}`;
25
+ },
26
+ pattern: 'Formato inválido',
27
+ }; }
28
+ constructor(ngControl) {
29
+ this.ngControl = ngControl;
7
30
  this.type = 'text';
8
31
  this.placeholder = '';
9
32
  this.required = false;
@@ -12,6 +35,8 @@ export class BrInputComponent {
12
35
  this.highlight = false;
13
36
  this.inline = false;
14
37
  this.rows = 3;
38
+ /** Mensagens customizadas para erros de validação */
39
+ this.errorMessages = {};
15
40
  this.fileSelected = new EventEmitter();
16
41
  this.searchSubmit = new EventEmitter();
17
42
  this.value = '';
@@ -21,8 +46,10 @@ export class BrInputComponent {
21
46
  this.inputId = `br-input-${BrInputComponent.nextId++}`;
22
47
  this.onChange = () => { };
23
48
  this.onTouched = () => { };
49
+ if (this.ngControl) {
50
+ this.ngControl.valueAccessor = this;
51
+ }
24
52
  }
25
- static { this.nextId = 0; }
26
53
  get isStandardInput() {
27
54
  return this.type !== 'select' && this.type !== 'file' && this.type !== 'textarea';
28
55
  }
@@ -37,6 +64,42 @@ export class BrInputComponent {
37
64
  get containerClasses() {
38
65
  return this.type !== 'file' ? 'relative' : '';
39
66
  }
67
+ /**
68
+ * Retorna a mensagem de erro a ser exibida.
69
+ * Prioridade: error manual > erro do validator (se touched/dirty)
70
+ */
71
+ get displayError() {
72
+ // Erro manual sempre tem prioridade
73
+ if (this.error) {
74
+ return this.error;
75
+ }
76
+ // Se não tem NgControl, não há erro automático
77
+ if (!this.ngControl?.control) {
78
+ return undefined;
79
+ }
80
+ const control = this.ngControl.control;
81
+ // Só mostra erro se o controle foi tocado ou modificado E está inválido
82
+ if (!control.errors || (!control.touched && !control.dirty)) {
83
+ return undefined;
84
+ }
85
+ // Busca a primeira mensagem de erro
86
+ return this.getFirstErrorMessage(control.errors);
87
+ }
88
+ getFirstErrorMessage(errors) {
89
+ for (const key of Object.keys(errors)) {
90
+ // Primeiro verifica mensagens customizadas do usuário
91
+ if (this.errorMessages[key]) {
92
+ return this.errorMessages[key];
93
+ }
94
+ // Depois verifica mensagens padrão
95
+ const defaultMsg = BrInputComponent.defaultErrorMessages[key];
96
+ if (defaultMsg) {
97
+ return typeof defaultMsg === 'function' ? defaultMsg(errors[key]) : defaultMsg;
98
+ }
99
+ }
100
+ // Fallback genérico
101
+ return 'Campo inválido';
102
+ }
40
103
  get inputClasses() {
41
104
  const densityStyles = {
42
105
  small: 'text-govbr-sm',
@@ -57,7 +120,7 @@ export class BrInputComponent {
57
120
  const highlightCls = this.highlight
58
121
  ? 'h-[var(--govbr-input-highlight)] bg-govbr-gray-2 border-transparent text-govbr-lg px-6'
59
122
  : '';
60
- const statusCls = this.error
123
+ const statusCls = this.displayError
61
124
  ? 'govbr-input--danger'
62
125
  : this.successMessage
63
126
  ? 'govbr-input--success'
@@ -108,14 +171,8 @@ export class BrInputComponent {
108
171
  this.fileSelected.emit(file);
109
172
  }
110
173
  }
111
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BrInputComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
112
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: BrInputComponent, isStandalone: true, selector: "br-input", inputs: { label: "label", type: "type", placeholder: "placeholder", required: "required", error: "error", successMessage: "successMessage", warningMessage: "warningMessage", infoMessage: "infoMessage", helperText: "helperText", icon: "icon", iconRight: "iconRight", density: "density", highlight: "highlight", inline: "inline", maxLength: "maxLength", rows: "rows" }, outputs: { fileSelected: "fileSelected", searchSubmit: "searchSubmit" }, providers: [
113
- {
114
- provide: NG_VALUE_ACCESSOR,
115
- useExisting: forwardRef(() => BrInputComponent),
116
- multi: true,
117
- },
118
- ], ngImport: i0, template: `
174
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BrInputComponent, deps: [{ token: i1.NgControl, optional: true, self: true }], target: i0.ɵɵFactoryTarget.Component }); }
175
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: BrInputComponent, isStandalone: true, selector: "br-input", inputs: { label: "label", type: "type", placeholder: "placeholder", required: "required", error: "error", successMessage: "successMessage", warningMessage: "warningMessage", infoMessage: "infoMessage", helperText: "helperText", icon: "icon", iconRight: "iconRight", density: "density", highlight: "highlight", inline: "inline", maxLength: "maxLength", rows: "rows", errorMessages: "errorMessages" }, outputs: { fileSelected: "fileSelected", searchSubmit: "searchSubmit" }, ngImport: i0, template: `
119
176
  <div class="mb-4 font-rawline" [class.inline-input]="inline">
120
177
  @if (label) {
121
178
  <label
@@ -256,25 +313,25 @@ export class BrInputComponent {
256
313
  }
257
314
 
258
315
  <!-- Feedback messages -->
259
- @if (error) {
316
+ @if (displayError) {
260
317
  <span class="flex items-center mt-1 text-govbr-sm text-govbr-danger">
261
- <i class="fa-solid fa-times-circle mr-1"></i>{{ error }}
318
+ <i class="fa-solid fa-times-circle mr-1"></i>{{ displayError }}
262
319
  </span>
263
320
  }
264
-
265
- @if (successMessage && !error) {
321
+
322
+ @if (successMessage && !displayError) {
266
323
  <span class="flex items-center mt-1 text-govbr-sm text-govbr-success">
267
324
  <i class="fa-solid fa-check-circle mr-1"></i>{{ successMessage }}
268
325
  </span>
269
326
  }
270
-
271
- @if (warningMessage && !error && !successMessage) {
327
+
328
+ @if (warningMessage && !displayError && !successMessage) {
272
329
  <span class="flex items-center mt-1 text-govbr-sm text-govbr-warning">
273
330
  <i class="fa-solid fa-exclamation-triangle mr-1"></i>{{ warningMessage }}
274
331
  </span>
275
332
  }
276
-
277
- @if (infoMessage && !error && !successMessage && !warningMessage) {
333
+
334
+ @if (infoMessage && !displayError && !successMessage && !warningMessage) {
278
335
  <span class="flex items-center mt-1 text-govbr-sm text-govbr-info">
279
336
  <i class="fa-solid fa-info-circle mr-1"></i>{{ infoMessage }}
280
337
  </span>
@@ -289,13 +346,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
289
346
  standalone: true,
290
347
  imports: [FormsModule],
291
348
  encapsulation: ViewEncapsulation.None,
292
- providers: [
293
- {
294
- provide: NG_VALUE_ACCESSOR,
295
- useExisting: forwardRef(() => BrInputComponent),
296
- multi: true,
297
- },
298
- ],
299
349
  template: `
300
350
  <div class="mb-4 font-rawline" [class.inline-input]="inline">
301
351
  @if (label) {
@@ -437,25 +487,25 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
437
487
  }
438
488
 
439
489
  <!-- Feedback messages -->
440
- @if (error) {
490
+ @if (displayError) {
441
491
  <span class="flex items-center mt-1 text-govbr-sm text-govbr-danger">
442
- <i class="fa-solid fa-times-circle mr-1"></i>{{ error }}
492
+ <i class="fa-solid fa-times-circle mr-1"></i>{{ displayError }}
443
493
  </span>
444
494
  }
445
-
446
- @if (successMessage && !error) {
495
+
496
+ @if (successMessage && !displayError) {
447
497
  <span class="flex items-center mt-1 text-govbr-sm text-govbr-success">
448
498
  <i class="fa-solid fa-check-circle mr-1"></i>{{ successMessage }}
449
499
  </span>
450
500
  }
451
-
452
- @if (warningMessage && !error && !successMessage) {
501
+
502
+ @if (warningMessage && !displayError && !successMessage) {
453
503
  <span class="flex items-center mt-1 text-govbr-sm text-govbr-warning">
454
504
  <i class="fa-solid fa-exclamation-triangle mr-1"></i>{{ warningMessage }}
455
505
  </span>
456
506
  }
457
-
458
- @if (infoMessage && !error && !successMessage && !warningMessage) {
507
+
508
+ @if (infoMessage && !displayError && !successMessage && !warningMessage) {
459
509
  <span class="flex items-center mt-1 text-govbr-sm text-govbr-info">
460
510
  <i class="fa-solid fa-info-circle mr-1"></i>{{ infoMessage }}
461
511
  </span>
@@ -463,7 +513,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
463
513
  </div>
464
514
  `,
465
515
  }]
466
- }], propDecorators: { label: [{
516
+ }], ctorParameters: () => [{ type: i1.NgControl, decorators: [{
517
+ type: Self
518
+ }, {
519
+ type: Optional
520
+ }] }], propDecorators: { label: [{
467
521
  type: Input
468
522
  }], type: [{
469
523
  type: Input
@@ -495,9 +549,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
495
549
  type: Input
496
550
  }], rows: [{
497
551
  type: Input
552
+ }], errorMessages: [{
553
+ type: Input
498
554
  }], fileSelected: [{
499
555
  type: Output
500
556
  }], searchSubmit: [{
501
557
  type: Output
502
558
  }] } });
503
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"br-input.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-govbr-tw/src/lib/components/input/br-input.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  forwardRef,\n  Input,\n  Output,\n  ViewEncapsulation,\n} from '@angular/core';\n\nimport {\n  ControlValueAccessor,\n  NG_VALUE_ACCESSOR,\n  FormsModule,\n} from '@angular/forms';\n\n@Component({\n  selector: 'br-input',\n  standalone: true,\n  imports: [FormsModule],\n  encapsulation: ViewEncapsulation.None,\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => BrInputComponent),\n      multi: true,\n    },\n  ],\n  template: `\n    <div class=\"mb-4 font-rawline\" [class.inline-input]=\"inline\">\n      @if (label) {\n        <label\n          [for]=\"inputId\"\n          class=\"block text-govbr-sm font-semibold text-govbr-gray-80 mb-1\"\n          [class.sr-only]=\"inline\"\n          >\n          {{ label }}\n          @if (required) {\n            <span class=\"text-govbr-danger ml-0.5\">*</span>\n          }\n        </label>\n      }\n    \n      @if (helperText && !error && !successMessage) {\n        <span class=\"block text-govbr-xs text-govbr-gray-60 mb-1\">\n          {{ helperText }}\n        </span>\n      }\n    \n      <div [class]=\"containerClasses\">\n        <!-- Icon (left) -->\n        @if (icon && !iconRight) {\n          <i\n            [class]=\"'fa-solid fa-' + icon + ' absolute left-3 top-1/2 -translate-y-1/2 text-govbr-gray-40 pointer-events-none'\"\n          ></i>\n        }\n    \n        <!-- Select -->\n        @if (type === 'select') {\n          <select\n            [id]=\"inputId\"\n            [disabled]=\"isDisabled\"\n            [class]=\"inputClasses\"\n            [value]=\"value\"\n            (change)=\"onSelectChange($event)\"\n            (blur)=\"onTouched()\"\n            >\n            <option value=\"\" disabled [selected]=\"!value\">{{ placeholder }}</option>\n            <ng-content></ng-content>\n          </select>\n        }\n    \n        <!-- Textarea -->\n        @if (type === 'textarea') {\n          <textarea\n            [id]=\"inputId\"\n            [placeholder]=\"placeholder\"\n            [disabled]=\"isDisabled\"\n            [class]=\"inputClasses\"\n            [rows]=\"rows\"\n            [attr.maxlength]=\"maxLength || null\"\n            [value]=\"value\"\n            (input)=\"onInputChange($event)\"\n            (blur)=\"onTouched()\"\n          ></textarea>\n        }\n    \n        <!-- File -->\n        @if (type === 'file') {\n          <div class=\"flex items-center\">\n            <label\n              [for]=\"inputId\"\n              class=\"inline-flex items-center px-4 py-2 bg-govbr-primary text-white rounded-govbr-pill cursor-pointer hover:bg-govbr-primary-dark transition-colors text-govbr-sm font-semibold\"\n              [class.opacity-50]=\"isDisabled\"\n              [class.cursor-not-allowed]=\"isDisabled\"\n              >\n              <i class=\"fa-solid fa-upload mr-2\"></i>\n              {{ placeholder || 'Selecionar arquivo' }}\n            </label>\n            <input\n              [id]=\"inputId\"\n              type=\"file\"\n              class=\"hidden\"\n              [disabled]=\"isDisabled\"\n              (change)=\"onFileChange($event)\"\n              />\n            @if (fileName) {\n              <span class=\"ml-2 text-govbr-sm text-govbr-gray-60\">{{ fileName }}</span>\n            }\n          </div>\n        }\n    \n        <!-- Text / Email / Password / Number / Search -->\n        @if (isStandardInput) {\n          <input\n            [id]=\"inputId\"\n            [type]=\"actualType\"\n            [placeholder]=\"placeholder\"\n            [disabled]=\"isDisabled\"\n            [class]=\"inputClasses\"\n            [attr.maxlength]=\"maxLength || null\"\n            [value]=\"value\"\n            (input)=\"onInputChange($event)\"\n            (blur)=\"onTouched()\"\n            />\n        }\n    \n        <!-- Password toggle button -->\n        @if (type === 'password') {\n          <button\n            type=\"button\"\n            class=\"absolute right-3 top-1/2 -translate-y-1/2 text-govbr-primary hover:text-govbr-primary-dark govbr-focus-ring\"\n            [attr.aria-label]=\"showPassword ? 'Ocultar senha' : 'Mostrar senha'\"\n            (click)=\"togglePassword()\"\n            >\n            <i [class]=\"showPassword ? 'fa-solid fa-eye-slash' : 'fa-solid fa-eye'\"></i>\n          </button>\n        }\n    \n        <!-- Icon (right) -->\n        @if (icon && iconRight && type !== 'password') {\n          <i\n            [class]=\"'fa-solid fa-' + icon + ' absolute right-3 top-1/2 -translate-y-1/2 text-govbr-gray-40 pointer-events-none'\"\n          ></i>\n        }\n    \n        <!-- Search button -->\n        @if (type === 'search') {\n          <button\n            type=\"button\"\n            class=\"absolute right-1 top-1/2 -translate-y-1/2 w-8 h-8 rounded-full flex items-center justify-center text-govbr-primary hover:bg-govbr-primary-lightest govbr-focus-ring transition-colors\"\n            aria-label=\"Buscar\"\n            (click)=\"searchSubmit.emit(value)\"\n            >\n            <i class=\"fa-solid fa-magnifying-glass\"></i>\n          </button>\n        }\n      </div>\n    \n      <!-- Character counter -->\n      @if (maxLength && (type === 'textarea' || isStandardInput)) {\n        <div class=\"flex justify-end mt-0.5\">\n          <span class=\"text-govbr-xs text-govbr-gray-40\">\n            {{ value?.length || 0 }}/{{ maxLength }}\n          </span>\n        </div>\n      }\n    \n      <!-- Feedback messages -->\n      @if (error) {\n        <span class=\"flex items-center mt-1 text-govbr-sm text-govbr-danger\">\n          <i class=\"fa-solid fa-times-circle mr-1\"></i>{{ error }}\n        </span>\n      }\n    \n      @if (successMessage && !error) {\n        <span class=\"flex items-center mt-1 text-govbr-sm text-govbr-success\">\n          <i class=\"fa-solid fa-check-circle mr-1\"></i>{{ successMessage }}\n        </span>\n      }\n    \n      @if (warningMessage && !error && !successMessage) {\n        <span class=\"flex items-center mt-1 text-govbr-sm text-govbr-warning\">\n          <i class=\"fa-solid fa-exclamation-triangle mr-1\"></i>{{ warningMessage }}\n        </span>\n      }\n    \n      @if (infoMessage && !error && !successMessage && !warningMessage) {\n        <span class=\"flex items-center mt-1 text-govbr-sm text-govbr-info\">\n          <i class=\"fa-solid fa-info-circle mr-1\"></i>{{ infoMessage }}\n        </span>\n      }\n    </div>\n    `,\n})\nexport class BrInputComponent implements ControlValueAccessor {\n  @Input() label?: string;\n  @Input() type:\n    | 'text'\n    | 'email'\n    | 'password'\n    | 'select'\n    | 'file'\n    | 'number'\n    | 'search'\n    | 'textarea' = 'text';\n  @Input() placeholder = '';\n  @Input() required = false;\n  @Input() error?: string;\n  @Input() successMessage?: string;\n  @Input() warningMessage?: string;\n  @Input() infoMessage?: string;\n  @Input() helperText?: string;\n  @Input() icon?: string;\n  @Input() iconRight = false;\n  @Input() density: 'small' | 'medium' | 'large' = 'medium';\n  @Input() highlight = false;\n  @Input() inline = false;\n  @Input() maxLength?: number;\n  @Input() rows = 3;\n\n  @Output() fileSelected = new EventEmitter<File>();\n  @Output() searchSubmit = new EventEmitter<string>();\n\n  value: string = '';\n  isDisabled = false;\n  fileName = '';\n  showPassword = false;\n\n  private static nextId = 0;\n  inputId = `br-input-${BrInputComponent.nextId++}`;\n\n  onChange: (value: string) => void = () => {};\n  onTouched: () => void = () => {};\n\n  get isStandardInput(): boolean {\n    return this.type !== 'select' && this.type !== 'file' && this.type !== 'textarea';\n  }\n\n  get actualType(): string {\n    if (this.type === 'password') {\n      return this.showPassword ? 'text' : 'password';\n    }\n    if (this.type === 'search') return 'text';\n    return this.type;\n  }\n\n  get containerClasses(): string {\n    return this.type !== 'file' ? 'relative' : '';\n  }\n\n  get inputClasses(): string {\n    const densityStyles = {\n      small: 'text-govbr-sm',\n      medium: 'text-govbr-base',\n      large: 'text-govbr-lg',\n    } as const;\n\n    const densityHeights = {\n      small: 'h-[var(--govbr-input-small)]',\n      medium: 'h-[var(--govbr-input-medium)]',\n      large: 'h-[var(--govbr-input-large)]',\n    } as const;\n\n    const base = [\n      'w-full px-4 font-rawline govbr-input',\n      this.type === 'textarea' ? 'py-2' : '',\n    ].join(' ');\n\n    const density = densityStyles[this.density];\n    const height = this.type !== 'textarea' ? densityHeights[this.density] : '';\n\n    const highlightCls = this.highlight\n      ? 'h-[var(--govbr-input-highlight)] bg-govbr-gray-2 border-transparent text-govbr-lg px-6'\n      : '';\n\n    const statusCls = this.error\n      ? 'govbr-input--danger'\n      : this.successMessage\n        ? 'govbr-input--success'\n        : this.warningMessage\n          ? 'govbr-input--warning'\n          : '';\n\n    const disabledCls = this.isDisabled\n      ? 'bg-govbr-gray-2 text-govbr-gray-40 cursor-not-allowed'\n      : 'bg-white text-govbr-gray-80';\n\n    const iconPaddingLeft = this.icon && !this.iconRight ? 'pl-10' : '';\n    const iconPaddingRight =\n      (this.icon && this.iconRight) || this.type === 'password' || this.type === 'search'\n        ? 'pr-10'\n        : '';\n\n    return [base, density, height, highlightCls, statusCls, disabledCls, iconPaddingLeft, iconPaddingRight]\n      .filter(Boolean)\n      .join(' ');\n  }\n\n  writeValue(value: string): void {\n    this.value = value ?? '';\n  }\n\n  registerOnChange(fn: (value: string) => void): void {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: () => void): void {\n    this.onTouched = fn;\n  }\n\n  setDisabledState(isDisabled: boolean): void {\n    this.isDisabled = isDisabled;\n  }\n\n  togglePassword(): void {\n    this.showPassword = !this.showPassword;\n  }\n\n  onInputChange(event: Event): void {\n    const target = event.target as HTMLInputElement | HTMLTextAreaElement;\n    this.value = target.value;\n    this.onChange(target.value);\n  }\n\n  onSelectChange(event: Event): void {\n    const val = (event.target as HTMLSelectElement).value;\n    this.value = val;\n    this.onChange(val);\n  }\n\n  onFileChange(event: Event): void {\n    const input = event.target as HTMLInputElement;\n    if (input.files && input.files.length > 0) {\n      const file = input.files[0];\n      this.fileName = file.name;\n      this.fileSelected.emit(file);\n    }\n  }\n}\n"]}
559
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"br-input.component.js","sourceRoot":"","sources":["../../../../../../projects/ng-govbr-tw/src/lib/components/input/br-input.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  forwardRef,\n  Input,\n  Optional,\n  Output,\n  Self,\n  ViewEncapsulation,\n} from '@angular/core';\n\nimport {\n  ControlValueAccessor,\n  NG_VALUE_ACCESSOR,\n  FormsModule,\n  NgControl,\n  ValidationErrors,\n} from '@angular/forms';\n\n@Component({\n  selector: 'br-input',\n  standalone: true,\n  imports: [FormsModule],\n  encapsulation: ViewEncapsulation.None,\n  template: `\n    <div class=\"mb-4 font-rawline\" [class.inline-input]=\"inline\">\n      @if (label) {\n        <label\n          [for]=\"inputId\"\n          class=\"block text-govbr-sm font-semibold text-govbr-gray-80 mb-1\"\n          [class.sr-only]=\"inline\"\n          >\n          {{ label }}\n          @if (required) {\n            <span class=\"text-govbr-danger ml-0.5\">*</span>\n          }\n        </label>\n      }\n    \n      @if (helperText && !error && !successMessage) {\n        <span class=\"block text-govbr-xs text-govbr-gray-60 mb-1\">\n          {{ helperText }}\n        </span>\n      }\n    \n      <div [class]=\"containerClasses\">\n        <!-- Icon (left) -->\n        @if (icon && !iconRight) {\n          <i\n            [class]=\"'fa-solid fa-' + icon + ' absolute left-3 top-1/2 -translate-y-1/2 text-govbr-gray-40 pointer-events-none'\"\n          ></i>\n        }\n    \n        <!-- Select -->\n        @if (type === 'select') {\n          <select\n            [id]=\"inputId\"\n            [disabled]=\"isDisabled\"\n            [class]=\"inputClasses\"\n            [value]=\"value\"\n            (change)=\"onSelectChange($event)\"\n            (blur)=\"onTouched()\"\n            >\n            <option value=\"\" disabled [selected]=\"!value\">{{ placeholder }}</option>\n            <ng-content></ng-content>\n          </select>\n        }\n    \n        <!-- Textarea -->\n        @if (type === 'textarea') {\n          <textarea\n            [id]=\"inputId\"\n            [placeholder]=\"placeholder\"\n            [disabled]=\"isDisabled\"\n            [class]=\"inputClasses\"\n            [rows]=\"rows\"\n            [attr.maxlength]=\"maxLength || null\"\n            [value]=\"value\"\n            (input)=\"onInputChange($event)\"\n            (blur)=\"onTouched()\"\n          ></textarea>\n        }\n    \n        <!-- File -->\n        @if (type === 'file') {\n          <div class=\"flex items-center\">\n            <label\n              [for]=\"inputId\"\n              class=\"inline-flex items-center px-4 py-2 bg-govbr-primary text-white rounded-govbr-pill cursor-pointer hover:bg-govbr-primary-dark transition-colors text-govbr-sm font-semibold\"\n              [class.opacity-50]=\"isDisabled\"\n              [class.cursor-not-allowed]=\"isDisabled\"\n              >\n              <i class=\"fa-solid fa-upload mr-2\"></i>\n              {{ placeholder || 'Selecionar arquivo' }}\n            </label>\n            <input\n              [id]=\"inputId\"\n              type=\"file\"\n              class=\"hidden\"\n              [disabled]=\"isDisabled\"\n              (change)=\"onFileChange($event)\"\n              />\n            @if (fileName) {\n              <span class=\"ml-2 text-govbr-sm text-govbr-gray-60\">{{ fileName }}</span>\n            }\n          </div>\n        }\n    \n        <!-- Text / Email / Password / Number / Search -->\n        @if (isStandardInput) {\n          <input\n            [id]=\"inputId\"\n            [type]=\"actualType\"\n            [placeholder]=\"placeholder\"\n            [disabled]=\"isDisabled\"\n            [class]=\"inputClasses\"\n            [attr.maxlength]=\"maxLength || null\"\n            [value]=\"value\"\n            (input)=\"onInputChange($event)\"\n            (blur)=\"onTouched()\"\n            />\n        }\n    \n        <!-- Password toggle button -->\n        @if (type === 'password') {\n          <button\n            type=\"button\"\n            class=\"absolute right-3 top-1/2 -translate-y-1/2 text-govbr-primary hover:text-govbr-primary-dark govbr-focus-ring\"\n            [attr.aria-label]=\"showPassword ? 'Ocultar senha' : 'Mostrar senha'\"\n            (click)=\"togglePassword()\"\n            >\n            <i [class]=\"showPassword ? 'fa-solid fa-eye-slash' : 'fa-solid fa-eye'\"></i>\n          </button>\n        }\n    \n        <!-- Icon (right) -->\n        @if (icon && iconRight && type !== 'password') {\n          <i\n            [class]=\"'fa-solid fa-' + icon + ' absolute right-3 top-1/2 -translate-y-1/2 text-govbr-gray-40 pointer-events-none'\"\n          ></i>\n        }\n    \n        <!-- Search button -->\n        @if (type === 'search') {\n          <button\n            type=\"button\"\n            class=\"absolute right-1 top-1/2 -translate-y-1/2 w-8 h-8 rounded-full flex items-center justify-center text-govbr-primary hover:bg-govbr-primary-lightest govbr-focus-ring transition-colors\"\n            aria-label=\"Buscar\"\n            (click)=\"searchSubmit.emit(value)\"\n            >\n            <i class=\"fa-solid fa-magnifying-glass\"></i>\n          </button>\n        }\n      </div>\n    \n      <!-- Character counter -->\n      @if (maxLength && (type === 'textarea' || isStandardInput)) {\n        <div class=\"flex justify-end mt-0.5\">\n          <span class=\"text-govbr-xs text-govbr-gray-40\">\n            {{ value?.length || 0 }}/{{ maxLength }}\n          </span>\n        </div>\n      }\n    \n      <!-- Feedback messages -->\n      @if (displayError) {\n        <span class=\"flex items-center mt-1 text-govbr-sm text-govbr-danger\">\n          <i class=\"fa-solid fa-times-circle mr-1\"></i>{{ displayError }}\n        </span>\n      }\n\n      @if (successMessage && !displayError) {\n        <span class=\"flex items-center mt-1 text-govbr-sm text-govbr-success\">\n          <i class=\"fa-solid fa-check-circle mr-1\"></i>{{ successMessage }}\n        </span>\n      }\n\n      @if (warningMessage && !displayError && !successMessage) {\n        <span class=\"flex items-center mt-1 text-govbr-sm text-govbr-warning\">\n          <i class=\"fa-solid fa-exclamation-triangle mr-1\"></i>{{ warningMessage }}\n        </span>\n      }\n\n      @if (infoMessage && !displayError && !successMessage && !warningMessage) {\n        <span class=\"flex items-center mt-1 text-govbr-sm text-govbr-info\">\n          <i class=\"fa-solid fa-info-circle mr-1\"></i>{{ infoMessage }}\n        </span>\n      }\n    </div>\n    `,\n})\nexport class BrInputComponent implements ControlValueAccessor {\n  @Input() label?: string;\n  @Input() type:\n    | 'text'\n    | 'email'\n    | 'password'\n    | 'select'\n    | 'file'\n    | 'number'\n    | 'search'\n    | 'textarea' = 'text';\n  @Input() placeholder = '';\n  @Input() required = false;\n  /** Mensagem de erro manual (sobrescreve erros automáticos do validator) */\n  @Input() error?: string;\n  @Input() successMessage?: string;\n  @Input() warningMessage?: string;\n  @Input() infoMessage?: string;\n  @Input() helperText?: string;\n  @Input() icon?: string;\n  @Input() iconRight = false;\n  @Input() density: 'small' | 'medium' | 'large' = 'medium';\n  @Input() highlight = false;\n  @Input() inline = false;\n  @Input() maxLength?: number;\n  @Input() rows = 3;\n\n  /** Mensagens customizadas para erros de validação */\n  @Input() errorMessages: Record<string, string> = {};\n\n  @Output() fileSelected = new EventEmitter<File>();\n  @Output() searchSubmit = new EventEmitter<string>();\n\n  value: string = '';\n  isDisabled = false;\n  fileName = '';\n  showPassword = false;\n\n  private static nextId = 0;\n  inputId = `br-input-${BrInputComponent.nextId++}`;\n\n  private static readonly defaultErrorMessages: Record<string, string | ((err: unknown) => string)> = {\n    required: 'Campo obrigatório',\n    email: 'Email inválido',\n    minlength: (err: unknown) => {\n      const e = err as { requiredLength: number };\n      return `Mínimo de ${e.requiredLength} caracteres`;\n    },\n    maxlength: (err: unknown) => {\n      const e = err as { requiredLength: number };\n      return `Máximo de ${e.requiredLength} caracteres`;\n    },\n    min: (err: unknown) => {\n      const e = err as { min: number };\n      return `Valor mínimo: ${e.min}`;\n    },\n    max: (err: unknown) => {\n      const e = err as { max: number };\n      return `Valor máximo: ${e.max}`;\n    },\n    pattern: 'Formato inválido',\n  };\n\n  constructor(@Self() @Optional() private ngControl: NgControl) {\n    if (this.ngControl) {\n      this.ngControl.valueAccessor = this;\n    }\n  }\n\n  onChange: (value: string) => void = () => {};\n  onTouched: () => void = () => {};\n\n  get isStandardInput(): boolean {\n    return this.type !== 'select' && this.type !== 'file' && this.type !== 'textarea';\n  }\n\n  get actualType(): string {\n    if (this.type === 'password') {\n      return this.showPassword ? 'text' : 'password';\n    }\n    if (this.type === 'search') return 'text';\n    return this.type;\n  }\n\n  get containerClasses(): string {\n    return this.type !== 'file' ? 'relative' : '';\n  }\n\n  /**\n   * Retorna a mensagem de erro a ser exibida.\n   * Prioridade: error manual > erro do validator (se touched/dirty)\n   */\n  get displayError(): string | undefined {\n    // Erro manual sempre tem prioridade\n    if (this.error) {\n      return this.error;\n    }\n\n    // Se não tem NgControl, não há erro automático\n    if (!this.ngControl?.control) {\n      return undefined;\n    }\n\n    const control = this.ngControl.control;\n\n    // Só mostra erro se o controle foi tocado ou modificado E está inválido\n    if (!control.errors || (!control.touched && !control.dirty)) {\n      return undefined;\n    }\n\n    // Busca a primeira mensagem de erro\n    return this.getFirstErrorMessage(control.errors);\n  }\n\n  private getFirstErrorMessage(errors: ValidationErrors): string {\n    for (const key of Object.keys(errors)) {\n      // Primeiro verifica mensagens customizadas do usuário\n      if (this.errorMessages[key]) {\n        return this.errorMessages[key];\n      }\n\n      // Depois verifica mensagens padrão\n      const defaultMsg = BrInputComponent.defaultErrorMessages[key];\n      if (defaultMsg) {\n        return typeof defaultMsg === 'function' ? defaultMsg(errors[key]) : defaultMsg;\n      }\n    }\n\n    // Fallback genérico\n    return 'Campo inválido';\n  }\n\n  get inputClasses(): string {\n    const densityStyles = {\n      small: 'text-govbr-sm',\n      medium: 'text-govbr-base',\n      large: 'text-govbr-lg',\n    } as const;\n\n    const densityHeights = {\n      small: 'h-[var(--govbr-input-small)]',\n      medium: 'h-[var(--govbr-input-medium)]',\n      large: 'h-[var(--govbr-input-large)]',\n    } as const;\n\n    const base = [\n      'w-full px-4 font-rawline govbr-input',\n      this.type === 'textarea' ? 'py-2' : '',\n    ].join(' ');\n\n    const density = densityStyles[this.density];\n    const height = this.type !== 'textarea' ? densityHeights[this.density] : '';\n\n    const highlightCls = this.highlight\n      ? 'h-[var(--govbr-input-highlight)] bg-govbr-gray-2 border-transparent text-govbr-lg px-6'\n      : '';\n\n    const statusCls = this.displayError\n      ? 'govbr-input--danger'\n      : this.successMessage\n        ? 'govbr-input--success'\n        : this.warningMessage\n          ? 'govbr-input--warning'\n          : '';\n\n    const disabledCls = this.isDisabled\n      ? 'bg-govbr-gray-2 text-govbr-gray-40 cursor-not-allowed'\n      : 'bg-white text-govbr-gray-80';\n\n    const iconPaddingLeft = this.icon && !this.iconRight ? 'pl-10' : '';\n    const iconPaddingRight =\n      (this.icon && this.iconRight) || this.type === 'password' || this.type === 'search'\n        ? 'pr-10'\n        : '';\n\n    return [base, density, height, highlightCls, statusCls, disabledCls, iconPaddingLeft, iconPaddingRight]\n      .filter(Boolean)\n      .join(' ');\n  }\n\n  writeValue(value: string): void {\n    this.value = value ?? '';\n  }\n\n  registerOnChange(fn: (value: string) => void): void {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: () => void): void {\n    this.onTouched = fn;\n  }\n\n  setDisabledState(isDisabled: boolean): void {\n    this.isDisabled = isDisabled;\n  }\n\n  togglePassword(): void {\n    this.showPassword = !this.showPassword;\n  }\n\n  onInputChange(event: Event): void {\n    const target = event.target as HTMLInputElement | HTMLTextAreaElement;\n    this.value = target.value;\n    this.onChange(target.value);\n  }\n\n  onSelectChange(event: Event): void {\n    const val = (event.target as HTMLSelectElement).value;\n    this.value = val;\n    this.onChange(val);\n  }\n\n  onFileChange(event: Event): void {\n    const input = event.target as HTMLInputElement;\n    if (input.files && input.files.length > 0) {\n      const file = input.files[0];\n      this.fileName = file.name;\n      this.fileSelected.emit(file);\n    }\n  }\n}\n"]}
@@ -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"]}