cfc-ds
Version:
Design System do Conselho Federal de Contabilidade baseado no govbr-ds
242 lines • 45.9 kB
JavaScript
import { Component, EventEmitter, Input, Output, ViewChild } from '@angular/core';
import * as i0 from "@angular/core";
import * as i1 from "@angular/common";
import * as i2 from "@angular/forms";
import * as i3 from "@angular/flex-layout/extended";
export class UploadComponent {
config;
parentForm;
filesChanged = new EventEmitter();
fileInput;
files = [];
isDragging = false;
isInvalid = false;
uploading = false;
message = null;
get acceptedFileTypesString() {
return this.config?.acceptedFileTypes?.join(',') || '';
}
constructor() { }
ngOnInit() {
if (!this.config) {
console.error('A configuração do componente Upload é obrigatória');
}
}
openFileSelector() {
this.fileInput.nativeElement.click();
}
onDragOver(event) {
event.preventDefault();
event.stopPropagation();
this.isDragging = true;
}
onDragLeave(event) {
event.preventDefault();
event.stopPropagation();
this.isDragging = false;
}
onDrop(event) {
event.preventDefault();
event.stopPropagation();
this.isDragging = false;
if (event.dataTransfer?.files) {
this.handleFiles(event.dataTransfer.files);
}
}
onFileSelected(event) {
const input = event.target;
if (input.files) {
this.handleFiles(input.files);
}
}
handleFiles(fileList) {
if (!this.config)
return;
// Verificar se é permitido múltiplos arquivos
if (!this.config.multiple && fileList.length > 1) {
this.showMessage({
type: 'error',
text: this.config.messages?.error || 'É permitido o envio de apenas um arquivo'
});
return;
}
// Verificar número máximo de arquivos
if (this.config.maxFiles && this.files.length + fileList.length > this.config.maxFiles) {
this.showMessage({
type: 'error',
text: this.config.messages?.maxFilesError ||
`É permitido o envio de no máximo ${this.config.maxFiles} arquivos`
});
return;
}
// Validar e processar cada arquivo
const newFiles = [];
let hasInvalidFiles = false;
this.uploading = true;
// Array para armazenar promessas de processamento de arquivos
const fileProcessingPromises = [];
for (let i = 0; i < fileList.length; i++) {
const file = fileList[i];
const isValid = this.validateFile(file);
if (isValid) {
// Criar uma promessa para cada arquivo
const filePromise = new Promise((resolve) => {
const reader = new FileReader();
reader.onload = () => {
// O arquivo foi lido completamente
if (!this.config.multiple) {
// Se não permite múltiplos arquivos, substitui o arquivo existente
if (this.files.length > 0) {
this.showMessage({
type: 'warning',
text: 'O arquivo enviado anteriormente foi substituído'
});
this.files = [file];
newFiles.push(file);
}
else {
this.files.push(file);
newFiles.push(file);
}
}
else {
this.files.push(file);
newFiles.push(file);
}
resolve();
};
// Em caso de erro na leitura
reader.onerror = () => {
hasInvalidFiles = true;
this.showMessage({
type: 'error',
text: `Erro ao processar o arquivo ${file.name}`
});
resolve();
};
// Inicia a leitura do arquivo
reader.readAsArrayBuffer(file);
});
fileProcessingPromises.push(filePromise);
}
else {
hasInvalidFiles = true;
}
}
// Aguardar o processamento de todos os arquivos
Promise.all(fileProcessingPromises)
.then(() => {
this.uploading = false;
if (newFiles.length > 0 && !hasInvalidFiles) {
this.showMessage({
type: 'success',
text: this.config?.messages?.success || 'Campo preenchido corretamente'
});
}
this.filesChanged.emit(this.files);
})
.catch(error => {
this.uploading = false;
this.showMessage({
type: 'error',
text: 'Erro no processamento dos arquivos'
});
console.error('File processing error:', error);
});
}
validateFile(file) {
if (!this.config)
return false;
// Validar tipo de arquivo
if (this.config.acceptedFileTypes && this.config.acceptedFileTypes.length > 0) {
const fileNameParts = file.name.split('.');
const fileExtension = fileNameParts.length > 1 ? fileNameParts.pop()?.toLowerCase() : '';
const fileType = file.type;
const isValidType = this.config.acceptedFileTypes.some(type => {
if (type.startsWith('.')) {
// Validação por extensão
return fileExtension ? `.${fileExtension}` === type : false;
}
else {
// Validação por MIME type
return fileType === type || (type.includes('*') && fileType.startsWith(type.replace('*', '')));
}
});
if (!isValidType) {
this.showMessage({
type: 'error',
text: this.config.messages?.fileTypeError ||
`O arquivo ${file.name} possui formato inválido`
});
return false;
}
}
// Validar tamanho do arquivo
if (this.config.maxFileSize && file.size > this.config.maxFileSize) {
this.showMessage({
type: 'error',
text: this.config.messages?.fileSizeError ||
`O arquivo ${file.name} excede o tamanho máximo permitido`
});
return false;
}
return true;
}
removeFile(file) {
const index = this.files.indexOf(file);
if (index !== -1) {
this.files.splice(index, 1);
this.filesChanged.emit(this.files);
}
}
openFile(file) {
// Abrir o arquivo em uma nova janela/guia do navegador
const url = URL.createObjectURL(file);
window.open(url, '_blank');
}
getTruncatedName(name) {
// Truncar o nome se for muito longo (por exemplo, 25 caracteres)
const maxLength = 25;
if (name.length <= maxLength) {
return name;
}
const extension = name.split('.').pop() || '';
const fileName = name.substring(0, name.length - extension.length - 1);
return `${fileName.substring(0, maxLength - extension.length - 4)}...${extension}`;
}
formatFileSize(bytes) {
if (bytes === 0)
return '0 B';
const k = 1024;
const sizes = ['B', 'KB', 'MB', 'GB'];
const i = Math.floor(Math.log(bytes) / Math.log(k));
return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];
}
showMessage(message) {
this.message = message;
this.isInvalid = message.type === 'error';
// Limpar a mensagem após 5 segundos
setTimeout(() => {
this.message = null;
if (message.type === 'error') {
this.isInvalid = false;
}
}, 5000);
}
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: UploadComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: UploadComponent, selector: "cfc-upload", inputs: { config: "config", parentForm: "parentForm" }, outputs: { filesChanged: "filesChanged" }, viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileInput"], descendants: true }], ngImport: i0, template: "<div class=\"upload-container\" [formGroup]=\"parentForm\" *ngIf=\"config\">\r\n <!-- Label (opcional) -->\r\n <label *ngIf=\"config.label\" [for]=\"config.id\" class=\"upload-label\">{{ config.label }}</label>\r\n\r\n <!-- \u00C1rea de upload -->\r\n <div class=\"upload-area\" [class.dragging]=\"isDragging\" [class.invalid]=\"isInvalid\" (dragover)=\"onDragOver($event)\"\r\n (dragleave)=\"onDragLeave($event)\" (drop)=\"onDrop($event)\" (click)=\"openFileSelector()\">\r\n\r\n <!-- \u00CDcone de upload -->\r\n <i class=\"fa fa-upload upload-icon\"></i>\r\n\r\n <!-- Placeholder -->\r\n <span class=\"upload-placeholder\">\r\n {{ config.multiple ? 'Selecione o(s) arquivo(s)' : 'Selecione o arquivo' }}\r\n </span>\r\n\r\n <!-- Input file oculto -->\r\n <input type=\"file\" [id]=\"config.id\" class=\"file-input\" [multiple]=\"config.multiple\"\r\n [accept]=\"acceptedFileTypesString\" (change)=\"onFileSelected($event)\" #fileInput>\r\n </div>\r\n\r\n <!-- Loading durante upload -->\r\n <div *ngIf=\"uploading\" class=\"upload-loading\">\r\n <div class=\"loading-spinner\"></div>\r\n <span>Carregando...</span>\r\n </div>\r\n\r\n <!-- Mensagem de erro/sucesso -->\r\n <div *ngIf=\"message\" class=\"upload-message\" [ngClass]=\"message.type\">\r\n <i [class]=\"'fa ' + (message.type === 'error' ? 'fa-exclamation-circle' : 'fa-check-circle')\"></i>\r\n {{ message.text }}\r\n </div>\r\n\r\n <!-- Texto auxiliar -->\r\n <div *ngIf=\"config.helperText\" class=\"upload-helper-text\">\r\n {{ config.helperText }}\r\n </div>\r\n\r\n <!-- Lista de arquivos -->\r\n <div *ngIf=\"config.showFileList && files.length > 0\" class=\"file-list\">\r\n <div *ngFor=\"let file of files\" class=\"file-item\">\r\n <span class=\"file-name\" (click)=\"openFile(file)\">{{ getTruncatedName(file.name) }}</span>\r\n <span class=\"file-size\">{{ formatFileSize(file.size) }}</span>\r\n <button class=\"delete-button\" (click)=\"removeFile(file)\">\r\n <i class=\"fa fa-trash\"></i>\r\n </button>\r\n </div>\r\n </div>\r\n</div>", styles: [".upload-container{display:flex;flex-direction:column;width:100%;font-family:Arial,sans-serif}.upload-label{margin-bottom:8px;font-size:14px;font-weight:500;color:#333}.upload-area{position:relative;display:flex;align-items:center;padding:16px;min-height:56px;border:2px dashed #ccc;border-radius:4px;background-color:#f9f9f9;cursor:pointer;transition:all .3s ease}.upload-area:hover{border-color:#999}.upload-area.dragging{border-color:#28a745;background-color:#28a7450d}.upload-area.invalid{border-color:#dc3545;background-color:#dc35450d}.upload-icon{color:#06c;margin-right:12px;font-size:20px}.upload-placeholder{color:#666;font-size:14px}.file-input{display:none}.upload-message{display:flex;align-items:center;margin-top:8px;padding:8px 12px;border-radius:4px;font-size:14px}.upload-message.success{background-color:#28a7451a;color:#28a745}.upload-message.error{background-color:#dc35451a;color:#dc3545}.upload-message.warning{background-color:#ffc1071a;color:#ffc107}.upload-message i{margin-right:8px}.upload-helper-text{margin-top:8px;font-size:12px;color:#666}.file-list{margin-top:16px;border:1px solid #eee;border-radius:4px}.file-item{display:flex;align-items:center;justify-content:space-between;padding:12px 16px;border-bottom:1px solid #eee}.file-item:last-child{border-bottom:none}.file-name{color:#06c;cursor:pointer;flex-grow:1;margin-right:16px}.file-name:hover{text-decoration:underline}.file-size{color:#666;margin-right:16px}.delete-button{background:none;border:none;color:#dc3545;cursor:pointer;padding:4px;font-size:16px}.delete-button:hover{color:#bd2130}.upload-loading{display:flex;align-items:center;margin-top:12px;color:#666;font-size:14px}.loading-spinner{width:20px;height:20px;border:2px solid rgba(0,102,204,.2);border-top-color:#06c;border-radius:50%;margin-right:8px;animation:spinner .8s linear infinite}@keyframes spinner{to{transform:rotate(360deg)}}@media screen and (max-width: 768px){.file-item{flex-wrap:wrap}.file-name{width:100%;margin-bottom:8px}}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i3.DefaultClassDirective, selector: " [ngClass], [ngClass.xs], [ngClass.sm], [ngClass.md], [ngClass.lg], [ngClass.xl], [ngClass.lt-sm], [ngClass.lt-md], [ngClass.lt-lg], [ngClass.lt-xl], [ngClass.gt-xs], [ngClass.gt-sm], [ngClass.gt-md], [ngClass.gt-lg]", inputs: ["ngClass", "ngClass.xs", "ngClass.sm", "ngClass.md", "ngClass.lg", "ngClass.xl", "ngClass.lt-sm", "ngClass.lt-md", "ngClass.lt-lg", "ngClass.lt-xl", "ngClass.gt-xs", "ngClass.gt-sm", "ngClass.gt-md", "ngClass.gt-lg"] }] });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: UploadComponent, decorators: [{
type: Component,
args: [{ selector: 'cfc-upload', template: "<div class=\"upload-container\" [formGroup]=\"parentForm\" *ngIf=\"config\">\r\n <!-- Label (opcional) -->\r\n <label *ngIf=\"config.label\" [for]=\"config.id\" class=\"upload-label\">{{ config.label }}</label>\r\n\r\n <!-- \u00C1rea de upload -->\r\n <div class=\"upload-area\" [class.dragging]=\"isDragging\" [class.invalid]=\"isInvalid\" (dragover)=\"onDragOver($event)\"\r\n (dragleave)=\"onDragLeave($event)\" (drop)=\"onDrop($event)\" (click)=\"openFileSelector()\">\r\n\r\n <!-- \u00CDcone de upload -->\r\n <i class=\"fa fa-upload upload-icon\"></i>\r\n\r\n <!-- Placeholder -->\r\n <span class=\"upload-placeholder\">\r\n {{ config.multiple ? 'Selecione o(s) arquivo(s)' : 'Selecione o arquivo' }}\r\n </span>\r\n\r\n <!-- Input file oculto -->\r\n <input type=\"file\" [id]=\"config.id\" class=\"file-input\" [multiple]=\"config.multiple\"\r\n [accept]=\"acceptedFileTypesString\" (change)=\"onFileSelected($event)\" #fileInput>\r\n </div>\r\n\r\n <!-- Loading durante upload -->\r\n <div *ngIf=\"uploading\" class=\"upload-loading\">\r\n <div class=\"loading-spinner\"></div>\r\n <span>Carregando...</span>\r\n </div>\r\n\r\n <!-- Mensagem de erro/sucesso -->\r\n <div *ngIf=\"message\" class=\"upload-message\" [ngClass]=\"message.type\">\r\n <i [class]=\"'fa ' + (message.type === 'error' ? 'fa-exclamation-circle' : 'fa-check-circle')\"></i>\r\n {{ message.text }}\r\n </div>\r\n\r\n <!-- Texto auxiliar -->\r\n <div *ngIf=\"config.helperText\" class=\"upload-helper-text\">\r\n {{ config.helperText }}\r\n </div>\r\n\r\n <!-- Lista de arquivos -->\r\n <div *ngIf=\"config.showFileList && files.length > 0\" class=\"file-list\">\r\n <div *ngFor=\"let file of files\" class=\"file-item\">\r\n <span class=\"file-name\" (click)=\"openFile(file)\">{{ getTruncatedName(file.name) }}</span>\r\n <span class=\"file-size\">{{ formatFileSize(file.size) }}</span>\r\n <button class=\"delete-button\" (click)=\"removeFile(file)\">\r\n <i class=\"fa fa-trash\"></i>\r\n </button>\r\n </div>\r\n </div>\r\n</div>", styles: [".upload-container{display:flex;flex-direction:column;width:100%;font-family:Arial,sans-serif}.upload-label{margin-bottom:8px;font-size:14px;font-weight:500;color:#333}.upload-area{position:relative;display:flex;align-items:center;padding:16px;min-height:56px;border:2px dashed #ccc;border-radius:4px;background-color:#f9f9f9;cursor:pointer;transition:all .3s ease}.upload-area:hover{border-color:#999}.upload-area.dragging{border-color:#28a745;background-color:#28a7450d}.upload-area.invalid{border-color:#dc3545;background-color:#dc35450d}.upload-icon{color:#06c;margin-right:12px;font-size:20px}.upload-placeholder{color:#666;font-size:14px}.file-input{display:none}.upload-message{display:flex;align-items:center;margin-top:8px;padding:8px 12px;border-radius:4px;font-size:14px}.upload-message.success{background-color:#28a7451a;color:#28a745}.upload-message.error{background-color:#dc35451a;color:#dc3545}.upload-message.warning{background-color:#ffc1071a;color:#ffc107}.upload-message i{margin-right:8px}.upload-helper-text{margin-top:8px;font-size:12px;color:#666}.file-list{margin-top:16px;border:1px solid #eee;border-radius:4px}.file-item{display:flex;align-items:center;justify-content:space-between;padding:12px 16px;border-bottom:1px solid #eee}.file-item:last-child{border-bottom:none}.file-name{color:#06c;cursor:pointer;flex-grow:1;margin-right:16px}.file-name:hover{text-decoration:underline}.file-size{color:#666;margin-right:16px}.delete-button{background:none;border:none;color:#dc3545;cursor:pointer;padding:4px;font-size:16px}.delete-button:hover{color:#bd2130}.upload-loading{display:flex;align-items:center;margin-top:12px;color:#666;font-size:14px}.loading-spinner{width:20px;height:20px;border:2px solid rgba(0,102,204,.2);border-top-color:#06c;border-radius:50%;margin-right:8px;animation:spinner .8s linear infinite}@keyframes spinner{to{transform:rotate(360deg)}}@media screen and (max-width: 768px){.file-item{flex-wrap:wrap}.file-name{width:100%;margin-bottom:8px}}\n"] }]
}], ctorParameters: () => [], propDecorators: { config: [{
type: Input
}], parentForm: [{
type: Input
}], filesChanged: [{
type: Output
}], fileInput: [{
type: ViewChild,
args: ['fileInput']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"upload.component.js","sourceRoot":"","sources":["../../../../../../projects/cfc-ds/src/lib/components/upload/upload.component.ts","../../../../../../projects/cfc-ds/src/lib/components/upload/upload.component.html"],"names":[],"mappings":"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","sourcesContent":["\r\n\r\n\r\n\r\n\r\n\r\n\r\n    \r\n    import { Component, ElementRef, EventEmitter, Input, Output, ViewChild } from '@angular/core';\r\n    import { FormGroup } from '@angular/forms';\r\n    \r\n    export interface UploadConfig {\r\n      id: string;\r\n      label?: string;\r\n      multiple: boolean;\r\n      acceptedFileTypes?: string[];\r\n      maxFileSize?: number; // em bytes\r\n      helperText?: string;\r\n      required?: boolean;\r\n      maxFiles?: number;\r\n      showFileList?: boolean;\r\n      messages?: {\r\n        success?: string;\r\n        error?: string;\r\n        fileTypeError?: string;\r\n        fileSizeError?: string;\r\n        maxFilesError?: string;\r\n      };\r\n    }\r\n    \r\n    interface FileMessage {\r\n      type: 'success' | 'error' | 'warning';\r\n      text: string;\r\n    }\r\n    \r\n    @Component({\r\n      selector: 'cfc-upload',\r\n      templateUrl: './upload.component.html',\r\n      styleUrl: './upload.component.scss'\r\n    })\r\n    export class UploadComponent {\r\n      @Input() config!: UploadConfig;\r\n      @Input() parentForm!: FormGroup;\r\n      @Output() filesChanged = new EventEmitter<File[]>();\r\n      \r\n      @ViewChild('fileInput') fileInput!: ElementRef;\r\n      \r\n      files: File[] = [];\r\n      isDragging = false;\r\n      isInvalid = false;\r\n      uploading = false;\r\n      message: FileMessage | null = null;\r\n      \r\n      get acceptedFileTypesString(): string {\r\n        return this.config?.acceptedFileTypes?.join(',') || '';\r\n      }\r\n      \r\n      constructor() { }\r\n      \r\n      ngOnInit(): void {\r\n        if (!this.config) {\r\n          console.error('A configuração do componente Upload é obrigatória');\r\n        }\r\n      }\r\n      \r\n      openFileSelector(): void {\r\n        this.fileInput.nativeElement.click();\r\n      }\r\n      \r\n      onDragOver(event: DragEvent): void {\r\n        event.preventDefault();\r\n        event.stopPropagation();\r\n        this.isDragging = true;\r\n      }\r\n      \r\n      onDragLeave(event: DragEvent): void {\r\n        event.preventDefault();\r\n        event.stopPropagation();\r\n        this.isDragging = false;\r\n      }\r\n      \r\n      onDrop(event: DragEvent): void {\r\n        event.preventDefault();\r\n        event.stopPropagation();\r\n        this.isDragging = false;\r\n        \r\n        if (event.dataTransfer?.files) {\r\n          this.handleFiles(event.dataTransfer.files);\r\n        }\r\n      }\r\n      \r\n      onFileSelected(event: Event): void {\r\n        const input = event.target as HTMLInputElement;\r\n        if (input.files) {\r\n          this.handleFiles(input.files);\r\n        }\r\n      }\r\n      \r\n      handleFiles(fileList: FileList): void {\r\n        if (!this.config) return;\r\n        \r\n        // Verificar se é permitido múltiplos arquivos\r\n        if (!this.config.multiple && fileList.length > 1) {\r\n          this.showMessage({\r\n            type: 'error',\r\n            text: this.config.messages?.error || 'É permitido o envio de apenas um arquivo'\r\n          });\r\n          return;\r\n        }\r\n        \r\n        // Verificar número máximo de arquivos\r\n        if (this.config.maxFiles && this.files.length + fileList.length > this.config.maxFiles) {\r\n          this.showMessage({\r\n            type: 'error',\r\n            text: this.config.messages?.maxFilesError || \r\n                  `É permitido o envio de no máximo ${this.config.maxFiles} arquivos`\r\n          });\r\n          return;\r\n        }\r\n        \r\n        // Validar e processar cada arquivo\r\n        const newFiles: File[] = [];\r\n        let hasInvalidFiles = false;\r\n        \r\n        this.uploading = true;\r\n        \r\n        // Array para armazenar promessas de processamento de arquivos\r\n        const fileProcessingPromises: Promise<void>[] = [];\r\n      \r\n        for (let i = 0; i < fileList.length; i++) {\r\n          const file = fileList[i];\r\n          const isValid = this.validateFile(file);\r\n          \r\n          if (isValid) {\r\n            // Criar uma promessa para cada arquivo\r\n            const filePromise = new Promise<void>((resolve) => {\r\n              const reader = new FileReader();\r\n              \r\n              reader.onload = () => {\r\n                // O arquivo foi lido completamente\r\n                if (!this.config.multiple) {\r\n                  // Se não permite múltiplos arquivos, substitui o arquivo existente\r\n                  if (this.files.length > 0) {\r\n                    this.showMessage({\r\n                      type: 'warning',\r\n                      text: 'O arquivo enviado anteriormente foi substituído'\r\n                    });\r\n                    this.files = [file];\r\n                    newFiles.push(file);\r\n                  } else {\r\n                    this.files.push(file);\r\n                    newFiles.push(file);\r\n                  }\r\n                } else {\r\n                  this.files.push(file);\r\n                  newFiles.push(file);\r\n                }\r\n                resolve();\r\n              };\r\n              \r\n              // Em caso de erro na leitura\r\n              reader.onerror = () => {\r\n                hasInvalidFiles = true;\r\n                this.showMessage({\r\n                  type: 'error',\r\n                  text: `Erro ao processar o arquivo ${file.name}`\r\n                });\r\n                resolve();\r\n              };\r\n              \r\n              // Inicia a leitura do arquivo\r\n              reader.readAsArrayBuffer(file);\r\n            });\r\n            \r\n            fileProcessingPromises.push(filePromise);\r\n          } else {\r\n            hasInvalidFiles = true;\r\n          }\r\n        }\r\n        \r\n        // Aguardar o processamento de todos os arquivos\r\n        Promise.all(fileProcessingPromises)\r\n          .then(() => {\r\n            this.uploading = false;\r\n            \r\n            if (newFiles.length > 0 && !hasInvalidFiles) {\r\n              this.showMessage({\r\n                type: 'success',\r\n                text: this.config?.messages?.success || 'Campo preenchido corretamente'\r\n              });\r\n            }\r\n            \r\n            this.filesChanged.emit(this.files);\r\n          })\r\n          .catch(error => {\r\n            this.uploading = false;\r\n            this.showMessage({\r\n              type: 'error',\r\n              text: 'Erro no processamento dos arquivos'\r\n            });\r\n            console.error('File processing error:', error);\r\n          });\r\n      }\r\n      \r\n      validateFile(file: File): boolean {\r\n        if (!this.config) return false;\r\n        \r\n        // Validar tipo de arquivo\r\n        if (this.config.acceptedFileTypes && this.config.acceptedFileTypes.length > 0) {\r\n          const fileNameParts = file.name.split('.');\r\n          const fileExtension = fileNameParts.length > 1 ? fileNameParts.pop()?.toLowerCase() : '';\r\n          const fileType = file.type;\r\n          \r\n          const isValidType = this.config.acceptedFileTypes.some(type => {\r\n            if (type.startsWith('.')) {\r\n              // Validação por extensão\r\n              return fileExtension ? `.${fileExtension}` === type : false;\r\n            } else {\r\n              // Validação por MIME type\r\n              return fileType === type || (type.includes('*') && fileType.startsWith(type.replace('*', '')));\r\n            }\r\n          });\r\n          \r\n          if (!isValidType) {\r\n            this.showMessage({\r\n              type: 'error',\r\n              text: this.config.messages?.fileTypeError || \r\n                    `O arquivo ${file.name} possui formato inválido`\r\n            });\r\n            return false;\r\n          }\r\n        }\r\n        \r\n        // Validar tamanho do arquivo\r\n        if (this.config.maxFileSize && file.size > this.config.maxFileSize) {\r\n          this.showMessage({\r\n            type: 'error',\r\n            text: this.config.messages?.fileSizeError || \r\n                  `O arquivo ${file.name} excede o tamanho máximo permitido`\r\n          });\r\n          return false;\r\n        }\r\n        \r\n        return true;\r\n      }\r\n      \r\n      removeFile(file: File): void {\r\n        const index = this.files.indexOf(file);\r\n        if (index !== -1) {\r\n          this.files.splice(index, 1);\r\n          this.filesChanged.emit(this.files);\r\n        }\r\n      }\r\n      \r\n      openFile(file: File): void {\r\n        // Abrir o arquivo em uma nova janela/guia do navegador\r\n        const url = URL.createObjectURL(file);\r\n        window.open(url, '_blank');\r\n      }\r\n      \r\n      getTruncatedName(name: string): string {\r\n        // Truncar o nome se for muito longo (por exemplo, 25 caracteres)\r\n        const maxLength = 25;\r\n        if (name.length <= maxLength) {\r\n          return name;\r\n        }\r\n        \r\n        const extension = name.split('.').pop() || '';\r\n        const fileName = name.substring(0, name.length - extension.length - 1);\r\n        \r\n        return `${fileName.substring(0, maxLength - extension.length - 4)}...${extension}`;\r\n      }\r\n      \r\n      formatFileSize(bytes: number): string {\r\n        if (bytes === 0) return '0 B';\r\n        \r\n        const k = 1024;\r\n        const sizes = ['B', 'KB', 'MB', 'GB'];\r\n        const i = Math.floor(Math.log(bytes) / Math.log(k));\r\n        \r\n        return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];\r\n      }\r\n      \r\n      showMessage(message: FileMessage): void {\r\n        this.message = message;\r\n        this.isInvalid = message.type === 'error';\r\n        \r\n        // Limpar a mensagem após 5 segundos\r\n        setTimeout(() => {\r\n          this.message = null;\r\n          if (message.type === 'error') {\r\n            this.isInvalid = false;\r\n          }\r\n        }, 5000);\r\n      }\r\n    }\r\n    ","<div class=\"upload-container\" [formGroup]=\"parentForm\" *ngIf=\"config\">\r\n    <!-- Label (opcional) -->\r\n    <label *ngIf=\"config.label\" [for]=\"config.id\" class=\"upload-label\">{{ config.label }}</label>\r\n\r\n    <!-- Área de upload -->\r\n    <div class=\"upload-area\" [class.dragging]=\"isDragging\" [class.invalid]=\"isInvalid\" (dragover)=\"onDragOver($event)\"\r\n        (dragleave)=\"onDragLeave($event)\" (drop)=\"onDrop($event)\" (click)=\"openFileSelector()\">\r\n\r\n        <!-- Ícone de upload -->\r\n        <i class=\"fa fa-upload upload-icon\"></i>\r\n\r\n        <!-- Placeholder -->\r\n        <span class=\"upload-placeholder\">\r\n            {{ config.multiple ? 'Selecione o(s) arquivo(s)' : 'Selecione o arquivo' }}\r\n        </span>\r\n\r\n        <!-- Input file oculto -->\r\n        <input type=\"file\" [id]=\"config.id\" class=\"file-input\" [multiple]=\"config.multiple\"\r\n            [accept]=\"acceptedFileTypesString\" (change)=\"onFileSelected($event)\" #fileInput>\r\n    </div>\r\n\r\n    <!-- Loading durante upload -->\r\n    <div *ngIf=\"uploading\" class=\"upload-loading\">\r\n        <div class=\"loading-spinner\"></div>\r\n        <span>Carregando...</span>\r\n    </div>\r\n\r\n    <!-- Mensagem de erro/sucesso -->\r\n    <div *ngIf=\"message\" class=\"upload-message\" [ngClass]=\"message.type\">\r\n        <i [class]=\"'fa ' + (message.type === 'error' ? 'fa-exclamation-circle' : 'fa-check-circle')\"></i>\r\n        {{ message.text }}\r\n    </div>\r\n\r\n    <!-- Texto auxiliar -->\r\n    <div *ngIf=\"config.helperText\" class=\"upload-helper-text\">\r\n        {{ config.helperText }}\r\n    </div>\r\n\r\n    <!-- Lista de arquivos -->\r\n    <div *ngIf=\"config.showFileList && files.length > 0\" class=\"file-list\">\r\n        <div *ngFor=\"let file of files\" class=\"file-item\">\r\n            <span class=\"file-name\" (click)=\"openFile(file)\">{{ getTruncatedName(file.name) }}</span>\r\n            <span class=\"file-size\">{{ formatFileSize(file.size) }}</span>\r\n            <button class=\"delete-button\" (click)=\"removeFile(file)\">\r\n                <i class=\"fa fa-trash\"></i>\r\n            </button>\r\n        </div>\r\n    </div>\r\n</div>"]}