1 | import { NgModule, Component, Input, Output, EventEmitter, ContentChildren, ViewChild, ChangeDetectionStrategy, ViewEncapsulation } from '@angular/core';
|
---|
2 | import { CommonModule } from '@angular/common';
|
---|
3 | import { ButtonModule } from 'primeng/button';
|
---|
4 | import { MessagesModule } from 'primeng/messages';
|
---|
5 | import { ProgressBarModule } from 'primeng/progressbar';
|
---|
6 | import { DomHandler } from 'primeng/dom';
|
---|
7 | import { TranslationKeys } from 'primeng/api';
|
---|
8 | import { PrimeTemplate, SharedModule } from 'primeng/api';
|
---|
9 | import { RippleModule } from 'primeng/ripple';
|
---|
10 | import { HttpEventType } from "@angular/common/http";
|
---|
11 | import * as i0 from "@angular/core";
|
---|
12 | import * as i1 from "@angular/platform-browser";
|
---|
13 | import * as i2 from "@angular/common/http";
|
---|
14 | import * as i3 from "primeng/api";
|
---|
15 | import * as i4 from "primeng/button";
|
---|
16 | import * as i5 from "primeng/progressbar";
|
---|
17 | import * as i6 from "primeng/messages";
|
---|
18 | import * as i7 from "@angular/common";
|
---|
19 | import * as i8 from "primeng/ripple";
|
---|
20 | export class FileUpload {
|
---|
21 | constructor(el, sanitizer, zone, http, cd, config) {
|
---|
22 | this.el = el;
|
---|
23 | this.sanitizer = sanitizer;
|
---|
24 | this.zone = zone;
|
---|
25 | this.http = http;
|
---|
26 | this.cd = cd;
|
---|
27 | this.config = config;
|
---|
28 | this.method = 'post';
|
---|
29 | this.invalidFileSizeMessageSummary = '{0}: Invalid file size, ';
|
---|
30 | this.invalidFileSizeMessageDetail = 'maximum upload size is {0}.';
|
---|
31 | this.invalidFileTypeMessageSummary = '{0}: Invalid file type, ';
|
---|
32 | this.invalidFileTypeMessageDetail = 'allowed file types: {0}.';
|
---|
33 | this.invalidFileLimitMessageDetail = 'limit is {0} at most.';
|
---|
34 | this.invalidFileLimitMessageSummary = 'Maximum number of files exceeded, ';
|
---|
35 | this.previewWidth = 50;
|
---|
36 | this.chooseIcon = 'pi pi-plus';
|
---|
37 | this.uploadIcon = 'pi pi-upload';
|
---|
38 | this.cancelIcon = 'pi pi-times';
|
---|
39 | this.showUploadButton = true;
|
---|
40 | this.showCancelButton = true;
|
---|
41 | this.mode = 'advanced';
|
---|
42 | this.onBeforeUpload = new EventEmitter();
|
---|
43 | this.onSend = new EventEmitter();
|
---|
44 | this.onUpload = new EventEmitter();
|
---|
45 | this.onError = new EventEmitter();
|
---|
46 | this.onClear = new EventEmitter();
|
---|
47 | this.onRemove = new EventEmitter();
|
---|
48 | this.onSelect = new EventEmitter();
|
---|
49 | this.onProgress = new EventEmitter();
|
---|
50 | this.uploadHandler = new EventEmitter();
|
---|
51 | this._files = [];
|
---|
52 | this.progress = 0;
|
---|
53 | this.uploadedFileCount = 0;
|
---|
54 | }
|
---|
55 | set files(files) {
|
---|
56 | this._files = [];
|
---|
57 | for (let i = 0; i < files.length; i++) {
|
---|
58 | let file = files[i];
|
---|
59 | if (this.validate(file)) {
|
---|
60 | if (this.isImage(file)) {
|
---|
61 | file.objectURL = this.sanitizer.bypassSecurityTrustUrl((window.URL.createObjectURL(files[i])));
|
---|
62 | }
|
---|
63 | this._files.push(files[i]);
|
---|
64 | }
|
---|
65 | }
|
---|
66 | }
|
---|
67 | get files() {
|
---|
68 | return this._files;
|
---|
69 | }
|
---|
70 | ngAfterContentInit() {
|
---|
71 | this.templates.forEach((item) => {
|
---|
72 | switch (item.getType()) {
|
---|
73 | case 'file':
|
---|
74 | this.fileTemplate = item.template;
|
---|
75 | break;
|
---|
76 | case 'content':
|
---|
77 | this.contentTemplate = item.template;
|
---|
78 | break;
|
---|
79 | case 'toolbar':
|
---|
80 | this.toolbarTemplate = item.template;
|
---|
81 | break;
|
---|
82 | default:
|
---|
83 | this.fileTemplate = item.template;
|
---|
84 | break;
|
---|
85 | }
|
---|
86 | });
|
---|
87 | }
|
---|
88 | ngOnInit() {
|
---|
89 | this.translationSubscription = this.config.translationObserver.subscribe(() => {
|
---|
90 | this.cd.markForCheck();
|
---|
91 | });
|
---|
92 | }
|
---|
93 | ngAfterViewInit() {
|
---|
94 | if (this.mode === 'advanced') {
|
---|
95 | this.zone.runOutsideAngular(() => {
|
---|
96 | if (this.content)
|
---|
97 | this.content.nativeElement.addEventListener('dragover', this.onDragOver.bind(this));
|
---|
98 | });
|
---|
99 | }
|
---|
100 | }
|
---|
101 | choose() {
|
---|
102 | this.advancedFileInput.nativeElement.click();
|
---|
103 | }
|
---|
104 | onFileSelect(event) {
|
---|
105 | if (event.type !== 'drop' && this.isIE11() && this.duplicateIEEvent) {
|
---|
106 | this.duplicateIEEvent = false;
|
---|
107 | return;
|
---|
108 | }
|
---|
109 | this.msgs = [];
|
---|
110 | if (!this.multiple) {
|
---|
111 | this.files = [];
|
---|
112 | }
|
---|
113 | let files = event.dataTransfer ? event.dataTransfer.files : event.target.files;
|
---|
114 | for (let i = 0; i < files.length; i++) {
|
---|
115 | let file = files[i];
|
---|
116 | if (!this.isFileSelected(file)) {
|
---|
117 | if (this.validate(file)) {
|
---|
118 | if (this.isImage(file)) {
|
---|
119 | file.objectURL = this.sanitizer.bypassSecurityTrustUrl((window.URL.createObjectURL(files[i])));
|
---|
120 | }
|
---|
121 | this.files.push(files[i]);
|
---|
122 | }
|
---|
123 | }
|
---|
124 | }
|
---|
125 | this.onSelect.emit({ originalEvent: event, files: files, currentFiles: this.files });
|
---|
126 | if (this.fileLimit && this.mode == "advanced") {
|
---|
127 | this.checkFileLimit();
|
---|
128 | }
|
---|
129 | if (this.hasFiles() && this.auto && (!(this.mode === "advanced") || !this.isFileLimitExceeded())) {
|
---|
130 | this.upload();
|
---|
131 | }
|
---|
132 | if (event.type !== 'drop' && this.isIE11()) {
|
---|
133 | this.clearIEInput();
|
---|
134 | }
|
---|
135 | else {
|
---|
136 | this.clearInputElement();
|
---|
137 | }
|
---|
138 | }
|
---|
139 | isFileSelected(file) {
|
---|
140 | for (let sFile of this.files) {
|
---|
141 | if ((sFile.name + sFile.type + sFile.size) === (file.name + file.type + file.size)) {
|
---|
142 | return true;
|
---|
143 | }
|
---|
144 | }
|
---|
145 | return false;
|
---|
146 | }
|
---|
147 | isIE11() {
|
---|
148 | return !!window['MSInputMethodContext'] && !!document['documentMode'];
|
---|
149 | }
|
---|
150 | validate(file) {
|
---|
151 | if (this.accept && !this.isFileTypeValid(file)) {
|
---|
152 | this.msgs.push({
|
---|
153 | severity: 'error',
|
---|
154 | summary: this.invalidFileTypeMessageSummary.replace('{0}', file.name),
|
---|
155 | detail: this.invalidFileTypeMessageDetail.replace('{0}', this.accept)
|
---|
156 | });
|
---|
157 | return false;
|
---|
158 | }
|
---|
159 | if (this.maxFileSize && file.size > this.maxFileSize) {
|
---|
160 | this.msgs.push({
|
---|
161 | severity: 'error',
|
---|
162 | summary: this.invalidFileSizeMessageSummary.replace('{0}', file.name),
|
---|
163 | detail: this.invalidFileSizeMessageDetail.replace('{0}', this.formatSize(this.maxFileSize))
|
---|
164 | });
|
---|
165 | return false;
|
---|
166 | }
|
---|
167 | return true;
|
---|
168 | }
|
---|
169 | isFileTypeValid(file) {
|
---|
170 | let acceptableTypes = this.accept.split(',').map(type => type.trim());
|
---|
171 | for (let type of acceptableTypes) {
|
---|
172 | let acceptable = this.isWildcard(type) ? this.getTypeClass(file.type) === this.getTypeClass(type)
|
---|
173 | : file.type == type || this.getFileExtension(file).toLowerCase() === type.toLowerCase();
|
---|
174 | if (acceptable) {
|
---|
175 | return true;
|
---|
176 | }
|
---|
177 | }
|
---|
178 | return false;
|
---|
179 | }
|
---|
180 | getTypeClass(fileType) {
|
---|
181 | return fileType.substring(0, fileType.indexOf('/'));
|
---|
182 | }
|
---|
183 | isWildcard(fileType) {
|
---|
184 | return fileType.indexOf('*') !== -1;
|
---|
185 | }
|
---|
186 | getFileExtension(file) {
|
---|
187 | return '.' + file.name.split('.').pop();
|
---|
188 | }
|
---|
189 | isImage(file) {
|
---|
190 | return /^image\//.test(file.type);
|
---|
191 | }
|
---|
192 | onImageLoad(img) {
|
---|
193 | window.URL.revokeObjectURL(img.src);
|
---|
194 | }
|
---|
195 | upload() {
|
---|
196 | if (this.customUpload) {
|
---|
197 | if (this.fileLimit) {
|
---|
198 | this.uploadedFileCount += this.files.length;
|
---|
199 | }
|
---|
200 | this.uploadHandler.emit({
|
---|
201 | files: this.files
|
---|
202 | });
|
---|
203 | this.cd.markForCheck();
|
---|
204 | }
|
---|
205 | else {
|
---|
206 | this.uploading = true;
|
---|
207 | this.msgs = [];
|
---|
208 | let formData = new FormData();
|
---|
209 | this.onBeforeUpload.emit({
|
---|
210 | 'formData': formData
|
---|
211 | });
|
---|
212 | for (let i = 0; i < this.files.length; i++) {
|
---|
213 | formData.append(this.name, this.files[i], this.files[i].name);
|
---|
214 | }
|
---|
215 | this.http[this.method](this.url, formData, {
|
---|
216 | headers: this.headers, reportProgress: true, observe: 'events', withCredentials: this.withCredentials
|
---|
217 | }).subscribe((event) => {
|
---|
218 | switch (event.type) {
|
---|
219 | case HttpEventType.Sent:
|
---|
220 | this.onSend.emit({
|
---|
221 | originalEvent: event,
|
---|
222 | 'formData': formData
|
---|
223 | });
|
---|
224 | break;
|
---|
225 | case HttpEventType.Response:
|
---|
226 | this.uploading = false;
|
---|
227 | this.progress = 0;
|
---|
228 | if (event['status'] >= 200 && event['status'] < 300) {
|
---|
229 | if (this.fileLimit) {
|
---|
230 | this.uploadedFileCount += this.files.length;
|
---|
231 | }
|
---|
232 | this.onUpload.emit({ originalEvent: event, files: this.files });
|
---|
233 | }
|
---|
234 | else {
|
---|
235 | this.onError.emit({ files: this.files });
|
---|
236 | }
|
---|
237 | this.clear();
|
---|
238 | break;
|
---|
239 | case HttpEventType.UploadProgress: {
|
---|
240 | if (event['loaded']) {
|
---|
241 | this.progress = Math.round((event['loaded'] * 100) / event['total']);
|
---|
242 | }
|
---|
243 | this.onProgress.emit({ originalEvent: event, progress: this.progress });
|
---|
244 | break;
|
---|
245 | }
|
---|
246 | }
|
---|
247 | this.cd.markForCheck();
|
---|
248 | }, error => {
|
---|
249 | this.uploading = false;
|
---|
250 | this.onError.emit({ files: this.files, error: error });
|
---|
251 | });
|
---|
252 | }
|
---|
253 | }
|
---|
254 | clear() {
|
---|
255 | this.files = [];
|
---|
256 | this.onClear.emit();
|
---|
257 | this.clearInputElement();
|
---|
258 | this.cd.markForCheck();
|
---|
259 | }
|
---|
260 | remove(event, index) {
|
---|
261 | this.clearInputElement();
|
---|
262 | this.onRemove.emit({ originalEvent: event, file: this.files[index] });
|
---|
263 | this.files.splice(index, 1);
|
---|
264 | }
|
---|
265 | isFileLimitExceeded() {
|
---|
266 | if (this.fileLimit && this.fileLimit <= this.files.length + this.uploadedFileCount && this.focus) {
|
---|
267 | this.focus = false;
|
---|
268 | }
|
---|
269 | return this.fileLimit && this.fileLimit < this.files.length + this.uploadedFileCount;
|
---|
270 | }
|
---|
271 | isChooseDisabled() {
|
---|
272 | return this.fileLimit && this.fileLimit <= this.files.length + this.uploadedFileCount;
|
---|
273 | }
|
---|
274 | checkFileLimit() {
|
---|
275 | if (this.isFileLimitExceeded()) {
|
---|
276 | this.msgs.push({
|
---|
277 | severity: 'error',
|
---|
278 | summary: this.invalidFileLimitMessageSummary.replace('{0}', this.fileLimit.toString()),
|
---|
279 | detail: this.invalidFileLimitMessageDetail.replace('{0}', this.fileLimit.toString())
|
---|
280 | });
|
---|
281 | }
|
---|
282 | }
|
---|
283 | clearInputElement() {
|
---|
284 | if (this.advancedFileInput && this.advancedFileInput.nativeElement) {
|
---|
285 | this.advancedFileInput.nativeElement.value = '';
|
---|
286 | }
|
---|
287 | if (this.basicFileInput && this.basicFileInput.nativeElement) {
|
---|
288 | this.basicFileInput.nativeElement.value = '';
|
---|
289 | }
|
---|
290 | }
|
---|
291 | clearIEInput() {
|
---|
292 | if (this.advancedFileInput && this.advancedFileInput.nativeElement) {
|
---|
293 | this.duplicateIEEvent = true; //IE11 fix to prevent onFileChange trigger again
|
---|
294 | this.advancedFileInput.nativeElement.value = '';
|
---|
295 | }
|
---|
296 | }
|
---|
297 | hasFiles() {
|
---|
298 | return this.files && this.files.length > 0;
|
---|
299 | }
|
---|
300 | onDragEnter(e) {
|
---|
301 | if (!this.disabled) {
|
---|
302 | e.stopPropagation();
|
---|
303 | e.preventDefault();
|
---|
304 | }
|
---|
305 | }
|
---|
306 | onDragOver(e) {
|
---|
307 | if (!this.disabled) {
|
---|
308 | DomHandler.addClass(this.content.nativeElement, 'p-fileupload-highlight');
|
---|
309 | this.dragHighlight = true;
|
---|
310 | e.stopPropagation();
|
---|
311 | e.preventDefault();
|
---|
312 | }
|
---|
313 | }
|
---|
314 | onDragLeave(event) {
|
---|
315 | if (!this.disabled) {
|
---|
316 | DomHandler.removeClass(this.content.nativeElement, 'p-fileupload-highlight');
|
---|
317 | }
|
---|
318 | }
|
---|
319 | onDrop(event) {
|
---|
320 | if (!this.disabled) {
|
---|
321 | DomHandler.removeClass(this.content.nativeElement, 'p-fileupload-highlight');
|
---|
322 | event.stopPropagation();
|
---|
323 | event.preventDefault();
|
---|
324 | let files = event.dataTransfer ? event.dataTransfer.files : event.target.files;
|
---|
325 | let allowDrop = this.multiple || (files && files.length === 1);
|
---|
326 | if (allowDrop) {
|
---|
327 | this.onFileSelect(event);
|
---|
328 | }
|
---|
329 | }
|
---|
330 | }
|
---|
331 | onFocus() {
|
---|
332 | this.focus = true;
|
---|
333 | }
|
---|
334 | onBlur() {
|
---|
335 | this.focus = false;
|
---|
336 | }
|
---|
337 | formatSize(bytes) {
|
---|
338 | if (bytes == 0) {
|
---|
339 | return '0 B';
|
---|
340 | }
|
---|
341 | let k = 1000, dm = 3, sizes = ['B', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'], i = Math.floor(Math.log(bytes) / Math.log(k));
|
---|
342 | return parseFloat((bytes / Math.pow(k, i)).toFixed(dm)) + ' ' + sizes[i];
|
---|
343 | }
|
---|
344 | onBasicUploaderClick() {
|
---|
345 | if (this.hasFiles())
|
---|
346 | this.upload();
|
---|
347 | else
|
---|
348 | this.basicFileInput.nativeElement.click();
|
---|
349 | }
|
---|
350 | getBlockableElement() {
|
---|
351 | return this.el.nativeElement.children[0];
|
---|
352 | }
|
---|
353 | get chooseButtonLabel() {
|
---|
354 | return this.chooseLabel || this.config.getTranslation(TranslationKeys.CHOOSE);
|
---|
355 | }
|
---|
356 | get uploadButtonLabel() {
|
---|
357 | return this.uploadLabel || this.config.getTranslation(TranslationKeys.UPLOAD);
|
---|
358 | }
|
---|
359 | get cancelButtonLabel() {
|
---|
360 | return this.cancelLabel || this.config.getTranslation(TranslationKeys.CANCEL);
|
---|
361 | }
|
---|
362 | ngOnDestroy() {
|
---|
363 | if (this.content && this.content.nativeElement) {
|
---|
364 | this.content.nativeElement.removeEventListener('dragover', this.onDragOver);
|
---|
365 | }
|
---|
366 | if (this.translationSubscription) {
|
---|
367 | this.translationSubscription.unsubscribe();
|
---|
368 | }
|
---|
369 | }
|
---|
370 | }
|
---|
371 | FileUpload.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.0.0", ngImport: i0, type: FileUpload, deps: [{ token: i0.ElementRef }, { token: i1.DomSanitizer }, { token: i0.NgZone }, { token: i2.HttpClient }, { token: i0.ChangeDetectorRef }, { token: i3.PrimeNGConfig }], target: i0.ɵɵFactoryTarget.Component });
|
---|
372 | FileUpload.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.0.0", type: FileUpload, selector: "p-fileUpload", inputs: { name: "name", url: "url", method: "method", multiple: "multiple", accept: "accept", disabled: "disabled", auto: "auto", withCredentials: "withCredentials", maxFileSize: "maxFileSize", invalidFileSizeMessageSummary: "invalidFileSizeMessageSummary", invalidFileSizeMessageDetail: "invalidFileSizeMessageDetail", invalidFileTypeMessageSummary: "invalidFileTypeMessageSummary", invalidFileTypeMessageDetail: "invalidFileTypeMessageDetail", invalidFileLimitMessageDetail: "invalidFileLimitMessageDetail", invalidFileLimitMessageSummary: "invalidFileLimitMessageSummary", style: "style", styleClass: "styleClass", previewWidth: "previewWidth", chooseLabel: "chooseLabel", uploadLabel: "uploadLabel", cancelLabel: "cancelLabel", chooseIcon: "chooseIcon", uploadIcon: "uploadIcon", cancelIcon: "cancelIcon", showUploadButton: "showUploadButton", showCancelButton: "showCancelButton", mode: "mode", headers: "headers", customUpload: "customUpload", fileLimit: "fileLimit", files: "files" }, outputs: { onBeforeUpload: "onBeforeUpload", onSend: "onSend", onUpload: "onUpload", onError: "onError", onClear: "onClear", onRemove: "onRemove", onSelect: "onSelect", onProgress: "onProgress", uploadHandler: "uploadHandler" }, host: { classAttribute: "p-element" }, queries: [{ propertyName: "templates", predicate: PrimeTemplate }], viewQueries: [{ propertyName: "advancedFileInput", first: true, predicate: ["advancedfileinput"], descendants: true }, { propertyName: "basicFileInput", first: true, predicate: ["basicfileinput"], descendants: true }, { propertyName: "content", first: true, predicate: ["content"], descendants: true }], ngImport: i0, template: `
|
---|
373 | <div [ngClass]="'p-fileupload p-fileupload-advanced p-component'" [ngStyle]="style" [class]="styleClass" *ngIf="mode === 'advanced'">
|
---|
374 | <div class="p-fileupload-buttonbar">
|
---|
375 | <span class="p-button p-component p-fileupload-choose" [ngClass]="{'p-focus': focus, 'p-disabled':disabled || isChooseDisabled()}" (focus)="onFocus()" (blur)="onBlur()" pRipple
|
---|
376 | (click)="choose()" (keydown.enter)="choose()" tabindex="0">
|
---|
377 | <input #advancedfileinput type="file" (change)="onFileSelect($event)" [multiple]="multiple" [accept]="accept" [disabled]="disabled || isChooseDisabled()" [attr.title]="''">
|
---|
378 | <span [ngClass]="'p-button-icon p-button-icon-left'" [class]="chooseIcon"></span>
|
---|
379 | <span class="p-button-label">{{chooseButtonLabel}}</span>
|
---|
380 | </span>
|
---|
381 |
|
---|
382 | <p-button *ngIf="!auto&&showUploadButton" type="button" [label]="uploadButtonLabel" [icon]="uploadIcon" (onClick)="upload()" [disabled]="!hasFiles() || isFileLimitExceeded()"></p-button>
|
---|
383 | <p-button *ngIf="!auto&&showCancelButton" type="button" [label]="cancelButtonLabel" [icon]="cancelIcon" (onClick)="clear()" [disabled]="!hasFiles() || uploading"></p-button>
|
---|
384 |
|
---|
385 | <ng-container *ngTemplateOutlet="toolbarTemplate"></ng-container>
|
---|
386 | </div>
|
---|
387 | <div #content class="p-fileupload-content" (dragenter)="onDragEnter($event)" (dragleave)="onDragLeave($event)" (drop)="onDrop($event)">
|
---|
388 | <p-progressBar [value]="progress" [showValue]="false" *ngIf="hasFiles()"></p-progressBar>
|
---|
389 |
|
---|
390 | <p-messages [value]="msgs" [enableService]="false"></p-messages>
|
---|
391 |
|
---|
392 | <div class="p-fileupload-files" *ngIf="hasFiles()">
|
---|
393 | <div *ngIf="!fileTemplate">
|
---|
394 | <div class="p-fileupload-row" *ngFor="let file of files; let i = index;">
|
---|
395 | <div><img [src]="file.objectURL" *ngIf="isImage(file)" [width]="previewWidth" /></div>
|
---|
396 | <div class="p-fileupload-filename">{{file.name}}</div>
|
---|
397 | <div>{{formatSize(file.size)}}</div>
|
---|
398 | <div>
|
---|
399 | <button type="button" icon="pi pi-times" pButton (click)="remove($event,i)" [disabled]="uploading"></button>
|
---|
400 | </div>
|
---|
401 | </div>
|
---|
402 | </div>
|
---|
403 | <div *ngIf="fileTemplate">
|
---|
404 | <ng-template ngFor [ngForOf]="files" [ngForTemplate]="fileTemplate"></ng-template>
|
---|
405 | </div>
|
---|
406 | </div>
|
---|
407 | <ng-container *ngTemplateOutlet="contentTemplate; context: {$implicit: files}"></ng-container>
|
---|
408 | </div>
|
---|
409 | </div>
|
---|
410 | <div class="p-fileupload p-fileupload-basic p-component" *ngIf="mode === 'basic'">
|
---|
411 | <p-messages [value]="msgs" [enableService]="false"></p-messages>
|
---|
412 | <span [ngClass]="{'p-button p-component p-fileupload-choose': true, 'p-button-icon-only': !chooseLabel, 'p-fileupload-choose-selected': hasFiles(),'p-focus': focus, 'p-disabled':disabled}"
|
---|
413 | [ngStyle]="style" [class]="styleClass" (mouseup)="onBasicUploaderClick()" (keydown)="onBasicUploaderClick()" tabindex="0" pRipple>
|
---|
414 | <span class="p-button-icon p-button-icon-left pi" [ngClass]="hasFiles()&&!auto ? uploadIcon : chooseIcon"></span>
|
---|
415 | <span class="p-button-label">{{auto ? chooseLabel : hasFiles() ? files[0].name : chooseLabel}}</span>
|
---|
416 | <input #basicfileinput type="file" [accept]="accept" [multiple]="multiple" [disabled]="disabled"
|
---|
417 | (change)="onFileSelect($event)" *ngIf="!hasFiles()" (focus)="onFocus()" (blur)="onBlur()">
|
---|
418 | </span>
|
---|
419 | </div>
|
---|
420 | `, isInline: true, styles: [".p-fileupload-content{position:relative}.p-fileupload-row{display:flex;align-items:center}.p-fileupload-row>div{flex:1 1 auto;width:25%}.p-fileupload-row>div:last-child{text-align:right}.p-fileupload-content .p-progressbar{width:100%;position:absolute;top:0;left:0}.p-button.p-fileupload-choose{position:relative;overflow:hidden}.p-button.p-fileupload-choose input[type=file],.p-fileupload-choose.p-fileupload-choose-selected input[type=file]{display:none}.p-fluid .p-fileupload .p-button{width:auto}.p-fileupload-filename{word-break:break-all}\n"], components: [{ type: i4.Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "style", "styleClass", "badgeClass"], outputs: ["onClick", "onFocus", "onBlur"] }, { type: i5.ProgressBar, selector: "p-progressBar", inputs: ["value", "showValue", "style", "styleClass", "unit", "mode"] }, { type: i6.Messages, selector: "p-messages", inputs: ["value", "closable", "style", "styleClass", "enableService", "key", "escape", "severity", "showTransitionOptions", "hideTransitionOptions"], outputs: ["valueChange"] }], directives: [{ type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i7.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i7.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i8.Ripple, selector: "[pRipple]" }, { type: i7.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i4.ButtonDirective, selector: "[pButton]", inputs: ["iconPos", "loadingIcon", "label", "icon", "loading"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
|
---|
421 | i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.0.0", ngImport: i0, type: FileUpload, decorators: [{
|
---|
422 | type: Component,
|
---|
423 | args: [{ selector: 'p-fileUpload', template: `
|
---|
424 | <div [ngClass]="'p-fileupload p-fileupload-advanced p-component'" [ngStyle]="style" [class]="styleClass" *ngIf="mode === 'advanced'">
|
---|
425 | <div class="p-fileupload-buttonbar">
|
---|
426 | <span class="p-button p-component p-fileupload-choose" [ngClass]="{'p-focus': focus, 'p-disabled':disabled || isChooseDisabled()}" (focus)="onFocus()" (blur)="onBlur()" pRipple
|
---|
427 | (click)="choose()" (keydown.enter)="choose()" tabindex="0">
|
---|
428 | <input #advancedfileinput type="file" (change)="onFileSelect($event)" [multiple]="multiple" [accept]="accept" [disabled]="disabled || isChooseDisabled()" [attr.title]="''">
|
---|
429 | <span [ngClass]="'p-button-icon p-button-icon-left'" [class]="chooseIcon"></span>
|
---|
430 | <span class="p-button-label">{{chooseButtonLabel}}</span>
|
---|
431 | </span>
|
---|
432 |
|
---|
433 | <p-button *ngIf="!auto&&showUploadButton" type="button" [label]="uploadButtonLabel" [icon]="uploadIcon" (onClick)="upload()" [disabled]="!hasFiles() || isFileLimitExceeded()"></p-button>
|
---|
434 | <p-button *ngIf="!auto&&showCancelButton" type="button" [label]="cancelButtonLabel" [icon]="cancelIcon" (onClick)="clear()" [disabled]="!hasFiles() || uploading"></p-button>
|
---|
435 |
|
---|
436 | <ng-container *ngTemplateOutlet="toolbarTemplate"></ng-container>
|
---|
437 | </div>
|
---|
438 | <div #content class="p-fileupload-content" (dragenter)="onDragEnter($event)" (dragleave)="onDragLeave($event)" (drop)="onDrop($event)">
|
---|
439 | <p-progressBar [value]="progress" [showValue]="false" *ngIf="hasFiles()"></p-progressBar>
|
---|
440 |
|
---|
441 | <p-messages [value]="msgs" [enableService]="false"></p-messages>
|
---|
442 |
|
---|
443 | <div class="p-fileupload-files" *ngIf="hasFiles()">
|
---|
444 | <div *ngIf="!fileTemplate">
|
---|
445 | <div class="p-fileupload-row" *ngFor="let file of files; let i = index;">
|
---|
446 | <div><img [src]="file.objectURL" *ngIf="isImage(file)" [width]="previewWidth" /></div>
|
---|
447 | <div class="p-fileupload-filename">{{file.name}}</div>
|
---|
448 | <div>{{formatSize(file.size)}}</div>
|
---|
449 | <div>
|
---|
450 | <button type="button" icon="pi pi-times" pButton (click)="remove($event,i)" [disabled]="uploading"></button>
|
---|
451 | </div>
|
---|
452 | </div>
|
---|
453 | </div>
|
---|
454 | <div *ngIf="fileTemplate">
|
---|
455 | <ng-template ngFor [ngForOf]="files" [ngForTemplate]="fileTemplate"></ng-template>
|
---|
456 | </div>
|
---|
457 | </div>
|
---|
458 | <ng-container *ngTemplateOutlet="contentTemplate; context: {$implicit: files}"></ng-container>
|
---|
459 | </div>
|
---|
460 | </div>
|
---|
461 | <div class="p-fileupload p-fileupload-basic p-component" *ngIf="mode === 'basic'">
|
---|
462 | <p-messages [value]="msgs" [enableService]="false"></p-messages>
|
---|
463 | <span [ngClass]="{'p-button p-component p-fileupload-choose': true, 'p-button-icon-only': !chooseLabel, 'p-fileupload-choose-selected': hasFiles(),'p-focus': focus, 'p-disabled':disabled}"
|
---|
464 | [ngStyle]="style" [class]="styleClass" (mouseup)="onBasicUploaderClick()" (keydown)="onBasicUploaderClick()" tabindex="0" pRipple>
|
---|
465 | <span class="p-button-icon p-button-icon-left pi" [ngClass]="hasFiles()&&!auto ? uploadIcon : chooseIcon"></span>
|
---|
466 | <span class="p-button-label">{{auto ? chooseLabel : hasFiles() ? files[0].name : chooseLabel}}</span>
|
---|
467 | <input #basicfileinput type="file" [accept]="accept" [multiple]="multiple" [disabled]="disabled"
|
---|
468 | (change)="onFileSelect($event)" *ngIf="!hasFiles()" (focus)="onFocus()" (blur)="onBlur()">
|
---|
469 | </span>
|
---|
470 | </div>
|
---|
471 | `, changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, host: {
|
---|
472 | 'class': 'p-element'
|
---|
473 | }, styles: [".p-fileupload-content{position:relative}.p-fileupload-row{display:flex;align-items:center}.p-fileupload-row>div{flex:1 1 auto;width:25%}.p-fileupload-row>div:last-child{text-align:right}.p-fileupload-content .p-progressbar{width:100%;position:absolute;top:0;left:0}.p-button.p-fileupload-choose{position:relative;overflow:hidden}.p-button.p-fileupload-choose input[type=file],.p-fileupload-choose.p-fileupload-choose-selected input[type=file]{display:none}.p-fluid .p-fileupload .p-button{width:auto}.p-fileupload-filename{word-break:break-all}\n"] }]
|
---|
474 | }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i1.DomSanitizer }, { type: i0.NgZone }, { type: i2.HttpClient }, { type: i0.ChangeDetectorRef }, { type: i3.PrimeNGConfig }]; }, propDecorators: { name: [{
|
---|
475 | type: Input
|
---|
476 | }], url: [{
|
---|
477 | type: Input
|
---|
478 | }], method: [{
|
---|
479 | type: Input
|
---|
480 | }], multiple: [{
|
---|
481 | type: Input
|
---|
482 | }], accept: [{
|
---|
483 | type: Input
|
---|
484 | }], disabled: [{
|
---|
485 | type: Input
|
---|
486 | }], auto: [{
|
---|
487 | type: Input
|
---|
488 | }], withCredentials: [{
|
---|
489 | type: Input
|
---|
490 | }], maxFileSize: [{
|
---|
491 | type: Input
|
---|
492 | }], invalidFileSizeMessageSummary: [{
|
---|
493 | type: Input
|
---|
494 | }], invalidFileSizeMessageDetail: [{
|
---|
495 | type: Input
|
---|
496 | }], invalidFileTypeMessageSummary: [{
|
---|
497 | type: Input
|
---|
498 | }], invalidFileTypeMessageDetail: [{
|
---|
499 | type: Input
|
---|
500 | }], invalidFileLimitMessageDetail: [{
|
---|
501 | type: Input
|
---|
502 | }], invalidFileLimitMessageSummary: [{
|
---|
503 | type: Input
|
---|
504 | }], style: [{
|
---|
505 | type: Input
|
---|
506 | }], styleClass: [{
|
---|
507 | type: Input
|
---|
508 | }], previewWidth: [{
|
---|
509 | type: Input
|
---|
510 | }], chooseLabel: [{
|
---|
511 | type: Input
|
---|
512 | }], uploadLabel: [{
|
---|
513 | type: Input
|
---|
514 | }], cancelLabel: [{
|
---|
515 | type: Input
|
---|
516 | }], chooseIcon: [{
|
---|
517 | type: Input
|
---|
518 | }], uploadIcon: [{
|
---|
519 | type: Input
|
---|
520 | }], cancelIcon: [{
|
---|
521 | type: Input
|
---|
522 | }], showUploadButton: [{
|
---|
523 | type: Input
|
---|
524 | }], showCancelButton: [{
|
---|
525 | type: Input
|
---|
526 | }], mode: [{
|
---|
527 | type: Input
|
---|
528 | }], headers: [{
|
---|
529 | type: Input
|
---|
530 | }], customUpload: [{
|
---|
531 | type: Input
|
---|
532 | }], fileLimit: [{
|
---|
533 | type: Input
|
---|
534 | }], onBeforeUpload: [{
|
---|
535 | type: Output
|
---|
536 | }], onSend: [{
|
---|
537 | type: Output
|
---|
538 | }], onUpload: [{
|
---|
539 | type: Output
|
---|
540 | }], onError: [{
|
---|
541 | type: Output
|
---|
542 | }], onClear: [{
|
---|
543 | type: Output
|
---|
544 | }], onRemove: [{
|
---|
545 | type: Output
|
---|
546 | }], onSelect: [{
|
---|
547 | type: Output
|
---|
548 | }], onProgress: [{
|
---|
549 | type: Output
|
---|
550 | }], uploadHandler: [{
|
---|
551 | type: Output
|
---|
552 | }], templates: [{
|
---|
553 | type: ContentChildren,
|
---|
554 | args: [PrimeTemplate]
|
---|
555 | }], advancedFileInput: [{
|
---|
556 | type: ViewChild,
|
---|
557 | args: ['advancedfileinput']
|
---|
558 | }], basicFileInput: [{
|
---|
559 | type: ViewChild,
|
---|
560 | args: ['basicfileinput']
|
---|
561 | }], content: [{
|
---|
562 | type: ViewChild,
|
---|
563 | args: ['content']
|
---|
564 | }], files: [{
|
---|
565 | type: Input
|
---|
566 | }] } });
|
---|
567 | export class FileUploadModule {
|
---|
568 | }
|
---|
569 | FileUploadModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.0.0", ngImport: i0, type: FileUploadModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
|
---|
570 | FileUploadModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "12.0.0", version: "13.0.0", ngImport: i0, type: FileUploadModule, declarations: [FileUpload], imports: [CommonModule, SharedModule, ButtonModule, ProgressBarModule, MessagesModule, RippleModule], exports: [FileUpload, SharedModule, ButtonModule, ProgressBarModule, MessagesModule] });
|
---|
571 | FileUploadModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "13.0.0", ngImport: i0, type: FileUploadModule, imports: [[CommonModule, SharedModule, ButtonModule, ProgressBarModule, MessagesModule, RippleModule], SharedModule, ButtonModule, ProgressBarModule, MessagesModule] });
|
---|
572 | i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.0.0", ngImport: i0, type: FileUploadModule, decorators: [{
|
---|
573 | type: NgModule,
|
---|
574 | args: [{
|
---|
575 | imports: [CommonModule, SharedModule, ButtonModule, ProgressBarModule, MessagesModule, RippleModule],
|
---|
576 | exports: [FileUpload, SharedModule, ButtonModule, ProgressBarModule, MessagesModule],
|
---|
577 | declarations: [FileUpload]
|
---|
578 | }]
|
---|
579 | }] });
|
---|
580 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fileupload.js","sourceRoot":"","sources":["../../../src/app/components/fileupload/fileupload.ts"],"names":[],"mappings":"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","sourcesContent":["import {NgModule,Component,OnDestroy,Input,Output,EventEmitter,TemplateRef,AfterViewInit,AfterContentInit,\n            ContentChildren,QueryList,ViewChild,ElementRef,NgZone,ChangeDetectionStrategy, ViewEncapsulation, ChangeDetectorRef, OnInit} from '@angular/core';\nimport {CommonModule} from '@angular/common';\nimport {DomSanitizer} from '@angular/platform-browser';\nimport {ButtonModule} from 'primeng/button';\nimport {MessagesModule} from 'primeng/messages';\nimport {ProgressBarModule} from 'primeng/progressbar';\nimport {DomHandler} from 'primeng/dom';\nimport {Message, TranslationKeys} from 'primeng/api';\nimport {PrimeTemplate,SharedModule,PrimeNGConfig} from 'primeng/api';\nimport {BlockableUI} from 'primeng/api';\nimport {RippleModule} from 'primeng/ripple';\nimport {HttpClient, HttpEvent, HttpEventType, HttpHeaders} from \"@angular/common/http\";\nimport {Subscription} from 'rxjs';\n\n@Component({\n    selector: 'p-fileUpload',\n    template: `\n        <div [ngClass]=\"'p-fileupload p-fileupload-advanced p-component'\" [ngStyle]=\"style\" [class]=\"styleClass\" *ngIf=\"mode === 'advanced'\">\n            <div class=\"p-fileupload-buttonbar\">\n                <span class=\"p-button p-component p-fileupload-choose\" [ngClass]=\"{'p-focus': focus, 'p-disabled':disabled || isChooseDisabled()}\" (focus)=\"onFocus()\" (blur)=\"onBlur()\" pRipple\n                    (click)=\"choose()\" (keydown.enter)=\"choose()\" tabindex=\"0\">\n                    <input #advancedfileinput type=\"file\" (change)=\"onFileSelect($event)\" [multiple]=\"multiple\" [accept]=\"accept\" [disabled]=\"disabled || isChooseDisabled()\" [attr.title]=\"''\">\n                    <span [ngClass]=\"'p-button-icon p-button-icon-left'\" [class]=\"chooseIcon\"></span>\n                    <span class=\"p-button-label\">{{chooseButtonLabel}}</span>\n                </span>\n\n                <p-button *ngIf=\"!auto&&showUploadButton\" type=\"button\" [label]=\"uploadButtonLabel\" [icon]=\"uploadIcon\" (onClick)=\"upload()\" [disabled]=\"!hasFiles() || isFileLimitExceeded()\"></p-button>\n                <p-button *ngIf=\"!auto&&showCancelButton\" type=\"button\" [label]=\"cancelButtonLabel\" [icon]=\"cancelIcon\" (onClick)=\"clear()\" [disabled]=\"!hasFiles() || uploading\"></p-button>\n\n                <ng-container *ngTemplateOutlet=\"toolbarTemplate\"></ng-container>\n            </div>\n            <div #content class=\"p-fileupload-content\" (dragenter)=\"onDragEnter($event)\" (dragleave)=\"onDragLeave($event)\" (drop)=\"onDrop($event)\">\n                <p-progressBar [value]=\"progress\" [showValue]=\"false\" *ngIf=\"hasFiles()\"></p-progressBar>\n\n                <p-messages [value]=\"msgs\" [enableService]=\"false\"></p-messages>\n\n                <div class=\"p-fileupload-files\" *ngIf=\"hasFiles()\">\n                    <div *ngIf=\"!fileTemplate\">\n                        <div class=\"p-fileupload-row\" *ngFor=\"let file of files; let i = index;\">\n                            <div><img [src]=\"file.objectURL\" *ngIf=\"isImage(file)\" [width]=\"previewWidth\" /></div>\n                            <div class=\"p-fileupload-filename\">{{file.name}}</div>\n                            <div>{{formatSize(file.size)}}</div>\n                            <div>\n                                <button type=\"button\" icon=\"pi pi-times\" pButton (click)=\"remove($event,i)\" [disabled]=\"uploading\"></button>\n                            </div>\n                        </div>\n                    </div>\n                    <div *ngIf=\"fileTemplate\">\n                        <ng-template ngFor [ngForOf]=\"files\" [ngForTemplate]=\"fileTemplate\"></ng-template>\n                    </div>\n                </div>\n                <ng-container *ngTemplateOutlet=\"contentTemplate; context: {$implicit: files}\"></ng-container>\n            </div>\n        </div>\n        <div class=\"p-fileupload p-fileupload-basic p-component\" *ngIf=\"mode === 'basic'\">\n            <p-messages [value]=\"msgs\" [enableService]=\"false\"></p-messages>\n            <span [ngClass]=\"{'p-button p-component p-fileupload-choose': true, 'p-button-icon-only': !chooseLabel, 'p-fileupload-choose-selected': hasFiles(),'p-focus': focus, 'p-disabled':disabled}\"\n                [ngStyle]=\"style\" [class]=\"styleClass\" (mouseup)=\"onBasicUploaderClick()\" (keydown)=\"onBasicUploaderClick()\" tabindex=\"0\" pRipple>\n                <span class=\"p-button-icon p-button-icon-left pi\" [ngClass]=\"hasFiles()&&!auto ? uploadIcon : chooseIcon\"></span>\n                <span class=\"p-button-label\">{{auto ? chooseLabel : hasFiles() ? files[0].name : chooseLabel}}</span>\n                <input #basicfileinput type=\"file\" [accept]=\"accept\" [multiple]=\"multiple\" [disabled]=\"disabled\"\n                    (change)=\"onFileSelect($event)\" *ngIf=\"!hasFiles()\" (focus)=\"onFocus()\" (blur)=\"onBlur()\">\n            </span>\n        </div>\n    `,\n    changeDetection: ChangeDetectionStrategy.OnPush,\n    encapsulation: ViewEncapsulation.None,\n    styleUrls: ['./fileupload.css'],\n    host: {\n        'class': 'p-element'\n    }\n})\nexport class FileUpload implements AfterViewInit,AfterContentInit,OnInit,OnDestroy,BlockableUI {\n\n    @Input() name: string;\n\n    @Input() url: string;\n\n    @Input() method: string = 'post';\n\n    @Input() multiple: boolean;\n\n    @Input() accept: string;\n\n    @Input() disabled: boolean;\n\n    @Input() auto: boolean;\n\n    @Input() withCredentials: boolean;\n\n    @Input() maxFileSize: number;\n\n    @Input() invalidFileSizeMessageSummary: string = '{0}: Invalid file size, ';\n\n    @Input() invalidFileSizeMessageDetail: string = 'maximum upload size is {0}.';\n\n    @Input() invalidFileTypeMessageSummary: string = '{0}: Invalid file type, ';\n\n    @Input() invalidFileTypeMessageDetail: string = 'allowed file types: {0}.';\n\n    @Input() invalidFileLimitMessageDetail: string = 'limit is {0} at most.';\n\n    @Input() invalidFileLimitMessageSummary: string = 'Maximum number of files exceeded, ';\n\n    @Input() style: any;\n\n    @Input() styleClass: string;\n\n    @Input() previewWidth: number = 50;\n\n    @Input() chooseLabel: string;\n\n    @Input() uploadLabel: string;\n\n    @Input() cancelLabel: string;\n\n    @Input() chooseIcon: string = 'pi pi-plus';\n\n    @Input() uploadIcon: string = 'pi pi-upload';\n\n    @Input() cancelIcon: string = 'pi pi-times';\n\n    @Input() showUploadButton: boolean = true;\n\n    @Input() showCancelButton: boolean = true;\n\n    @Input() mode: string = 'advanced';\n\n    @Input() headers: HttpHeaders;\n\n    @Input() customUpload: boolean;\n\n    @Input() fileLimit: number;\n\n    @Output() onBeforeUpload: EventEmitter<any> = new EventEmitter();\n\n    @Output() onSend: EventEmitter<any> = new EventEmitter();\n\n    @Output() onUpload: EventEmitter<any> = new EventEmitter();\n\n    @Output() onError: EventEmitter<any> = new EventEmitter();\n\n    @Output() onClear: EventEmitter<any> = new EventEmitter();\n\n    @Output() onRemove: EventEmitter<any> = new EventEmitter();\n\n    @Output() onSelect: EventEmitter<any> = new EventEmitter();\n\n    @Output() onProgress: EventEmitter<any> = new EventEmitter();\n\n    @Output() uploadHandler: EventEmitter<any> = new EventEmitter();\n\n    @ContentChildren(PrimeTemplate) templates: QueryList<any>;\n\n    @ViewChild('advancedfileinput') advancedFileInput: ElementRef;\n\n    @ViewChild('basicfileinput') basicFileInput: ElementRef;\n\n    @ViewChild('content') content: ElementRef;\n\n    @Input() set files(files) {\n        this._files = [];\n\n        for(let i = 0; i < files.length; i++) {\n            let file = files[i];\n\n            if (this.validate(file)) {\n                if (this.isImage(file)) {\n                    (<any>file).objectURL = this.sanitizer.bypassSecurityTrustUrl((window.URL.createObjectURL(files[i])));\n                }\n\n                this._files.push(files[i]);\n            }\n        }\n    }\n\n    get files(): File[] {\n        return this._files;\n    }\n\n    public _files: File[] = [];\n\n    public progress: number = 0;\n\n    public dragHighlight: boolean;\n\n    public msgs: Message[];\n\n    public fileTemplate: TemplateRef<any>;\n\n    public contentTemplate: TemplateRef<any>;\n\n    public toolbarTemplate: TemplateRef<any>;\n\n    public uploadedFileCount: number = 0;\n\n    focus: boolean;\n\n    uploading: boolean;\n\n    duplicateIEEvent: boolean;  // flag to recognize duplicate onchange event for file input\n\n    translationSubscription: Subscription;\n\n    constructor(private el: ElementRef, public sanitizer: DomSanitizer, public zone: NgZone, private http: HttpClient, public cd: ChangeDetectorRef, public config: PrimeNGConfig){}\n\n    ngAfterContentInit() {\n        this.templates.forEach((item) => {\n            switch(item.getType()) {\n                case 'file':\n                    this.fileTemplate = item.template;\n                break;\n\n                case 'content':\n                    this.contentTemplate = item.template;\n                break;\n\n                case 'toolbar':\n                    this.toolbarTemplate = item.template;\n                break;\n\n                default:\n                    this.fileTemplate = item.template;\n                break;\n            }\n        });\n    }\n\n\n    ngOnInit() {\n        this.translationSubscription = this.config.translationObserver.subscribe(() => {\n            this.cd.markForCheck();\n        });\n    }\n\n    ngAfterViewInit() {\n        if (this.mode === 'advanced') {\n            this.zone.runOutsideAngular(() => {\n                if (this.content)\n                    this.content.nativeElement.addEventListener('dragover', this.onDragOver.bind(this));\n            });\n        }\n    }\n\n    choose() {\n        this.advancedFileInput.nativeElement.click();\n    }\n\n    onFileSelect(event) {\n        if (event.type !== 'drop' && this.isIE11() && this.duplicateIEEvent) {\n            this.duplicateIEEvent = false;\n            return;\n        }\n\n        this.msgs = [];\n        if (!this.multiple) {\n            this.files = [];\n        }\n\n        let files = event.dataTransfer ? event.dataTransfer.files : event.target.files;\n        for(let i = 0; i < files.length; i++) {\n            let file = files[i];\n\n            if (!this.isFileSelected(file)){\n              if (this.validate(file)) {\n                  if (this.isImage(file)) {\n                      file.objectURL = this.sanitizer.bypassSecurityTrustUrl((window.URL.createObjectURL(files[i])));\n                  }\n\n                  this.files.push(files[i]);\n              }\n            }\n        }\n\n        this.onSelect.emit({originalEvent: event, files: files, currentFiles: this.files});\n\n        if (this.fileLimit && this.mode == \"advanced\") {\n            this.checkFileLimit();\n        }\n\n        if (this.hasFiles() && this.auto && (!(this.mode === \"advanced\") || !this.isFileLimitExceeded())) {\n            this.upload();\n        }\n\n        if (event.type !== 'drop' && this.isIE11()) {\n          this.clearIEInput();\n        } else {\n          this.clearInputElement();\n        }\n    }\n\n    isFileSelected(file: File): boolean{\n        for(let sFile of this.files){\n            if ((sFile.name + sFile.type + sFile.size) === (file.name + file.type+file.size)) {\n                return true;\n            }\n        }\n\n        return false;\n    }\n\n    isIE11() {\n        return !!window['MSInputMethodContext'] && !!document['documentMode'];\n    }\n\n    validate(file: File): boolean {\n        if (this.accept && !this.isFileTypeValid(file)) {\n            this.msgs.push({\n                severity: 'error',\n                summary: this.invalidFileTypeMessageSummary.replace('{0}', file.name),\n                detail: this.invalidFileTypeMessageDetail.replace('{0}', this.accept)\n            });\n            return false;\n        }\n\n        if (this.maxFileSize  && file.size > this.maxFileSize) {\n            this.msgs.push({\n                severity: 'error',\n                summary: this.invalidFileSizeMessageSummary.replace('{0}', file.name),\n                detail: this.invalidFileSizeMessageDetail.replace('{0}', this.formatSize(this.maxFileSize))\n            });\n            return false;\n        }\n\n        return true;\n    }\n\n    private isFileTypeValid(file: File): boolean {\n        let acceptableTypes = this.accept.split(',').map(type => type.trim());\n        for(let type of acceptableTypes) {\n            let acceptable = this.isWildcard(type) ? this.getTypeClass(file.type) === this.getTypeClass(type)\n                                                    : file.type == type || this.getFileExtension(file).toLowerCase() === type.toLowerCase();\n\n            if (acceptable) {\n                return true;\n            }\n        }\n\n        return false;\n    }\n\n    getTypeClass(fileType: string): string {\n        return fileType.substring(0, fileType.indexOf('/'));\n    }\n\n    isWildcard(fileType: string): boolean {\n        return fileType.indexOf('*') !== -1;\n    }\n\n    getFileExtension(file: File): string {\n        return '.' + file.name.split('.').pop();\n    }\n\n    isImage(file: File): boolean {\n        return /^image\\//.test(file.type);\n    }\n\n    onImageLoad(img: any) {\n        window.URL.revokeObjectURL(img.src);\n    }\n\n    upload() {\n        if (this.customUpload) {\n            if (this.fileLimit) {\n                this.uploadedFileCount += this.files.length;\n            }\n\n            this.uploadHandler.emit({\n                files: this.files\n            });\n\n            this.cd.markForCheck();\n        }\n        else {\n            this.uploading = true;\n            this.msgs = [];\n            let formData = new FormData();\n\n            this.onBeforeUpload.emit({\n                'formData': formData\n            });\n\n            for (let i = 0; i < this.files.length; i++) {\n                formData.append(this.name, this.files[i], this.files[i].name);\n            }\n\n            this.http[this.method](this.url, formData, {\n                headers: this.headers, reportProgress: true, observe: 'events', withCredentials: this.withCredentials\n            }).subscribe( (event: HttpEvent<any>) => {\n                    switch (event.type) {\n                        case HttpEventType.Sent:\n                            this.onSend.emit({\n                                originalEvent: event,\n                                'formData': formData\n                            });\n                            break;\n                        case HttpEventType.Response:\n                            this.uploading = false;\n                            this.progress = 0;\n\n                            if (event['status'] >= 200 && event['status'] < 300) {\n                                if (this.fileLimit) {\n                                    this.uploadedFileCount += this.files.length;\n                                }\n\n                                this.onUpload.emit({originalEvent: event, files: this.files});\n                            } else {\n                                this.onError.emit({files: this.files});\n                            }\n\n                            this.clear();\n                            break;\n                        case HttpEventType.UploadProgress: {\n                            if (event['loaded']) {\n                                this.progress = Math.round((event['loaded'] * 100) / event['total']);\n                            }\n\n                            this.onProgress.emit({originalEvent: event, progress: this.progress});\n                            break;\n                        }\n                    }\n\n                    this.cd.markForCheck();\n                },\n                error => {\n                    this.uploading = false;\n                    this.onError.emit({files: this.files, error: error});\n                });\n        }\n    }\n\n    clear() {\n        this.files = [];\n        this.onClear.emit();\n        this.clearInputElement();\n        this.cd.markForCheck();\n    }\n\n    remove(event: Event, index: number) {\n        this.clearInputElement();\n        this.onRemove.emit({originalEvent: event, file: this.files[index]});\n        this.files.splice(index, 1);\n    }\n\n    isFileLimitExceeded() {\n        if (this.fileLimit && this.fileLimit <= this.files.length + this.uploadedFileCount && this.focus) {\n            this.focus = false;\n        }\n\n        return this.fileLimit && this.fileLimit < this.files.length + this.uploadedFileCount;\n    }\n\n    isChooseDisabled() {\n        return this.fileLimit && this.fileLimit <= this.files.length + this.uploadedFileCount;\n    }\n\n    checkFileLimit() {\n        if (this.isFileLimitExceeded()) {\n            this.msgs.push({\n                severity: 'error',\n                summary: this.invalidFileLimitMessageSummary.replace('{0}', this.fileLimit.toString()),\n                detail: this.invalidFileLimitMessageDetail.replace('{0}', this.fileLimit.toString())\n            });\n        }\n    }\n\n    clearInputElement() {\n        if (this.advancedFileInput && this.advancedFileInput.nativeElement) {\n            this.advancedFileInput.nativeElement.value = '';\n        }\n\n        if (this.basicFileInput && this.basicFileInput.nativeElement) {\n            this.basicFileInput.nativeElement.value = '';\n        }\n    }\n\n    clearIEInput() {\n        if (this.advancedFileInput && this.advancedFileInput.nativeElement) {\n            this.duplicateIEEvent = true; //IE11 fix to prevent onFileChange trigger again\n            this.advancedFileInput.nativeElement.value = '';\n        }\n    }\n\n    hasFiles(): boolean {\n        return this.files && this.files.length > 0;\n    }\n\n    onDragEnter(e) {\n        if (!this.disabled) {\n            e.stopPropagation();\n            e.preventDefault();\n        }\n    }\n\n    onDragOver(e) {\n        if (!this.disabled) {\n            DomHandler.addClass(this.content.nativeElement, 'p-fileupload-highlight');\n            this.dragHighlight = true;\n            e.stopPropagation();\n            e.preventDefault();\n        }\n    }\n\n    onDragLeave(event) {\n        if (!this.disabled) {\n            DomHandler.removeClass(this.content.nativeElement, 'p-fileupload-highlight');\n        }\n    }\n\n    onDrop(event) {\n        if (!this.disabled) {\n            DomHandler.removeClass(this.content.nativeElement, 'p-fileupload-highlight');\n            event.stopPropagation();\n            event.preventDefault();\n\n            let files = event.dataTransfer ? event.dataTransfer.files : event.target.files;\n            let allowDrop = this.multiple||(files && files.length === 1);\n\n            if (allowDrop) {\n                this.onFileSelect(event);\n            }\n        }\n    }\n\n    onFocus() {\n        this.focus = true;\n    }\n\n    onBlur() {\n        this.focus = false;\n    }\n\n    formatSize(bytes) {\n        if (bytes == 0) {\n            return '0 B';\n        }\n        let k = 1000,\n        dm = 3,\n        sizes = ['B', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'],\n        i = Math.floor(Math.log(bytes) / Math.log(k));\n\n        return parseFloat((bytes / Math.pow(k, i)).toFixed(dm)) + ' ' + sizes[i];\n    }\n\n    onBasicUploaderClick() {\n        if (this.hasFiles())\n            this.upload();\n        else\n            this.basicFileInput.nativeElement.click();\n    }\n\n    getBlockableElement(): HTMLElement {\n      return this.el.nativeElement.children[0];\n    }\n\n    get chooseButtonLabel(): string {\n        return this.chooseLabel || this.config.getTranslation(TranslationKeys.CHOOSE);\n    }\n\n    get uploadButtonLabel(): string {\n        return this.uploadLabel || this.config.getTranslation(TranslationKeys.UPLOAD);\n    }\n\n    get cancelButtonLabel(): string {\n        return this.cancelLabel || this.config.getTranslation(TranslationKeys.CANCEL);\n    }\n\n    ngOnDestroy() {\n        if (this.content && this.content.nativeElement) {\n            this.content.nativeElement.removeEventListener('dragover', this.onDragOver);\n        }\n\n        if (this.translationSubscription) {\n            this.translationSubscription.unsubscribe();\n        }\n    }\n}\n\n@NgModule({\n    imports: [CommonModule,SharedModule,ButtonModule,ProgressBarModule,MessagesModule,RippleModule],\n    exports: [FileUpload,SharedModule,ButtonModule,ProgressBarModule,MessagesModule],\n    declarations: [FileUpload]\n})\nexport class FileUploadModule { }\n"]} |
---|