lib-portal-angular 0.0.86 → 0.0.88
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/components/components.module.mjs +65 -40
- package/esm2022/lib/components/csv-importer/csv-importer.component.mjs +92 -0
- package/esm2022/lib/components/data-excel-exporter/IExcelExportConfig.mjs +2 -0
- package/esm2022/lib/components/data-excel-exporter/data-excel-exporter.component.mjs +126 -0
- package/esm2022/lib/components/data-excel-import/data-excel-importer.component.mjs +198 -0
- package/esm2022/lib/components/multi-select/multi-select.component.mjs +162 -114
- package/esm2022/lib/components/pdf-data-handler/argenta-pdf-data-handler.component.mjs +147 -0
- package/esm2022/lib/components/pdf-data-handler/function/formatToBrazilianNumber.mjs +7 -0
- package/esm2022/lib/components/pdf-data-handler/function/parseBrazilianNumber.mjs +9 -0
- package/esm2022/lib/components/pdf-data-handler/interface/DataRow.mjs +2 -0
- package/esm2022/lib/components/pdf-data-handler/interface/PdfExportConfig.mjs +2 -0
- package/esm2022/lib/components/pdf-data-handler/interface/PdfTableStyles.mjs +2 -0
- package/esm2022/lib/components/pdf-data-handler/interface/RowCalculation.mjs +2 -0
- package/esm2022/lib/components/pdf-data-handler/interface/TotalizerConfig.mjs +2 -0
- package/esm2022/lib/components/pdf-download/argenta-pdf-download.component.mjs +51 -0
- package/esm2022/public-api.mjs +36 -5
- package/fesm2022/lib-portal-angular.mjs +1093 -422
- package/fesm2022/lib-portal-angular.mjs.map +1 -1
- package/lib/components/components.module.d.ts +10 -5
- package/lib/components/csv-importer/csv-importer.component.d.ts +19 -0
- package/lib/components/data-excel-exporter/IExcelExportConfig.d.ts +21 -0
- package/lib/components/data-excel-exporter/data-excel-exporter.component.d.ts +9 -0
- package/lib/components/data-excel-import/data-excel-importer.component.d.ts +44 -0
- package/lib/components/multi-select/multi-select.component.d.ts +20 -9
- package/lib/components/pdf-data-handler/argenta-pdf-data-handler.component.d.ts +15 -0
- package/lib/components/pdf-data-handler/function/formatToBrazilianNumber.d.ts +1 -0
- package/lib/components/pdf-data-handler/function/parseBrazilianNumber.d.ts +1 -0
- package/lib/components/pdf-data-handler/interface/DataRow.d.ts +3 -0
- package/lib/components/pdf-data-handler/interface/PdfExportConfig.d.ts +12 -0
- package/lib/components/pdf-data-handler/interface/PdfTableStyles.d.ts +28 -0
- package/lib/components/pdf-data-handler/interface/RowCalculation.d.ts +4 -0
- package/lib/components/pdf-data-handler/interface/TotalizerConfig.d.ts +8 -0
- package/lib/components/pdf-download/argenta-pdf-download.component.d.ts +13 -0
- package/package.json +1 -1
- package/public-api.d.ts +34 -3
@@ -0,0 +1,198 @@
|
|
1
|
+
import { Component, EventEmitter, Output } from "@angular/core";
|
2
|
+
import * as XLSX from "xlsx";
|
3
|
+
import * as i0 from "@angular/core";
|
4
|
+
import * as i1 from "@angular/common";
|
5
|
+
import * as i2 from "@angular/forms";
|
6
|
+
import * as i3 from "lucide-angular";
|
7
|
+
export class DataExcelImporterComponent {
|
8
|
+
constructor() {
|
9
|
+
this.excelData = new EventEmitter(); // Evento para enviar dados processados
|
10
|
+
// Configurações padrão
|
11
|
+
this.config = {
|
12
|
+
acceptedFileTypes: [".xlsx", ".xls", ".csv"],
|
13
|
+
onImport: (data) => this.defaultImportHandler(data),
|
14
|
+
excludeRows: [],
|
15
|
+
excludeColumns: [],
|
16
|
+
hasHeader: true,
|
17
|
+
headerRowIndex: 1,
|
18
|
+
titleRowIndex: 1,
|
19
|
+
};
|
20
|
+
// Controle do modal e inputs de configuração
|
21
|
+
this.userConfig = { ...this.config };
|
22
|
+
this.excludeRowsInput = ""; // Para capturar linhas a excluir
|
23
|
+
this.excludeColumnsInput = ""; // Para capturar colunas a excluir
|
24
|
+
this.showConfigModal = false;
|
25
|
+
this.buttonLabel = "Importar Dados";
|
26
|
+
}
|
27
|
+
// Abre o modal de configurações
|
28
|
+
openModal() {
|
29
|
+
this.showConfigModal = true;
|
30
|
+
}
|
31
|
+
// Fecha o modal de configurações
|
32
|
+
closeModal() {
|
33
|
+
this.showConfigModal = false;
|
34
|
+
}
|
35
|
+
// Salva as configurações ajustadas no modal
|
36
|
+
saveConfig() {
|
37
|
+
this.config = {
|
38
|
+
...this.userConfig,
|
39
|
+
titleRowIndex: (this.userConfig.titleRowIndex ?? 1) - 1,
|
40
|
+
headerRowIndex: (this.userConfig.headerRowIndex ?? 2) - 1,
|
41
|
+
excludeRows: this.excludeRowsInput
|
42
|
+
.split(",")
|
43
|
+
.map((row) => parseInt(row.trim(), 10) - 1),
|
44
|
+
excludeColumns: this.excludeColumnsInput
|
45
|
+
.split(",")
|
46
|
+
.map((col) => this.columnLetterToIndex(col.trim())), // Converte letras para índices
|
47
|
+
};
|
48
|
+
this.closeModal();
|
49
|
+
}
|
50
|
+
// Valida que as linhas informadas sejam pelo menos 1
|
51
|
+
validateRowInput(field) {
|
52
|
+
if (field === "headerRow" && this.userConfig.headerRowIndex < 1) {
|
53
|
+
this.userConfig.headerRowIndex = 1;
|
54
|
+
}
|
55
|
+
if (field === "titleRow" && this.userConfig.titleRowIndex < 1) {
|
56
|
+
this.userConfig.titleRowIndex = 1;
|
57
|
+
}
|
58
|
+
}
|
59
|
+
// Processa o arquivo selecionado
|
60
|
+
handleFileInput(event) {
|
61
|
+
const file = event.target.files[0];
|
62
|
+
if (file) {
|
63
|
+
const fileType = file.name.split(".").pop()?.toLowerCase();
|
64
|
+
if (fileType === "csv") {
|
65
|
+
this.processCSVFile(file); // Processa o CSV de forma isolada
|
66
|
+
}
|
67
|
+
else {
|
68
|
+
this.processExcelFile(file); // Mantém o fluxo atual para Excel
|
69
|
+
}
|
70
|
+
event.target.value = ""; // Reseta o input para permitir reimportação
|
71
|
+
}
|
72
|
+
}
|
73
|
+
processExcelFile(file) {
|
74
|
+
const reader = new FileReader();
|
75
|
+
reader.onload = (e) => {
|
76
|
+
const data = new Uint8Array(e.target.result);
|
77
|
+
const workbook = XLSX.read(data, { type: "array" });
|
78
|
+
const sheetName = workbook.SheetNames[0];
|
79
|
+
const worksheet = workbook.Sheets[sheetName];
|
80
|
+
const titleRowIndex = this.config.titleRowIndex ?? 0;
|
81
|
+
const headerRowIndex = this.config.headerRowIndex ?? titleRowIndex + 1;
|
82
|
+
if (worksheet["!ref"]) {
|
83
|
+
worksheet["!ref"] = this.adjustRange(worksheet["!ref"], headerRowIndex + 1);
|
84
|
+
}
|
85
|
+
else {
|
86
|
+
console.error("Intervalo da planilha ('!ref') não encontrado.");
|
87
|
+
return;
|
88
|
+
}
|
89
|
+
const jsonData = XLSX.utils.sheet_to_json(worksheet, {
|
90
|
+
header: 1,
|
91
|
+
defval: "",
|
92
|
+
blankrows: false,
|
93
|
+
});
|
94
|
+
this.processData(jsonData);
|
95
|
+
};
|
96
|
+
reader.readAsArrayBuffer(file); // Lê o Excel como array buffer
|
97
|
+
}
|
98
|
+
processCSVFile(file) {
|
99
|
+
if (this.userConfig.hasHeader) {
|
100
|
+
if (this.config?.headerRowIndex !== undefined) {
|
101
|
+
this.config.headerRowIndex =
|
102
|
+
this.config.headerRowIndex === 0 ? 1 : this.config.headerRowIndex + 1;
|
103
|
+
}
|
104
|
+
const sum = (this.config.excludeRows ?? []).reduce((acc, value) => acc + value, 0) +
|
105
|
+
1;
|
106
|
+
this.config.excludeRows = [sum];
|
107
|
+
}
|
108
|
+
const reader = new FileReader();
|
109
|
+
reader.onload = (e) => {
|
110
|
+
const csvData = e.target.result;
|
111
|
+
// Converte CSV em JSON usando XLSX
|
112
|
+
const workbook = XLSX.read(csvData, { type: "string" });
|
113
|
+
const sheetName = workbook.SheetNames[0];
|
114
|
+
const worksheet = workbook.Sheets[sheetName];
|
115
|
+
const jsonData = XLSX.utils.sheet_to_json(worksheet, {
|
116
|
+
header: 1,
|
117
|
+
defval: "",
|
118
|
+
blankrows: false,
|
119
|
+
});
|
120
|
+
this.processData(jsonData);
|
121
|
+
};
|
122
|
+
reader.readAsText(file); // Lê o CSV como texto
|
123
|
+
}
|
124
|
+
processData(jsonData) {
|
125
|
+
const titleRowIndex = this.config.titleRowIndex ?? 0;
|
126
|
+
const headerRowIndex = this.config.headerRowIndex ?? titleRowIndex + 1;
|
127
|
+
const headers = jsonData[0];
|
128
|
+
if (!headers) {
|
129
|
+
console.error("Cabeçalhos não encontrados. Verifique o headerRowIndex.");
|
130
|
+
return;
|
131
|
+
}
|
132
|
+
const normalizedHeaders = headers.map((header, index) => this.normalizeHeader(header || `col_${index}`));
|
133
|
+
const dataRows = jsonData.slice(1); // Remove a linha de cabeçalho
|
134
|
+
// Filtra as linhas baseando-se no índice informado pelo usuário (1-based)
|
135
|
+
const filteredRows = dataRows.filter((_, index) => !this.config.excludeRows?.includes(index + headerRowIndex + 1));
|
136
|
+
const formattedData = filteredRows.map((row) => {
|
137
|
+
const record = {};
|
138
|
+
normalizedHeaders.forEach((header, index) => {
|
139
|
+
record[header] = row[index] ?? "";
|
140
|
+
});
|
141
|
+
return record;
|
142
|
+
});
|
143
|
+
const finalData = this.filterColumns(formattedData);
|
144
|
+
this.config.onImport(finalData); // Chama o callback com os dados processados
|
145
|
+
this.excelData.emit(finalData); // Emite os dados processados para o cliente
|
146
|
+
}
|
147
|
+
// Ajusta o intervalo da planilha com base na linha do cabeçalho
|
148
|
+
adjustRange(range, startRow) {
|
149
|
+
const [start, end] = range.split(":");
|
150
|
+
const startCol = start.replace(/[0-9]/g, "");
|
151
|
+
const endCol = end.replace(/[0-9]/g, "");
|
152
|
+
const endRow = end.replace(/[A-Za-z]/g, "");
|
153
|
+
return `${startCol}${startRow}:${endCol}${endRow}`;
|
154
|
+
}
|
155
|
+
// Normaliza os cabeçalhos removendo acentos e espaços
|
156
|
+
normalizeHeader(header) {
|
157
|
+
return header
|
158
|
+
.normalize("NFD")
|
159
|
+
.replace(/[\u0300-\u036f]/g, "")
|
160
|
+
.replace(/\s+/g, "_")
|
161
|
+
.toLowerCase();
|
162
|
+
}
|
163
|
+
// Filtra as colunas configuradas para exclusão
|
164
|
+
filterColumns(data) {
|
165
|
+
const { excludeColumns } = this.config;
|
166
|
+
return data.map((row) => {
|
167
|
+
const filteredRow = {};
|
168
|
+
Object.keys(row).forEach((key, index) => {
|
169
|
+
if (!excludeColumns?.includes(index)) {
|
170
|
+
filteredRow[key] = row[key];
|
171
|
+
}
|
172
|
+
});
|
173
|
+
return filteredRow;
|
174
|
+
});
|
175
|
+
}
|
176
|
+
// Converte letra da coluna para índice (A -> 0, B -> 1, etc.)
|
177
|
+
columnLetterToIndex(letter) {
|
178
|
+
let index = 0;
|
179
|
+
for (let i = 0; i < letter.length; i++) {
|
180
|
+
index = index * 26 + (letter.charCodeAt(i) - "A".charCodeAt(0) + 1);
|
181
|
+
}
|
182
|
+
return index - 1;
|
183
|
+
}
|
184
|
+
// Callback padrão para manipular dados importados
|
185
|
+
defaultImportHandler(data) { }
|
186
|
+
transformToUppercase() {
|
187
|
+
this.excludeColumnsInput = this.excludeColumnsInput.toUpperCase();
|
188
|
+
}
|
189
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DataExcelImporterComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
190
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: DataExcelImporterComponent, selector: "argenta-excel-importer", outputs: { excelData: "excelData" }, ngImport: i0, template: "<div class=\"excel-importer\">\n <input\n type=\"file\"\n (change)=\"handleFileInput($event)\"\n [accept]=\"config.acceptedFileTypes?.join(',')\"\n style=\"display: none\"\n #fileInput\n />\n <button (click)=\"fileInput.click()\">{{ buttonLabel }}</button>\n <lucide-icon\n name=\"settings\"\n [size]=\"24\"\n [strokeWidth]=\"2.25\"\n class=\"settings-icon\"\n (click)=\"openModal()\"\n ></lucide-icon>\n</div>\n\n<div class=\"modal-container\" *ngIf=\"showConfigModal\">\n <div class=\"modal-content\">\n <h4 class=\"titulo\">Configura\u00E7\u00E3o de Importa\u00E7\u00E3o</h4>\n <form (ngSubmit)=\"saveConfig()\">\n <div class=\"form-group\">\n <label for=\"acceptedFileTypes\">Tipos de Arquivo Permitidos</label>\n <input\n id=\"acceptedFileTypes\"\n type=\"text\"\n [(ngModel)]=\"userConfig.acceptedFileTypes\"\n name=\"acceptedFileTypes\"\n placeholder=\".xlsx, .xls\"\n />\n </div>\n <div class=\"form-group\" style=\"display: none\">\n <label for=\"titleRow\">Linha do T\u00EDtulo (se aplic\u00E1vel)</label>\n <input\n id=\"titleRow\"\n type=\"number\"\n [(ngModel)]=\"userConfig.titleRowIndex\"\n name=\"titleRow\"\n placeholder=\"Ex: 1\"\n (change)=\"validateRowInput('titleRow')\"\n />\n </div>\n <div class=\"form-group checkbox-group\">\n <div class=\"row\">\n <div class=\"col-md-8\">\n <label for=\"hasHeader\">A planilha cont\u00E9m cabe\u00E7alho?</label>\n </div>\n <div class=\"col-md-4\">\n <input\n id=\"hasHeader\"\n class=\"input-checkbox\"\n type=\"checkbox\"\n [(ngModel)]=\"userConfig.hasHeader\"\n name=\"hasHeader\"\n />\n </div>\n </div>\n </div>\n\n <!-- Exibe a linha do cabe\u00E7alho apenas se o checkbox estiver marcado -->\n <div class=\"form-group\" *ngIf=\"userConfig.hasHeader\">\n <label for=\"headerRow\">Linha do Cabe\u00E7alho</label>\n <input\n id=\"headerRow\"\n type=\"number\"\n [(ngModel)]=\"userConfig.headerRowIndex\"\n name=\"headerRow\"\n placeholder=\"Ex: 2\"\n (change)=\"validateRowInput('headerRow')\"\n />\n </div>\n\n <div class=\"form-group\">\n <label for=\"excludeRows\">Linhas a Excluir</label>\n <input\n id=\"excludeRows\"\n type=\"text\"\n [(ngModel)]=\"excludeRowsInput\"\n name=\"excludeRows\"\n placeholder=\"Ex: 1, 3\"\n />\n </div>\n <div class=\"form-group\">\n <label for=\"excludeColumns\">Colunas a Excluir</label>\n <input\n id=\"excludeColumns\"\n type=\"text\"\n [(ngModel)]=\"excludeColumnsInput\"\n name=\"excludeColumns\"\n placeholder=\"Ex: A, C\"\n (input)=\"transformToUppercase()\"\n />\n </div>\n <button type=\"submit\">Salvar Configura\u00E7\u00F5es</button>\n <button type=\"button\" (click)=\"closeModal()\">Fechar</button>\n </form>\n </div>\n</div>\n", styles: [".excel-importer{display:flex;align-items:center;gap:10px}.excel-importer button{background-color:var(--primary-color);color:var(--text-color);font-family:var(--font-family);padding:10px 20px;border:none;cursor:pointer;display:flex;align-items:center}.excel-importer button:hover{background-color:var(--secondary-color)}.excel-importer .settings-icon{color:var(--text-color);background-color:var(--primary-color);border-radius:50%;padding:5px;cursor:pointer;transition:background-color .3s,color .3s}.excel-importer .settings-icon:hover{background-color:var(--secondary-color);color:#fff}.modal-container{position:fixed;top:0;left:0;width:100%;height:100%;background-color:#00000080;display:flex;justify-content:center;align-items:center}.modal-content{background:#fff;padding:20px;border-radius:5px;width:400px}.modal-content h2{margin-bottom:20px}.modal-content .form-group{margin-bottom:15px}.modal-content .form-group label{display:block;margin-bottom:5px}.modal-content .form-group input{width:100%;padding:8px;border:1px solid #ccc;border-radius:4px}.modal-content button{margin-right:10px;padding:8px 15px;border:none;cursor:pointer}.modal-content button:first-of-type{background-color:var(--primary-color);color:#fff}.modal-content button:last-of-type{background-color:#ccc}input#excludeColumns{text-transform:uppercase}.small-label{font-size:.7rem;color:#555}.input-checkbox{margin-left:-5rem}.titulo{margin-bottom:2rem}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i2.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i2.NgForm, selector: "form:not([ngNoForm]):not([formGroup]),ng-form,[ngForm]", inputs: ["ngFormOptions"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "component", type: i3.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }] }); }
|
191
|
+
}
|
192
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DataExcelImporterComponent, decorators: [{
|
193
|
+
type: Component,
|
194
|
+
args: [{ selector: "argenta-excel-importer", template: "<div class=\"excel-importer\">\n <input\n type=\"file\"\n (change)=\"handleFileInput($event)\"\n [accept]=\"config.acceptedFileTypes?.join(',')\"\n style=\"display: none\"\n #fileInput\n />\n <button (click)=\"fileInput.click()\">{{ buttonLabel }}</button>\n <lucide-icon\n name=\"settings\"\n [size]=\"24\"\n [strokeWidth]=\"2.25\"\n class=\"settings-icon\"\n (click)=\"openModal()\"\n ></lucide-icon>\n</div>\n\n<div class=\"modal-container\" *ngIf=\"showConfigModal\">\n <div class=\"modal-content\">\n <h4 class=\"titulo\">Configura\u00E7\u00E3o de Importa\u00E7\u00E3o</h4>\n <form (ngSubmit)=\"saveConfig()\">\n <div class=\"form-group\">\n <label for=\"acceptedFileTypes\">Tipos de Arquivo Permitidos</label>\n <input\n id=\"acceptedFileTypes\"\n type=\"text\"\n [(ngModel)]=\"userConfig.acceptedFileTypes\"\n name=\"acceptedFileTypes\"\n placeholder=\".xlsx, .xls\"\n />\n </div>\n <div class=\"form-group\" style=\"display: none\">\n <label for=\"titleRow\">Linha do T\u00EDtulo (se aplic\u00E1vel)</label>\n <input\n id=\"titleRow\"\n type=\"number\"\n [(ngModel)]=\"userConfig.titleRowIndex\"\n name=\"titleRow\"\n placeholder=\"Ex: 1\"\n (change)=\"validateRowInput('titleRow')\"\n />\n </div>\n <div class=\"form-group checkbox-group\">\n <div class=\"row\">\n <div class=\"col-md-8\">\n <label for=\"hasHeader\">A planilha cont\u00E9m cabe\u00E7alho?</label>\n </div>\n <div class=\"col-md-4\">\n <input\n id=\"hasHeader\"\n class=\"input-checkbox\"\n type=\"checkbox\"\n [(ngModel)]=\"userConfig.hasHeader\"\n name=\"hasHeader\"\n />\n </div>\n </div>\n </div>\n\n <!-- Exibe a linha do cabe\u00E7alho apenas se o checkbox estiver marcado -->\n <div class=\"form-group\" *ngIf=\"userConfig.hasHeader\">\n <label for=\"headerRow\">Linha do Cabe\u00E7alho</label>\n <input\n id=\"headerRow\"\n type=\"number\"\n [(ngModel)]=\"userConfig.headerRowIndex\"\n name=\"headerRow\"\n placeholder=\"Ex: 2\"\n (change)=\"validateRowInput('headerRow')\"\n />\n </div>\n\n <div class=\"form-group\">\n <label for=\"excludeRows\">Linhas a Excluir</label>\n <input\n id=\"excludeRows\"\n type=\"text\"\n [(ngModel)]=\"excludeRowsInput\"\n name=\"excludeRows\"\n placeholder=\"Ex: 1, 3\"\n />\n </div>\n <div class=\"form-group\">\n <label for=\"excludeColumns\">Colunas a Excluir</label>\n <input\n id=\"excludeColumns\"\n type=\"text\"\n [(ngModel)]=\"excludeColumnsInput\"\n name=\"excludeColumns\"\n placeholder=\"Ex: A, C\"\n (input)=\"transformToUppercase()\"\n />\n </div>\n <button type=\"submit\">Salvar Configura\u00E7\u00F5es</button>\n <button type=\"button\" (click)=\"closeModal()\">Fechar</button>\n </form>\n </div>\n</div>\n", styles: [".excel-importer{display:flex;align-items:center;gap:10px}.excel-importer button{background-color:var(--primary-color);color:var(--text-color);font-family:var(--font-family);padding:10px 20px;border:none;cursor:pointer;display:flex;align-items:center}.excel-importer button:hover{background-color:var(--secondary-color)}.excel-importer .settings-icon{color:var(--text-color);background-color:var(--primary-color);border-radius:50%;padding:5px;cursor:pointer;transition:background-color .3s,color .3s}.excel-importer .settings-icon:hover{background-color:var(--secondary-color);color:#fff}.modal-container{position:fixed;top:0;left:0;width:100%;height:100%;background-color:#00000080;display:flex;justify-content:center;align-items:center}.modal-content{background:#fff;padding:20px;border-radius:5px;width:400px}.modal-content h2{margin-bottom:20px}.modal-content .form-group{margin-bottom:15px}.modal-content .form-group label{display:block;margin-bottom:5px}.modal-content .form-group input{width:100%;padding:8px;border:1px solid #ccc;border-radius:4px}.modal-content button{margin-right:10px;padding:8px 15px;border:none;cursor:pointer}.modal-content button:first-of-type{background-color:var(--primary-color);color:#fff}.modal-content button:last-of-type{background-color:#ccc}input#excludeColumns{text-transform:uppercase}.small-label{font-size:.7rem;color:#555}.input-checkbox{margin-left:-5rem}.titulo{margin-bottom:2rem}\n"] }]
|
195
|
+
}], propDecorators: { excelData: [{
|
196
|
+
type: Output
|
197
|
+
}] } });
|
198
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-excel-importer.component.js","sourceRoot":"","sources":["../../../../../../projects/lib-portal-angular/src/lib/components/data-excel-import/data-excel-importer.component.ts","../../../../../../projects/lib-portal-angular/src/lib/components/data-excel-import/data-excel-importer.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Output } from \"@angular/core\";\nimport * as XLSX from \"xlsx\";\n\nexport interface ExcelImporterConfig {\n  acceptedFileTypes: string[];\n  onImport: (data: any[]) => void;\n  excludeRows?: number[];\n  excludeColumns?: number[];\n  hasHeader?: boolean;\n  headerRowIndex?: number;\n  titleRowIndex?: number;\n}\n\n@Component({\n  selector: \"argenta-excel-importer\",\n  templateUrl: \"./data-excel-importer.component.html\",\n  styleUrls: [\"./data-excel-importer.component.scss\"],\n})\nexport class DataExcelImporterComponent {\n  @Output() excelData = new EventEmitter<any[]>(); // Evento para enviar dados processados\n  // Configurações padrão\n  config: ExcelImporterConfig = {\n    acceptedFileTypes: [\".xlsx\", \".xls\", \".csv\"],\n    onImport: (data) => this.defaultImportHandler(data),\n    excludeRows: [],\n    excludeColumns: [],\n    hasHeader: true,\n    headerRowIndex: 1,\n    titleRowIndex: 1,\n  };\n\n  // Controle do modal e inputs de configuração\n  userConfig = { ...this.config };\n  excludeRowsInput: string = \"\"; // Para capturar linhas a excluir\n  excludeColumnsInput: string = \"\"; // Para capturar colunas a excluir\n  showConfigModal = false;\n  buttonLabel = \"Importar Dados\";\n\n  // Abre o modal de configurações\n  openModal(): void {\n    this.showConfigModal = true;\n  }\n\n  // Fecha o modal de configurações\n  closeModal(): void {\n    this.showConfigModal = false;\n  }\n\n  // Salva as configurações ajustadas no modal\n  saveConfig(): void {\n    this.config = {\n      ...this.userConfig,\n      titleRowIndex: (this.userConfig.titleRowIndex ?? 1) - 1, // Converte para índice zero-based\n      headerRowIndex: (this.userConfig.headerRowIndex ?? 2) - 1, // Converte para índice zero-based\n      excludeRows: this.excludeRowsInput\n        .split(\",\")\n        .map((row) => parseInt(row.trim(), 10) - 1), // Converte para zero-based\n      excludeColumns: this.excludeColumnsInput\n        .split(\",\")\n        .map((col) => this.columnLetterToIndex(col.trim())), // Converte letras para índices\n    };\n    this.closeModal();\n  }\n\n  // Valida que as linhas informadas sejam pelo menos 1\n  validateRowInput(field: \"headerRow\" | \"titleRow\"): void {\n    if (field === \"headerRow\" && this.userConfig.headerRowIndex! < 1) {\n      this.userConfig.headerRowIndex = 1;\n    }\n    if (field === \"titleRow\" && this.userConfig.titleRowIndex! < 1) {\n      this.userConfig.titleRowIndex = 1;\n    }\n  }\n\n  // Processa o arquivo selecionado\n  handleFileInput(event: any): void {\n    const file = event.target.files[0];\n    if (file) {\n      const fileType = file.name.split(\".\").pop()?.toLowerCase();\n\n      if (fileType === \"csv\") {\n        this.processCSVFile(file); // Processa o CSV de forma isolada\n      } else {\n        this.processExcelFile(file); // Mantém o fluxo atual para Excel\n      }\n      event.target.value = \"\"; // Reseta o input para permitir reimportação\n    }\n  }\n\n  private processExcelFile(file: File): void {\n    const reader = new FileReader();\n    reader.onload = (e: any) => {\n      const data = new Uint8Array(e.target.result);\n      const workbook = XLSX.read(data, { type: \"array\" });\n      const sheetName = workbook.SheetNames[0];\n      const worksheet = workbook.Sheets[sheetName];\n\n      const titleRowIndex = this.config.titleRowIndex ?? 0;\n      const headerRowIndex = this.config.headerRowIndex ?? titleRowIndex + 1;\n\n      if (worksheet[\"!ref\"]) {\n        worksheet[\"!ref\"] = this.adjustRange(\n          worksheet[\"!ref\"],\n          headerRowIndex + 1\n        );\n      } else {\n        console.error(\"Intervalo da planilha ('!ref') não encontrado.\");\n        return;\n      }\n\n      const jsonData = XLSX.utils.sheet_to_json<any[]>(worksheet, {\n        header: 1,\n        defval: \"\",\n        blankrows: false,\n      });\n\n      this.processData(jsonData);\n    };\n    reader.readAsArrayBuffer(file); // Lê o Excel como array buffer\n  }\n\n  private processCSVFile(file: File): void {\n    if (this.userConfig.hasHeader) {\n      if (this.config?.headerRowIndex !== undefined) {\n        this.config.headerRowIndex =\n          this.config.headerRowIndex === 0 ? 1 : this.config.headerRowIndex + 1;\n      }\n\n      const sum =\n        (this.config.excludeRows ?? []).reduce((acc, value) => acc + value, 0) +\n        1;\n\n      this.config.excludeRows = [sum];\n    }\n\n    const reader = new FileReader();\n    reader.onload = (e: any) => {\n      const csvData = e.target.result;\n\n      // Converte CSV em JSON usando XLSX\n      const workbook = XLSX.read(csvData, { type: \"string\" });\n      const sheetName = workbook.SheetNames[0];\n      const worksheet = workbook.Sheets[sheetName];\n\n      const jsonData = XLSX.utils.sheet_to_json<any[]>(worksheet, {\n        header: 1,\n        defval: \"\",\n        blankrows: false,\n      });\n\n      this.processData(jsonData);\n    };\n    reader.readAsText(file); // Lê o CSV como texto\n  }\n\n  private processData(jsonData: any[]): void {\n    const titleRowIndex = this.config.titleRowIndex ?? 0;\n    const headerRowIndex = this.config.headerRowIndex ?? titleRowIndex + 1;\n\n    const headers = jsonData[0];\n    if (!headers) {\n      console.error(\"Cabeçalhos não encontrados. Verifique o headerRowIndex.\");\n      return;\n    }\n\n    const normalizedHeaders = headers.map((header: any, index: any) =>\n      this.normalizeHeader(header || `col_${index}`)\n    );\n\n    const dataRows = jsonData.slice(1); // Remove a linha de cabeçalho\n\n    // Filtra as linhas baseando-se no índice informado pelo usuário (1-based)\n    const filteredRows = dataRows.filter(\n      (_, index) =>\n        !this.config.excludeRows?.includes(index + headerRowIndex + 1)\n    );\n\n    const formattedData = filteredRows.map((row) => {\n      const record: any = {};\n      normalizedHeaders.forEach((header: any, index: any) => {\n        record[header] = row[index] ?? \"\";\n      });\n      return record;\n    });\n\n    const finalData = this.filterColumns(formattedData);\n    this.config.onImport(finalData); // Chama o callback com os dados processados\n    this.excelData.emit(finalData); // Emite os dados processados para o cliente\n  }\n\n  // Ajusta o intervalo da planilha com base na linha do cabeçalho\n  private adjustRange(range: string, startRow: number): string {\n    const [start, end] = range.split(\":\");\n    const startCol = start.replace(/[0-9]/g, \"\");\n    const endCol = end.replace(/[0-9]/g, \"\");\n    const endRow = end.replace(/[A-Za-z]/g, \"\");\n    return `${startCol}${startRow}:${endCol}${endRow}`;\n  }\n\n  // Normaliza os cabeçalhos removendo acentos e espaços\n  private normalizeHeader(header: string): string {\n    return header\n      .normalize(\"NFD\")\n      .replace(/[\\u0300-\\u036f]/g, \"\")\n      .replace(/\\s+/g, \"_\")\n      .toLowerCase();\n  }\n\n  // Filtra as colunas configuradas para exclusão\n  private filterColumns(data: any[]): any[] {\n    const { excludeColumns } = this.config;\n\n    return data.map((row) => {\n      const filteredRow: any = {};\n      Object.keys(row).forEach((key, index) => {\n        if (!excludeColumns?.includes(index)) {\n          filteredRow[key] = row[key];\n        }\n      });\n      return filteredRow;\n    });\n  }\n\n  // Converte letra da coluna para índice (A -> 0, B -> 1, etc.)\n  private columnLetterToIndex(letter: string): number {\n    let index = 0;\n    for (let i = 0; i < letter.length; i++) {\n      index = index * 26 + (letter.charCodeAt(i) - \"A\".charCodeAt(0) + 1);\n    }\n    return index - 1;\n  }\n\n  // Callback padrão para manipular dados importados\n  private defaultImportHandler(data: any[]): void {}\n\n  transformToUppercase(): void {\n    this.excludeColumnsInput = this.excludeColumnsInput.toUpperCase();\n  }\n}\n","<div class=\"excel-importer\">\n  <input\n    type=\"file\"\n    (change)=\"handleFileInput($event)\"\n    [accept]=\"config.acceptedFileTypes?.join(',')\"\n    style=\"display: none\"\n    #fileInput\n  />\n  <button (click)=\"fileInput.click()\">{{ buttonLabel }}</button>\n  <lucide-icon\n    name=\"settings\"\n    [size]=\"24\"\n    [strokeWidth]=\"2.25\"\n    class=\"settings-icon\"\n    (click)=\"openModal()\"\n  ></lucide-icon>\n</div>\n\n<div class=\"modal-container\" *ngIf=\"showConfigModal\">\n  <div class=\"modal-content\">\n    <h4 class=\"titulo\">Configuração de Importação</h4>\n    <form (ngSubmit)=\"saveConfig()\">\n      <div class=\"form-group\">\n        <label for=\"acceptedFileTypes\">Tipos de Arquivo Permitidos</label>\n        <input\n          id=\"acceptedFileTypes\"\n          type=\"text\"\n          [(ngModel)]=\"userConfig.acceptedFileTypes\"\n          name=\"acceptedFileTypes\"\n          placeholder=\".xlsx, .xls\"\n        />\n      </div>\n      <div class=\"form-group\" style=\"display: none\">\n        <label for=\"titleRow\">Linha do Título (se aplicável)</label>\n        <input\n          id=\"titleRow\"\n          type=\"number\"\n          [(ngModel)]=\"userConfig.titleRowIndex\"\n          name=\"titleRow\"\n          placeholder=\"Ex: 1\"\n          (change)=\"validateRowInput('titleRow')\"\n        />\n      </div>\n      <div class=\"form-group checkbox-group\">\n        <div class=\"row\">\n          <div class=\"col-md-8\">\n            <label for=\"hasHeader\">A planilha contém cabeçalho?</label>\n          </div>\n          <div class=\"col-md-4\">\n            <input\n              id=\"hasHeader\"\n              class=\"input-checkbox\"\n              type=\"checkbox\"\n              [(ngModel)]=\"userConfig.hasHeader\"\n              name=\"hasHeader\"\n            />\n          </div>\n        </div>\n      </div>\n\n      <!-- Exibe a linha do cabeçalho apenas se o checkbox estiver marcado -->\n      <div class=\"form-group\" *ngIf=\"userConfig.hasHeader\">\n        <label for=\"headerRow\">Linha do Cabeçalho</label>\n        <input\n          id=\"headerRow\"\n          type=\"number\"\n          [(ngModel)]=\"userConfig.headerRowIndex\"\n          name=\"headerRow\"\n          placeholder=\"Ex: 2\"\n          (change)=\"validateRowInput('headerRow')\"\n        />\n      </div>\n\n      <div class=\"form-group\">\n        <label for=\"excludeRows\">Linhas a Excluir</label>\n        <input\n          id=\"excludeRows\"\n          type=\"text\"\n          [(ngModel)]=\"excludeRowsInput\"\n          name=\"excludeRows\"\n          placeholder=\"Ex: 1, 3\"\n        />\n      </div>\n      <div class=\"form-group\">\n        <label for=\"excludeColumns\">Colunas a Excluir</label>\n        <input\n          id=\"excludeColumns\"\n          type=\"text\"\n          [(ngModel)]=\"excludeColumnsInput\"\n          name=\"excludeColumns\"\n          placeholder=\"Ex: A, C\"\n          (input)=\"transformToUppercase()\"\n        />\n      </div>\n      <button type=\"submit\">Salvar Configurações</button>\n      <button type=\"button\" (click)=\"closeModal()\">Fechar</button>\n    </form>\n  </div>\n</div>\n"]}
|