ng-zorro-antd
Version:
An enterprise-class UI components based on Ant Design and Angular
1 lines • 71.5 kB
Source Map (JSON)
{"version":3,"file":"ng-zorro-antd-upload.mjs","sources":["../../components/upload/interface.ts","../../components/upload/upload-btn.component.ts","../../components/upload/upload-btn.component.html","../../components/upload/upload-list.component.ts","../../components/upload/upload-list.component.html","../../components/upload/upload.component.ts","../../components/upload/upload.component.html","../../components/upload/upload.module.ts","../../components/upload/public-api.ts","../../components/upload/ng-zorro-antd-upload.ts"],"sourcesContent":["/**\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://github.com/NG-ZORRO/ng-zorro-antd/blob/master/LICENSE\n */\n\nimport { TemplateRef } from '@angular/core';\nimport { Observable, Subscription } from 'rxjs';\n\nimport { IndexableObject, NzSafeAny } from 'ng-zorro-antd/core/types';\n\n/** Status */\nexport type UploadFileStatus = 'error' | 'success' | 'done' | 'uploading' | 'removed';\n\nexport type NzUploadType = 'select' | 'drag';\n\n/** Built-in styles of the uploading list. */\nexport type NzUploadListType = 'text' | 'picture' | 'picture-card';\n\nexport interface NzUploadFile {\n uid: string;\n size?: number;\n name: string;\n filename?: string;\n lastModified?: string;\n lastModifiedDate?: Date;\n url?: string;\n status?: UploadFileStatus;\n originFileObj?: File;\n percent?: number;\n thumbUrl?: string;\n response?: NzSafeAny;\n error?: NzSafeAny;\n linkProps?: { download: string };\n type?: string;\n\n [key: string]: NzSafeAny;\n}\n\nexport interface NzUploadChangeParam {\n file: NzUploadFile;\n fileList: NzUploadFile[];\n event?: { percent: number };\n /** Callback type. */\n type?: string;\n}\n\nexport interface NzShowUploadList {\n showRemoveIcon?: boolean;\n showPreviewIcon?: boolean;\n showDownloadIcon?: boolean;\n}\n\nexport type NzUploadTransformFileType = string | Blob | NzUploadFile | Observable<string | Blob | File>;\n\nexport interface ZipButtonOptions {\n disabled?: boolean;\n accept?: string | string[];\n action?: string | ((file: NzUploadFile) => string | Observable<string>);\n directory?: boolean;\n openFileDialogOnClick?: boolean;\n beforeUpload?(file: NzUploadFile, fileList: NzUploadFile[]): boolean | Observable<NzSafeAny>;\n customRequest?(item: NzSafeAny): Subscription;\n data?: {} | ((file: NzUploadFile) => {} | Observable<{}>);\n headers?: {} | ((file: NzUploadFile) => {} | Observable<{}>);\n name?: string;\n multiple?: boolean;\n withCredentials?: boolean;\n filters?: UploadFilter[];\n transformFile?(file: NzUploadFile): NzUploadTransformFileType;\n onStart?(file: NzUploadFile): void;\n onProgress?(e: NzSafeAny, file: NzUploadFile): void;\n onSuccess?(ret: NzSafeAny, file: NzUploadFile, xhr: NzSafeAny): void;\n onError?(err: NzSafeAny, file: NzUploadFile): void;\n}\n\nexport interface UploadFilter {\n name: string;\n fn(fileList: NzUploadFile[]): NzUploadFile[] | Observable<NzUploadFile[]>;\n}\n\nexport interface NzUploadXHRArgs {\n action?: string;\n name?: string;\n headers?: IndexableObject;\n file: NzUploadFile;\n postFile: string | Blob | File | NzUploadFile;\n data?: IndexableObject;\n withCredentials?: boolean;\n onProgress?(e: NzSafeAny, file: NzUploadFile): void;\n onSuccess?(ret: NzSafeAny, file: NzUploadFile, xhr: NzSafeAny): void;\n onError?(err: NzSafeAny, file: NzUploadFile): void;\n}\n\nexport type NzIconRenderTemplate = TemplateRef<{ $implicit: NzUploadFile }>;\n","/**\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://github.com/NG-ZORRO/ng-zorro-antd/blob/master/LICENSE\n */\n\nimport { ENTER } from '@angular/cdk/keycodes';\nimport { HttpClient, HttpEvent, HttpEventType, HttpHeaders, HttpRequest, HttpResponse } from '@angular/common/http';\nimport {\n Component,\n ElementRef,\n Input,\n NgZone,\n OnInit,\n OnDestroy,\n Optional,\n ViewChild,\n ViewEncapsulation\n} from '@angular/core';\nimport { fromEvent, Observable, of, Subject, Subscription } from 'rxjs';\nimport { map, switchMap, takeUntil, tap } from 'rxjs/operators';\n\nimport { warn } from 'ng-zorro-antd/core/logger';\nimport { NzSafeAny } from 'ng-zorro-antd/core/types';\n\nimport { NzUploadFile, NzUploadXHRArgs, ZipButtonOptions } from './interface';\n\n@Component({\n selector: '[nz-upload-btn]',\n exportAs: 'nzUploadBtn',\n templateUrl: './upload-btn.component.html',\n host: {\n class: 'ant-upload',\n '[attr.tabindex]': '\"0\"',\n '[attr.role]': '\"button\"',\n '[class.ant-upload-disabled]': 'options.disabled',\n '(drop)': 'onFileDrop($event)',\n '(dragover)': 'onFileDrop($event)'\n },\n preserveWhitespaces: false,\n encapsulation: ViewEncapsulation.None\n})\nexport class NzUploadBtnComponent implements OnInit, OnDestroy {\n reqs: { [key: string]: Subscription } = {};\n private destroy = false;\n private destroy$ = new Subject<void>();\n @ViewChild('file', { static: true }) file!: ElementRef<HTMLInputElement>;\n @Input() options!: ZipButtonOptions;\n\n onClick(): void {\n if (this.options.disabled || !this.options.openFileDialogOnClick) {\n return;\n }\n this.file.nativeElement.click();\n }\n\n // skip safari bug\n onFileDrop(e: DragEvent): void {\n if (this.options.disabled || e.type === 'dragover') {\n e.preventDefault();\n return;\n }\n if (this.options.directory) {\n this.traverseFileTree(e.dataTransfer!.items);\n } else {\n const files: File[] = Array.prototype.slice\n .call(e.dataTransfer!.files)\n .filter((file: File) => this.attrAccept(file, this.options.accept));\n if (files.length) {\n this.uploadFiles(files);\n }\n }\n\n e.preventDefault();\n }\n\n onChange(e: Event): void {\n if (this.options.disabled) {\n return;\n }\n const hie = e.target as HTMLInputElement;\n this.uploadFiles(hie.files!);\n hie.value = '';\n }\n\n private traverseFileTree(files: DataTransferItemList): void {\n const _traverseFileTree = (item: NzSafeAny, path: string): void => {\n if (item.isFile) {\n item.file((file: File) => {\n if (this.attrAccept(file, this.options.accept)) {\n this.uploadFiles([file]);\n }\n });\n } else if (item.isDirectory) {\n const dirReader = item.createReader();\n\n dirReader.readEntries((entries: NzSafeAny) => {\n for (const entrieItem of entries) {\n _traverseFileTree(entrieItem, `${path}${item.name}/`);\n }\n });\n }\n };\n\n for (const file of files as NzSafeAny) {\n _traverseFileTree(file.webkitGetAsEntry(), '');\n }\n }\n\n private attrAccept(file: File, acceptedFiles?: string | string[]): boolean {\n if (file && acceptedFiles) {\n const acceptedFilesArray = Array.isArray(acceptedFiles) ? acceptedFiles : acceptedFiles.split(',');\n const fileName = `${file.name}`;\n const mimeType = `${file.type}`;\n const baseMimeType = mimeType.replace(/\\/.*$/, '');\n\n return acceptedFilesArray.some(type => {\n const validType = type.trim();\n if (validType.charAt(0) === '.') {\n return (\n fileName\n .toLowerCase()\n .indexOf(validType.toLowerCase(), fileName.toLowerCase().length - validType.toLowerCase().length) !== -1\n );\n } else if (/\\/\\*$/.test(validType)) {\n // This is something like a image/* mime type\n return baseMimeType === validType.replace(/\\/.*$/, '');\n }\n return mimeType === validType;\n });\n }\n return true;\n }\n\n private attachUid(file: NzUploadFile): NzUploadFile {\n if (!file.uid) {\n file.uid = Math.random().toString(36).substring(2);\n }\n return file;\n }\n\n uploadFiles(fileList: FileList | File[]): void {\n let filters$: Observable<NzUploadFile[]> = of(Array.prototype.slice.call(fileList));\n if (this.options.filters) {\n this.options.filters.forEach(f => {\n filters$ = filters$.pipe(\n switchMap(list => {\n const fnRes = f.fn(list);\n return fnRes instanceof Observable ? fnRes : of(fnRes);\n })\n );\n });\n }\n filters$.subscribe(\n list => {\n list.forEach((file: NzUploadFile) => {\n this.attachUid(file);\n this.upload(file, list);\n });\n },\n e => {\n warn(`Unhandled upload filter error`, e);\n }\n );\n }\n\n private upload(file: NzUploadFile, fileList: NzUploadFile[]): void {\n if (!this.options.beforeUpload) {\n return this.post(file);\n }\n const before = this.options.beforeUpload(file, fileList);\n if (before instanceof Observable) {\n before.subscribe(\n (processedFile: NzUploadFile) => {\n const processedFileType = Object.prototype.toString.call(processedFile);\n if (processedFileType === '[object File]' || processedFileType === '[object Blob]') {\n this.attachUid(processedFile);\n this.post(processedFile);\n } else if (typeof processedFile === 'boolean' && processedFile !== false) {\n this.post(file);\n }\n },\n e => {\n warn(`Unhandled upload beforeUpload error`, e);\n }\n );\n } else if (before !== false) {\n return this.post(file);\n }\n }\n\n private post(file: NzUploadFile): void {\n if (this.destroy) {\n return;\n }\n let process$: Observable<string | Blob | File | NzUploadFile> = of(file);\n let transformedFile: string | Blob | File | NzUploadFile | undefined;\n const opt = this.options;\n const { uid } = file;\n const { action, data, headers, transformFile } = opt;\n\n const args: NzUploadXHRArgs = {\n action: typeof action === 'string' ? action : '',\n name: opt.name,\n headers,\n file,\n postFile: file,\n data,\n withCredentials: opt.withCredentials,\n onProgress: opt.onProgress\n ? e => {\n opt.onProgress!(e, file);\n }\n : undefined,\n onSuccess: (ret, xhr) => {\n this.clean(uid);\n opt.onSuccess!(ret, file, xhr);\n },\n onError: xhr => {\n this.clean(uid);\n opt.onError!(xhr, file);\n }\n };\n\n if (typeof action === 'function') {\n const actionResult = (action as (file: NzUploadFile) => string | Observable<string>)(file);\n if (actionResult instanceof Observable) {\n process$ = process$.pipe(\n switchMap(() => actionResult),\n map(res => {\n args.action = res;\n return file;\n })\n );\n } else {\n args.action = actionResult;\n }\n }\n\n if (typeof transformFile === 'function') {\n const transformResult = transformFile(file);\n process$ = process$.pipe(\n switchMap(() => (transformResult instanceof Observable ? transformResult : of(transformResult))),\n tap(newFile => (transformedFile = newFile))\n );\n }\n\n if (typeof data === 'function') {\n const dataResult = (data as (file: NzUploadFile) => {} | Observable<{}>)(file);\n if (dataResult instanceof Observable) {\n process$ = process$.pipe(\n switchMap(() => dataResult),\n map(res => {\n args.data = res;\n return transformedFile ?? file;\n })\n );\n } else {\n args.data = dataResult;\n }\n }\n\n if (typeof headers === 'function') {\n const headersResult = (headers as (file: NzUploadFile) => {} | Observable<{}>)(file);\n if (headersResult instanceof Observable) {\n process$ = process$.pipe(\n switchMap(() => headersResult),\n map(res => {\n args.headers = res;\n return transformedFile ?? file;\n })\n );\n } else {\n args.headers = headersResult;\n }\n }\n\n process$.subscribe(newFile => {\n args.postFile = newFile;\n const req$ = (opt.customRequest || this.xhr).call(this, args);\n if (!(req$ instanceof Subscription)) {\n warn(`Must return Subscription type in '[nzCustomRequest]' property`);\n }\n this.reqs[uid] = req$;\n opt.onStart!(file);\n });\n }\n\n private xhr(args: NzUploadXHRArgs): Subscription {\n const formData = new FormData();\n\n if (args.data) {\n Object.keys(args.data).map(key => {\n formData.append(key, args.data![key]);\n });\n }\n\n formData.append(args.name!, args.postFile as NzSafeAny);\n\n if (!args.headers) {\n args.headers = {};\n }\n if (args.headers['X-Requested-With'] !== null) {\n args.headers['X-Requested-With'] = `XMLHttpRequest`;\n } else {\n delete args.headers['X-Requested-With'];\n }\n const req = new HttpRequest('POST', args.action!, formData, {\n reportProgress: true,\n withCredentials: args.withCredentials,\n headers: new HttpHeaders(args.headers)\n });\n return this.http.request(req).subscribe(\n (event: HttpEvent<NzSafeAny>) => {\n if (event.type === HttpEventType.UploadProgress) {\n if (event.total! > 0) {\n (event as NzSafeAny).percent = (event.loaded / event.total!) * 100;\n }\n args.onProgress!(event, args.file);\n } else if (event instanceof HttpResponse) {\n args.onSuccess!(event.body, args.file, event);\n }\n },\n err => {\n this.abort(args.file);\n args.onError!(err, args.file);\n }\n );\n }\n\n private clean(uid: string): void {\n const req$ = this.reqs[uid];\n if (req$ instanceof Subscription) {\n req$.unsubscribe();\n }\n delete this.reqs[uid];\n }\n\n abort(file?: NzUploadFile): void {\n if (file) {\n this.clean(file && file.uid);\n } else {\n Object.keys(this.reqs).forEach(uid => this.clean(uid));\n }\n }\n\n constructor(private ngZone: NgZone, @Optional() private http: HttpClient, private elementRef: ElementRef) {\n if (!http) {\n throw new Error(`Not found 'HttpClient', You can import 'HttpClientModule' in your root module.`);\n }\n }\n\n ngOnInit(): void {\n // Caretaker note: `input[type=file].click()` will open a native OS file picker,\n // it doesn't require Angular to run `ApplicationRef.tick()`.\n this.ngZone.runOutsideAngular(() => {\n fromEvent(this.elementRef.nativeElement, 'click')\n .pipe(takeUntil(this.destroy$))\n .subscribe(() => this.onClick());\n\n fromEvent<KeyboardEvent>(this.elementRef.nativeElement, 'keydown')\n .pipe(takeUntil(this.destroy$))\n .subscribe(event => {\n if (this.options.disabled) {\n return;\n }\n if (event.key === 'Enter' || event.keyCode === ENTER) {\n this.onClick();\n }\n });\n });\n }\n\n ngOnDestroy(): void {\n this.destroy = true;\n this.destroy$.next();\n this.abort();\n }\n}\n","<input\n type=\"file\"\n #file\n (change)=\"onChange($event)\"\n [attr.accept]=\"options.accept\"\n [attr.directory]=\"options.directory ? 'directory' : null\"\n [attr.webkitdirectory]=\"options.directory ? 'webkitdirectory' : null\"\n [multiple]=\"options.multiple\"\n style=\"display: none\"\n/>\n<ng-content></ng-content>\n","/**\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://github.com/NG-ZORRO/ng-zorro-antd/blob/master/LICENSE\n */\n\nimport { animate, style, transition, trigger } from '@angular/animations';\nimport { Direction } from '@angular/cdk/bidi';\nimport { Platform } from '@angular/cdk/platform';\nimport { DOCUMENT } from '@angular/common';\nimport {\n ChangeDetectionStrategy,\n ChangeDetectorRef,\n Component,\n Inject,\n Input,\n NgZone,\n OnChanges,\n OnDestroy,\n ViewEncapsulation\n} from '@angular/core';\nimport { fromEvent, Observable, of, Subject } from 'rxjs';\nimport { takeUntil, map } from 'rxjs/operators';\n\nimport { NzSafeAny } from 'ng-zorro-antd/core/types';\n\nimport { NzIconRenderTemplate, NzShowUploadList, NzUploadFile, NzUploadListType } from './interface';\n\nconst isImageFileType = (type: string): boolean => !!type && type.indexOf('image/') === 0;\n\nconst MEASURE_SIZE = 200;\n\ntype UploadListIconType = '' | 'uploading' | 'thumbnail';\n\ninterface UploadListFile extends NzUploadFile {\n isImageUrl?: boolean;\n isUploading?: boolean;\n iconType?: UploadListIconType;\n showDownload?: boolean;\n}\n\n@Component({\n selector: 'nz-upload-list',\n exportAs: 'nzUploadList',\n templateUrl: './upload-list.component.html',\n animations: [\n trigger('itemState', [\n transition(':enter', [\n style({ height: '0', width: '0', opacity: 0 }),\n animate(150, style({ height: '*', width: '*', opacity: 1 }))\n ]),\n transition(':leave', [animate(150, style({ height: '0', width: '0', opacity: 0 }))])\n ])\n ],\n host: {\n class: 'ant-upload-list',\n '[class.ant-upload-list-rtl]': `dir === 'rtl'`,\n '[class.ant-upload-list-text]': `listType === 'text'`,\n '[class.ant-upload-list-picture]': `listType === 'picture'`,\n '[class.ant-upload-list-picture-card]': `listType === 'picture-card'`\n },\n preserveWhitespaces: false,\n encapsulation: ViewEncapsulation.None,\n changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class NzUploadListComponent implements OnChanges, OnDestroy {\n list: UploadListFile[] = [];\n\n private get showPic(): boolean {\n return this.listType === 'picture' || this.listType === 'picture-card';\n }\n\n @Input() locale: NzSafeAny = {};\n @Input() listType!: NzUploadListType;\n @Input()\n set items(list: NzUploadFile[]) {\n this.list = list;\n }\n @Input() icons!: NzShowUploadList;\n @Input() onPreview?: (file: NzUploadFile) => void;\n @Input() onRemove!: (file: NzUploadFile) => void;\n @Input() onDownload?: (file: NzUploadFile) => void;\n @Input() previewFile?: (file: NzUploadFile) => Observable<string>;\n @Input() previewIsImage?: (file: NzUploadFile) => boolean;\n @Input() iconRender: NzIconRenderTemplate | null = null;\n @Input() dir: Direction = 'ltr';\n\n private destroy$ = new Subject<void>();\n\n private genErr(file: NzUploadFile): string {\n if (file.response && typeof file.response === 'string') {\n return file.response;\n }\n return (file.error && file.error.statusText) || this.locale.uploadError;\n }\n\n private extname(url: string): string {\n const temp = url.split('/');\n const filename = temp[temp.length - 1];\n const filenameWithoutSuffix = filename.split(/#|\\?/)[0];\n return (/\\.[^./\\\\]*$/.exec(filenameWithoutSuffix) || [''])[0];\n }\n\n isImageUrl(file: NzUploadFile): boolean {\n if (isImageFileType(file.type!)) {\n return true;\n }\n const url: string = (file.thumbUrl || file.url || '') as string;\n if (!url) {\n return false;\n }\n const extension = this.extname(url);\n if (/^data:image\\//.test(url) || /(webp|svg|png|gif|jpg|jpeg|jfif|bmp|dpg)$/i.test(extension)) {\n return true;\n } else if (/^data:/.test(url)) {\n // other file types of base64\n return false;\n } else if (extension) {\n // other file types which have extension\n return false;\n }\n return true;\n }\n\n private getIconType(file: UploadListFile): UploadListIconType {\n if (!this.showPic) {\n return '';\n }\n if (file.isUploading || (!file.thumbUrl && !file.url)) {\n return 'uploading';\n } else {\n return 'thumbnail';\n }\n }\n\n private previewImage(file: File | Blob): Observable<string> {\n if (!isImageFileType(file.type) || !this.platform.isBrowser) {\n return of('');\n }\n\n const canvas = this.doc.createElement('canvas');\n canvas.width = MEASURE_SIZE;\n canvas.height = MEASURE_SIZE;\n canvas.style.cssText = `position: fixed; left: 0; top: 0; width: ${MEASURE_SIZE}px; height: ${MEASURE_SIZE}px; z-index: 9999; display: none;`;\n this.doc.body.appendChild(canvas);\n const ctx = canvas.getContext('2d');\n const img = new Image();\n const objectUrl = URL.createObjectURL(file);\n img.src = objectUrl;\n return fromEvent(img, 'load').pipe(\n map(() => {\n const { width, height } = img;\n\n let drawWidth = MEASURE_SIZE;\n let drawHeight = MEASURE_SIZE;\n let offsetX = 0;\n let offsetY = 0;\n\n if (width < height) {\n drawHeight = height * (MEASURE_SIZE / width);\n offsetY = -(drawHeight - drawWidth) / 2;\n } else {\n drawWidth = width * (MEASURE_SIZE / height);\n offsetX = -(drawWidth - drawHeight) / 2;\n }\n\n try {\n ctx!.drawImage(img, offsetX, offsetY, drawWidth, drawHeight);\n } catch {}\n const dataURL = canvas.toDataURL();\n this.doc.body.removeChild(canvas);\n\n URL.revokeObjectURL(objectUrl);\n return dataURL;\n })\n );\n }\n\n private genThumb(): void {\n if (!this.platform.isBrowser) {\n return;\n }\n\n const win = window as NzSafeAny;\n if (\n !this.showPic ||\n typeof document === 'undefined' ||\n typeof win === 'undefined' ||\n !win.FileReader ||\n !win.File\n ) {\n return;\n }\n this.list\n .filter(file => file.originFileObj instanceof File && file.thumbUrl === undefined)\n .forEach(file => {\n file.thumbUrl = '';\n // Caretaker note: we shouldn't use promises here since they're not cancellable.\n // A promise microtask can be resolved after the view is destroyed. Thus running `detectChanges()`\n // will cause a runtime exception (`detectChanges()` cannot be run on destroyed views).\n const dataUrl$ = (this.previewFile ? this.previewFile(file) : this.previewImage(file.originFileObj!)).pipe(\n takeUntil(this.destroy$)\n );\n this.ngZone.runOutsideAngular(() => {\n dataUrl$.subscribe(dataUrl => {\n this.ngZone.run(() => {\n file.thumbUrl = dataUrl;\n this.detectChanges();\n });\n });\n });\n });\n }\n\n private showDownload(file: NzUploadFile): boolean {\n return !!(this.icons.showDownloadIcon && file.status === 'done');\n }\n\n private fixData(): void {\n this.list.forEach(file => {\n file.isUploading = file.status === 'uploading';\n file.message = this.genErr(file);\n file.linkProps = typeof file.linkProps === 'string' ? JSON.parse(file.linkProps) : file.linkProps;\n file.isImageUrl = this.previewIsImage ? this.previewIsImage(file) : this.isImageUrl(file);\n file.iconType = this.getIconType(file);\n file.showDownload = this.showDownload(file);\n });\n }\n\n handlePreview(file: NzUploadFile, e: Event): void {\n if (!this.onPreview) {\n return;\n }\n\n e.preventDefault();\n return this.onPreview(file);\n }\n\n handleRemove(file: NzUploadFile, e: Event): void {\n e.preventDefault();\n if (this.onRemove) {\n this.onRemove(file);\n }\n return;\n }\n\n handleDownload(file: NzUploadFile): void {\n if (typeof this.onDownload === 'function') {\n this.onDownload(file);\n } else if (file.url) {\n window.open(file.url);\n }\n }\n\n // #endregion\n\n constructor(\n private cdr: ChangeDetectorRef,\n @Inject(DOCUMENT) private doc: NzSafeAny,\n private ngZone: NgZone,\n private platform: Platform\n ) {}\n\n detectChanges(): void {\n this.fixData();\n this.cdr.detectChanges();\n }\n\n ngOnChanges(): void {\n this.fixData();\n this.genThumb();\n }\n\n ngOnDestroy(): void {\n this.destroy$.next();\n }\n}\n","<div *ngFor=\"let file of list\" class=\"ant-upload-list-{{ listType }}-container\">\n <div\n class=\"ant-upload-list-item ant-upload-list-item-{{ file.status }} ant-upload-list-item-list-type-{{ listType }}\"\n [attr.data-key]=\"file.key\"\n @itemState\n nz-tooltip\n [nzTooltipTitle]=\"file.status === 'error' ? file.message : null\"\n >\n <ng-template #icon>\n <ng-container [ngSwitch]=\"file.iconType\">\n <div\n *ngSwitchCase=\"'uploading'\"\n class=\"ant-upload-list-item-thumbnail\"\n [class.ant-upload-list-item-file]=\"!file.isUploading\"\n >\n <ng-template [ngTemplateOutlet]=\"iconNode\" [ngTemplateOutletContext]=\"{ $implicit: file }\"></ng-template>\n </div>\n <a\n *ngSwitchCase=\"'thumbnail'\"\n class=\"ant-upload-list-item-thumbnail\"\n [class.ant-upload-list-item-file]=\"!file.isImageUrl\"\n target=\"_blank\"\n rel=\"noopener noreferrer\"\n [href]=\"file.url || file.thumbUrl\"\n (click)=\"handlePreview(file, $event)\"\n >\n <img\n *ngIf=\"file.isImageUrl; else noImageThumbTpl\"\n class=\"ant-upload-list-item-image\"\n [src]=\"file.thumbUrl || file.url\"\n [attr.alt]=\"file.name\"\n />\n </a>\n <div *ngSwitchDefault class=\"ant-upload-text-icon\">\n <ng-template [ngTemplateOutlet]=\"iconNode\" [ngTemplateOutletContext]=\"{ $implicit: file }\"></ng-template>\n </div>\n </ng-container>\n <ng-template #noImageThumbTpl>\n <ng-template [ngTemplateOutlet]=\"iconNode\" [ngTemplateOutletContext]=\"{ $implicit: file }\"></ng-template>\n </ng-template>\n </ng-template>\n <ng-template #iconNode let-file>\n <ng-container *ngIf=\"!iconRender; else customIconRender\">\n <ng-container [ngSwitch]=\"listType\">\n <ng-container *ngSwitchCase=\"'picture'\">\n <ng-container *ngIf=\"file.isUploading; else iconNodeFileIcon\">\n <i nz-icon nzType=\"loading\"></i>\n </ng-container>\n </ng-container>\n <ng-container *ngSwitchCase=\"'picture-card'\">\n <ng-container *ngIf=\"file.isUploading; else iconNodeFileIcon\">\n {{ locale.uploading }}\n </ng-container>\n </ng-container>\n <i *ngSwitchDefault nz-icon [nzType]=\"file.isUploading ? 'loading' : 'paper-clip'\"></i>\n </ng-container>\n </ng-container>\n <ng-template\n #customIconRender\n [ngTemplateOutlet]=\"iconRender\"\n [ngTemplateOutletContext]=\"{ $implicit: file }\"\n ></ng-template>\n <ng-template #iconNodeFileIcon>\n <i nz-icon [nzType]=\"file.isImageUrl ? 'picture' : 'file'\" nzTheme=\"twotone\"></i>\n </ng-template>\n </ng-template>\n <ng-template #removeIcon>\n <button\n *ngIf=\"icons.showRemoveIcon\"\n type=\"button\"\n nz-button\n nzType=\"text\"\n nzSize=\"small\"\n (click)=\"handleRemove(file, $event)\"\n [attr.title]=\"locale.removeFile\"\n class=\"ant-upload-list-item-card-actions-btn\"\n >\n <i nz-icon nzType=\"delete\"></i>\n </button>\n </ng-template>\n <ng-template #downloadIcon>\n <button\n *ngIf=\"file.showDownload\"\n type=\"button\"\n nz-button\n nzType=\"text\"\n nzSize=\"small\"\n (click)=\"handleDownload(file)\"\n [attr.title]=\"locale.downloadFile\"\n class=\"ant-upload-list-item-card-actions-btn\"\n >\n <i nz-icon nzType=\"download\"></i>\n </button>\n </ng-template>\n <ng-template #downloadOrDelete>\n <span\n *ngIf=\"listType !== 'picture-card'\"\n class=\"ant-upload-list-item-card-actions {{ listType === 'picture' ? 'picture' : '' }}\"\n >\n <ng-template [ngTemplateOutlet]=\"downloadIcon\"></ng-template>\n <ng-template [ngTemplateOutlet]=\"removeIcon\"></ng-template>\n </span>\n </ng-template>\n <ng-template #preview>\n <a\n *ngIf=\"file.url\"\n target=\"_blank\"\n rel=\"noopener noreferrer\"\n class=\"ant-upload-list-item-name\"\n [attr.title]=\"file.name\"\n [href]=\"file.url\"\n [attr.download]=\"file.linkProps && file.linkProps.download\"\n (click)=\"handlePreview(file, $event)\"\n >\n {{ file.name }}\n </a>\n <span\n *ngIf=\"!file.url\"\n class=\"ant-upload-list-item-name\"\n [attr.title]=\"file.name\"\n (click)=\"handlePreview(file, $event)\"\n >\n {{ file.name }}\n </span>\n <ng-template [ngTemplateOutlet]=\"downloadOrDelete\"></ng-template>\n </ng-template>\n <div class=\"ant-upload-list-item-info\">\n <span class=\"ant-upload-span\">\n <ng-template [ngTemplateOutlet]=\"icon\"></ng-template>\n <ng-template [ngTemplateOutlet]=\"preview\"></ng-template>\n </span>\n </div>\n <span *ngIf=\"listType === 'picture-card' && !file.isUploading\" class=\"ant-upload-list-item-actions\">\n <a\n *ngIf=\"icons.showPreviewIcon\"\n [href]=\"file.url || file.thumbUrl\"\n target=\"_blank\"\n rel=\"noopener noreferrer\"\n [attr.title]=\"locale.previewFile\"\n [ngStyle]=\"!(file.url || file.thumbUrl) ? { opacity: 0.5, 'pointer-events': 'none' } : null\"\n (click)=\"handlePreview(file, $event)\"\n >\n <i nz-icon nzType=\"eye\"></i>\n </a>\n <ng-container *ngIf=\"file.status === 'done'\">\n <ng-template [ngTemplateOutlet]=\"downloadIcon\"></ng-template>\n </ng-container>\n <ng-template [ngTemplateOutlet]=\"removeIcon\"></ng-template>\n </span>\n <div *ngIf=\"file.isUploading\" class=\"ant-upload-list-item-progress\">\n <nz-progress [nzPercent]=\"file.percent!\" nzType=\"line\" [nzShowInfo]=\"false\" [nzStrokeWidth]=\"2\"></nz-progress>\n </div>\n </div>\n</div>\n","/**\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://github.com/NG-ZORRO/ng-zorro-antd/blob/master/LICENSE\n */\n\nimport { Direction, Directionality } from '@angular/cdk/bidi';\nimport {\n ChangeDetectionStrategy,\n ChangeDetectorRef,\n Component,\n EventEmitter,\n Input,\n OnChanges,\n OnDestroy,\n OnInit,\n Optional,\n Output,\n TemplateRef,\n ViewChild,\n ViewEncapsulation\n} from '@angular/core';\nimport { Observable, of, Subject, Subscription } from 'rxjs';\nimport { filter, takeUntil } from 'rxjs/operators';\n\nimport { BooleanInput, NumberInput, NzSafeAny } from 'ng-zorro-antd/core/types';\nimport { InputBoolean, InputNumber, toBoolean } from 'ng-zorro-antd/core/util';\nimport { NzI18nService, NzUploadI18nInterface } from 'ng-zorro-antd/i18n';\n\nimport {\n NzIconRenderTemplate,\n NzShowUploadList,\n NzUploadChangeParam,\n NzUploadFile,\n NzUploadListType,\n NzUploadTransformFileType,\n NzUploadType,\n NzUploadXHRArgs,\n UploadFilter,\n ZipButtonOptions\n} from './interface';\nimport { NzUploadBtnComponent } from './upload-btn.component';\nimport { NzUploadListComponent } from './upload-list.component';\n\n@Component({\n selector: 'nz-upload',\n exportAs: 'nzUpload',\n templateUrl: './upload.component.html',\n preserveWhitespaces: false,\n encapsulation: ViewEncapsulation.None,\n changeDetection: ChangeDetectionStrategy.OnPush,\n host: {\n '[class.ant-upload-picture-card-wrapper]': 'nzListType === \"picture-card\"'\n }\n})\nexport class NzUploadComponent implements OnInit, OnChanges, OnDestroy {\n static ngAcceptInputType_nzLimit: NumberInput;\n static ngAcceptInputType_nzSize: NumberInput;\n static ngAcceptInputType_nzDirectory: BooleanInput;\n static ngAcceptInputType_nzOpenFileDialogOnClick: BooleanInput;\n static ngAcceptInputType_nzDisabled: BooleanInput;\n static ngAcceptInputType_nzMultiple: BooleanInput;\n static ngAcceptInputType_nzShowUploadList: BooleanInput | NzShowUploadList;\n static ngAcceptInputType_nzShowButton: BooleanInput;\n static ngAcceptInputType_nzWithCredentials: BooleanInput;\n\n private destroy$ = new Subject<void>();\n @ViewChild('uploadComp', { static: false }) uploadComp!: NzUploadBtnComponent;\n @ViewChild('listComp', { static: false }) listComp!: NzUploadListComponent;\n\n locale!: NzUploadI18nInterface;\n dir: Direction = 'ltr';\n\n // #region fields\n\n @Input() nzType: NzUploadType = 'select';\n @Input() @InputNumber() nzLimit = 0;\n @Input() @InputNumber() nzSize = 0;\n\n @Input() nzFileType?: string;\n @Input() nzAccept?: string | string[];\n @Input() nzAction?: string | ((file: NzUploadFile) => string | Observable<string>);\n @Input() @InputBoolean() nzDirectory = false;\n @Input() @InputBoolean() nzOpenFileDialogOnClick = true;\n @Input() nzBeforeUpload?: (file: NzUploadFile, fileList: NzUploadFile[]) => boolean | Observable<boolean>;\n @Input() nzCustomRequest?: (item: NzUploadXHRArgs) => Subscription;\n @Input() nzData?: {} | ((file: NzUploadFile) => {} | Observable<{}>);\n @Input() nzFilter: UploadFilter[] = [];\n @Input() nzFileList: NzUploadFile[] = [];\n @Input() @InputBoolean() nzDisabled = false;\n @Input() nzHeaders?: {} | ((file: NzUploadFile) => {} | Observable<{}>);\n @Input() nzListType: NzUploadListType = 'text';\n @Input() @InputBoolean() nzMultiple = false;\n @Input() nzName = 'file';\n\n private _showUploadList: boolean | NzShowUploadList = true;\n\n @Input()\n set nzShowUploadList(value: boolean | NzShowUploadList) {\n this._showUploadList = typeof value === 'boolean' ? toBoolean(value) : value;\n }\n\n get nzShowUploadList(): boolean | NzShowUploadList {\n return this._showUploadList;\n }\n\n @Input() @InputBoolean() nzShowButton = true;\n @Input() @InputBoolean() nzWithCredentials = false;\n\n @Input() nzRemove?: (file: NzUploadFile) => boolean | Observable<boolean>;\n @Input() nzPreview?: (file: NzUploadFile) => void;\n @Input() nzPreviewFile?: (file: NzUploadFile) => Observable<string>;\n @Input() nzPreviewIsImage?: (file: NzUploadFile) => boolean;\n @Input() nzTransformFile?: (file: NzUploadFile) => NzUploadTransformFileType;\n @Input() nzDownload?: (file: NzUploadFile) => void;\n @Input() nzIconRender: NzIconRenderTemplate | null = null;\n @Input() nzFileListRender: TemplateRef<void> | null = null;\n\n @Output() readonly nzChange: EventEmitter<NzUploadChangeParam> = new EventEmitter<NzUploadChangeParam>();\n @Output() readonly nzFileListChange: EventEmitter<NzUploadFile[]> = new EventEmitter<NzUploadFile[]>();\n\n _btnOptions?: ZipButtonOptions;\n\n private zipOptions(): this {\n if (typeof this.nzShowUploadList === 'boolean' && this.nzShowUploadList) {\n this.nzShowUploadList = {\n showPreviewIcon: true,\n showRemoveIcon: true,\n showDownloadIcon: true\n };\n }\n // filters\n const filters: UploadFilter[] = this.nzFilter.slice();\n if (this.nzMultiple && this.nzLimit > 0 && filters.findIndex(w => w.name === 'limit') === -1) {\n filters.push({\n name: 'limit',\n fn: (fileList: NzUploadFile[]) => fileList.slice(-this.nzLimit)\n });\n }\n if (this.nzSize > 0 && filters.findIndex(w => w.name === 'size') === -1) {\n filters.push({\n name: 'size',\n fn: (fileList: NzUploadFile[]) => fileList.filter(w => w.size! / 1024 <= this.nzSize)\n });\n }\n if (this.nzFileType && this.nzFileType.length > 0 && filters.findIndex(w => w.name === 'type') === -1) {\n const types = this.nzFileType.split(',');\n filters.push({\n name: 'type',\n fn: (fileList: NzUploadFile[]) => fileList.filter(w => ~types.indexOf(w.type!))\n });\n }\n this._btnOptions = {\n disabled: this.nzDisabled,\n accept: this.nzAccept,\n action: this.nzAction,\n directory: this.nzDirectory,\n openFileDialogOnClick: this.nzOpenFileDialogOnClick,\n beforeUpload: this.nzBeforeUpload,\n customRequest: this.nzCustomRequest,\n data: this.nzData,\n headers: this.nzHeaders,\n name: this.nzName,\n multiple: this.nzMultiple,\n withCredentials: this.nzWithCredentials,\n filters,\n transformFile: this.nzTransformFile,\n onStart: this.onStart,\n onProgress: this.onProgress,\n onSuccess: this.onSuccess,\n onError: this.onError\n };\n return this;\n }\n\n // #endregion\n\n constructor(\n private cdr: ChangeDetectorRef,\n private i18n: NzI18nService,\n @Optional() private directionality: Directionality\n ) {}\n\n // #region upload\n\n private fileToObject(file: NzUploadFile): NzUploadFile {\n return {\n lastModified: file.lastModified,\n lastModifiedDate: file.lastModifiedDate,\n name: file.filename || file.name,\n size: file.size,\n type: file.type,\n uid: file.uid,\n response: file.response,\n error: file.error,\n percent: 0,\n originFileObj: file as NzSafeAny\n };\n }\n\n private getFileItem(file: NzUploadFile, fileList: NzUploadFile[]): NzUploadFile {\n return fileList.filter(item => item.uid === file.uid)[0];\n }\n\n private removeFileItem(file: NzUploadFile, fileList: NzUploadFile[]): NzUploadFile[] {\n return fileList.filter(item => item.uid !== file.uid);\n }\n\n private onStart = (file: NzUploadFile): void => {\n if (!this.nzFileList) {\n this.nzFileList = [];\n }\n const targetItem = this.fileToObject(file);\n targetItem.status = 'uploading';\n this.nzFileList = this.nzFileList.concat(targetItem);\n this.nzFileListChange.emit(this.nzFileList);\n this.nzChange.emit({ file: targetItem, fileList: this.nzFileList, type: 'start' });\n this.detectChangesList();\n };\n\n private onProgress = (e: { percent: number }, file: NzUploadFile): void => {\n const fileList = this.nzFileList;\n const targetItem = this.getFileItem(file, fileList);\n targetItem.percent = e.percent;\n this.nzChange.emit({\n event: e,\n file: { ...targetItem },\n fileList: this.nzFileList,\n type: 'progress'\n });\n this.detectChangesList();\n };\n\n private onSuccess = (res: {}, file: NzUploadFile): void => {\n const fileList = this.nzFileList;\n const targetItem = this.getFileItem(file, fileList);\n targetItem.status = 'done';\n targetItem.response = res;\n this.nzChange.emit({\n file: { ...targetItem },\n fileList,\n type: 'success'\n });\n this.detectChangesList();\n };\n\n private onError = (err: {}, file: NzUploadFile): void => {\n const fileList = this.nzFileList;\n const targetItem = this.getFileItem(file, fileList);\n targetItem.error = err;\n targetItem.status = 'error';\n this.nzChange.emit({\n file: { ...targetItem },\n fileList,\n type: 'error'\n });\n this.detectChangesList();\n };\n\n // #endregion\n\n // #region drag\n\n private dragState?: string;\n\n // skip safari bug\n fileDrop(e: DragEvent): void {\n if (e.type === this.dragState) {\n return;\n }\n this.dragState = e.type;\n this.setClassMap();\n }\n\n // #endregion\n\n // #region list\n\n private detectChangesList(): void {\n this.cdr.detectChanges();\n this.listComp?.detectChanges();\n }\n\n onRemove = (file: NzUploadFile): void => {\n this.uploadComp.abort(file);\n file.status = 'removed';\n const fnRes =\n typeof this.nzRemove === 'function' ? this.nzRemove(file) : this.nzRemove == null ? true : this.nzRemove;\n (fnRes instanceof Observable ? fnRes : of(fnRes)).pipe(filter((res: boolean) => res)).subscribe(() => {\n this.nzFileList = this.removeFileItem(file, this.nzFileList);\n this.nzChange.emit({\n file,\n fileList: this.nzFileList,\n type: 'removed'\n });\n this.nzFileListChange.emit(this.nzFileList);\n this.cdr.detectChanges();\n });\n };\n\n // #endregion\n\n // #region styles\n\n private prefixCls = 'ant-upload';\n classList: string[] = [];\n\n private setClassMap(): void {\n let subCls: string[] = [];\n if (this.nzType === 'drag') {\n if (this.nzFileList.some(file => file.status === 'uploading')) {\n subCls.push(`${this.prefixCls}-drag-uploading`);\n }\n if (this.dragState === 'dragover') {\n subCls.push(`${this.prefixCls}-drag-hover`);\n }\n } else {\n subCls = [`${this.prefixCls}-select-${this.nzListType}`];\n }\n\n this.classList = [\n this.prefixCls,\n `${this.prefixCls}-${this.nzType}`,\n ...subCls,\n (this.nzDisabled && `${this.prefixCls}-disabled`) || '',\n (this.dir === 'rtl' && `${this.prefixCls}-rtl`) || ''\n ].filter(item => !!item);\n\n this.cdr.detectChanges();\n }\n\n // #endregion\n\n ngOnInit(): void {\n this.dir = this.directionality.value;\n this.directionality.change?.pipe(takeUntil(this.destroy$)).subscribe((direction: Direction) => {\n this.dir = direction;\n this.setClassMap();\n this.cdr.detectChanges();\n });\n\n this.i18n.localeChange.pipe(takeUntil(this.destroy$)).subscribe(() => {\n this.locale = this.i18n.getLocaleData('Upload');\n this.detectChangesList();\n });\n }\n\n ngOnChanges(): void {\n this.zipOptions().setClassMap();\n }\n\n ngOnDestroy(): void {\n this.destroy$.next();\n this.destroy$.complete();\n }\n}\n","<ng-template #list>\n <nz-upload-list\n *ngIf=\"locale && !nzFileListRender\"\n #listComp\n [style.display]=\"nzShowUploadList ? '' : 'none'\"\n [locale]=\"locale\"\n [listType]=\"nzListType\"\n [items]=\"nzFileList || []\"\n [icons]=\"$any(nzShowUploadList)\"\n [iconRender]=\"nzIconRender\"\n [previewFile]=\"nzPreviewFile\"\n [previewIsImage]=\"nzPreviewIsImage\"\n [onPreview]=\"nzPreview\"\n [onRemove]=\"onRemove\"\n [onDownload]=\"nzDownload\"\n [dir]=\"dir\"\n ></nz-upload-list>\n <ng-container *ngIf=\"nzFileListRender\">\n <ng-container *ngTemplateOutlet=\"nzFileListRender; context: { $implicit: nzFileList }\"></ng-container>\n </ng-container>\n</ng-template>\n<ng-template #con><ng-content></ng-content></ng-template>\n<ng-template #btn>\n <div [ngClass]=\"classList\" [style.display]=\"nzShowButton ? '' : 'none'\">\n <div nz-upload-btn #uploadComp [options]=\"_btnOptions!\">\n <ng-template [ngTemplateOutlet]=\"con\"></ng-template>\n </div>\n </div>\n</ng-template>\n<ng-container *ngIf=\"nzType === 'drag'; else select\">\n <div [ngClass]=\"classList\" (drop)=\"fileDrop($event)\" (dragover)=\"fileDrop($event)\" (dragleave)=\"fileDrop($event)\">\n <div nz-upload-btn #uploadComp [options]=\"_btnOptions!\" class=\"ant-upload-btn\">\n <div class=\"ant-upload-drag-container\">\n <ng-template [ngTemplateOutlet]=\"con\"></ng-template>\n </div>\n </div>\n </div>\n <ng-template [ngTemplateOutlet]=\"list\"></ng-template>\n</ng-container>\n<ng-template #select>\n <ng-container *ngIf=\"nzListType === 'picture-card'; else pic\">\n <ng-template [ngTemplateOutlet]=\"list\"></ng-template>\n <ng-template [ngTemplateOutlet]=\"btn\"></ng-template>\n </ng-container>\n</ng-template>\n<ng-template #pic>\n <ng-template [ngTemplateOutlet]=\"btn\"></ng-template>\n <ng-template [ngTemplateOutlet]=\"list\"></ng-template>\n</ng-template>\n","/**\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://github.com/NG-ZORRO/ng-zorro-antd/blob/master/LICENSE\n */\n\nimport { BidiModule } from '@angular/cdk/bidi';\nimport { PlatformModule } from '@angular/cdk/platform';\nimport { CommonModule } from '@angular/common';\nimport { NgModule } from '@angular/core';\nimport { FormsModule } from '@angular/forms';\n\nimport { NzButtonModule } from 'ng-zorro-antd/button';\nimport { NzI18nModule } from 'ng-zorro-antd/i18n';\nimport { NzIconModule } from 'ng-zorro-antd/icon';\nimport { NzProgressModule } from 'ng-zorro-antd/progress';\nimport { NzToolTipModule } from 'ng-zorro-antd/tooltip';\n\nimport { NzUploadBtnComponent } from './upload-btn.component';\nimport { NzUploadListComponent } from './upload-list.component';\nimport { NzUploadComponent } from './upload.component';\n\n@NgModule({\n imports: [\n BidiModule,\n CommonModule,\n FormsModule,\n PlatformModule,\n NzToolTipModule,\n NzProgressModule,\n NzI18nModule,\n NzIconModule,\n NzButtonModule\n ],\n declarations: [NzUploadComponent, NzUploadBtnComponent, NzUploadListComponent],\n exports: [NzUploadComponent]\n})\nexport class NzUploadModule {}\n","/**\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://github.com/NG-ZORRO/ng-zorro-antd/blob/master/LICENSE\n */\n\nexport * from './interface';\nexport * from './upload-btn.component';\nexport * from './upload-list.component';\nexport * from './upload.component';\nexport * from './upload.module';\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './public-api';\n"],"names":[],"mappings":"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