UNPKG

ngx-ui-hero

Version:

Simple, fast and reliable utilities for Angular.

348 lines 54.7 kB
import { FileUploader } from 'ng2-file-upload'; import { zip } from 'rxjs'; import { retry } from 'rxjs/operators'; import { HttpClient } from '@angular/common/http'; import { Component, EventEmitter, Inject, Input, Optional, Output } from '@angular/core'; import { INPUT_FORMS_CONFIG } from '../../input-forms-config.constants'; import * as i0 from "@angular/core"; import * as i1 from "@angular/common/http"; import * as i2 from "@angular/common"; import * as i3 from "@angular/forms"; import * as i4 from "ng2-file-upload"; import * as i5 from "ngx-bootstrap/progressbar"; import * as i6 from "ngx-bootstrap/tooltip"; let identifier = 0; export class InputUploadComponent { constructor(config, http) { this.http = http; this.placeholder = 'Select a file to upload...'; this.dropZonePlaceholder = 'Drag & drop a file to import.'; this.disabled = false; this.autoUpload = true; this.showDropZone = false; this.showQueue = false; this.chunk = false; this.chunkSize = 1048576; this.chunkRetries = 3; this.chunkRequestsCountInParallel = 50; this.maxFileSize = 0; this.withCredentials = false; this.selectButtonIcon = 'fa fa-folder'; this.selectButtonLabel = 'Select'; this.removeButtonIcon = 'fa fa-trash'; this.removeButtonLabel = 'Remove'; this.removeButtonAllowed = true; this.fileTypeErrorMessage = 'The file type [{extension}] is not allowed.'; this.fileSizeErrorMessage = 'This file exceeds the max file size allowed of {maxFileSize}MB.'; this.maxFileSizeLabel = 'Max file size:'; this.allowedExtensionsLabel = 'Allowed extensions:'; this.onFileAdded = new EventEmitter(); this.onUploadComplete = new EventEmitter(); this.onChunkFileUpload = new EventEmitter(); this.onError = new EventEmitter(); this.onClear = new EventEmitter(); this.identifier = `input-upload-${identifier++}`; this.onParallelChunkCompletes = new EventEmitter(); this.selectedFileName = ''; this.hasDropZoneOver = false; this.chunkProgress = 0; if (config && config.upload) { Object.assign(this, config.upload); } } ngOnInit() { this.uploader = new FileUploader({ url: this.url, autoUpload: false, maxFileSize: this.maxFileSize * 1000000, }); this.handleUploaderEvents(); } Clear() { this.chunkProgress = 0; this.selectedFileModel = null; this.selectedFileName = ''; this.errorMessage = null; this.chunks = null; this.uploader.clearQueue(); this.uploader.cancelAll(); this.onClear.emit(); } StartUploadManually() { let promise = new Promise((resolve, reject) => { if (!this.selectedFileBlob) { reject(); return; } this.onUploadComplete.subscribe(result => resolve(result), error => reject(error)); this.onError.subscribe(result => reject(result)); if (this.chunk && this.chunks && this.chunks.length > 0) { this.startChunkUpload(); } else { this.startSingleUpload(); } }); return promise; } SetSelectedFileName(fileName) { this.selectedFileName = fileName; } OnFileOver(e) { this.hasDropZoneOver = e; } OnFileChange(event) { if (event.target.files[0]) { this.addSelectedFileForManualUploading(event.target.files[0]); } } OnFileDrop(event) { if (event[0]) { this.addSelectedFileForManualUploading(event[0]); } } HasFile() { return this.selectedFileBlob != null && this.selectedFileBlob != undefined; } ResetState() { if (this.uploader.queue && this.uploader.queue.length > 0) { for (let i = 0; i < this.uploader.queue.length; i++) { this.uploader.queue[i].isError = false; this.uploader.queue[i].isUploaded = false; this.uploader.queue[i].isSuccess = false; } } } startSingleUpload() { this.uploader.uploadAll(); } startChunkUpload() { this.chunkProgress = 0; if (this.chunkRequestsCountInParallel > 0 && this.chunks.length > this.chunkRequestsCountInParallel) { this.sendGroupedPromisesSequentially(); this.onParallelChunkCompletes.subscribe(() => { this.chunkProgress = 100; this.onUploadComplete.emit({ chunkId: this.chunks[0].id }); }); } else { let chunksPromises = []; chunksPromises.push(this.sendChunks(this.chunks)); Promise.all(chunksPromises) .then(() => { this.chunkProgress = 100; this.onUploadComplete.emit({ chunkId: this.chunks[0].id }); }) .catch(error => { this.onError.emit(error); }); } } sendChunks(chunks) { let promise = new Promise((resolve, reject) => { let requests = []; for (let i = 0; i < chunks.length; i++) { let formData = new FormData(); formData.append("file", chunks[i].blob, chunks[i].name); requests.push(this.http.post(this.url, formData, { withCredentials: this.withCredentials }) .pipe(retry(this.chunkRetries))); } zip(...requests).subscribe(() => { this.chunkProgress += (100 / this.chunks.length) * chunks.length; resolve(); }, error => reject(error)); }); return promise; } sendGroupedPromisesSequentially(index = 0) { let start = index * this.chunkRequestsCountInParallel; let end = ((index + 1) * this.chunkRequestsCountInParallel) - 1; let lastIndex = end + 1 >= this.chunks.length; if (end > this.chunks.length - 1) { end = this.chunks.length - 1; } this.sendChunks(this.chunks.slice(start, end + 1)) .then(() => { if (!lastIndex) { this.sendGroupedPromisesSequentially(index + 1); } else { this.onParallelChunkCompletes.emit(); } }) .catch(error => { this.onError.emit(error); this.Clear(); }); } splitSelectedFileInChunks() { this.chunks = []; let file = this.selectedFileBlob; let fileSize = file.size; let start = 0; let end = this.chunkSize; let chunksCount = 0; let chunkGuid = Math.random() .toString() .replace("0.", ""); if (fileSize % this.chunkSize == 0) { chunksCount = fileSize / this.chunkSize; } else { chunksCount = Math.floor(fileSize / this.chunkSize) + 1; } for (let i = 0; i < chunksCount; i++) { this.chunks.push({ id: chunkGuid, name: `${chunkGuid}_${i}`, blob: file.slice(start, end) }); start = end; end = start + this.chunkSize; } } handleUploaderEvents() { this.uploader.onBeforeUploadItem = (fileItem) => { fileItem.withCredentials = this.withCredentials; }; this.uploader.onSuccessItem = (item, response, status, headers) => { this.chunkProgress = 100; this.onUploadComplete.emit({ item, response }); this.ResetState(); }; this.uploader.onErrorItem = (item, response, status, headers) => { this.onError.emit({ item, response, status }); this.ResetState(); }; this.uploader.onWhenAddingFileFailed = (item, filter, options) => { this.onError.emit({ item, filter, options }); }; } addSelectedFileForManualUploading(file) { this.selectedFileBlob = null; if (!file) { return; } if (this.validate(file)) { this.selectedFileBlob = file; this.SetSelectedFileName(file.name); if (this.uploader.queue.length > 1) { this.uploader.removeFromQueue(this.uploader.queue[0]); } if (this.chunk) { this.splitSelectedFileInChunks(); } this.onFileAdded.emit(file); if (this.autoUpload) { this.StartUploadManually(); } } } validate(item) { this.selectedFileName = null; this.errorMessage = null; if (!this.validateFileType(item) || !this.validateFileSize(item)) { this.selectedFileModel = null; this.uploader.clearQueue(); return false; } return true; } validateFileType(file) { if (!this.allowedExtensions || this.allowedExtensions.length == 0) { return true; } let extensionArray = file.name.split('.'); let extension = extensionArray[extensionArray.length - 1].toLowerCase(); let result = this.allowedExtensions.find(x => x == extension); if (result == undefined || result == null) { this.errorMessage = this.fileTypeErrorMessage.replace('{extension}', extension); return false; } return true; } validateFileSize(file) { if (this.maxFileSize == 0) { return true; } if (file.size > this.maxFileSize * 1048576) { this.errorMessage = this.fileSizeErrorMessage.replace('{maxFileSize}', `${this.maxFileSize}`); return false; } return true; } } InputUploadComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: InputUploadComponent, deps: [{ token: INPUT_FORMS_CONFIG, optional: true }, { token: i1.HttpClient }], target: i0.ɵɵFactoryTarget.Component }); InputUploadComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: InputUploadComponent, selector: "input-upload", inputs: { url: "url", label: "label", help: "help", description: "description", placeholder: "placeholder", dropZonePlaceholder: "dropZonePlaceholder", disabled: "disabled", autoUpload: "autoUpload", showDropZone: "showDropZone", showQueue: "showQueue", chunk: "chunk", chunkSize: "chunkSize", chunkRetries: "chunkRetries", chunkRequestsCountInParallel: "chunkRequestsCountInParallel", maxFileSize: "maxFileSize", withCredentials: "withCredentials", selectButtonIcon: "selectButtonIcon", selectButtonLabel: "selectButtonLabel", removeButtonIcon: "removeButtonIcon", removeButtonLabel: "removeButtonLabel", removeButtonAllowed: "removeButtonAllowed", allowedExtensions: "allowedExtensions", fileTypeErrorMessage: "fileTypeErrorMessage", fileSizeErrorMessage: "fileSizeErrorMessage", maxFileSizeLabel: "maxFileSizeLabel", allowedExtensionsLabel: "allowedExtensionsLabel" }, outputs: { onFileAdded: "onFileAdded", onUploadComplete: "onUploadComplete", onChunkFileUpload: "onChunkFileUpload", onError: "onError", onClear: "onClear" }, ngImport: i0, template: "<label *ngIf=\"label\">\n {{label}}\n <i class=\"fa fa-question-circle ml-2\" tooltip=\"{{help}}\" container=\"body\" *ngIf=\"help\"></i>\n</label>\n\n<input type=\"file\" class=\"d-none\" [(ngModel)]=\"selectedFileModel\" (change)=\"OnFileChange($event)\" #fileInput\n ng2FileSelect [uploader]=\"uploader\" [disabled]=\"disabled\" id=\"{{identifier}}-0\" />\n\n<div class=\"input-group\">\n <input type=\"text\" class=\"form-control\" [value]=\"selectedFileName\" placeholder=\"{{placeholder}}\" id=\"{{identifier}}-1\"\n (click)=\"fileInput.click()\" readonly />\n <div class=\"input-group-append\">\n <button class=\"btn btn-primary\" type=\"button\" (click)=\"fileInput.click()\" [disabled]=\"disabled\"><i\n *ngIf=\"selectButtonIcon\" class=\"{{selectButtonIcon}}\"></i> {{selectButtonLabel}}</button>\n <button class=\"btn btn-outline-danger\" type=\"button\" *ngIf=\"selectedFileName && removeButtonAllowed\"\n (click)=\"Clear()\" [disabled]=\"disabled\"><i *ngIf=\"removeButtonIcon\" class=\"{{removeButtonIcon}}\"></i>\n {{removeButtonLabel}}</button>\n </div>\n</div>\n\n<p class=\"mb-0\" *ngIf=\"description\">\n <small class=\"text-muted\">{{description}}</small>\n</p>\n\n<small class=\"text-muted\" *ngIf=\"maxFileSize > 0\">\n {{maxFileSizeLabel}} <b>{{maxFileSize}}MB</b>.\n</small>\n<small class=\"text-muted\" *ngIf=\"allowedExtensions?.length > 0\">\n {{allowedExtensionsLabel}}\n <span *ngFor=\"let item of allowedExtensions; let first = first;\">\n <span *ngIf=\"!first\">, </span>\n <b>{{item}}</b>\n </span>\n <span>.</span>\n</small>\n\n<div *ngIf=\"errorMessage\">\n <small class=\"text-danger\">{{errorMessage}}</small>\n</div>\n\n<div class=\"chunks-queue\" *ngIf=\"chunk && chunks?.length > 0 && showQueue\">\n <div class=\"chunk\">\n <div class=\"d-flex\">\n <div class=\"icon mr-2\">\n <i class=\"fa fa-file fa-2x\"></i>\n </div>\n <div class=\"data flex-grow-1\">\n <div class=\"d-flex align-items-center\">\n <small class=\"file-name\"><b>{{selectedFileName}} ({{chunks.length}})</b></small>\n <small class=\"file-size ml-auto\">{{selectedFileBlob?.size / 1000 | number:'1.2-2'}}KB</small>\n </div>\n <progressbar [value]=\"chunkProgress\">\n {{chunkProgress}}%\n </progressbar>\n </div>\n </div>\n </div>\n</div>\n\n<div ng2FileDrop *ngIf=\"showDropZone\" [ngClass]=\"{'mouse-over': hasDropZoneOver}\" (fileOver)=\"OnFileOver($event)\"\n (onFileDrop)=\"OnFileDrop($event)\" [uploader]=\"uploader\" class=\"dropzone\">\n <i class=\"fa fa-cloud-download fa-4x\"></i><br />\n <span>{{dropZonePlaceholder}}</span>\n</div>\n", styles: [".dropzone{background:white;border-radius:5px;border:2px dashed #CCC;margin:10px 0;padding:40px;text-align:center}.dropzone i{color:#ccc}.dropzone span{color:#999}.dropzone,.dropzone>*{transition:color .35s ease-in-out,background-color .35s ease-in-out}.dropzone.mouse-over{background-color:#f0f0f0;border:2px solid #CCC;opacity:.5}.dropzone.mouse-over i,.dropzone.mouse-over span{color:#292929}.chunks-queue{border:1px solid #CCC;border-radius:6px;margin:10px 0}.chunks-queue .chunk{padding:10px;border-bottom:1px solid #CCC}.chunks-queue .chunk .icon i{font-size:2.2em}.chunks-queue .chunk .icon i,.chunks-queue .chunk .data .file-name{color:#00559a}.chunks-queue .chunk:last-child{border-bottom:0}.chunks-queue.scrollable{max-height:300px;overflow-y:scroll}\n"], dependencies: [{ kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.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: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i4.FileDropDirective, selector: "[ng2FileDrop]", inputs: ["uploader"], outputs: ["fileOver", "onFileDrop"] }, { kind: "directive", type: i4.FileSelectDirective, selector: "[ng2FileSelect]", inputs: ["uploader"], outputs: ["onFileSelected"] }, { kind: "component", type: i5.ProgressbarComponent, selector: "progressbar", inputs: ["animate", "striped", "value", "max", "type"] }, { kind: "directive", type: i6.TooltipDirective, selector: "[tooltip], [tooltipHtml]", inputs: ["containerClass", "tooltipAnimation", "tooltipFadeDuration", "isOpen", "tooltipHtml", "tooltip", "tooltipPlacement", "placement", "tooltipIsOpen", "tooltipEnable", "isDisabled", "tooltipAppendToBody", "container", "tooltipClass", "tooltipContext", "tooltipPopupDelay", "delay", "tooltipTrigger", "triggers", "adaptivePosition"], outputs: ["tooltipChange", "tooltipStateChanged", "onShown", "onHidden"], exportAs: ["bs-tooltip"] }, { kind: "pipe", type: i2.DecimalPipe, name: "number" }] }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: InputUploadComponent, decorators: [{ type: Component, args: [{ selector: 'input-upload', template: "<label *ngIf=\"label\">\n {{label}}\n <i class=\"fa fa-question-circle ml-2\" tooltip=\"{{help}}\" container=\"body\" *ngIf=\"help\"></i>\n</label>\n\n<input type=\"file\" class=\"d-none\" [(ngModel)]=\"selectedFileModel\" (change)=\"OnFileChange($event)\" #fileInput\n ng2FileSelect [uploader]=\"uploader\" [disabled]=\"disabled\" id=\"{{identifier}}-0\" />\n\n<div class=\"input-group\">\n <input type=\"text\" class=\"form-control\" [value]=\"selectedFileName\" placeholder=\"{{placeholder}}\" id=\"{{identifier}}-1\"\n (click)=\"fileInput.click()\" readonly />\n <div class=\"input-group-append\">\n <button class=\"btn btn-primary\" type=\"button\" (click)=\"fileInput.click()\" [disabled]=\"disabled\"><i\n *ngIf=\"selectButtonIcon\" class=\"{{selectButtonIcon}}\"></i> {{selectButtonLabel}}</button>\n <button class=\"btn btn-outline-danger\" type=\"button\" *ngIf=\"selectedFileName && removeButtonAllowed\"\n (click)=\"Clear()\" [disabled]=\"disabled\"><i *ngIf=\"removeButtonIcon\" class=\"{{removeButtonIcon}}\"></i>\n {{removeButtonLabel}}</button>\n </div>\n</div>\n\n<p class=\"mb-0\" *ngIf=\"description\">\n <small class=\"text-muted\">{{description}}</small>\n</p>\n\n<small class=\"text-muted\" *ngIf=\"maxFileSize > 0\">\n {{maxFileSizeLabel}} <b>{{maxFileSize}}MB</b>.\n</small>\n<small class=\"text-muted\" *ngIf=\"allowedExtensions?.length > 0\">\n {{allowedExtensionsLabel}}\n <span *ngFor=\"let item of allowedExtensions; let first = first;\">\n <span *ngIf=\"!first\">, </span>\n <b>{{item}}</b>\n </span>\n <span>.</span>\n</small>\n\n<div *ngIf=\"errorMessage\">\n <small class=\"text-danger\">{{errorMessage}}</small>\n</div>\n\n<div class=\"chunks-queue\" *ngIf=\"chunk && chunks?.length > 0 && showQueue\">\n <div class=\"chunk\">\n <div class=\"d-flex\">\n <div class=\"icon mr-2\">\n <i class=\"fa fa-file fa-2x\"></i>\n </div>\n <div class=\"data flex-grow-1\">\n <div class=\"d-flex align-items-center\">\n <small class=\"file-name\"><b>{{selectedFileName}} ({{chunks.length}})</b></small>\n <small class=\"file-size ml-auto\">{{selectedFileBlob?.size / 1000 | number:'1.2-2'}}KB</small>\n </div>\n <progressbar [value]=\"chunkProgress\">\n {{chunkProgress}}%\n </progressbar>\n </div>\n </div>\n </div>\n</div>\n\n<div ng2FileDrop *ngIf=\"showDropZone\" [ngClass]=\"{'mouse-over': hasDropZoneOver}\" (fileOver)=\"OnFileOver($event)\"\n (onFileDrop)=\"OnFileDrop($event)\" [uploader]=\"uploader\" class=\"dropzone\">\n <i class=\"fa fa-cloud-download fa-4x\"></i><br />\n <span>{{dropZonePlaceholder}}</span>\n</div>\n", styles: [".dropzone{background:white;border-radius:5px;border:2px dashed #CCC;margin:10px 0;padding:40px;text-align:center}.dropzone i{color:#ccc}.dropzone span{color:#999}.dropzone,.dropzone>*{transition:color .35s ease-in-out,background-color .35s ease-in-out}.dropzone.mouse-over{background-color:#f0f0f0;border:2px solid #CCC;opacity:.5}.dropzone.mouse-over i,.dropzone.mouse-over span{color:#292929}.chunks-queue{border:1px solid #CCC;border-radius:6px;margin:10px 0}.chunks-queue .chunk{padding:10px;border-bottom:1px solid #CCC}.chunks-queue .chunk .icon i{font-size:2.2em}.chunks-queue .chunk .icon i,.chunks-queue .chunk .data .file-name{color:#00559a}.chunks-queue .chunk:last-child{border-bottom:0}.chunks-queue.scrollable{max-height:300px;overflow-y:scroll}\n"] }] }], ctorParameters: function () { return [{ type: undefined, decorators: [{ type: Inject, args: [INPUT_FORMS_CONFIG] }, { type: Optional }] }, { type: i1.HttpClient }]; }, propDecorators: { url: [{ type: Input }], label: [{ type: Input }], help: [{ type: Input }], description: [{ type: Input }], placeholder: [{ type: Input }], dropZonePlaceholder: [{ type: Input }], disabled: [{ type: Input }], autoUpload: [{ type: Input }], showDropZone: [{ type: Input }], showQueue: [{ type: Input }], chunk: [{ type: Input }], chunkSize: [{ type: Input }], chunkRetries: [{ type: Input }], chunkRequestsCountInParallel: [{ type: Input }], maxFileSize: [{ type: Input }], withCredentials: [{ type: Input }], selectButtonIcon: [{ type: Input }], selectButtonLabel: [{ type: Input }], removeButtonIcon: [{ type: Input }], removeButtonLabel: [{ type: Input }], removeButtonAllowed: [{ type: Input }], allowedExtensions: [{ type: Input }], fileTypeErrorMessage: [{ type: Input }], fileSizeErrorMessage: [{ type: Input }], maxFileSizeLabel: [{ type: Input }], allowedExtensionsLabel: [{ type: Input }], onFileAdded: [{ type: Output }], onUploadComplete: [{ type: Output }], onChunkFileUpload: [{ type: Output }], onError: [{ type: Output }], onClear: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"input-upload.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-ui-hero/src/lib/input-forms/components/input-upload/input-upload.component.ts","../../../../../../../projects/ngx-ui-hero/src/lib/input-forms/components/input-upload/input-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import { FileItem, FileUploader } from 'ng2-file-upload';\nimport { Observable, zip } from 'rxjs';\nimport { retry } from 'rxjs/operators';\n\nimport { HttpClient } from '@angular/common/http';\nimport { Component, EventEmitter, Inject, Input, OnInit, Optional, Output } from '@angular/core';\n\nimport { InputFormsConfig } from '../../input-forms-config';\nimport { INPUT_FORMS_CONFIG } from '../../input-forms-config.constants';\n\nlet identifier = 0;\n\n@Component({\n  selector: 'input-upload',\n  templateUrl: 'input-upload.component.html',\n  styleUrls: ['input-upload.component.scss']\n})\nexport class InputUploadComponent implements OnInit {\n  @Input() url: string;\n  @Input() label?: string;\n  @Input() help: string;\n  @Input() description: string;\n  @Input() placeholder?: string = 'Select a file to upload...';\n  @Input() dropZonePlaceholder?: string = 'Drag & drop a file to import.';\n  @Input() disabled?: boolean = false;\n  @Input() autoUpload?: boolean = true;\n  @Input() showDropZone?: boolean = false;\n  @Input() showQueue?: boolean = false;\n  @Input() chunk?: boolean = false;\n  @Input() chunkSize?: number = 1048576;\n  @Input() chunkRetries?: number = 3;\n  @Input() chunkRequestsCountInParallel?: number = 50;\n  @Input() maxFileSize?: number = 0;\n  @Input() withCredentials?: boolean = false;\n  @Input() selectButtonIcon?: string = 'fa fa-folder';\n  @Input() selectButtonLabel?: string = 'Select';\n  @Input() removeButtonIcon?: string = 'fa fa-trash';\n  @Input() removeButtonLabel?: string = 'Remove';\n  @Input() removeButtonAllowed?: boolean = true;\n  @Input() allowedExtensions?: Array<string>;\n  @Input() fileTypeErrorMessage?: string = 'The file type [{extension}] is not allowed.';\n  @Input() fileSizeErrorMessage?: string = 'This file exceeds the max file size allowed of {maxFileSize}MB.';\n  @Input() maxFileSizeLabel?: string = 'Max file size:';\n  @Input() allowedExtensionsLabel?: string = 'Allowed extensions:';\n  @Output() onFileAdded = new EventEmitter<any>();\n  @Output() onUploadComplete = new EventEmitter<any>();\n  @Output() onChunkFileUpload = new EventEmitter<any>();\n  @Output() onError = new EventEmitter<any>();\n  @Output() onClear = new EventEmitter();\n\n  public identifier = `input-upload-${identifier++}`;\n  private onParallelChunkCompletes = new EventEmitter();\n\n  selectedFileBlob: any;\n  selectedFileModel: any;\n  selectedFileName: string = '';\n  errorMessage: string;\n  uploader: FileUploader;\n  hasDropZoneOver: boolean = false;\n  chunks: any[];\n  chunkProgress: number = 0;\n\n  constructor(\n    @Inject(INPUT_FORMS_CONFIG) @Optional() config: InputFormsConfig,\n    private http: HttpClient\n  ) {\n    if (config && config.upload) {\n      Object.assign(this, config.upload);\n    }\n  }\n\n  ngOnInit() {\n    this.uploader = new FileUploader({\n      url: this.url,\n      autoUpload: false,\n      maxFileSize: this.maxFileSize * 1000000,\n    });\n\n    this.handleUploaderEvents();\n  }\n\n  Clear(): void {\n    this.chunkProgress = 0;\n    this.selectedFileModel = null;\n    this.selectedFileName = '';\n    this.errorMessage = null;\n    this.chunks = null;\n    this.uploader.clearQueue();\n    this.uploader.cancelAll();\n    this.onClear.emit();\n  }\n\n  StartUploadManually(): Promise<any> {\n    let promise = new Promise<any>((resolve, reject) => {\n      if (!this.selectedFileBlob) {\n        reject();\n        return;\n      }\n\n      this.onUploadComplete.subscribe(result => resolve(result), error => reject(error));\n      this.onError.subscribe(result => reject(result));\n\n      if (this.chunk && this.chunks && this.chunks.length > 0) {\n        this.startChunkUpload();\n      } else {\n        this.startSingleUpload();\n      }\n    });\n\n    return promise;\n  }\n\n  SetSelectedFileName(fileName: string): void {\n    this.selectedFileName = fileName;\n  }\n\n  OnFileOver(e: any): void {\n    this.hasDropZoneOver = e;\n  }\n  OnFileChange(event: any): void {\n    if (event.target.files[0]) {\n      this.addSelectedFileForManualUploading(event.target.files[0]);\n    }\n  }\n  OnFileDrop(event: any): void {\n    if (event[0]) {\n      this.addSelectedFileForManualUploading(event[0]);\n    }\n  }\n\n  HasFile(): boolean {\n    return this.selectedFileBlob != null && this.selectedFileBlob != undefined;\n  }\n\n  ResetState(): void {\n    if (this.uploader.queue && this.uploader.queue.length > 0) {\n      for (let i = 0; i < this.uploader.queue.length; i++) {\n        this.uploader.queue[i].isError = false;\n        this.uploader.queue[i].isUploaded = false;\n        this.uploader.queue[i].isSuccess = false;\n      }\n    }\n  }\n\n  private startSingleUpload(): void {\n    this.uploader.uploadAll();\n  }\n  private startChunkUpload(): void {\n    this.chunkProgress = 0;\n\n    if (this.chunkRequestsCountInParallel > 0 && this.chunks.length > this.chunkRequestsCountInParallel) {\n      this.sendGroupedPromisesSequentially();\n\n      this.onParallelChunkCompletes.subscribe(() => {\n        this.chunkProgress = 100;\n        this.onUploadComplete.emit({ chunkId: this.chunks[0].id });\n      });\n    } else {\n      let chunksPromises: Array<Promise<void>> = [];\n      chunksPromises.push(this.sendChunks(this.chunks));\n\n      Promise.all(chunksPromises)\n        .then(() => {\n          this.chunkProgress = 100;\n          this.onUploadComplete.emit({ chunkId: this.chunks[0].id });\n        })\n        .catch(error => {\n          this.onError.emit(error);\n        });\n    }\n  }\n  private sendChunks(chunks: any[]): Promise<void> {\n    let promise = new Promise<void>((resolve, reject) => {\n      let requests: Array<Observable<any>> = [];\n\n      for (let i = 0; i < chunks.length; i++) {\n        let formData = new FormData();\n        formData.append(\"file\", chunks[i].blob, chunks[i].name);\n\n        requests.push(this.http.post(this.url, formData, { withCredentials: this.withCredentials })\n          .pipe(retry(this.chunkRetries)));\n      }\n\n      zip(...requests).subscribe(\n        () => {\n          this.chunkProgress += (100 / this.chunks.length) * chunks.length;\n          resolve();\n        },\n        error => reject(error)\n      );\n    });\n\n    return promise;\n  }\n  private sendGroupedPromisesSequentially(index: number = 0): void {\n    let start = index * this.chunkRequestsCountInParallel;\n    let end = ((index + 1) * this.chunkRequestsCountInParallel) - 1;\n    let lastIndex = end + 1 >= this.chunks.length;\n\n    if (end > this.chunks.length - 1) {\n      end = this.chunks.length - 1;\n    }\n\n    this.sendChunks(this.chunks.slice(start, end + 1))\n      .then(() => {\n        if (!lastIndex) {\n          this.sendGroupedPromisesSequentially(index + 1);\n        } else {\n          this.onParallelChunkCompletes.emit();\n        }\n      })\n      .catch(error => {\n        this.onError.emit(error);\n        this.Clear();\n      });\n  }\n  private splitSelectedFileInChunks(): void {\n    this.chunks = [];\n    let file: File = this.selectedFileBlob;\n    let fileSize = file.size;\n    let start = 0;\n    let end = this.chunkSize;\n    let chunksCount = 0;\n    let chunkGuid = Math.random()\n      .toString()\n      .replace(\"0.\", \"\");\n\n    if (fileSize % this.chunkSize == 0) {\n      chunksCount = fileSize / this.chunkSize;\n    } else {\n      chunksCount = Math.floor(fileSize / this.chunkSize) + 1;\n    }\n\n    for (let i = 0; i < chunksCount; i++) {\n      this.chunks.push({\n        id: chunkGuid,\n        name: `${chunkGuid}_${i}`,\n        blob: file.slice(start, end)\n      });\n\n      start = end;\n      end = start + this.chunkSize;\n    }\n  }\n  private handleUploaderEvents(): void {\n    this.uploader.onBeforeUploadItem = (fileItem: FileItem) => {\n      fileItem.withCredentials = this.withCredentials;\n    };\n    this.uploader.onSuccessItem = (item: any, response: any, status: any, headers: any) => {\n      this.chunkProgress = 100;\n      this.onUploadComplete.emit({ item, response });\n      this.ResetState();\n    };\n    this.uploader.onErrorItem = (item: any, response: any, status: any, headers: any) => {\n      this.onError.emit({ item, response, status });\n      this.ResetState();\n    };\n    this.uploader.onWhenAddingFileFailed = (item: any, filter: any, options: any) => {\n      this.onError.emit({ item, filter, options });\n    };\n  }\n  private addSelectedFileForManualUploading(file: any): void {\n    this.selectedFileBlob = null;\n\n    if (!file) {\n      return;\n    }\n\n    if (this.validate(file)) {\n      this.selectedFileBlob = file;\n      this.SetSelectedFileName(file.name);\n\n      if (this.uploader.queue.length > 1) {\n        this.uploader.removeFromQueue(this.uploader.queue[0]);\n      }\n\n      if (this.chunk) {\n        this.splitSelectedFileInChunks();\n      }\n\n      this.onFileAdded.emit(file);\n\n      if (this.autoUpload) {\n        this.StartUploadManually();\n      }\n    }\n  }\n  private validate(item: any): boolean {\n    this.selectedFileName = null;\n    this.errorMessage = null;\n\n    if (!this.validateFileType(item) || !this.validateFileSize(item)) {\n      this.selectedFileModel = null;\n      this.uploader.clearQueue();\n      return false;\n    }\n\n    return true;\n  }\n  private validateFileType(file: any): boolean {\n    if (!this.allowedExtensions ||