cps-ui-kit 0.154.0 → 0.156.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -12,6 +12,7 @@
12
12
  - Datepicker
13
13
  - Dialog
14
14
  - Expansion panel
15
+ - File upload
15
16
  - Icon
16
17
  - Info circle
17
18
  - Input
@@ -0,0 +1,138 @@
1
+ import { Component, EventEmitter, Input, Output } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { convertSize } from '../../utils/internal/size-utils';
4
+ import { CpsIconComponent } from '../cps-icon/cps-icon.component';
5
+ import { CpsProgressLinearComponent } from '../cps-progress-linear/cps-progress-linear.component';
6
+ import { catchError, of, take } from 'rxjs';
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "@angular/common";
9
+ /**
10
+ * CpsFileUploadComponent is an advanced uploader with dragdrop support.
11
+ * @group Components
12
+ */
13
+ export class CpsFileUploadComponent {
14
+ constructor() {
15
+ /**
16
+ * Expected extensions of a file to be uploaded. E.g. 'doc or .doc'.
17
+ * @group Props
18
+ */
19
+ this.extensions = [];
20
+ /**
21
+ * Expected file description. E.g. 'Word document'.
22
+ * @group Props
23
+ */
24
+ this.fileDesc = 'Any file';
25
+ /**
26
+ * Width of the component, a number denoting pixels or a string.
27
+ * @group Props
28
+ */
29
+ this.width = '100%';
30
+ /**
31
+ * Callback for uploaded file processing.
32
+ * @group Props
33
+ */
34
+ this.fileProcessingCallback = undefined;
35
+ /**
36
+ * Callback to invoke when file is uploaded.
37
+ * @param {File} File
38
+ * @group Emits
39
+ */
40
+ this.fileUploaded = new EventEmitter();
41
+ /**
42
+ * Callback to invoke when file upload fails.
43
+ * @param {string} - file name
44
+ * @group Emits
45
+ */
46
+ this.fileUploadFailed = new EventEmitter();
47
+ /**
48
+ * Callback to invoke when uploaded file is removed.
49
+ * @param {string} - file name
50
+ * @group Emits
51
+ */
52
+ this.uploadedFileRemoved = new EventEmitter();
53
+ this.isDragoverFile = false;
54
+ this.extensionsString = '';
55
+ this.extensionsStringAsterisks = '';
56
+ this.isProcessingFile = false;
57
+ }
58
+ ngOnInit() {
59
+ this.width = convertSize(this.width);
60
+ this.extensions = this.extensions.map((ext) => ext.startsWith('.') ? ext : '.' + ext);
61
+ this.extensionsString = this.extensions.join(', ');
62
+ this.extensionsStringAsterisks = this.extensions
63
+ .map((ext) => '*' + ext)
64
+ .join(', ');
65
+ }
66
+ tryUploadFile(event) {
67
+ event.preventDefault();
68
+ event.stopPropagation();
69
+ this.isDragoverFile = false;
70
+ let file;
71
+ if (event.type === 'drop') {
72
+ file = event.dataTransfer?.files.item(0) ?? undefined;
73
+ }
74
+ else {
75
+ const files = event.target.files;
76
+ if (!files || files.length < 1)
77
+ return;
78
+ file = files.item(0) ?? undefined;
79
+ }
80
+ if (!this._isFileExtensionValid(file)) {
81
+ this.fileUploadFailed.emit(file?.name ?? '');
82
+ return;
83
+ }
84
+ this.uploadedFile = file;
85
+ if (this.uploadedFile) {
86
+ this.fileUploaded.emit(this.uploadedFile);
87
+ if (this.fileProcessingCallback) {
88
+ this.isProcessingFile = true;
89
+ this.fileProcessingCallback(this.uploadedFile)
90
+ .pipe(take(1), catchError(() => {
91
+ return of(false);
92
+ }))
93
+ .subscribe((res) => {
94
+ if (!res)
95
+ this.removeUploadedFile();
96
+ this.isProcessingFile = false;
97
+ });
98
+ }
99
+ }
100
+ }
101
+ removeUploadedFile() {
102
+ const name = this.uploadedFile?.name ?? '';
103
+ this.uploadedFile = undefined;
104
+ this.uploadedFileRemoved.emit(name);
105
+ }
106
+ _isFileExtensionValid(file) {
107
+ if (!file)
108
+ return false;
109
+ if (this.extensions.length < 1)
110
+ return true;
111
+ for (const ext of this.extensions) {
112
+ if (file.name.endsWith(ext))
113
+ return true;
114
+ }
115
+ return false;
116
+ }
117
+ }
118
+ CpsFileUploadComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: CpsFileUploadComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
119
+ CpsFileUploadComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: CpsFileUploadComponent, isStandalone: true, selector: "cps-file-upload", inputs: { extensions: "extensions", fileDesc: "fileDesc", width: "width", fileProcessingCallback: "fileProcessingCallback" }, outputs: { fileUploaded: "fileUploaded", fileUploadFailed: "fileUploadFailed", uploadedFileRemoved: "uploadedFileRemoved" }, ngImport: i0, template: "<div class=\"cps-file-upload\" [ngStyle]=\"{ width: width }\">\n <div\n class=\"cps-file-upload-dropzone\"\n [ngClass]=\"{\n 'dragged-over': isDragoverFile,\n 'with-uploads': uploadedFile?.name\n }\"\n (dragend)=\"isDragoverFile = false\"\n (dragenter)=\"isDragoverFile = true\"\n (dragleave)=\"isDragoverFile = false\"\n (dragover)=\"isDragoverFile = true\"\n (mouseleave)=\"isDragoverFile = false\"\n (drop)=\"tryUploadFile($event)\">\n <input\n (change)=\"tryUploadFile($event)\"\n type=\"file\"\n [accept]=\"extensionsString\"\n title=\"\" />\n <cps-icon icon=\"export\" size=\"large\" color=\"text-darkest\"></cps-icon>\n <div class=\"cps-file-upload-dropzone-title\">\n Drag&Drop or choose a file to upload\n </div>\n <span\n class=\"cps-file-upload-dropzone-file-desc\"\n *ngIf=\"fileDesc || extensionsStringAsterisks\"\n >{{ fileDesc }}\n {{\n extensionsStringAsterisks ? '(' + extensionsStringAsterisks + ')' : ''\n }}</span\n >\n <cps-progress-linear\n *ngIf=\"isProcessingFile && uploadedFile\"\n height=\"3\"\n radius=\"4\"\n opacity=\"0.3\"\n class=\"cps-file-upload-progress-bar\"\n bgColor=\"transparent\">\n </cps-progress-linear>\n </div>\n <div *ngIf=\"uploadedFile\" class=\"cps-file-upload-uploaded-files\">\n <div class=\"cps-file-upload-uploaded-file\">\n <div class=\"cps-file-upload-uploaded-file-title\">\n <cps-icon\n class=\"cps-file-upload-uploaded-file-status-icon\"\n [icon]=\"isProcessingFile ? 'pending' : 'toast-success'\"\n [color]=\"isProcessingFile ? 'warn' : 'success'\">\n </cps-icon>\n <div class=\"cps-file-upload-uploaded-file-name\">\n {{ uploadedFile.name }}\n </div>\n </div>\n <cps-icon\n *ngIf=\"!isProcessingFile\"\n class=\"cps-file-upload-uploaded-file-remove-icon\"\n icon=\"remove\"\n color=\"error\"\n (click)=\"removeUploadedFile()\">\n </cps-icon>\n </div>\n </div>\n</div>\n", styles: [":host{display:flex}:host .cps-file-upload .cps-file-upload-dropzone{font-family:Source Sans Pro,sans-serif;min-height:20px;padding:2rem;text-align:center;border:2px dashed var(--cps-color-calm);position:relative;border-radius:1rem}:host .cps-file-upload .cps-file-upload-dropzone:hover{background-color:var(--cps-color-highlight-hover)}:host .cps-file-upload .cps-file-upload-dropzone:active{background-color:var(--cps-color-highlight-selected)}:host .cps-file-upload .cps-file-upload-dropzone.dragged-over{background-color:var(--cps-color-highlight-selected)}:host .cps-file-upload .cps-file-upload-dropzone.with-uploads{border-bottom-right-radius:0;border-bottom-left-radius:0;border-bottom:1px dashed var(--cps-color-calm)}:host .cps-file-upload .cps-file-upload-dropzone input{cursor:pointer;opacity:0;position:absolute;width:100%;height:100%;top:0;left:0}:host .cps-file-upload .cps-file-upload-dropzone .cps-file-upload-dropzone-title{font-size:16px;margin:8px 0;color:var(--cps-color-calm);font-weight:600}:host .cps-file-upload .cps-file-upload-dropzone .cps-file-upload-dropzone-file-desc{font-size:18px;color:var(--cps-color-text-dark)}:host .cps-file-upload .cps-file-upload-dropzone .cps-file-upload-progress-bar{position:absolute;bottom:-2px;left:0}:host .cps-file-upload .cps-file-upload-uploaded-files{border-bottom:2px dashed var(--cps-color-calm);border-left:2px dashed var(--cps-color-calm);border-right:2px dashed var(--cps-color-calm);border-bottom-right-radius:16px;border-bottom-left-radius:16px;background-color:var(--cps-color-highlight-hover)}:host .cps-file-upload .cps-file-upload-uploaded-files .cps-file-upload-uploaded-file{cursor:default;display:flex;align-items:center;justify-content:space-between;padding:8px 12px}:host .cps-file-upload .cps-file-upload-uploaded-files .cps-file-upload-uploaded-file .cps-file-upload-uploaded-file-title{display:flex;align-items:center}:host .cps-file-upload .cps-file-upload-uploaded-files .cps-file-upload-uploaded-file .cps-file-upload-uploaded-file-title .cps-file-upload-uploaded-file-name{color:var(--cps-color-text-darkest);font-size:20px;margin:0 8px}:host .cps-file-upload .cps-file-upload-uploaded-files .cps-file-upload-uploaded-file .cps-file-upload-uploaded-file-remove-icon{cursor:pointer}:host .cps-file-upload .cps-file-upload-uploaded-files .cps-file-upload-uploaded-file .cps-file-upload-uploaded-file-remove-icon:hover ::ng-deep .cps-icon{color:var(--cps-color-error-darken1)!important}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: CpsIconComponent, selector: "cps-icon", inputs: ["icon", "size", "color"] }, { kind: "component", type: CpsProgressLinearComponent, selector: "cps-progress-linear", inputs: ["width", "height", "color", "bgColor", "opacity", "radius"] }] });
120
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: CpsFileUploadComponent, decorators: [{
121
+ type: Component,
122
+ args: [{ selector: 'cps-file-upload', standalone: true, imports: [CommonModule, CpsIconComponent, CpsProgressLinearComponent], template: "<div class=\"cps-file-upload\" [ngStyle]=\"{ width: width }\">\n <div\n class=\"cps-file-upload-dropzone\"\n [ngClass]=\"{\n 'dragged-over': isDragoverFile,\n 'with-uploads': uploadedFile?.name\n }\"\n (dragend)=\"isDragoverFile = false\"\n (dragenter)=\"isDragoverFile = true\"\n (dragleave)=\"isDragoverFile = false\"\n (dragover)=\"isDragoverFile = true\"\n (mouseleave)=\"isDragoverFile = false\"\n (drop)=\"tryUploadFile($event)\">\n <input\n (change)=\"tryUploadFile($event)\"\n type=\"file\"\n [accept]=\"extensionsString\"\n title=\"\" />\n <cps-icon icon=\"export\" size=\"large\" color=\"text-darkest\"></cps-icon>\n <div class=\"cps-file-upload-dropzone-title\">\n Drag&Drop or choose a file to upload\n </div>\n <span\n class=\"cps-file-upload-dropzone-file-desc\"\n *ngIf=\"fileDesc || extensionsStringAsterisks\"\n >{{ fileDesc }}\n {{\n extensionsStringAsterisks ? '(' + extensionsStringAsterisks + ')' : ''\n }}</span\n >\n <cps-progress-linear\n *ngIf=\"isProcessingFile && uploadedFile\"\n height=\"3\"\n radius=\"4\"\n opacity=\"0.3\"\n class=\"cps-file-upload-progress-bar\"\n bgColor=\"transparent\">\n </cps-progress-linear>\n </div>\n <div *ngIf=\"uploadedFile\" class=\"cps-file-upload-uploaded-files\">\n <div class=\"cps-file-upload-uploaded-file\">\n <div class=\"cps-file-upload-uploaded-file-title\">\n <cps-icon\n class=\"cps-file-upload-uploaded-file-status-icon\"\n [icon]=\"isProcessingFile ? 'pending' : 'toast-success'\"\n [color]=\"isProcessingFile ? 'warn' : 'success'\">\n </cps-icon>\n <div class=\"cps-file-upload-uploaded-file-name\">\n {{ uploadedFile.name }}\n </div>\n </div>\n <cps-icon\n *ngIf=\"!isProcessingFile\"\n class=\"cps-file-upload-uploaded-file-remove-icon\"\n icon=\"remove\"\n color=\"error\"\n (click)=\"removeUploadedFile()\">\n </cps-icon>\n </div>\n </div>\n</div>\n", styles: [":host{display:flex}:host .cps-file-upload .cps-file-upload-dropzone{font-family:Source Sans Pro,sans-serif;min-height:20px;padding:2rem;text-align:center;border:2px dashed var(--cps-color-calm);position:relative;border-radius:1rem}:host .cps-file-upload .cps-file-upload-dropzone:hover{background-color:var(--cps-color-highlight-hover)}:host .cps-file-upload .cps-file-upload-dropzone:active{background-color:var(--cps-color-highlight-selected)}:host .cps-file-upload .cps-file-upload-dropzone.dragged-over{background-color:var(--cps-color-highlight-selected)}:host .cps-file-upload .cps-file-upload-dropzone.with-uploads{border-bottom-right-radius:0;border-bottom-left-radius:0;border-bottom:1px dashed var(--cps-color-calm)}:host .cps-file-upload .cps-file-upload-dropzone input{cursor:pointer;opacity:0;position:absolute;width:100%;height:100%;top:0;left:0}:host .cps-file-upload .cps-file-upload-dropzone .cps-file-upload-dropzone-title{font-size:16px;margin:8px 0;color:var(--cps-color-calm);font-weight:600}:host .cps-file-upload .cps-file-upload-dropzone .cps-file-upload-dropzone-file-desc{font-size:18px;color:var(--cps-color-text-dark)}:host .cps-file-upload .cps-file-upload-dropzone .cps-file-upload-progress-bar{position:absolute;bottom:-2px;left:0}:host .cps-file-upload .cps-file-upload-uploaded-files{border-bottom:2px dashed var(--cps-color-calm);border-left:2px dashed var(--cps-color-calm);border-right:2px dashed var(--cps-color-calm);border-bottom-right-radius:16px;border-bottom-left-radius:16px;background-color:var(--cps-color-highlight-hover)}:host .cps-file-upload .cps-file-upload-uploaded-files .cps-file-upload-uploaded-file{cursor:default;display:flex;align-items:center;justify-content:space-between;padding:8px 12px}:host .cps-file-upload .cps-file-upload-uploaded-files .cps-file-upload-uploaded-file .cps-file-upload-uploaded-file-title{display:flex;align-items:center}:host .cps-file-upload .cps-file-upload-uploaded-files .cps-file-upload-uploaded-file .cps-file-upload-uploaded-file-title .cps-file-upload-uploaded-file-name{color:var(--cps-color-text-darkest);font-size:20px;margin:0 8px}:host .cps-file-upload .cps-file-upload-uploaded-files .cps-file-upload-uploaded-file .cps-file-upload-uploaded-file-remove-icon{cursor:pointer}:host .cps-file-upload .cps-file-upload-uploaded-files .cps-file-upload-uploaded-file .cps-file-upload-uploaded-file-remove-icon:hover ::ng-deep .cps-icon{color:var(--cps-color-error-darken1)!important}\n"] }]
123
+ }], propDecorators: { extensions: [{
124
+ type: Input
125
+ }], fileDesc: [{
126
+ type: Input
127
+ }], width: [{
128
+ type: Input
129
+ }], fileProcessingCallback: [{
130
+ type: Input
131
+ }], fileUploaded: [{
132
+ type: Output
133
+ }], fileUploadFailed: [{
134
+ type: Output
135
+ }], uploadedFileRemoved: [{
136
+ type: Output
137
+ }] } });
138
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cps-file-upload.component.js","sourceRoot":"","sources":["../../../../../../projects/cps-ui-kit/src/lib/components/cps-file-upload/cps-file-upload.component.ts","../../../../../../projects/cps-ui-kit/src/lib/components/cps-file-upload/cps-file-upload.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { convertSize } from '../../utils/internal/size-utils';\nimport { CpsIconComponent } from '../cps-icon/cps-icon.component';\nimport { CpsProgressLinearComponent } from '../cps-progress-linear/cps-progress-linear.component';\nimport { Observable, catchError, of, take } from 'rxjs';\n\n/**\n * CpsFileUploadComponent is an advanced uploader with dragdrop support.\n * @group Components\n */\n@Component({\n  selector: 'cps-file-upload',\n  standalone: true,\n  imports: [CommonModule, CpsIconComponent, CpsProgressLinearComponent],\n  templateUrl: './cps-file-upload.component.html',\n  styleUrls: ['./cps-file-upload.component.scss']\n})\nexport class CpsFileUploadComponent implements OnInit {\n  /**\n   * Expected extensions of a file to be uploaded. E.g. 'doc or .doc'.\n   * @group Props\n   */\n  @Input() extensions: string[] = [];\n\n  /**\n   * Expected file description. E.g. 'Word document'.\n   * @group Props\n   */\n  @Input() fileDesc = 'Any file';\n\n  /**\n   * Width of the component, a number denoting pixels or a string.\n   * @group Props\n   */\n  @Input() width: number | string = '100%';\n\n  /**\n   * Callback for uploaded file processing.\n   * @group Props\n   */\n  @Input() fileProcessingCallback?: (file: File) => Observable<boolean> =\n    undefined;\n\n  /**\n   * Callback to invoke when file is uploaded.\n   * @param {File} File\n   * @group Emits\n   */\n  @Output() fileUploaded = new EventEmitter<File>();\n\n  /**\n   * Callback to invoke when file upload fails.\n   * @param {string} - file name\n   * @group Emits\n   */\n  @Output() fileUploadFailed = new EventEmitter<string>();\n\n  /**\n   * Callback to invoke when uploaded file is removed.\n   * @param {string} - file name\n   * @group Emits\n   */\n  @Output() uploadedFileRemoved = new EventEmitter<string>();\n\n  isDragoverFile = false;\n  uploadedFile?: File;\n  extensionsString = '';\n  extensionsStringAsterisks = '';\n\n  isProcessingFile = false;\n\n  ngOnInit(): void {\n    this.width = convertSize(this.width);\n    this.extensions = this.extensions.map((ext) =>\n      ext.startsWith('.') ? ext : '.' + ext\n    );\n    this.extensionsString = this.extensions.join(', ');\n    this.extensionsStringAsterisks = this.extensions\n      .map((ext) => '*' + ext)\n      .join(', ');\n  }\n\n  tryUploadFile(event: Event) {\n    event.preventDefault();\n    event.stopPropagation();\n\n    this.isDragoverFile = false;\n    let file: File | undefined;\n\n    if (event.type === 'drop') {\n      file = (event as DragEvent).dataTransfer?.files.item(0) ?? undefined;\n    } else {\n      const files = (event.target as HTMLInputElement).files;\n      if (!files || files.length < 1) return;\n      file = files.item(0) ?? undefined;\n    }\n\n    if (!this._isFileExtensionValid(file)) {\n      this.fileUploadFailed.emit(file?.name ?? '');\n      return;\n    }\n\n    this.uploadedFile = file;\n    if (this.uploadedFile) {\n      this.fileUploaded.emit(this.uploadedFile);\n      if (this.fileProcessingCallback) {\n        this.isProcessingFile = true;\n        this.fileProcessingCallback(this.uploadedFile)\n          .pipe(\n            take(1),\n            catchError(() => {\n              return of(false);\n            })\n          )\n          .subscribe((res) => {\n            if (!res) this.removeUploadedFile();\n            this.isProcessingFile = false;\n          });\n      }\n    }\n  }\n\n  removeUploadedFile() {\n    const name = this.uploadedFile?.name ?? '';\n    this.uploadedFile = undefined;\n    this.uploadedFileRemoved.emit(name);\n  }\n\n  private _isFileExtensionValid(file?: File) {\n    if (!file) return false;\n\n    if (this.extensions.length < 1) return true;\n\n    for (const ext of this.extensions) {\n      if (file.name.endsWith(ext)) return true;\n    }\n\n    return false;\n  }\n}\n","<div class=\"cps-file-upload\" [ngStyle]=\"{ width: width }\">\n  <div\n    class=\"cps-file-upload-dropzone\"\n    [ngClass]=\"{\n      'dragged-over': isDragoverFile,\n      'with-uploads': uploadedFile?.name\n    }\"\n    (dragend)=\"isDragoverFile = false\"\n    (dragenter)=\"isDragoverFile = true\"\n    (dragleave)=\"isDragoverFile = false\"\n    (dragover)=\"isDragoverFile = true\"\n    (mouseleave)=\"isDragoverFile = false\"\n    (drop)=\"tryUploadFile($event)\">\n    <input\n      (change)=\"tryUploadFile($event)\"\n      type=\"file\"\n      [accept]=\"extensionsString\"\n      title=\"\" />\n    <cps-icon icon=\"export\" size=\"large\" color=\"text-darkest\"></cps-icon>\n    <div class=\"cps-file-upload-dropzone-title\">\n      Drag&Drop or choose a file to upload\n    </div>\n    <span\n      class=\"cps-file-upload-dropzone-file-desc\"\n      *ngIf=\"fileDesc || extensionsStringAsterisks\"\n      >{{ fileDesc }}\n      {{\n        extensionsStringAsterisks ? '(' + extensionsStringAsterisks + ')' : ''\n      }}</span\n    >\n    <cps-progress-linear\n      *ngIf=\"isProcessingFile && uploadedFile\"\n      height=\"3\"\n      radius=\"4\"\n      opacity=\"0.3\"\n      class=\"cps-file-upload-progress-bar\"\n      bgColor=\"transparent\">\n    </cps-progress-linear>\n  </div>\n  <div *ngIf=\"uploadedFile\" class=\"cps-file-upload-uploaded-files\">\n    <div class=\"cps-file-upload-uploaded-file\">\n      <div class=\"cps-file-upload-uploaded-file-title\">\n        <cps-icon\n          class=\"cps-file-upload-uploaded-file-status-icon\"\n          [icon]=\"isProcessingFile ? 'pending' : 'toast-success'\"\n          [color]=\"isProcessingFile ? 'warn' : 'success'\">\n        </cps-icon>\n        <div class=\"cps-file-upload-uploaded-file-name\">\n          {{ uploadedFile.name }}\n        </div>\n      </div>\n      <cps-icon\n        *ngIf=\"!isProcessingFile\"\n        class=\"cps-file-upload-uploaded-file-remove-icon\"\n        icon=\"remove\"\n        color=\"error\"\n        (click)=\"removeUploadedFile()\">\n      </cps-icon>\n    </div>\n  </div>\n</div>\n"]}