@seniorsistemas/components-ai 1.0.0 → 1.0.1

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.
@@ -7,8 +7,8 @@ import { RadioButtonModule } from 'primeng/radiobutton';
7
7
  import { CheckboxModule } from 'primeng/checkbox';
8
8
  import { DividerModule } from 'primeng/divider';
9
9
  import * as XLSX from 'xlsx';
10
- import { jsPDF } from 'jspdf';
11
- import autoTable from 'jspdf-autotable';
10
+ import jsPDF from 'jspdf';
11
+ import 'jspdf-autotable';
12
12
  import { TranslatePipe } from '../../pipes/translate.pipe';
13
13
  import * as i0 from "@angular/core";
14
14
  import * as i1 from "../../services/translation.service";
@@ -156,7 +156,7 @@ export class ExportDialogComponent {
156
156
  });
157
157
  });
158
158
  // Tabela
159
- autoTable(doc, {
159
+ doc.autoTable({
160
160
  head: [headers],
161
161
  body: data,
162
162
  startY: 40,
@@ -208,4 +208,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
208
208
  }], visibleChange: [{
209
209
  type: Output
210
210
  }] } });
211
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"export-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/components-ai/src/lib/components/export-dialog/export-dialog.component.ts","../../../../../projects/components-ai/src/lib/components/export-dialog/export-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, Output, OnChanges } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { FormsModule } from '@angular/forms';\nimport { DialogModule } from 'primeng/dialog';\nimport { ButtonModule } from 'primeng/button';\nimport { RadioButtonModule } from 'primeng/radiobutton';\nimport { CheckboxModule } from 'primeng/checkbox';\nimport { DividerModule } from 'primeng/divider';\nimport * as XLSX from 'xlsx';\nimport { jsPDF } from 'jspdf';\nimport autoTable from 'jspdf-autotable';\nimport { TranslatePipe } from '../../pipes/translate.pipe';\nimport { TranslationService } from '../../services/translation.service';\n\nexport interface ColumnOption {\n  field: string;\n  header: string;\n  selected: boolean;\n}\n\ninterface ExportFormat {\n  value: 'xlsx' | 'pdf';\n  label: string;\n  icon: string;\n  description: string;\n}\n\ninterface ExportScope {\n  value: 'all' | 'selected';\n  label: string;\n  icon: string;\n  description: string;\n}\n\n@Component({\n  selector: 'sia-export-dialog',\n  standalone: true,\n  imports: [\n    CommonModule,\n    FormsModule,\n    DialogModule,\n    ButtonModule,\n    RadioButtonModule,\n    CheckboxModule,\n    DividerModule,\n    TranslatePipe\n  ],\n  templateUrl: './export-dialog.component.html',\n  styleUrl: './export-dialog.component.scss'\n})\nexport class ExportDialogComponent implements OnChanges {\n  @Input() visible: boolean = false;\n  @Input() allRecords: any[] = [];\n  @Input() selectedRecords: any[] = [];\n  @Input() statusLabels?: { [key: string]: string };\n  @Input() translationPrefix: string = 'design.components_ai';\n  @Input() set availableColumns(columns: ColumnOption[]) {\n    if (columns && columns.length > 0) {\n      this.columns = columns;\n    }\n  }\n  @Output() visibleChange = new EventEmitter<boolean>();\n\n  exportFormat: 'xlsx' | 'pdf' = 'xlsx';\n  exportScope: 'all' | 'selected' = 'all';\n\n  formats: ExportFormat[] = [];\n  scopes: ExportScope[] = [];\n  columns: ColumnOption[] = [];\n\n  constructor(private translationService: TranslationService) {\n    this.initializeFormats();\n    this.initializeScopes();\n  }\n\n  ngOnChanges(): void {\n    this.initializeFormats();\n    this.initializeScopes();\n  }\n\n  private initializeFormats(): void {\n    this.formats = [\n      {\n        value: 'xlsx',\n        label: 'Excel',\n        icon: 'pi-file-excel',\n        description: this.translationService.translate('design.components_ai.export_excel_description')\n      },\n      {\n        value: 'pdf',\n        label: 'PDF',\n        icon: 'pi-file-pdf',\n        description: this.translationService.translate('design.components_ai.export_pdf_description')\n      }\n    ];\n  }\n\n  private initializeScopes(): void {\n    this.scopes = [\n      {\n        value: 'all',\n        label: this.translationService.translate('design.components_ai.export_all_records'),\n        icon: 'pi-list',\n        description: this.translationService.translate('design.components_ai.export_all_records_description')\n      },\n      {\n        value: 'selected',\n        label: this.translationService.translate('design.components_ai.export_selected_records'),\n        icon: 'pi-check-square',\n        description: this.translationService.translate('design.components_ai.export_selected_records_description')\n      }\n    ];\n  }\n\n  get recordsToExport(): any[] {\n    return this.exportScope === 'all' ? this.allRecords : this.selectedRecords;\n  }\n\n  get selectedColumnsCount(): number {\n    return this.columns.filter(col => col.selected).length;\n  }\n\n  get canExport(): boolean {\n    return this.selectedColumnsCount > 0 && this.recordsToExport.length > 0;\n  }\n\n  onHide(): void {\n    this.visibleChange.emit(false);\n  }\n\n  selectAllColumns(): void {\n    this.columns.forEach(col => col.selected = true);\n  }\n\n  deselectAllColumns(): void {\n    this.columns.forEach(col => col.selected = false);\n  }\n\n  export(): void {\n    if (!this.canExport) return;\n\n    if (this.exportFormat === 'xlsx') {\n      this.exportToExcel();\n    } else {\n      this.exportToPDF();\n    }\n\n    this.onHide();\n  }\n\n  private exportToExcel(): void {\n    const selectedColumns = this.columns.filter(col => col.selected);\n    const data = this.recordsToExport.map(record => {\n      const row: any = {};\n      selectedColumns.forEach(col => {\n        if (col.field === 'status' && record.status && this.statusLabels) {\n          row[col.header] = this.statusLabels[record.status];\n        } else {\n          row[col.header] = record[col.field];\n        }\n      });\n      return row;\n    });\n\n    const worksheet = XLSX.utils.json_to_sheet(data);\n    const workbook = XLSX.utils.book_new();\n    XLSX.utils.book_append_sheet(workbook, worksheet, this.translationService.translate('design.components_ai.export_data_sheet_name'));\n\n    // Ajustar largura das colunas\n    const maxWidth = 30;\n    const colWidths = selectedColumns.map(col => ({\n      wch: Math.min(col.header.length + 5, maxWidth)\n    }));\n    worksheet['!cols'] = colWidths;\n\n    const fileName = `export-${new Date().getTime()}.xlsx`;\n    XLSX.writeFile(workbook, fileName);\n  }\n\n  private exportToPDF(): void {\n    const doc = new jsPDF();\n    const selectedColumns = this.columns.filter(col => col.selected);\n\n    // Título\n    doc.setFontSize(18);\n    doc.setTextColor(18, 168, 133);\n    doc.text(this.translationService.translate('design.components_ai.export_pdf_title'), 14, 20);\n\n    // Subtítulo\n    doc.setFontSize(10);\n    doc.setTextColor(100);\n    doc.text(`${this.translationService.translate('design.components_ai.export_exported_at')}: ${new Date().toLocaleString('pt-BR')}`, 14, 28);\n    doc.text(`${this.translationService.translate('design.components_ai.export_total_records')}: ${this.recordsToExport.length}`, 14, 34);\n\n    // Preparar dados\n    const headers = selectedColumns.map(col => col.header);\n    const data = this.recordsToExport.map(record => {\n      return selectedColumns.map(col => {\n        if (col.field === 'status' && record.status && this.statusLabels) {\n          return this.statusLabels[record.status];\n        }\n        return String(record[col.field] || '');\n      });\n    });\n\n    // Tabela\n    autoTable(doc, {\n      head: [headers],\n      body: data,\n      startY: 40,\n      theme: 'grid',\n      headStyles: {\n        fillColor: [18, 168, 133],\n        textColor: [255, 255, 255],\n        fontStyle: 'bold',\n        halign: 'center'\n      },\n      styles: {\n        fontSize: 9,\n        cellPadding: 3\n      },\n      alternateRowStyles: {\n        fillColor: [248, 255, 254]\n      }\n    });\n\n    const fileName = `export-${new Date().getTime()}.pdf`;\n    doc.save(fileName);\n  }\n}\n","<p-dialog\n  [(visible)]=\"visible\"\n  (onHide)=\"onHide()\"\n  [modal]=\"true\"\n  [draggable]=\"false\"\n  [resizable]=\"false\"\n  [style]=\"{ width: '700px' }\"\n  styleClass=\"export-dialog\"\n>\n  <ng-template pTemplate=\"header\">\n    <div class=\"custom-header\">\n      <i class=\"pi pi-download header-icon\"></i>\n      <span>{{ 'design.components_ai.export_data_title' | translate }}</span>\n    </div>\n  </ng-template>\n\n  <div class=\"export-content\">\n    <!-- Formato de Exportação -->\n    <div class=\"export-section\">\n      <div class=\"section-header\">\n        <div class=\"section-icon\">\n          <i class=\"pi pi-file\"></i>\n        </div>\n        <h3>{{ 'design.components_ai.export_format_title' | translate }}</h3>\n      </div>\n      <div class=\"format-options\">\n        <div\n          *ngFor=\"let format of formats\"\n          class=\"format-card\"\n          [class.selected]=\"exportFormat === format.value\"\n          (click)=\"exportFormat = format.value\"\n        >\n          <div class=\"card-radio\">\n            <p-radioButton\n              [value]=\"format.value\"\n              [(ngModel)]=\"exportFormat\"\n              [inputId]=\"format.value\"\n            ></p-radioButton>\n          </div>\n          <div class=\"card-icon\" [class.excel]=\"format.value === 'xlsx'\" [class.pdf]=\"format.value === 'pdf'\">\n            <i class=\"pi\" [ngClass]=\"format.icon\"></i>\n          </div>\n          <div class=\"card-info\">\n            <label [for]=\"format.value\" class=\"card-label\">{{ format.label }}</label>\n            <span class=\"card-description\">{{ format.description }}</span>\n          </div>\n        </div>\n      </div>\n    </div>\n\n    <!-- Escopo de Exportação -->\n    <div class=\"export-section\">\n      <div class=\"section-header\">\n        <div class=\"section-icon\">\n          <i class=\"pi pi-filter\"></i>\n        </div>\n        <h3>{{ 'design.components_ai.export_scope_title' | translate }}</h3>\n      </div>\n      <div class=\"scope-options\">\n        <div\n          *ngFor=\"let scope of scopes\"\n          class=\"scope-card\"\n          [class.selected]=\"exportScope === scope.value\"\n          [class.disabled]=\"scope.value === 'selected' && selectedRecords.length === 0\"\n          (click)=\"scope.value === 'selected' && selectedRecords.length === 0 ? null : exportScope = scope.value\"\n        >\n          <div class=\"card-radio\">\n            <p-radioButton\n              [value]=\"scope.value\"\n              [(ngModel)]=\"exportScope\"\n              [inputId]=\"scope.value\"\n              [disabled]=\"scope.value === 'selected' && selectedRecords.length === 0\"\n            ></p-radioButton>\n          </div>\n          <div class=\"card-icon\">\n            <i class=\"pi\" [ngClass]=\"scope.icon\"></i>\n          </div>\n          <div class=\"card-info\">\n            <label [for]=\"scope.value\" class=\"card-label\">{{ scope.label }}</label>\n            <span class=\"card-description\">{{ scope.description }}</span>\n            <span class=\"card-count\">\n              <i class=\"pi pi-circle-fill\"></i>\n              {{ scope.value === 'all' ? allRecords.length : selectedRecords.length }} {{ 'design.components_ai.export_records_count' | translate }}\n            </span>\n          </div>\n        </div>\n      </div>\n    </div>\n\n    <!-- Seleção de Colunas -->\n    <div class=\"export-section\">\n      <div class=\"section-header\">\n        <div class=\"section-icon\">\n          <i class=\"pi pi-table\"></i>\n        </div>\n        <h3>{{ 'design.components_ai.export_columns_title' | translate }}</h3>\n        <div class=\"column-actions\">\n          <button type=\"button\" class=\"action-link\" (click)=\"selectAllColumns()\">\n            {{ 'design.components_ai.export_select_all_columns' | translate }}\n          </button>\n          <span class=\"separator\">|</span>\n          <button type=\"button\" class=\"action-link\" (click)=\"deselectAllColumns()\">\n            {{ 'design.components_ai.export_clear_selection' | translate }}\n          </button>\n        </div>\n      </div>\n      <div class=\"columns-grid\">\n        <div\n          *ngFor=\"let column of columns\"\n          class=\"column-item\"\n          [class.selected]=\"column.selected\"\n          (click)=\"column.selected = !column.selected\"\n        >\n          <p-checkbox\n            [(ngModel)]=\"column.selected\"\n            [binary]=\"true\"\n            [inputId]=\"column.field\"\n          ></p-checkbox>\n          <label [for]=\"column.field\" class=\"column-label\">\n            {{ column.header }}\n          </label>\n        </div>\n      </div>\n      <div class=\"columns-summary\">\n        <i class=\"pi pi-info-circle\"></i>\n        <span>{{ selectedColumnsCount }} {{ 'design.components_ai.export_columns_summary' | translate }} {{ columns.length }} {{ 'design.components_ai.export_columns_selected' | translate }}</span>\n      </div>\n    </div>\n  </div>\n\n  <ng-template pTemplate=\"footer\">\n    <p-button\n      [label]=\"'design.components_ai.cancel' | translate\"\n      icon=\"pi pi-times\"\n      severity=\"secondary\"\n      [text]=\"true\"\n      (onClick)=\"onHide()\"\n    ></p-button>\n    <p-button\n      [label]=\"'design.components_ai.export_button' | translate\"\n      icon=\"pi pi-download\"\n      [disabled]=\"!canExport\"\n      (onClick)=\"export()\"\n    ></p-button>\n  </ng-template>\n</p-dialog>\n"]}
211
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"export-dialog.component.js","sourceRoot":"","sources":["../../../../../projects/components-ai/src/lib/components/export-dialog/export-dialog.component.ts","../../../../../projects/components-ai/src/lib/components/export-dialog/export-dialog.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAE,MAAM,EAAa,MAAM,eAAe,CAAC;AAClF,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;AAC9C,OAAO,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;AAC9C,OAAO,EAAE,iBAAiB,EAAE,MAAM,qBAAqB,CAAC;AACxD,OAAO,EAAE,cAAc,EAAE,MAAM,kBAAkB,CAAC;AAClD,OAAO,EAAE,aAAa,EAAE,MAAM,iBAAiB,CAAC;AAChD,OAAO,KAAK,IAAI,MAAM,MAAM,CAAC;AAC7B,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,iBAAiB,CAAC;AACzB,OAAO,EAAE,aAAa,EAAE,MAAM,4BAA4B,CAAC;;;;;;;;;;AAuC3D,MAAM,OAAO,qBAAqB;IAoBZ;IAnBX,OAAO,GAAY,KAAK,CAAC;IACzB,UAAU,GAAU,EAAE,CAAC;IACvB,eAAe,GAAU,EAAE,CAAC;IAC5B,YAAY,CAA6B;IACzC,iBAAiB,GAAW,sBAAsB,CAAC;IAC5D,IAAa,gBAAgB,CAAC,OAAuB;QACnD,IAAI,OAAO,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAClC,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;QACzB,CAAC;IACH,CAAC;IACS,aAAa,GAAG,IAAI,YAAY,EAAW,CAAC;IAEtD,YAAY,GAAmB,MAAM,CAAC;IACtC,WAAW,GAAuB,KAAK,CAAC;IAExC,OAAO,GAAmB,EAAE,CAAC;IAC7B,MAAM,GAAkB,EAAE,CAAC;IAC3B,OAAO,GAAmB,EAAE,CAAC;IAE7B,YAAoB,kBAAsC;QAAtC,uBAAkB,GAAlB,kBAAkB,CAAoB;QACxD,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,IAAI,CAAC,gBAAgB,EAAE,CAAC;IAC1B,CAAC;IAED,WAAW;QACT,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,IAAI,CAAC,gBAAgB,EAAE,CAAC;IAC1B,CAAC;IAEO,iBAAiB;QACvB,IAAI,CAAC,OAAO,GAAG;YACb;gBACE,KAAK,EAAE,MAAM;gBACb,KAAK,EAAE,OAAO;gBACd,IAAI,EAAE,eAAe;gBACrB,WAAW,EAAE,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,+CAA+C,CAAC;aAChG;YACD;gBACE,KAAK,EAAE,KAAK;gBACZ,KAAK,EAAE,KAAK;gBACZ,IAAI,EAAE,aAAa;gBACnB,WAAW,EAAE,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,6CAA6C,CAAC;aAC9F;SACF,CAAC;IACJ,CAAC;IAEO,gBAAgB;QACtB,IAAI,CAAC,MAAM,GAAG;YACZ;gBACE,KAAK,EAAE,KAAK;gBACZ,KAAK,EAAE,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,yCAAyC,CAAC;gBACnF,IAAI,EAAE,SAAS;gBACf,WAAW,EAAE,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,qDAAqD,CAAC;aACtG;YACD;gBACE,KAAK,EAAE,UAAU;gBACjB,KAAK,EAAE,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,8CAA8C,CAAC;gBACxF,IAAI,EAAE,iBAAiB;gBACvB,WAAW,EAAE,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,0DAA0D,CAAC;aAC3G;SACF,CAAC;IACJ,CAAC;IAED,IAAI,eAAe;QACjB,OAAO,IAAI,CAAC,WAAW,KAAK,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC;IAC7E,CAAC;IAED,IAAI,oBAAoB;QACtB,OAAO,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC;IACzD,CAAC;IAED,IAAI,SAAS;QACX,OAAO,IAAI,CAAC,oBAAoB,GAAG,CAAC,IAAI,IAAI,CAAC,eAAe,CAAC,MAAM,GAAG,CAAC,CAAC;IAC1E,CAAC;IAED,MAAM;QACJ,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACjC,CAAC;IAED,gBAAgB;QACd,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,QAAQ,GAAG,IAAI,CAAC,CAAC;IACnD,CAAC;IAED,kBAAkB;QAChB,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,QAAQ,GAAG,KAAK,CAAC,CAAC;IACpD,CAAC;IAED,MAAM;QACJ,IAAI,CAAC,IAAI,CAAC,SAAS;YAAE,OAAO;QAE5B,IAAI,IAAI,CAAC,YAAY,KAAK,MAAM,EAAE,CAAC;YACjC,IAAI,CAAC,aAAa,EAAE,CAAC;QACvB,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,WAAW,EAAE,CAAC;QACrB,CAAC;QAED,IAAI,CAAC,MAAM,EAAE,CAAC;IAChB,CAAC;IAEO,aAAa;QACnB,MAAM,eAAe,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QACjE,MAAM,IAAI,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE;YAC7C,MAAM,GAAG,GAAQ,EAAE,CAAC;YACpB,eAAe,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE;gBAC5B,IAAI,GAAG,CAAC,KAAK,KAAK,QAAQ,IAAI,MAAM,CAAC,MAAM,IAAI,IAAI,CAAC,YAAY,EAAE,CAAC;oBACjE,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;gBACrD,CAAC;qBAAM,CAAC;oBACN,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,GAAG,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;gBACtC,CAAC;YACH,CAAC,CAAC,CAAC;YACH,OAAO,GAAG,CAAC;QACb,CAAC,CAAC,CAAC;QAEH,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;QACjD,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC;QACvC,IAAI,CAAC,KAAK,CAAC,iBAAiB,CAAC,QAAQ,EAAE,SAAS,EAAE,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,6CAA6C,CAAC,CAAC,CAAC;QAEpI,8BAA8B;QAC9B,MAAM,QAAQ,GAAG,EAAE,CAAC;QACpB,MAAM,SAAS,GAAG,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YAC5C,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,QAAQ,CAAC;SAC/C,CAAC,CAAC,CAAC;QACJ,SAAS,CAAC,OAAO,CAAC,GAAG,SAAS,CAAC;QAE/B,MAAM,QAAQ,GAAG,UAAU,IAAI,IAAI,EAAE,CAAC,OAAO,EAAE,OAAO,CAAC;QACvD,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;IACrC,CAAC;IAEO,WAAW;QACjB,MAAM,GAAG,GAAG,IAAI,KAAK,EAAE,CAAC;QACxB,MAAM,eAAe,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAEjE,SAAS;QACT,GAAG,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;QACpB,GAAG,CAAC,YAAY,CAAC,EAAE,EAAE,GAAG,EAAE,GAAG,CAAC,CAAC;QAC/B,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,uCAAuC,CAAC,EAAE,EAAE,EAAE,EAAE,CAAC,CAAC;QAE7F,YAAY;QACZ,GAAG,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;QACpB,GAAG,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;QACtB,GAAG,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,yCAAyC,CAAC,KAAK,IAAI,IAAI,EAAE,CAAC,cAAc,CAAC,OAAO,CAAC,EAAE,EAAE,EAAE,EAAE,EAAE,CAAC,CAAC;QAC3I,GAAG,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,kBAAkB,CAAC,SAAS,CAAC,2CAA2C,CAAC,KAAK,IAAI,CAAC,eAAe,CAAC,MAAM,EAAE,EAAE,EAAE,EAAE,EAAE,CAAC,CAAC;QAEtI,iBAAiB;QACjB,MAAM,OAAO,GAAG,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QACvD,MAAM,IAAI,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE;YAC7C,OAAO,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE;gBAC/B,IAAI,GAAG,CAAC,KAAK,KAAK,QAAQ,IAAI,MAAM,CAAC,MAAM,IAAI,IAAI,CAAC,YAAY,EAAE,CAAC;oBACjE,OAAO,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;gBAC1C,CAAC;gBACD,OAAO,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC,CAAC;YACzC,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,SAAS;QACR,GAAW,CAAC,SAAS,CAAC;YACrB,IAAI,EAAE,CAAC,OAAO,CAAC;YACf,IAAI,EAAE,IAAI;YACV,MAAM,EAAE,EAAE;YACV,KAAK,EAAE,MAAM;YACb,UAAU,EAAE;gBACV,SAAS,EAAE,CAAC,EAAE,EAAE,GAAG,EAAE,GAAG,CAAC;gBACzB,SAAS,EAAE,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC;gBAC1B,SAAS,EAAE,MAAM;gBACjB,MAAM,EAAE,QAAQ;aACjB;YACD,MAAM,EAAE;gBACN,QAAQ,EAAE,CAAC;gBACX,WAAW,EAAE,CAAC;aACf;YACD,kBAAkB,EAAE;gBAClB,SAAS,EAAE,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC;aAC3B;SACF,CAAC,CAAC;QAEH,MAAM,QAAQ,GAAG,UAAU,IAAI,IAAI,EAAE,CAAC,OAAO,EAAE,MAAM,CAAC;QACtD,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IACrB,CAAC;wGAlLU,qBAAqB;4FAArB,qBAAqB,yVClDlC,i7KAkJA,01ND5GI,YAAY,yPACZ,WAAW,8VACX,YAAY,0gCACZ,YAAY,ibACZ,iBAAiB,iWACjB,cAAc,kcACd,aAAa,0BACb,aAAa;;4FAKJ,qBAAqB;kBAhBjC,SAAS;+BACE,mBAAmB,cACjB,IAAI,WACP;wBACP,YAAY;wBACZ,WAAW;wBACX,YAAY;wBACZ,YAAY;wBACZ,iBAAiB;wBACjB,cAAc;wBACd,aAAa;wBACb,aAAa;qBACd;uFAKQ,OAAO;sBAAf,KAAK;gBACG,UAAU;sBAAlB,KAAK;gBACG,eAAe;sBAAvB,KAAK;gBACG,YAAY;sBAApB,KAAK;gBACG,iBAAiB;sBAAzB,KAAK;gBACO,gBAAgB;sBAA5B,KAAK;gBAKI,aAAa;sBAAtB,MAAM","sourcesContent":["import { Component, EventEmitter, Input, Output, OnChanges } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { FormsModule } from '@angular/forms';\nimport { DialogModule } from 'primeng/dialog';\nimport { ButtonModule } from 'primeng/button';\nimport { RadioButtonModule } from 'primeng/radiobutton';\nimport { CheckboxModule } from 'primeng/checkbox';\nimport { DividerModule } from 'primeng/divider';\nimport * as XLSX from 'xlsx';\nimport jsPDF from 'jspdf';\nimport 'jspdf-autotable';\nimport { TranslatePipe } from '../../pipes/translate.pipe';\nimport { TranslationService } from '../../services/translation.service';\n\nexport interface ColumnOption {\n  field: string;\n  header: string;\n  selected: boolean;\n}\n\ninterface ExportFormat {\n  value: 'xlsx' | 'pdf';\n  label: string;\n  icon: string;\n  description: string;\n}\n\ninterface ExportScope {\n  value: 'all' | 'selected';\n  label: string;\n  icon: string;\n  description: string;\n}\n\n@Component({\n  selector: 'sia-export-dialog',\n  standalone: true,\n  imports: [\n    CommonModule,\n    FormsModule,\n    DialogModule,\n    ButtonModule,\n    RadioButtonModule,\n    CheckboxModule,\n    DividerModule,\n    TranslatePipe\n  ],\n  templateUrl: './export-dialog.component.html',\n  styleUrl: './export-dialog.component.scss'\n})\nexport class ExportDialogComponent implements OnChanges {\n  @Input() visible: boolean = false;\n  @Input() allRecords: any[] = [];\n  @Input() selectedRecords: any[] = [];\n  @Input() statusLabels?: { [key: string]: string };\n  @Input() translationPrefix: string = 'design.components_ai';\n  @Input() set availableColumns(columns: ColumnOption[]) {\n    if (columns && columns.length > 0) {\n      this.columns = columns;\n    }\n  }\n  @Output() visibleChange = new EventEmitter<boolean>();\n\n  exportFormat: 'xlsx' | 'pdf' = 'xlsx';\n  exportScope: 'all' | 'selected' = 'all';\n\n  formats: ExportFormat[] = [];\n  scopes: ExportScope[] = [];\n  columns: ColumnOption[] = [];\n\n  constructor(private translationService: TranslationService) {\n    this.initializeFormats();\n    this.initializeScopes();\n  }\n\n  ngOnChanges(): void {\n    this.initializeFormats();\n    this.initializeScopes();\n  }\n\n  private initializeFormats(): void {\n    this.formats = [\n      {\n        value: 'xlsx',\n        label: 'Excel',\n        icon: 'pi-file-excel',\n        description: this.translationService.translate('design.components_ai.export_excel_description')\n      },\n      {\n        value: 'pdf',\n        label: 'PDF',\n        icon: 'pi-file-pdf',\n        description: this.translationService.translate('design.components_ai.export_pdf_description')\n      }\n    ];\n  }\n\n  private initializeScopes(): void {\n    this.scopes = [\n      {\n        value: 'all',\n        label: this.translationService.translate('design.components_ai.export_all_records'),\n        icon: 'pi-list',\n        description: this.translationService.translate('design.components_ai.export_all_records_description')\n      },\n      {\n        value: 'selected',\n        label: this.translationService.translate('design.components_ai.export_selected_records'),\n        icon: 'pi-check-square',\n        description: this.translationService.translate('design.components_ai.export_selected_records_description')\n      }\n    ];\n  }\n\n  get recordsToExport(): any[] {\n    return this.exportScope === 'all' ? this.allRecords : this.selectedRecords;\n  }\n\n  get selectedColumnsCount(): number {\n    return this.columns.filter(col => col.selected).length;\n  }\n\n  get canExport(): boolean {\n    return this.selectedColumnsCount > 0 && this.recordsToExport.length > 0;\n  }\n\n  onHide(): void {\n    this.visibleChange.emit(false);\n  }\n\n  selectAllColumns(): void {\n    this.columns.forEach(col => col.selected = true);\n  }\n\n  deselectAllColumns(): void {\n    this.columns.forEach(col => col.selected = false);\n  }\n\n  export(): void {\n    if (!this.canExport) return;\n\n    if (this.exportFormat === 'xlsx') {\n      this.exportToExcel();\n    } else {\n      this.exportToPDF();\n    }\n\n    this.onHide();\n  }\n\n  private exportToExcel(): void {\n    const selectedColumns = this.columns.filter(col => col.selected);\n    const data = this.recordsToExport.map(record => {\n      const row: any = {};\n      selectedColumns.forEach(col => {\n        if (col.field === 'status' && record.status && this.statusLabels) {\n          row[col.header] = this.statusLabels[record.status];\n        } else {\n          row[col.header] = record[col.field];\n        }\n      });\n      return row;\n    });\n\n    const worksheet = XLSX.utils.json_to_sheet(data);\n    const workbook = XLSX.utils.book_new();\n    XLSX.utils.book_append_sheet(workbook, worksheet, this.translationService.translate('design.components_ai.export_data_sheet_name'));\n\n    // Ajustar largura das colunas\n    const maxWidth = 30;\n    const colWidths = selectedColumns.map(col => ({\n      wch: Math.min(col.header.length + 5, maxWidth)\n    }));\n    worksheet['!cols'] = colWidths;\n\n    const fileName = `export-${new Date().getTime()}.xlsx`;\n    XLSX.writeFile(workbook, fileName);\n  }\n\n  private exportToPDF(): void {\n    const doc = new jsPDF();\n    const selectedColumns = this.columns.filter(col => col.selected);\n\n    // Título\n    doc.setFontSize(18);\n    doc.setTextColor(18, 168, 133);\n    doc.text(this.translationService.translate('design.components_ai.export_pdf_title'), 14, 20);\n\n    // Subtítulo\n    doc.setFontSize(10);\n    doc.setTextColor(100);\n    doc.text(`${this.translationService.translate('design.components_ai.export_exported_at')}: ${new Date().toLocaleString('pt-BR')}`, 14, 28);\n    doc.text(`${this.translationService.translate('design.components_ai.export_total_records')}: ${this.recordsToExport.length}`, 14, 34);\n\n    // Preparar dados\n    const headers = selectedColumns.map(col => col.header);\n    const data = this.recordsToExport.map(record => {\n      return selectedColumns.map(col => {\n        if (col.field === 'status' && record.status && this.statusLabels) {\n          return this.statusLabels[record.status];\n        }\n        return String(record[col.field] || '');\n      });\n    });\n\n    // Tabela\n    (doc as any).autoTable({\n      head: [headers],\n      body: data,\n      startY: 40,\n      theme: 'grid',\n      headStyles: {\n        fillColor: [18, 168, 133],\n        textColor: [255, 255, 255],\n        fontStyle: 'bold',\n        halign: 'center'\n      },\n      styles: {\n        fontSize: 9,\n        cellPadding: 3\n      },\n      alternateRowStyles: {\n        fillColor: [248, 255, 254]\n      }\n    });\n\n    const fileName = `export-${new Date().getTime()}.pdf`;\n    doc.save(fileName);\n  }\n}\n","<p-dialog\n  [(visible)]=\"visible\"\n  (onHide)=\"onHide()\"\n  [modal]=\"true\"\n  [draggable]=\"false\"\n  [resizable]=\"false\"\n  [style]=\"{ width: '700px' }\"\n  styleClass=\"export-dialog\"\n>\n  <ng-template pTemplate=\"header\">\n    <div class=\"custom-header\">\n      <i class=\"pi pi-download header-icon\"></i>\n      <span>{{ 'design.components_ai.export_data_title' | translate }}</span>\n    </div>\n  </ng-template>\n\n  <div class=\"export-content\">\n    <!-- Formato de Exportação -->\n    <div class=\"export-section\">\n      <div class=\"section-header\">\n        <div class=\"section-icon\">\n          <i class=\"pi pi-file\"></i>\n        </div>\n        <h3>{{ 'design.components_ai.export_format_title' | translate }}</h3>\n      </div>\n      <div class=\"format-options\">\n        <div\n          *ngFor=\"let format of formats\"\n          class=\"format-card\"\n          [class.selected]=\"exportFormat === format.value\"\n          (click)=\"exportFormat = format.value\"\n        >\n          <div class=\"card-radio\">\n            <p-radioButton\n              [value]=\"format.value\"\n              [(ngModel)]=\"exportFormat\"\n              [inputId]=\"format.value\"\n            ></p-radioButton>\n          </div>\n          <div class=\"card-icon\" [class.excel]=\"format.value === 'xlsx'\" [class.pdf]=\"format.value === 'pdf'\">\n            <i class=\"pi\" [ngClass]=\"format.icon\"></i>\n          </div>\n          <div class=\"card-info\">\n            <label [for]=\"format.value\" class=\"card-label\">{{ format.label }}</label>\n            <span class=\"card-description\">{{ format.description }}</span>\n          </div>\n        </div>\n      </div>\n    </div>\n\n    <!-- Escopo de Exportação -->\n    <div class=\"export-section\">\n      <div class=\"section-header\">\n        <div class=\"section-icon\">\n          <i class=\"pi pi-filter\"></i>\n        </div>\n        <h3>{{ 'design.components_ai.export_scope_title' | translate }}</h3>\n      </div>\n      <div class=\"scope-options\">\n        <div\n          *ngFor=\"let scope of scopes\"\n          class=\"scope-card\"\n          [class.selected]=\"exportScope === scope.value\"\n          [class.disabled]=\"scope.value === 'selected' && selectedRecords.length === 0\"\n          (click)=\"scope.value === 'selected' && selectedRecords.length === 0 ? null : exportScope = scope.value\"\n        >\n          <div class=\"card-radio\">\n            <p-radioButton\n              [value]=\"scope.value\"\n              [(ngModel)]=\"exportScope\"\n              [inputId]=\"scope.value\"\n              [disabled]=\"scope.value === 'selected' && selectedRecords.length === 0\"\n            ></p-radioButton>\n          </div>\n          <div class=\"card-icon\">\n            <i class=\"pi\" [ngClass]=\"scope.icon\"></i>\n          </div>\n          <div class=\"card-info\">\n            <label [for]=\"scope.value\" class=\"card-label\">{{ scope.label }}</label>\n            <span class=\"card-description\">{{ scope.description }}</span>\n            <span class=\"card-count\">\n              <i class=\"pi pi-circle-fill\"></i>\n              {{ scope.value === 'all' ? allRecords.length : selectedRecords.length }} {{ 'design.components_ai.export_records_count' | translate }}\n            </span>\n          </div>\n        </div>\n      </div>\n    </div>\n\n    <!-- Seleção de Colunas -->\n    <div class=\"export-section\">\n      <div class=\"section-header\">\n        <div class=\"section-icon\">\n          <i class=\"pi pi-table\"></i>\n        </div>\n        <h3>{{ 'design.components_ai.export_columns_title' | translate }}</h3>\n        <div class=\"column-actions\">\n          <button type=\"button\" class=\"action-link\" (click)=\"selectAllColumns()\">\n            {{ 'design.components_ai.export_select_all_columns' | translate }}\n          </button>\n          <span class=\"separator\">|</span>\n          <button type=\"button\" class=\"action-link\" (click)=\"deselectAllColumns()\">\n            {{ 'design.components_ai.export_clear_selection' | translate }}\n          </button>\n        </div>\n      </div>\n      <div class=\"columns-grid\">\n        <div\n          *ngFor=\"let column of columns\"\n          class=\"column-item\"\n          [class.selected]=\"column.selected\"\n          (click)=\"column.selected = !column.selected\"\n        >\n          <p-checkbox\n            [(ngModel)]=\"column.selected\"\n            [binary]=\"true\"\n            [inputId]=\"column.field\"\n          ></p-checkbox>\n          <label [for]=\"column.field\" class=\"column-label\">\n            {{ column.header }}\n          </label>\n        </div>\n      </div>\n      <div class=\"columns-summary\">\n        <i class=\"pi pi-info-circle\"></i>\n        <span>{{ selectedColumnsCount }} {{ 'design.components_ai.export_columns_summary' | translate }} {{ columns.length }} {{ 'design.components_ai.export_columns_selected' | translate }}</span>\n      </div>\n    </div>\n  </div>\n\n  <ng-template pTemplate=\"footer\">\n    <p-button\n      [label]=\"'design.components_ai.cancel' | translate\"\n      icon=\"pi pi-times\"\n      severity=\"secondary\"\n      [text]=\"true\"\n      (onClick)=\"onHide()\"\n    ></p-button>\n    <p-button\n      [label]=\"'design.components_ai.export_button' | translate\"\n      icon=\"pi pi-download\"\n      [disabled]=\"!canExport\"\n      (onClick)=\"export()\"\n    ></p-button>\n  </ng-template>\n</p-dialog>\n"]}
@@ -1,4 +1,7 @@
1
1
  {
2
+ "design.components_ai.error": "Oops, something went wrong",
3
+ "design.components_ai.server_error": "Server Error",
4
+ "design.components_ai.server_error_message": "A server error occurred. Please try again later.",
2
5
  "design.components_ai.bulk_delete_title": "Bulk Delete",
3
6
  "design.components_ai.bulk_delete_warning": "You are about to delete <strong>{{count}}</strong> record(s).",
4
7
  "design.components_ai.bulk_delete_description": "This action cannot be undone. Records will be deleted one by one.",
@@ -41,7 +44,6 @@
41
44
  "design.components_ai.close": "Close",
42
45
  "design.components_ai.filter_search_placeholder": "Search by {{label}}",
43
46
  "design.components_ai.filter_select_placeholder": "Select {{label}}",
44
- "design.components_ai.error": "Error",
45
47
  "design.components_ai.success": "Success",
46
48
  "design.components_ai.load_error": "Error loading {{entityType}}",
47
49
  "design.components_ai.confirm_delete": "Confirm Deletion",
@@ -1,4 +1,7 @@
1
1
  {
2
+ "design.components_ai.error": "Ups, algo salió mal",
3
+ "design.components_ai.server_error": "Error del Servidor",
4
+ "design.components_ai.server_error_message": "Ocurrió un error en el servidor. Inténtelo de nuevo más tarde.",
2
5
  "design.components_ai.bulk_delete_title": "Eliminación Masiva",
3
6
  "design.components_ai.bulk_delete_warning": "Está a punto de eliminar <strong>{{count}}</strong> registro(s).",
4
7
  "design.components_ai.bulk_delete_description": "Esta acción no se puede deshacer. Los registros se eliminarán uno por uno.",
@@ -41,7 +44,6 @@
41
44
  "design.components_ai.close": "Cerrar",
42
45
  "design.components_ai.filter_search_placeholder": "Buscar por {{label}}",
43
46
  "design.components_ai.filter_select_placeholder": "Seleccione {{label}}",
44
- "design.components_ai.error": "Error",
45
47
  "design.components_ai.success": "Éxito",
46
48
  "design.components_ai.load_error": "Error al cargar {{entityType}}",
47
49
  "design.components_ai.confirm_delete": "Confirmar Eliminación",
@@ -1,4 +1,7 @@
1
1
  {
2
+ "design.components_ai.error": "Ops, algo deu errado",
3
+ "design.components_ai.server_error": "Erro no Servidor",
4
+ "design.components_ai.server_error_message": "Ocorreu um erro no servidor. Tente novamente mais tarde.",
2
5
  "design.components_ai.bulk_delete_title": "Exclusão em Massa",
3
6
  "design.components_ai.bulk_delete_warning": "Você está prestes a excluir <strong>{{count}}</strong> registro(s).",
4
7
  "design.components_ai.bulk_delete_description": "Esta ação não pode ser desfeita. Os registros serão excluídos um por vez.",
@@ -41,7 +44,6 @@
41
44
  "design.components_ai.close": "Fechar",
42
45
  "design.components_ai.filter_search_placeholder": "Pesquisar por {{label}}",
43
46
  "design.components_ai.filter_select_placeholder": "Selecione {{label}}",
44
- "design.components_ai.error": "Erro",
45
47
  "design.components_ai.success": "Sucesso",
46
48
  "design.components_ai.load_error": "Erro ao carregar {{entityType}}",
47
49
  "design.components_ai.confirm_delete": "Confirmar Exclusão",
@@ -55,7 +55,7 @@ export const apiInterceptor = (req, next) => {
55
55
  const errorMessage = extractErrorMessage(error);
56
56
  messageService.add({
57
57
  severity: 'error',
58
- summary: translationService.translate('common.error') || 'Erro',
58
+ summary: translationService.translate('design.components_ai.error') || 'Erro',
59
59
  detail: errorMessage,
60
60
  life: 5000
61
61
  });
@@ -64,11 +64,10 @@ export const apiInterceptor = (req, next) => {
64
64
  // Handle 5xx errors
65
65
  else if (error.status >= 500) {
66
66
  if (messageService && translationService) {
67
- const errorMessage = extractErrorMessage(error);
68
67
  messageService.add({
69
68
  severity: 'error',
70
- summary: translationService.translate('common.server_error') || 'Erro no Servidor',
71
- detail: errorMessage || translationService.translate('common.server_error_message') || 'Ocorreu um erro no servidor. Tente novamente mais tarde.',
69
+ summary: translationService.translate('design.components_ai.server_error') || 'Erro no Servidor',
70
+ detail: translationService.translate('design.components_ai.server_error_message') || 'Ocorreu um erro no servidor. Tente novamente mais tarde.',
72
71
  life: 5000
73
72
  });
74
73
  }
@@ -120,4 +119,4 @@ function extractErrorMessage(error) {
120
119
  // Fallback to error.message or statusText
121
120
  return error.message || error.statusText || 'Erro desconhecido';
122
121
  }
123
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"api.interceptor.js","sourceRoot":"","sources":["../../../../projects/components-ai/src/lib/interceptors/api.interceptor.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AACjD,OAAO,EAAE,MAAM,EAAE,MAAM,iBAAiB,CAAC;AACzC,OAAO,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,MAAM,CAAC;AAC9C,OAAO,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;AAC7C,OAAO,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAC;AACvD,OAAO,EAAE,kBAAkB,EAAE,MAAM,iCAAiC,CAAC;AAErE,MAAM,CAAC,MAAM,cAAc,GAAsB,CAAC,GAAyB,EAAE,IAAmB,EAAE,EAAE;IAClG,MAAM,WAAW,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;IACxC,MAAM,QAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC,CAAC;IAElC,iDAAiD;IACjD,MAAM,WAAW,GAAG,WAAW,CAAC,cAAc,EAAE,CAAC;IACjD,MAAM,SAAS,GAAG,WAAW,CAAC,YAAY,EAAE,CAAC;IAC7C,MAAM,OAAO,GAAG,WAAW,CAAC,aAAa,EAAE,CAAC;IAE5C,IAAI,WAAW,GAAG,GAAG,CAAC;IAEtB,2CAA2C;IAC3C,IAAI,WAAW,IAAI,SAAS,EAAE,CAAC;QAC7B,MAAM,UAAU,GAAG,GAAG,SAAS,IAAI,WAAW,EAAE,CAAC;QACjD,WAAW,GAAG,WAAW,CAAC,KAAK,CAAC;YAC9B,UAAU,EAAE;gBACV,aAAa,EAAE,UAAU;aAC1B;SACF,CAAC,CAAC;IACL,CAAC;IAED,gEAAgE;IAChE,IAAI,OAAO,IAAI,CAAC,aAAa,CAAC,WAAW,CAAC,GAAG,CAAC,EAAE,CAAC;QAC/C,MAAM,OAAO,GAAG,WAAW,CAAC,OAAO,EAAE,WAAW,CAAC,GAAG,CAAC,CAAC;QACtD,WAAW,GAAG,WAAW,CAAC,KAAK,CAAC;YAC9B,GAAG,EAAE,OAAO;SACb,CAAC,CAAC;IACL,CAAC;IAED,OAAO,IAAI,CAAC,WAAW,CAAC,CAAC,IAAI,CAC3B,UAAU,CAAC,CAAC,KAAwB,EAAE,EAAE;QACtC,mDAAmD;QACnD,IAAI,cAAc,GAA0B,IAAI,CAAC;QACjD,IAAI,kBAAkB,GAA8B,IAAI,CAAC;QAEzD,IAAI,CAAC;YACH,cAAc,GAAG,QAAQ,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC;YAC9C,kBAAkB,GAAG,QAAQ,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC;QACxD,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,CAAC,IAAI,CAAC,oDAAoD,CAAC,CAAC;QACrE,CAAC;QAED,iCAAiC;QACjC,IAAI,KAAK,CAAC,MAAM,KAAK,GAAG,EAAE,CAAC;YACzB,IAAI,CAAC;gBACH,MAAM,MAAM,GAAG,QAAQ,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;gBACpC,MAAM,CAAC,QAAQ,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC;YAClC,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,OAAO,CAAC,IAAI,CAAC,kDAAkD,CAAC,CAAC;YACnE,CAAC;QACH,CAAC;QACD,iCAAiC;aAC5B,IAAI,KAAK,CAAC,MAAM,IAAI,GAAG,IAAI,KAAK,CAAC,MAAM,GAAG,GAAG,EAAE,CAAC;YACnD,IAAI,cAAc,IAAI,kBAAkB,EAAE,CAAC;gBACzC,MAAM,YAAY,GAAG,mBAAmB,CAAC,KAAK,CAAC,CAAC;gBAChD,cAAc,CAAC,GAAG,CAAC;oBACjB,QAAQ,EAAE,OAAO;oBACjB,OAAO,EAAE,kBAAkB,CAAC,SAAS,CAAC,cAAc,CAAC,IAAI,MAAM;oBAC/D,MAAM,EAAE,YAAY;oBACpB,IAAI,EAAE,IAAI;iBACX,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QACD,oBAAoB;aACf,IAAI,KAAK,CAAC,MAAM,IAAI,GAAG,EAAE,CAAC;YAC7B,IAAI,cAAc,IAAI,kBAAkB,EAAE,CAAC;gBACzC,MAAM,YAAY,GAAG,mBAAmB,CAAC,KAAK,CAAC,CAAC;gBAChD,cAAc,CAAC,GAAG,CAAC;oBACjB,QAAQ,EAAE,OAAO;oBACjB,OAAO,EAAE,kBAAkB,CAAC,SAAS,CAAC,qBAAqB,CAAC,IAAI,kBAAkB;oBAClF,MAAM,EAAE,YAAY,IAAI,kBAAkB,CAAC,SAAS,CAAC,6BAA6B,CAAC,IAAI,0DAA0D;oBACjJ,IAAI,EAAE,IAAI;iBACX,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAED,4DAA4D;QAC5D,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,CAAC;IACjC,CAAC,CAAC,CACH,CAAC;AACJ,CAAC,CAAC;AAEF;;GAEG;AACH,SAAS,aAAa,CAAC,GAAW;IAChC,OAAO,cAAc,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;AAClC,CAAC;AAED;;GAEG;AACH,SAAS,WAAW,CAAC,OAAe,EAAE,WAAmB;IACvD,sCAAsC;IACtC,MAAM,YAAY,GAAG,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;IAEhD,yCAAyC;IACzC,MAAM,gBAAgB,GAAG,WAAW,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;IAExD,OAAO,GAAG,YAAY,IAAI,gBAAgB,EAAE,CAAC;AAC/C,CAAC;AAED;;;GAGG;AACH,SAAS,mBAAmB,CAAC,KAAwB;IACnD,yDAAyD;IACzD,IAAI,KAAK,CAAC,KAAK,IAAI,OAAO,KAAK,CAAC,KAAK,KAAK,QAAQ,EAAE,CAAC;QACnD,6FAA6F;QAC7F,IAAI,KAAK,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC;YACxB,OAAO,KAAK,CAAC,KAAK,CAAC,OAAO,CAAC;QAC7B,CAAC;QAED,qDAAqD;QACrD,IAAI,KAAK,CAAC,KAAK,CAAC,iBAAiB,EAAE,CAAC;YAClC,OAAO,KAAK,CAAC,KAAK,CAAC,iBAAiB,CAAC;QACvC,CAAC;QAED,4BAA4B;QAC5B,IAAI,KAAK,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;YACvB,OAAO,KAAK,CAAC,KAAK,CAAC,MAAM,CAAC;QAC5B,CAAC;QAED,4BAA4B;QAC5B,IAAI,KAAK,CAAC,KAAK,CAAC,MAAM,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,KAAK,CAAC,KAAK,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC7F,OAAO,KAAK,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACvE,CAAC;IACH,CAAC;IAED,0CAA0C;IAC1C,OAAO,KAAK,CAAC,OAAO,IAAI,KAAK,CAAC,UAAU,IAAI,mBAAmB,CAAC;AAClE,CAAC","sourcesContent":["import { HttpInterceptorFn, HttpRequest, HttpHandlerFn, HttpErrorResponse } from '@angular/common/http';\nimport { inject, Injector } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { catchError, throwError } from 'rxjs';\nimport { MessageService } from 'primeng/api';\nimport { AuthService } from '../services/auth.service';\nimport { TranslationService } from '../services/translation.service';\n\nexport const apiInterceptor: HttpInterceptorFn = (req: HttpRequest<unknown>, next: HttpHandlerFn) => {\n  const authService = inject(AuthService);\n  const injector = inject(Injector);\n  \n  // Get the access token, token type, and base URL\n  const accessToken = authService.getAccessToken();\n  const tokenType = authService.getTokenType();\n  const baseUrl = authService.getApiBaseUrl();\n  \n  let modifiedReq = req;\n\n  // Add Authorization header if token exists\n  if (accessToken && tokenType) {\n    const authHeader = `${tokenType} ${accessToken}`;\n    modifiedReq = modifiedReq.clone({\n      setHeaders: {\n        Authorization: authHeader\n      }\n    });\n  }\n\n  // Prepend base URL if it exists and the request URL is relative\n  if (baseUrl && !isAbsoluteUrl(modifiedReq.url)) {\n    const fullUrl = combineUrls(baseUrl, modifiedReq.url);\n    modifiedReq = modifiedReq.clone({\n      url: fullUrl\n    });\n  }\n\n  return next(modifiedReq).pipe(\n    catchError((error: HttpErrorResponse) => {\n      // Try to get MessageService and TranslationService\n      let messageService: MessageService | null = null;\n      let translationService: TranslationService | null = null;\n      \n      try {\n        messageService = injector.get(MessageService);\n        translationService = injector.get(TranslationService);\n      } catch (e) {\n        console.warn('MessageService or TranslationService not available');\n      }\n\n      // Handle 401 Unauthorized errors\n      if (error.status === 401) {\n        try {\n          const router = injector.get(Router);\n          router.navigate(['/forbidden']);\n        } catch (e) {\n          console.warn('Router not available for navigation on 401 error');\n        }\n      }\n      // Handle 4xx errors (except 401)\n      else if (error.status >= 400 && error.status < 500) {\n        if (messageService && translationService) {\n          const errorMessage = extractErrorMessage(error);\n          messageService.add({\n            severity: 'error',\n            summary: translationService.translate('common.error') || 'Erro',\n            detail: errorMessage,\n            life: 5000\n          });\n        }\n      }\n      // Handle 5xx errors\n      else if (error.status >= 500) {\n        if (messageService && translationService) {\n          const errorMessage = extractErrorMessage(error);\n          messageService.add({\n            severity: 'error',\n            summary: translationService.translate('common.server_error') || 'Erro no Servidor',\n            detail: errorMessage || translationService.translate('common.server_error_message') || 'Ocorreu um erro no servidor. Tente novamente mais tarde.',\n            life: 5000\n          });\n        }\n      }\n      \n      // Re-throw the error so other error handlers can process it\n      return throwError(() => error);\n    })\n  );\n};\n\n/**\n * Check if URL is absolute\n */\nfunction isAbsoluteUrl(url: string): boolean {\n  return /^https?:\\/\\//.test(url);\n}\n\n/**\n * Combine base URL with relative URL\n */\nfunction combineUrls(baseUrl: string, relativeUrl: string): string {\n  // Remove trailing slash from base URL\n  const cleanBaseUrl = baseUrl.replace(/\\/$/, '');\n  \n  // Remove leading slash from relative URL\n  const cleanRelativeUrl = relativeUrl.replace(/^\\//, '');\n  \n  return `${cleanBaseUrl}/${cleanRelativeUrl}`;\n}\n\n/**\n * Extract error message from HttpErrorResponse\n * Supports multiple error response formats\n */\nfunction extractErrorMessage(error: HttpErrorResponse): string {\n  // Check if error.error exists and has a message property\n  if (error.error && typeof error.error === 'object') {\n    // Format: { message: \"...\", errorCode: \"...\", reason: \"...\", domain: \"...\", service: \"...\" }\n    if (error.error.message) {\n      return error.error.message;\n    }\n    \n    // Format: { error: \"...\", error_description: \"...\" }\n    if (error.error.error_description) {\n      return error.error.error_description;\n    }\n    \n    // Format: { detail: \"...\" }\n    if (error.error.detail) {\n      return error.error.detail;\n    }\n    \n    // Format: { errors: [...] }\n    if (error.error.errors && Array.isArray(error.error.errors) && error.error.errors.length > 0) {\n      return error.error.errors.map((e: any) => e.message || e).join(', ');\n    }\n  }\n  \n  // Fallback to error.message or statusText\n  return error.message || error.statusText || 'Erro desconhecido';\n}\n"]}
122
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"api.interceptor.js","sourceRoot":"","sources":["../../../../projects/components-ai/src/lib/interceptors/api.interceptor.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpInterceptorFn, HttpRequest, HttpHandlerFn, HttpErrorResponse } from '@angular/common/http';\nimport { inject, Injector } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { catchError, throwError } from 'rxjs';\nimport { MessageService } from 'primeng/api';\nimport { AuthService } from '../services/auth.service';\nimport { TranslationService } from '../services/translation.service';\n\nexport const apiInterceptor: HttpInterceptorFn = (req: HttpRequest<unknown>, next: HttpHandlerFn) => {\n  const authService = inject(AuthService);\n  const injector = inject(Injector);\n  \n  // Get the access token, token type, and base URL\n  const accessToken = authService.getAccessToken();\n  const tokenType = authService.getTokenType();\n  const baseUrl = authService.getApiBaseUrl();\n  \n  let modifiedReq = req;\n\n  // Add Authorization header if token exists\n  if (accessToken && tokenType) {\n    const authHeader = `${tokenType} ${accessToken}`;\n    modifiedReq = modifiedReq.clone({\n      setHeaders: {\n        Authorization: authHeader\n      }\n    });\n  }\n\n  // Prepend base URL if it exists and the request URL is relative\n  if (baseUrl && !isAbsoluteUrl(modifiedReq.url)) {\n    const fullUrl = combineUrls(baseUrl, modifiedReq.url);\n    modifiedReq = modifiedReq.clone({\n      url: fullUrl\n    });\n  }\n\n  return next(modifiedReq).pipe(\n    catchError((error: HttpErrorResponse) => {\n      // Try to get MessageService and TranslationService\n      let messageService: MessageService | null = null;\n      let translationService: TranslationService | null = null;\n      \n      try {\n        messageService = injector.get(MessageService);\n        translationService = injector.get(TranslationService);\n      } catch (e) {\n        console.warn('MessageService or TranslationService not available');\n      }\n\n      // Handle 401 Unauthorized errors\n      if (error.status === 401) {\n        try {\n          const router = injector.get(Router);\n          router.navigate(['/forbidden']);\n        } catch (e) {\n          console.warn('Router not available for navigation on 401 error');\n        }\n      }\n      // Handle 4xx errors (except 401)\n      else if (error.status >= 400 && error.status < 500) {\n        if (messageService && translationService) {\n          const errorMessage = extractErrorMessage(error);\n          messageService.add({\n            severity: 'error',\n            summary: translationService.translate('design.components_ai.error') || 'Erro',\n            detail: errorMessage,\n            life: 5000\n          });\n        }\n      }\n      // Handle 5xx errors\n      else if (error.status >= 500) {\n        if (messageService && translationService) {\n          messageService.add({\n            severity: 'error',\n            summary: translationService.translate('design.components_ai.server_error') || 'Erro no Servidor',\n            detail: translationService.translate('design.components_ai.server_error_message') || 'Ocorreu um erro no servidor. Tente novamente mais tarde.',\n            life: 5000\n          });\n        }\n      }\n      \n      // Re-throw the error so other error handlers can process it\n      return throwError(() => error);\n    })\n  );\n};\n\n/**\n * Check if URL is absolute\n */\nfunction isAbsoluteUrl(url: string): boolean {\n  return /^https?:\\/\\//.test(url);\n}\n\n/**\n * Combine base URL with relative URL\n */\nfunction combineUrls(baseUrl: string, relativeUrl: string): string {\n  // Remove trailing slash from base URL\n  const cleanBaseUrl = baseUrl.replace(/\\/$/, '');\n  \n  // Remove leading slash from relative URL\n  const cleanRelativeUrl = relativeUrl.replace(/^\\//, '');\n  \n  return `${cleanBaseUrl}/${cleanRelativeUrl}`;\n}\n\n/**\n * Extract error message from HttpErrorResponse\n * Supports multiple error response formats\n */\nfunction extractErrorMessage(error: HttpErrorResponse): string {\n  // Check if error.error exists and has a message property\n  if (error.error && typeof error.error === 'object') {\n    // Format: { message: \"...\", errorCode: \"...\", reason: \"...\", domain: \"...\", service: \"...\" }\n    if (error.error.message) {\n      return error.error.message;\n    }\n    \n    // Format: { error: \"...\", error_description: \"...\" }\n    if (error.error.error_description) {\n      return error.error.error_description;\n    }\n    \n    // Format: { detail: \"...\" }\n    if (error.error.detail) {\n      return error.error.detail;\n    }\n    \n    // Format: { errors: [...] }\n    if (error.error.errors && Array.isArray(error.error.errors) && error.error.errors.length > 0) {\n      return error.error.errors.map((e: any) => e.message || e).join(', ');\n    }\n  }\n  \n  // Fallback to error.message or statusText\n  return error.message || error.statusText || 'Erro desconhecido';\n}\n"]}
@@ -79,48 +79,13 @@ export class EntityService {
79
79
  }
80
80
  /**
81
81
  * Default error handler
82
+ * Note: This just logs the error and re-throws the original HttpErrorResponse
83
+ * so that the HTTP interceptor can handle it properly (show toast, etc.)
82
84
  */
83
85
  handleError = (error) => {
84
- let errorMessage = 'An error occurred';
85
- if (this.translationService) {
86
- errorMessage = this.translationService.translate('error.generic');
87
- }
88
- if (error.status) {
89
- switch (error.status) {
90
- case 401:
91
- errorMessage = this.translationService
92
- ? this.translationService.translate('error.unauthorized')
93
- : 'Unauthorized';
94
- break;
95
- case 403:
96
- errorMessage = this.translationService
97
- ? this.translationService.translate('error.forbidden')
98
- : 'Forbidden';
99
- break;
100
- case 404:
101
- errorMessage = this.translationService
102
- ? this.translationService.translate('error.not_found')
103
- : 'Not found';
104
- break;
105
- case 500:
106
- errorMessage = this.translationService
107
- ? this.translationService.translate('error.server_error')
108
- : 'Server error';
109
- break;
110
- default:
111
- if (error.error && error.error.message) {
112
- errorMessage = error.error.message;
113
- }
114
- else if (error.statusText) {
115
- errorMessage = error.statusText;
116
- }
117
- }
118
- }
119
- else if (error.message) {
120
- errorMessage = error.message;
121
- }
122
86
  console.error('EntityService Error:', error);
123
- return throwError(() => new Error(errorMessage));
87
+ // Re-throw the original error so the interceptor can handle it
88
+ return throwError(() => error);
124
89
  };
125
90
  /**
126
91
  * List entities with pagination, sorting, and filtering
@@ -205,4 +170,4 @@ export class EntityService {
205
170
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: EntityService, decorators: [{
206
171
  type: Injectable
207
172
  }], ctorParameters: () => [{ type: i1.HttpClient }, { type: undefined }] });
208
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"entity.service.js","sourceRoot":"","sources":["../../../../projects/components-ai/src/lib/services/entity.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient, HttpParams } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { Observable, throwError } from 'rxjs';\nimport { catchError, map } from 'rxjs/operators';\n\nexport interface SortMeta {\n  field: string;\n  order: number; // 1 for asc, -1 for desc\n}\n\nexport interface ListParams {\n  page?: number;\n  size?: number;\n  sort?: SortMeta[];\n  filterQuery?: string;\n  displayFields?: string[];\n}\n\nexport interface BodyParams {\n  offset?: number;\n  size?: number;\n  orderBy?: string;\n  filter?: string;\n  displayfields?: string;\n}\n\nexport interface EntityListResponse<T> {\n  contents: T[];\n  totalElements: number;\n  totalPages: number;\n  size?: number;\n  number?: number;\n  first?: boolean;\n  last?: boolean;\n}\n\n// Interface para o serviço de tradução (duck typing)\nexport interface TranslationService {\n  translate(key: string, params?: any): string;\n}\n\n@Injectable()\nexport abstract class EntityService<T> {\n  protected abstract entityUrl: string;\n  protected abstract actionsUrl: string;\n  protected queriesUrl: string = '';\n\n  constructor(\n    protected http: HttpClient,\n    protected translationService?: TranslationService\n  ) {\n    // Initialize queriesUrl in ngOnInit or a separate method\n    // since abstract properties can't be accessed in constructor\n  }\n\n  /**\n   * Initialize the service - call this after setting actionsUrl\n   */\n  protected initializeService(): void {\n    this.queriesUrl = this.actionsUrl.replace('actions', 'queries');\n  }\n\n  /**\n   * Convert ListParams to HttpParams for GET requests\n   */\n  protected getListQueryParams(listParams: ListParams): HttpParams {\n    const { page = 0, size = 10, sort = [], filterQuery = '', displayFields = [] } = listParams;\n    let params = new HttpParams();\n    \n    params = params.append('size', String(size));\n    params = params.append('offset', String(page));\n    \n    if (sort && sort.length) {\n      const orderBy = sort\n        .map((s) => {\n          let order = '';\n          if (s.order === 1) order = ' asc';\n          else if (s.order === -1) order = ' desc';\n          return `${s.field}${order}`;\n        })\n        .join(', ');\n      params = params.append('orderby', orderBy);\n    }\n\n    if (filterQuery) {\n      params = params.append('filter', filterQuery);\n    }\n\n    if (displayFields && displayFields.length) {\n      params = params.append('displayfields', displayFields.join(','));\n    }\n\n    return params;\n  }\n\n  /**\n   * Convert ListParams to body parameters for POST requests\n   */\n  protected getBodyParams(listParams: ListParams): BodyParams {\n    const { page = 0, size = 10, sort = [], filterQuery = '', displayFields = [] } = listParams;\n    const bodyParams: BodyParams = {};\n\n    bodyParams.size = size;\n    bodyParams.offset = page;\n\n    if (sort && sort.length) {\n      bodyParams.orderBy = sort\n        .map((s) => {\n          let order = '';\n          if (s.order === 1) order = ' asc';\n          else if (s.order === -1) order = ' desc';\n          return `${s.field}${order}`;\n        })\n        .join(', ');\n    }\n\n    if (filterQuery) {\n      bodyParams.filter = filterQuery;\n    }\n\n    if (displayFields && displayFields.length) {\n      bodyParams.displayfields = displayFields.join(',');\n    }\n\n    return bodyParams;\n  }\n\n  /**\n   * Default error handler\n   */\n  protected handleError = (error: any): Observable<never> => {\n    let errorMessage = 'An error occurred';\n    \n    if (this.translationService) {\n      errorMessage = this.translationService.translate('error.generic');\n    }\n    \n    if (error.status) {\n      switch (error.status) {\n        case 401:\n          errorMessage = this.translationService \n            ? this.translationService.translate('error.unauthorized')\n            : 'Unauthorized';\n          break;\n        case 403:\n          errorMessage = this.translationService\n            ? this.translationService.translate('error.forbidden')\n            : 'Forbidden';\n          break;\n        case 404:\n          errorMessage = this.translationService\n            ? this.translationService.translate('error.not_found')\n            : 'Not found';\n          break;\n        case 500:\n          errorMessage = this.translationService\n            ? this.translationService.translate('error.server_error')\n            : 'Server error';\n          break;\n        default:\n          if (error.error && error.error.message) {\n            errorMessage = error.error.message;\n          } else if (error.statusText) {\n            errorMessage = error.statusText;\n          }\n      }\n    } else if (error.message) {\n      errorMessage = error.message;\n    }\n\n    console.error('EntityService Error:', error);\n    return throwError(() => new Error(errorMessage));\n  };\n\n  /**\n   * List entities with pagination, sorting, and filtering\n   */\n  list(listParams: ListParams = {}): Observable<EntityListResponse<T>> {\n    const params = this.getListQueryParams(listParams);\n    return this.http.get<EntityListResponse<T>>(this.entityUrl, { params })\n      .pipe(catchError(this.handleError));\n  }\n\n  /**\n   * Get entity by ID\n   */\n  get(id: any): Observable<T> {\n    return this.http.get<T>(`${this.entityUrl}/${id}`)\n      .pipe(catchError(this.handleError));\n  }\n\n  /**\n   * Create new entity\n   */\n  insert(entity: T): Observable<T> {\n    return this.http.post<T>(this.entityUrl, entity)\n      .pipe(catchError(this.handleError));\n  }\n\n  /**\n   * Update existing entity\n   */\n  update(id: any, entity: T): Observable<T> {\n    return this.http.put<T>(`${this.entityUrl}/${id}`, entity)\n      .pipe(catchError(this.handleError));\n  }\n\n  /**\n   * Delete entity by ID\n   */\n  delete(id: any): Observable<void> {\n    return this.http.delete<void>(`${this.entityUrl}/${id}`, { observe: 'response' })\n      .pipe(\n        map(() => undefined),\n        catchError(this.handleError)\n      );\n  }\n\n  /**\n   * Execute custom filter action\n   */\n  listCustomFilter(listParams: ListParams, action: string): Observable<EntityListResponse<T>> {\n    const bodyParams = this.getBodyParams(listParams);\n    return this.http.post<EntityListResponse<T>>(`${this.actionsUrl}/${action}`, bodyParams)\n      .pipe(catchError(this.handleError));\n  }\n\n  /**\n   * Execute custom query\n   */\n  executeQuery(queryName: string, params: any = {}): Observable<any> {\n    return this.http.post<any>(`${this.queriesUrl}/${queryName}`, params)\n      .pipe(catchError(this.handleError));\n  }\n\n  /**\n   * Execute custom action\n   */\n  executeAction(actionName: string, params: any = {}): Observable<any> {\n    return this.http.post<any>(`${this.actionsUrl}/${actionName}`, params)\n      .pipe(catchError(this.handleError));\n  }\n\n  /**\n   * Check if entity exists by ID\n   */\n  exists(id: any): Observable<boolean> {\n    return this.http.head(`${this.entityUrl}/${id}`)\n      .pipe(\n        map(() => true),\n        catchError((error) => {\n          if (error.status === 404) {\n            return [false];\n          }\n          return this.handleError(error);\n        })\n      );\n  }\n\n  /**\n   * Bulk delete entities\n   */\n  bulkDelete(ids: any[]): Observable<void> {\n    return this.http.post<void>(`${this.actionsUrl}/bulkDelete`, { ids }, { observe: 'response' })\n      .pipe(\n        map(() => undefined),\n        catchError(this.handleError)\n      );\n  }\n}\n"]}
173
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"entity.service.js","sourceRoot":"","sources":["../../../../projects/components-ai/src/lib/services/entity.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient, HttpParams } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { Observable, throwError } from 'rxjs';\nimport { catchError, map } from 'rxjs/operators';\n\nexport interface SortMeta {\n  field: string;\n  order: number; // 1 for asc, -1 for desc\n}\n\nexport interface ListParams {\n  page?: number;\n  size?: number;\n  sort?: SortMeta[];\n  filterQuery?: string;\n  displayFields?: string[];\n}\n\nexport interface BodyParams {\n  offset?: number;\n  size?: number;\n  orderBy?: string;\n  filter?: string;\n  displayfields?: string;\n}\n\nexport interface EntityListResponse<T> {\n  contents: T[];\n  totalElements: number;\n  totalPages: number;\n  size?: number;\n  number?: number;\n  first?: boolean;\n  last?: boolean;\n}\n\n// Interface para o serviço de tradução (duck typing)\nexport interface TranslationService {\n  translate(key: string, params?: any): string;\n}\n\n@Injectable()\nexport abstract class EntityService<T> {\n  protected abstract entityUrl: string;\n  protected abstract actionsUrl: string;\n  protected queriesUrl: string = '';\n\n  constructor(\n    protected http: HttpClient,\n    protected translationService?: TranslationService\n  ) {\n    // Initialize queriesUrl in ngOnInit or a separate method\n    // since abstract properties can't be accessed in constructor\n  }\n\n  /**\n   * Initialize the service - call this after setting actionsUrl\n   */\n  protected initializeService(): void {\n    this.queriesUrl = this.actionsUrl.replace('actions', 'queries');\n  }\n\n  /**\n   * Convert ListParams to HttpParams for GET requests\n   */\n  protected getListQueryParams(listParams: ListParams): HttpParams {\n    const { page = 0, size = 10, sort = [], filterQuery = '', displayFields = [] } = listParams;\n    let params = new HttpParams();\n    \n    params = params.append('size', String(size));\n    params = params.append('offset', String(page));\n    \n    if (sort && sort.length) {\n      const orderBy = sort\n        .map((s) => {\n          let order = '';\n          if (s.order === 1) order = ' asc';\n          else if (s.order === -1) order = ' desc';\n          return `${s.field}${order}`;\n        })\n        .join(', ');\n      params = params.append('orderby', orderBy);\n    }\n\n    if (filterQuery) {\n      params = params.append('filter', filterQuery);\n    }\n\n    if (displayFields && displayFields.length) {\n      params = params.append('displayfields', displayFields.join(','));\n    }\n\n    return params;\n  }\n\n  /**\n   * Convert ListParams to body parameters for POST requests\n   */\n  protected getBodyParams(listParams: ListParams): BodyParams {\n    const { page = 0, size = 10, sort = [], filterQuery = '', displayFields = [] } = listParams;\n    const bodyParams: BodyParams = {};\n\n    bodyParams.size = size;\n    bodyParams.offset = page;\n\n    if (sort && sort.length) {\n      bodyParams.orderBy = sort\n        .map((s) => {\n          let order = '';\n          if (s.order === 1) order = ' asc';\n          else if (s.order === -1) order = ' desc';\n          return `${s.field}${order}`;\n        })\n        .join(', ');\n    }\n\n    if (filterQuery) {\n      bodyParams.filter = filterQuery;\n    }\n\n    if (displayFields && displayFields.length) {\n      bodyParams.displayfields = displayFields.join(',');\n    }\n\n    return bodyParams;\n  }\n\n  /**\n   * Default error handler\n   * Note: This just logs the error and re-throws the original HttpErrorResponse\n   * so that the HTTP interceptor can handle it properly (show toast, etc.)\n   */\n  protected handleError = (error: any): Observable<never> => {\n    console.error('EntityService Error:', error);\n    // Re-throw the original error so the interceptor can handle it\n    return throwError(() => error);\n  };\n\n  /**\n   * List entities with pagination, sorting, and filtering\n   */\n  list(listParams: ListParams = {}): Observable<EntityListResponse<T>> {\n    const params = this.getListQueryParams(listParams);\n    return this.http.get<EntityListResponse<T>>(this.entityUrl, { params })\n      .pipe(catchError(this.handleError));\n  }\n\n  /**\n   * Get entity by ID\n   */\n  get(id: any): Observable<T> {\n    return this.http.get<T>(`${this.entityUrl}/${id}`)\n      .pipe(catchError(this.handleError));\n  }\n\n  /**\n   * Create new entity\n   */\n  insert(entity: T): Observable<T> {\n    return this.http.post<T>(this.entityUrl, entity)\n      .pipe(catchError(this.handleError));\n  }\n\n  /**\n   * Update existing entity\n   */\n  update(id: any, entity: T): Observable<T> {\n    return this.http.put<T>(`${this.entityUrl}/${id}`, entity)\n      .pipe(catchError(this.handleError));\n  }\n\n  /**\n   * Delete entity by ID\n   */\n  delete(id: any): Observable<void> {\n    return this.http.delete<void>(`${this.entityUrl}/${id}`, { observe: 'response' })\n      .pipe(\n        map(() => undefined),\n        catchError(this.handleError)\n      );\n  }\n\n  /**\n   * Execute custom filter action\n   */\n  listCustomFilter(listParams: ListParams, action: string): Observable<EntityListResponse<T>> {\n    const bodyParams = this.getBodyParams(listParams);\n    return this.http.post<EntityListResponse<T>>(`${this.actionsUrl}/${action}`, bodyParams)\n      .pipe(catchError(this.handleError));\n  }\n\n  /**\n   * Execute custom query\n   */\n  executeQuery(queryName: string, params: any = {}): Observable<any> {\n    return this.http.post<any>(`${this.queriesUrl}/${queryName}`, params)\n      .pipe(catchError(this.handleError));\n  }\n\n  /**\n   * Execute custom action\n   */\n  executeAction(actionName: string, params: any = {}): Observable<any> {\n    return this.http.post<any>(`${this.actionsUrl}/${actionName}`, params)\n      .pipe(catchError(this.handleError));\n  }\n\n  /**\n   * Check if entity exists by ID\n   */\n  exists(id: any): Observable<boolean> {\n    return this.http.head(`${this.entityUrl}/${id}`)\n      .pipe(\n        map(() => true),\n        catchError((error) => {\n          if (error.status === 404) {\n            return [false];\n          }\n          return this.handleError(error);\n        })\n      );\n  }\n\n  /**\n   * Bulk delete entities\n   */\n  bulkDelete(ids: any[]): Observable<void> {\n    return this.http.post<void>(`${this.actionsUrl}/bulkDelete`, { ids }, { observe: 'response' })\n      .pipe(\n        map(() => undefined),\n        catchError(this.handleError)\n      );\n  }\n}\n"]}
@@ -18,8 +18,8 @@ import * as i8 from 'primeng/checkbox';
18
18
  import { CheckboxModule } from 'primeng/checkbox';
19
19
  import { DividerModule } from 'primeng/divider';
20
20
  import * as XLSX from 'xlsx';
21
- import { jsPDF } from 'jspdf';
22
- import autoTable from 'jspdf-autotable';
21
+ import jsPDF from 'jspdf';
22
+ import 'jspdf-autotable';
23
23
  import * as i2$2 from 'primeng/api';
24
24
  import { ConfirmationService, MessageService } from 'primeng/api';
25
25
  import * as i5 from 'primeng/progressbar';
@@ -146,48 +146,13 @@ class EntityService {
146
146
  }
147
147
  /**
148
148
  * Default error handler
149
+ * Note: This just logs the error and re-throws the original HttpErrorResponse
150
+ * so that the HTTP interceptor can handle it properly (show toast, etc.)
149
151
  */
150
152
  handleError = (error) => {
151
- let errorMessage = 'An error occurred';
152
- if (this.translationService) {
153
- errorMessage = this.translationService.translate('error.generic');
154
- }
155
- if (error.status) {
156
- switch (error.status) {
157
- case 401:
158
- errorMessage = this.translationService
159
- ? this.translationService.translate('error.unauthorized')
160
- : 'Unauthorized';
161
- break;
162
- case 403:
163
- errorMessage = this.translationService
164
- ? this.translationService.translate('error.forbidden')
165
- : 'Forbidden';
166
- break;
167
- case 404:
168
- errorMessage = this.translationService
169
- ? this.translationService.translate('error.not_found')
170
- : 'Not found';
171
- break;
172
- case 500:
173
- errorMessage = this.translationService
174
- ? this.translationService.translate('error.server_error')
175
- : 'Server error';
176
- break;
177
- default:
178
- if (error.error && error.error.message) {
179
- errorMessage = error.error.message;
180
- }
181
- else if (error.statusText) {
182
- errorMessage = error.statusText;
183
- }
184
- }
185
- }
186
- else if (error.message) {
187
- errorMessage = error.message;
188
- }
189
153
  console.error('EntityService Error:', error);
190
- return throwError(() => new Error(errorMessage));
154
+ // Re-throw the original error so the interceptor can handle it
155
+ return throwError(() => error);
191
156
  };
192
157
  /**
193
158
  * List entities with pagination, sorting, and filtering
@@ -1059,6 +1024,9 @@ function getLanguageInfo(code) {
1059
1024
  }
1060
1025
 
1061
1026
  var translationsPtBR = {
1027
+ "design.components_ai.error": "Ops, algo deu errado",
1028
+ "design.components_ai.server_error": "Erro no Servidor",
1029
+ "design.components_ai.server_error_message": "Ocorreu um erro no servidor. Tente novamente mais tarde.",
1062
1030
  "design.components_ai.bulk_delete_title": "Exclusão em Massa",
1063
1031
  "design.components_ai.bulk_delete_warning": "Você está prestes a excluir <strong>{{count}}</strong> registro(s).",
1064
1032
  "design.components_ai.bulk_delete_description": "Esta ação não pode ser desfeita. Os registros serão excluídos um por vez.",
@@ -1101,7 +1069,6 @@ var translationsPtBR = {
1101
1069
  "design.components_ai.close": "Fechar",
1102
1070
  "design.components_ai.filter_search_placeholder": "Pesquisar por {{label}}",
1103
1071
  "design.components_ai.filter_select_placeholder": "Selecione {{label}}",
1104
- "design.components_ai.error": "Erro",
1105
1072
  "design.components_ai.success": "Sucesso",
1106
1073
  "design.components_ai.load_error": "Erro ao carregar {{entityType}}",
1107
1074
  "design.components_ai.confirm_delete": "Confirmar Exclusão",
@@ -1113,6 +1080,9 @@ var translationsPtBR = {
1113
1080
  };
1114
1081
 
1115
1082
  var translationsEnUS = {
1083
+ "design.components_ai.error": "Oops, something went wrong",
1084
+ "design.components_ai.server_error": "Server Error",
1085
+ "design.components_ai.server_error_message": "A server error occurred. Please try again later.",
1116
1086
  "design.components_ai.bulk_delete_title": "Bulk Delete",
1117
1087
  "design.components_ai.bulk_delete_warning": "You are about to delete <strong>{{count}}</strong> record(s).",
1118
1088
  "design.components_ai.bulk_delete_description": "This action cannot be undone. Records will be deleted one by one.",
@@ -1155,7 +1125,6 @@ var translationsEnUS = {
1155
1125
  "design.components_ai.close": "Close",
1156
1126
  "design.components_ai.filter_search_placeholder": "Search by {{label}}",
1157
1127
  "design.components_ai.filter_select_placeholder": "Select {{label}}",
1158
- "design.components_ai.error": "Error",
1159
1128
  "design.components_ai.success": "Success",
1160
1129
  "design.components_ai.load_error": "Error loading {{entityType}}",
1161
1130
  "design.components_ai.confirm_delete": "Confirm Deletion",
@@ -1167,6 +1136,9 @@ var translationsEnUS = {
1167
1136
  };
1168
1137
 
1169
1138
  var translationsEsES = {
1139
+ "design.components_ai.error": "Ups, algo salió mal",
1140
+ "design.components_ai.server_error": "Error del Servidor",
1141
+ "design.components_ai.server_error_message": "Ocurrió un error en el servidor. Inténtelo de nuevo más tarde.",
1170
1142
  "design.components_ai.bulk_delete_title": "Eliminación Masiva",
1171
1143
  "design.components_ai.bulk_delete_warning": "Está a punto de eliminar <strong>{{count}}</strong> registro(s).",
1172
1144
  "design.components_ai.bulk_delete_description": "Esta acción no se puede deshacer. Los registros se eliminarán uno por uno.",
@@ -1209,7 +1181,6 @@ var translationsEsES = {
1209
1181
  "design.components_ai.close": "Cerrar",
1210
1182
  "design.components_ai.filter_search_placeholder": "Buscar por {{label}}",
1211
1183
  "design.components_ai.filter_select_placeholder": "Seleccione {{label}}",
1212
- "design.components_ai.error": "Error",
1213
1184
  "design.components_ai.success": "Éxito",
1214
1185
  "design.components_ai.load_error": "Error al cargar {{entityType}}",
1215
1186
  "design.components_ai.confirm_delete": "Confirmar Eliminación",
@@ -1704,7 +1675,7 @@ class ExportDialogComponent {
1704
1675
  });
1705
1676
  });
1706
1677
  // Tabela
1707
- autoTable(doc, {
1678
+ doc.autoTable({
1708
1679
  head: [headers],
1709
1680
  body: data,
1710
1681
  startY: 40,
@@ -4618,7 +4589,7 @@ const apiInterceptor = (req, next) => {
4618
4589
  const errorMessage = extractErrorMessage(error);
4619
4590
  messageService.add({
4620
4591
  severity: 'error',
4621
- summary: translationService.translate('common.error') || 'Erro',
4592
+ summary: translationService.translate('design.components_ai.error') || 'Erro',
4622
4593
  detail: errorMessage,
4623
4594
  life: 5000
4624
4595
  });
@@ -4627,11 +4598,10 @@ const apiInterceptor = (req, next) => {
4627
4598
  // Handle 5xx errors
4628
4599
  else if (error.status >= 500) {
4629
4600
  if (messageService && translationService) {
4630
- const errorMessage = extractErrorMessage(error);
4631
4601
  messageService.add({
4632
4602
  severity: 'error',
4633
- summary: translationService.translate('common.server_error') || 'Erro no Servidor',
4634
- detail: errorMessage || translationService.translate('common.server_error_message') || 'Ocorreu um erro no servidor. Tente novamente mais tarde.',
4603
+ summary: translationService.translate('design.components_ai.server_error') || 'Erro no Servidor',
4604
+ detail: translationService.translate('design.components_ai.server_error_message') || 'Ocorreu um erro no servidor. Tente novamente mais tarde.',
4635
4605
  life: 5000
4636
4606
  });
4637
4607
  }