@c8y/ngx-components 1018.0.82 → 1018.0.83

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.
Files changed (44) hide show
  1. package/core/common/files.service.d.ts +1 -1
  2. package/core/drop-area/drop-area.component.d.ts +3 -1
  3. package/esm2020/core/common/bytes.pipe.mjs +3 -3
  4. package/esm2020/core/common/files.service.mjs +4 -4
  5. package/esm2020/core/drop-area/drop-area.component.mjs +16 -5
  6. package/esm2020/core/dynamic-forms/file/file.type.component.mjs +4 -3
  7. package/esm2020/core/dynamic-forms/json-schema/c8y-json-schema.service.mjs +12 -1
  8. package/esm2020/core/file-picker/file-picker.component.mjs +1 -1
  9. package/esm2020/ecosystem/shared/add-application.component.mjs +1 -1
  10. package/esm2020/ecosystem/shared/upload-archive.component.mjs +1 -1
  11. package/esm2020/protocol-opcua/opcua-server-config.component.mjs +1 -1
  12. package/esm2020/sub-assets/asset-properties-item.component.mjs +3 -3
  13. package/esm2020/trusted-certificates/add-trusted-certificate.component.mjs +1 -1
  14. package/esm2020/widgets/implementations/image/image-widget-config/image-widget-config.component.mjs +2 -2
  15. package/esm2020/widgets/implementations/markdown/markdown-widget-config/markdown-widget-config.component.mjs +1 -1
  16. package/fesm2015/c8y-ngx-components-ecosystem-shared.mjs +2 -2
  17. package/fesm2015/c8y-ngx-components-ecosystem-shared.mjs.map +1 -1
  18. package/fesm2015/c8y-ngx-components-protocol-opcua.mjs +1 -1
  19. package/fesm2015/c8y-ngx-components-protocol-opcua.mjs.map +1 -1
  20. package/fesm2015/c8y-ngx-components-sub-assets.mjs +2 -2
  21. package/fesm2015/c8y-ngx-components-sub-assets.mjs.map +1 -1
  22. package/fesm2015/c8y-ngx-components-trusted-certificates.mjs +1 -1
  23. package/fesm2015/c8y-ngx-components-trusted-certificates.mjs.map +1 -1
  24. package/fesm2015/c8y-ngx-components-widgets-implementations-image.mjs +1 -1
  25. package/fesm2015/c8y-ngx-components-widgets-implementations-image.mjs.map +1 -1
  26. package/fesm2015/c8y-ngx-components-widgets-implementations-markdown.mjs +1 -1
  27. package/fesm2015/c8y-ngx-components-widgets-implementations-markdown.mjs.map +1 -1
  28. package/fesm2015/c8y-ngx-components.mjs +27 -11
  29. package/fesm2015/c8y-ngx-components.mjs.map +1 -1
  30. package/fesm2020/c8y-ngx-components-ecosystem-shared.mjs +2 -2
  31. package/fesm2020/c8y-ngx-components-ecosystem-shared.mjs.map +1 -1
  32. package/fesm2020/c8y-ngx-components-protocol-opcua.mjs +1 -1
  33. package/fesm2020/c8y-ngx-components-protocol-opcua.mjs.map +1 -1
  34. package/fesm2020/c8y-ngx-components-sub-assets.mjs +2 -2
  35. package/fesm2020/c8y-ngx-components-sub-assets.mjs.map +1 -1
  36. package/fesm2020/c8y-ngx-components-trusted-certificates.mjs +1 -1
  37. package/fesm2020/c8y-ngx-components-trusted-certificates.mjs.map +1 -1
  38. package/fesm2020/c8y-ngx-components-widgets-implementations-image.mjs +1 -1
  39. package/fesm2020/c8y-ngx-components-widgets-implementations-image.mjs.map +1 -1
  40. package/fesm2020/c8y-ngx-components-widgets-implementations-markdown.mjs +1 -1
  41. package/fesm2020/c8y-ngx-components-widgets-implementations-markdown.mjs.map +1 -1
  42. package/fesm2020/c8y-ngx-components.mjs +34 -11
  43. package/fesm2020/c8y-ngx-components.mjs.map +1 -1
  44. package/package.json +1 -1
@@ -38,7 +38,7 @@ export declare class FilesService {
38
38
  * @param files Files to check.
39
39
  * @returns Returns true if each file has the correct size.
40
40
  */
41
- haveValidSizes(files: FileList): Promise<boolean>;
41
+ haveValidSizes(files: FileList, maxFileSizeInBytes?: number): Promise<boolean>;
42
42
  /**
43
43
  * Checks the system file size limit, if not available returns the default value.
44
44
  * Default limit: [DEFAULT_BYTES_LIMIT]{@link DEFAULT_BYTES_LIMIT}
@@ -42,6 +42,7 @@ export declare class DropAreaComponent implements OnInit, ControlValueAccessor {
42
42
  dropped: EventEmitter<DroppedFile[]>;
43
43
  maxAllowedFiles: number;
44
44
  files: FileList;
45
+ maxFileSizeInMegaBytes: number;
45
46
  /** Specifies a filter for what file types the user can pick from the file input dialog box.
46
47
  * ## Example:
47
48
  *
@@ -122,6 +123,7 @@ export declare class DropAreaComponent implements OnInit, ControlValueAccessor {
122
123
  private onFileInvalidNameLength;
123
124
  private onFileInvalidType;
124
125
  private onFileInvalidSize;
126
+ private convertMegaBytesToBytes;
125
127
  private getFilesNamesAsString;
126
128
  private isFilesArrayEmpty;
127
129
  private isTooManyFiles;
@@ -134,7 +136,7 @@ export declare class DropAreaComponent implements OnInit, ControlValueAccessor {
134
136
  private read;
135
137
  private onLoad;
136
138
  static ɵfac: i0.ɵɵFactoryDeclaration<DropAreaComponent, never>;
137
- static ɵcmp: i0.ɵɵComponentDeclaration<DropAreaComponent, "c8y-drop-area", never, { "formControl": "formControl"; "title": "title"; "message": "message"; "icon": "icon"; "loadingMessage": "loadingMessage"; "forceHideList": "forceHideList"; "alwaysShow": "alwaysShow"; "clickToOpen": "clickToOpen"; "loading": "loading"; "progress": "progress"; "maxAllowedFiles": "maxAllowedFiles"; "files": "files"; "accept": "accept"; }, { "dropped": "dropped"; }, never, ["*"], false>;
139
+ static ɵcmp: i0.ɵɵComponentDeclaration<DropAreaComponent, "c8y-drop-area", never, { "formControl": "formControl"; "title": "title"; "message": "message"; "icon": "icon"; "loadingMessage": "loadingMessage"; "forceHideList": "forceHideList"; "alwaysShow": "alwaysShow"; "clickToOpen": "clickToOpen"; "loading": "loading"; "progress": "progress"; "maxAllowedFiles": "maxAllowedFiles"; "files": "files"; "maxFileSizeInMegaBytes": "maxFileSizeInMegaBytes"; "accept": "accept"; }, { "dropped": "dropped"; }, never, ["*"], false>;
138
140
  }
139
141
  export interface DroppedFile {
140
142
  file: File;
@@ -26,9 +26,9 @@ export class BytesPipe {
26
26
  return 0;
27
27
  }
28
28
  const units = ['bytes', 'kB', 'MB', 'GB', 'TB', 'PB'];
29
- const num = Math.floor(Math.log(bytes) / Math.log(1000));
29
+ const num = Math.floor(Math.log(bytes) / Math.log(1024));
30
30
  // eslint-disable-next-line no-mixed-operators
31
- const result = (bytes / 1000 ** Math.floor(num)).toFixed(precision);
31
+ const result = (bytes / 1024 ** Math.floor(num)).toFixed(precision);
32
32
  return `${result} ${units[num]}`;
33
33
  }
34
34
  transform(value, precision = 1) {
@@ -41,4 +41,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImpor
41
41
  type: Pipe,
42
42
  args: [{ name: 'bytes' }]
43
43
  }] });
44
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnl0ZXMucGlwZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL2NvcmUvY29tbW9uL2J5dGVzLnBpcGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLElBQUksRUFBaUIsTUFBTSxlQUFlLENBQUM7QUFDcEQsT0FBTyxFQUFFLEtBQUssRUFBRSxRQUFRLEVBQUUsTUFBTSxXQUFXLENBQUM7O0FBRTVDOzs7Ozs7Ozs7Ozs7Ozs7R0FlRztBQUVILE1BQU0sT0FBTyxTQUFTO0lBQ3BCLE1BQU0sQ0FBQyxLQUFLLENBQUMsS0FBYSxFQUFFLFNBQWlCO1FBQzNDLElBQUksS0FBSyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLEtBQUssQ0FBQyxFQUFFO1lBQ3BDLE9BQU8sR0FBRyxDQUFDO1NBQ1o7UUFDRCxJQUFJLEtBQUssS0FBSyxDQUFDLEVBQUU7WUFDZixPQUFPLENBQUMsQ0FBQztTQUNWO1FBQ0QsTUFBTSxLQUFLLEdBQUcsQ0FBQyxPQUFPLEVBQUUsSUFBSSxFQUFFLElBQUksRUFBRSxJQUFJLEVBQUUsSUFBSSxFQUFFLElBQUksQ0FBQyxDQUFDO1FBQ3RELE1BQU0sR0FBRyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUM7UUFDekQsOENBQThDO1FBQzlDLE1BQU0sTUFBTSxHQUFHLENBQUMsS0FBSyxHQUFHLElBQUksSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLFNBQVMsQ0FBQyxDQUFDO1FBRXBFLE9BQU8sR0FBRyxNQUFNLElBQUksS0FBSyxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUM7SUFDbkMsQ0FBQztJQUVELFNBQVMsQ0FBQyxLQUFhLEVBQUUsU0FBUyxHQUFHLENBQUM7UUFDcEMsT0FBTyxTQUFTLENBQUMsS0FBSyxDQUFDLEtBQUssRUFBRSxTQUFTLENBQUMsQ0FBQztJQUMzQyxDQUFDOztzR0FsQlUsU0FBUztvR0FBVCxTQUFTOzJGQUFULFNBQVM7a0JBRHJCLElBQUk7bUJBQUMsRUFBRSxJQUFJLEVBQUUsT0FBTyxFQUFFIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUGlwZSwgUGlwZVRyYW5zZm9ybSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgaXNOYU4sIGlzRmluaXRlIH0gZnJvbSAnbG9kYXNoLWVzJztcblxuLypcbiAqIENvbnZlcnRzIHRoZSBmaWxlIHNpemUgdG8gYSByZWFkYWJsZSBmb3JtYXQuXG4gKiBVc2FnZTpcbiAqXG4gKiBgYGBodG1sXG4gKiAge3sgbXlEYXRhIHwgYnl0ZXMgfX0gPCEtLSBlLmcuIDEwMC4xIE1CIC0tPlxuICpcbiAqIGBgYFxuICogRGVmYXVsdCBwcmVjaXNpb24gaXMgc2V0IHRvIDEgZGVjaW1hbCBwbGFjZS5cbiAqIFRvIGNoYW5nZSB0aGUgcHJlY2lzaW9uLCBwYXNzIGFuIGFkZGl0aW9uYWwgYXJndW1lbnQgdG8gcGlwZS5cblxuICogYGBgaHRtbFxuICoge3sgbXlEYXRhIHwgYnl0ZXM6IDUgfX0gPCEtLSBlLmcuIDEwMC4xMjM0NSBNQiAtLT5cbiAqIGBgYFxuICpcbiAqL1xuQFBpcGUoeyBuYW1lOiAnYnl0ZXMnIH0pXG5leHBvcnQgY2xhc3MgQnl0ZXNQaXBlIGltcGxlbWVudHMgUGlwZVRyYW5zZm9ybSB7XG4gIHN0YXRpYyBieXRlcyhieXRlczogbnVtYmVyLCBwcmVjaXNpb246IG51bWJlcik6IHN0cmluZyB8IG51bWJlciB7XG4gICAgaWYgKGlzTmFOKGJ5dGVzKSB8fCAhaXNGaW5pdGUoYnl0ZXMpKSB7XG4gICAgICByZXR1cm4gJy0nO1xuICAgIH1cbiAgICBpZiAoYnl0ZXMgPT09IDApIHtcbiAgICAgIHJldHVybiAwO1xuICAgIH1cbiAgICBjb25zdCB1bml0cyA9IFsnYnl0ZXMnLCAna0InLCAnTUInLCAnR0InLCAnVEInLCAnUEInXTtcbiAgICBjb25zdCBudW0gPSBNYXRoLmZsb29yKE1hdGgubG9nKGJ5dGVzKSAvIE1hdGgubG9nKDEwMDApKTtcbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgbm8tbWl4ZWQtb3BlcmF0b3JzXG4gICAgY29uc3QgcmVzdWx0ID0gKGJ5dGVzIC8gMTAwMCAqKiBNYXRoLmZsb29yKG51bSkpLnRvRml4ZWQocHJlY2lzaW9uKTtcblxuICAgIHJldHVybiBgJHtyZXN1bHR9ICR7dW5pdHNbbnVtXX1gO1xuICB9XG5cbiAgdHJhbnNmb3JtKHZhbHVlOiBudW1iZXIsIHByZWNpc2lvbiA9IDEpOiBzdHJpbmcgfCBudW1iZXIge1xuICAgIHJldHVybiBCeXRlc1BpcGUuYnl0ZXModmFsdWUsIHByZWNpc2lvbik7XG4gIH1cbn1cbiJdfQ==
44
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnl0ZXMucGlwZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL2NvcmUvY29tbW9uL2J5dGVzLnBpcGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLElBQUksRUFBaUIsTUFBTSxlQUFlLENBQUM7QUFDcEQsT0FBTyxFQUFFLEtBQUssRUFBRSxRQUFRLEVBQUUsTUFBTSxXQUFXLENBQUM7O0FBRTVDOzs7Ozs7Ozs7Ozs7Ozs7R0FlRztBQUVILE1BQU0sT0FBTyxTQUFTO0lBQ3BCLE1BQU0sQ0FBQyxLQUFLLENBQUMsS0FBYSxFQUFFLFNBQWlCO1FBQzNDLElBQUksS0FBSyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLEtBQUssQ0FBQyxFQUFFO1lBQ3BDLE9BQU8sR0FBRyxDQUFDO1NBQ1o7UUFDRCxJQUFJLEtBQUssS0FBSyxDQUFDLEVBQUU7WUFDZixPQUFPLENBQUMsQ0FBQztTQUNWO1FBQ0QsTUFBTSxLQUFLLEdBQUcsQ0FBQyxPQUFPLEVBQUUsSUFBSSxFQUFFLElBQUksRUFBRSxJQUFJLEVBQUUsSUFBSSxFQUFFLElBQUksQ0FBQyxDQUFDO1FBQ3RELE1BQU0sR0FBRyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUM7UUFDekQsOENBQThDO1FBQzlDLE1BQU0sTUFBTSxHQUFHLENBQUMsS0FBSyxHQUFHLElBQUksSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLFNBQVMsQ0FBQyxDQUFDO1FBRXBFLE9BQU8sR0FBRyxNQUFNLElBQUksS0FBSyxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUM7SUFDbkMsQ0FBQztJQUVELFNBQVMsQ0FBQyxLQUFhLEVBQUUsU0FBUyxHQUFHLENBQUM7UUFDcEMsT0FBTyxTQUFTLENBQUMsS0FBSyxDQUFDLEtBQUssRUFBRSxTQUFTLENBQUMsQ0FBQztJQUMzQyxDQUFDOztzR0FsQlUsU0FBUztvR0FBVCxTQUFTOzJGQUFULFNBQVM7a0JBRHJCLElBQUk7bUJBQUMsRUFBRSxJQUFJLEVBQUUsT0FBTyxFQUFFIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUGlwZSwgUGlwZVRyYW5zZm9ybSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgaXNOYU4sIGlzRmluaXRlIH0gZnJvbSAnbG9kYXNoLWVzJztcblxuLypcbiAqIENvbnZlcnRzIHRoZSBmaWxlIHNpemUgdG8gYSByZWFkYWJsZSBmb3JtYXQuXG4gKiBVc2FnZTpcbiAqXG4gKiBgYGBodG1sXG4gKiAge3sgbXlEYXRhIHwgYnl0ZXMgfX0gPCEtLSBlLmcuIDEwMC4xIE1CIC0tPlxuICpcbiAqIGBgYFxuICogRGVmYXVsdCBwcmVjaXNpb24gaXMgc2V0IHRvIDEgZGVjaW1hbCBwbGFjZS5cbiAqIFRvIGNoYW5nZSB0aGUgcHJlY2lzaW9uLCBwYXNzIGFuIGFkZGl0aW9uYWwgYXJndW1lbnQgdG8gcGlwZS5cblxuICogYGBgaHRtbFxuICoge3sgbXlEYXRhIHwgYnl0ZXM6IDUgfX0gPCEtLSBlLmcuIDEwMC4xMjM0NSBNQiAtLT5cbiAqIGBgYFxuICpcbiAqL1xuQFBpcGUoeyBuYW1lOiAnYnl0ZXMnIH0pXG5leHBvcnQgY2xhc3MgQnl0ZXNQaXBlIGltcGxlbWVudHMgUGlwZVRyYW5zZm9ybSB7XG4gIHN0YXRpYyBieXRlcyhieXRlczogbnVtYmVyLCBwcmVjaXNpb246IG51bWJlcik6IHN0cmluZyB8IG51bWJlciB7XG4gICAgaWYgKGlzTmFOKGJ5dGVzKSB8fCAhaXNGaW5pdGUoYnl0ZXMpKSB7XG4gICAgICByZXR1cm4gJy0nO1xuICAgIH1cbiAgICBpZiAoYnl0ZXMgPT09IDApIHtcbiAgICAgIHJldHVybiAwO1xuICAgIH1cbiAgICBjb25zdCB1bml0cyA9IFsnYnl0ZXMnLCAna0InLCAnTUInLCAnR0InLCAnVEInLCAnUEInXTtcbiAgICBjb25zdCBudW0gPSBNYXRoLmZsb29yKE1hdGgubG9nKGJ5dGVzKSAvIE1hdGgubG9nKDEwMjQpKTtcbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgbm8tbWl4ZWQtb3BlcmF0b3JzXG4gICAgY29uc3QgcmVzdWx0ID0gKGJ5dGVzIC8gMTAyNCAqKiBNYXRoLmZsb29yKG51bSkpLnRvRml4ZWQocHJlY2lzaW9uKTtcblxuICAgIHJldHVybiBgJHtyZXN1bHR9ICR7dW5pdHNbbnVtXX1gO1xuICB9XG5cbiAgdHJhbnNmb3JtKHZhbHVlOiBudW1iZXIsIHByZWNpc2lvbiA9IDEpOiBzdHJpbmcgfCBudW1iZXIge1xuICAgIHJldHVybiBCeXRlc1BpcGUuYnl0ZXModmFsdWUsIHByZWNpc2lvbik7XG4gIH1cbn1cbiJdfQ==
@@ -1,6 +1,6 @@
1
1
  import { Injectable } from '@angular/core';
2
2
  import { InventoryBinaryService, SystemOptionsService } from '@c8y/client';
3
- import { every, first, flatten, get, isNaN, isUndefined, keys, map, uniq } from 'lodash-es';
3
+ import { every, first, flatten, get, isNaN, isUndefined, keys, map, uniq, min } from 'lodash-es';
4
4
  import { saveAs } from 'file-saver';
5
5
  import { Observable, of, throwError } from 'rxjs';
6
6
  import { catchError, switchMap, startWith, share } from 'rxjs/operators';
@@ -75,8 +75,8 @@ export class FilesService {
75
75
  * @param files Files to check.
76
76
  * @returns Returns true if each file has the correct size.
77
77
  */
78
- async haveValidSizes(files) {
79
- const limit = await this.loadBytesSizeLimit();
78
+ async haveValidSizes(files, maxFileSizeInBytes) {
79
+ const limit = min([maxFileSizeInBytes, await this.loadBytesSizeLimit()]);
80
80
  return every(files, (f) => {
81
81
  return this.size(f) <= limit;
82
82
  });
@@ -327,4 +327,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImpor
327
327
  type: Injectable,
328
328
  args: [{ providedIn: 'root' }]
329
329
  }], ctorParameters: function () { return [{ type: i1.SystemOptionsService }, { type: i1.InventoryBinaryService }]; } });
330
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"files.service.js","sourceRoot":"","sources":["../../../../core/common/files.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport {\n  IFetchResponse,\n  IManagedObjectBinary,\n  InventoryBinaryService,\n  SystemOptionsService\n} from '@c8y/client';\nimport { every, first, flatten, get, isNaN, isUndefined, keys, map, uniq } from 'lodash-es';\nimport { saveAs } from 'file-saver';\nimport { Observable, of, throwError } from 'rxjs';\nimport { catchError, switchMap, startWith, share } from 'rxjs/operators';\n\nexport interface IFetchWithProgress {\n  totalBytes: number;\n  bufferedBytes: number;\n  percentage: number;\n  bytesPerSecond: number;\n  blob?: Blob;\n}\n\nexport enum GENERIC_FILE_TYPE {\n  ARCHIVE = 'archive',\n  AUDIO = 'audio',\n  CODE = 'code',\n  EXCEL = 'excel',\n  IMAGE = 'image',\n  PDF = 'pdf',\n  POWERPOINT = 'powerpoint',\n  TEXT = 'text',\n  VIDEO = 'video',\n  WORD = 'word',\n  EPL = 'epl'\n}\n@Injectable({ providedIn: 'root' })\nexport class FilesService {\n  readonly DEFAULT_BYTES_LIMIT = 52428800;\n  readonly FILENAME_MAX_LENGTH = 128;\n\n  fileTypeExtensionsMap: { [key: string]: { exts: string[] } } = {\n    [GENERIC_FILE_TYPE.ARCHIVE]: {\n      exts: ['7z', 'apk', 'cab', 'gz', 'iso', 'jar', 'rar', 'tar', 'zip']\n    },\n    [GENERIC_FILE_TYPE.AUDIO]: {\n      exts: ['3gp', 'aiff', 'aac', 'amr', 'm4a', 'm4p', 'mp3', 'oga', 'ogg', 'raw', 'wav', 'wma']\n    },\n    [GENERIC_FILE_TYPE.CODE]: {\n      exts: ['aspx', 'exe', 'htm', 'html', 'jad', 'js', 'json', 'jsp', 'php', 'xml']\n    },\n    [GENERIC_FILE_TYPE.EXCEL]: {\n      exts: ['xls', 'xlsx']\n    },\n    [GENERIC_FILE_TYPE.IMAGE]: {\n      exts: ['bmp', 'gif', 'jpeg', 'jpg', 'png', 'tiff', 'svg', 'ico', 'apng', 'webp']\n    },\n    [GENERIC_FILE_TYPE.PDF]: {\n      exts: ['pdf']\n    },\n    [GENERIC_FILE_TYPE.POWERPOINT]: {\n      exts: ['ppt', 'pptx']\n    },\n    [GENERIC_FILE_TYPE.TEXT]: {\n      exts: ['txt']\n    },\n    [GENERIC_FILE_TYPE.VIDEO]: {\n      exts: ['asf', 'avi', 'flv', 'mov', 'mp4', 'ogv', 'qt', 'rm', 'rmvb', 'wmv', '3gp']\n    },\n    [GENERIC_FILE_TYPE.WORD]: {\n      exts: ['doc', 'docx']\n    },\n    [GENERIC_FILE_TYPE.EPL]: {\n      exts: ['mon']\n    }\n  };\n\n  private fileSizeLimitCfg = {\n    systemOption: {\n      category: 'files',\n      key: 'max.size'\n    },\n    defaultBytesLimit: this.DEFAULT_BYTES_LIMIT,\n    actualBytesLimit: undefined\n  };\n\n  constructor(\n    private systemOptionsService: SystemOptionsService,\n    private inventoryBinaryService: InventoryBinaryService\n  ) {}\n\n  /**\n   * Checks if files have valid size.\n   * @param files Files to check.\n   * @returns Returns true if each file has the correct size.\n   */\n  async haveValidSizes(files: FileList): Promise<boolean> {\n    const limit = await this.loadBytesSizeLimit();\n\n    return every(files, (f: File) => {\n      return this.size(f) <= limit;\n    });\n  }\n\n  /**\n   * Checks the system file size limit, if not available returns the default value.\n   * Default limit: [DEFAULT_BYTES_LIMIT]{@link DEFAULT_BYTES_LIMIT}\n   * @returns Returns promise with the limit value.\n   */\n  async loadBytesSizeLimit(): Promise<number> {\n    let bytesLimit = this.DEFAULT_BYTES_LIMIT;\n    if (this.fileSizeLimitCfg.actualBytesLimit) {\n      return this.fileSizeLimitCfg.actualBytesLimit;\n    }\n    const { systemOption } = this.fileSizeLimitCfg;\n\n    try {\n      const {\n        data: { value: actualBytesLimit }\n      } = await this.systemOptionsService.detail(systemOption);\n\n      if (!actualBytesLimit) {\n        return bytesLimit;\n      }\n\n      const parsedActualBytesLimit = parseInt(actualBytesLimit, 10);\n      if (isNaN(parsedActualBytesLimit)) {\n        return bytesLimit;\n      }\n\n      this.fileSizeLimitCfg.actualBytesLimit = parsedActualBytesLimit;\n      bytesLimit = parsedActualBytesLimit;\n    } catch (error) {\n      // do nothing\n    }\n    return bytesLimit;\n  }\n\n  /**\n   * Checks the size of the file\n   * @param file File to check.\n   * @returns Returns size of the file in bytes.\n   */\n  size(file: File | IManagedObjectBinary): number {\n    const fileLength = get(file, 'length') || get(file, 'size');\n    const attachments = get(file, '_attachments');\n    const attachmentsObj = get(attachments, first(keys(attachments)));\n    return isUndefined(fileLength) ? get(attachmentsObj, 'length') : fileLength;\n  }\n\n  /**\n   * Checks whether files have allowed extensions.\n   * If the accept parameter is not specified, all extensions are accepted.\n   * @param files Files to check.\n   * @param accept String of comma separated file extensions and generic types ([GENERIC_FILE_TYPE]{@link GENERIC_FILE_TYPE}), e.g. .zip,.7z,excel.\n   * @returns  Returns true if each file has allowed extension.\n   */\n  haveValidExtensions(files: FileList | File | File[], accept: string): boolean {\n    if (!accept) {\n      return true;\n    }\n    const filesArray = (files as FileList).item\n      ? Array.from(files as FileList)\n      : Array.isArray(files)\n      ? files\n      : [files];\n\n    const filesExts = filesArray.map((file: File) => this.getFileExtension(file)?.toLowerCase());\n    const allowedExts = this.extractFileExtensions(accept);\n    return filesExts.every(ext => allowedExts.includes(ext));\n  }\n\n  /**\n   * Checks if each file has a valid filename length.\n   * @param files Files to check.\n   * @returns Returns true if each file has a valid filename length.\n   */\n  checkMaxLength(files: FileList): boolean {\n    return every(files, (f: File) => {\n      return this.FILENAME_MAX_LENGTH > f.name.length;\n    });\n  }\n\n  /**\n   * Extracts the file extension.\n   * @param file File from which the extension should be extracted.\n   * @returns Returns the file extension or undefined if the file has no extension.\n   */\n  getFileExtension(file: File): string | undefined {\n    const fileNameAndFileExt = file.name.split('.');\n    if (fileNameAndFileExt.length === 1) {\n      // no file ext\n      return undefined;\n    }\n    return fileNameAndFileExt.pop();\n  }\n\n  /**\n   * List of file extensions.\n   * @returns Returns list of file extensions.\n   */\n  getFileExtensions(): string[] {\n    return uniq(flatten(map(this.fileTypeExtensionsMap, ({ exts }) => exts)));\n  }\n\n  /**\n   * The list of generic file types.\n   * @returns Returns the list of generic file types.\n   */\n  getGenericFileTypes(): GENERIC_FILE_TYPE[] {\n    return Object.keys(this.fileTypeExtensionsMap) as GENERIC_FILE_TYPE[];\n  }\n\n  /**\n   * @ignore\n   */\n  mapGenericFileTypesToExtensions(genericFileTypes: GENERIC_FILE_TYPE[] = []): string[] {\n    const fileExts = genericFileTypes.map(gT => {\n      const { exts } = this.fileTypeExtensionsMap[gT];\n      return exts;\n    });\n\n    return uniq(flatten(fileExts));\n  }\n  /**\n   * Extracts a list of file extensions from a string.\n   * Can accept generic file types check: [GENERIC_FILE_TYPE]{@link GENERIC_FILE_TYPE}.\n   *\n   * @param str String from which the file extensions are extracted (comma separated values).\n   * Accepted string format:\n   * * \".zip,.iso\",\n   * * \"zip,ISO\",\n   * * \"archive\".\n   * Important: generic types cannot contain a dot. All values with a dot are treated as a normal extension.\n   * @returns Returns a list of the file extensions.\n   */\n  extractFileExtensions(str: string): string[] {\n    if (!str) {\n      return [];\n    }\n    const types = str.split(',').map(t => t.toLowerCase().trim());\n\n    const genericTypes = types.filter((t: string) => this.isGenericType(t));\n    const defaultFileExts = types.filter((t: string) => !this.isGenericType(t));\n\n    const allTypes = [\n      ...this.mapGenericFileTypesToExtensions(genericTypes as GENERIC_FILE_TYPE[]),\n      ...defaultFileExts\n    ].map(t => t.replace('.', ''));\n\n    return uniq(allTypes);\n  }\n\n  /**\n   * Converts a file to a base64 image string.\n   *\n   * @param file The file to convert to base 64.\n   * @returns The image string in base64 format.\n   */\n  toBase64(file: File): Promise<string> {\n    return new Promise((resolve, reject) => {\n      const reader = new FileReader();\n      reader.readAsDataURL(file);\n      reader.onload = () => resolve(String(reader.result));\n      reader.onerror = error => reject(error);\n    });\n  }\n\n  /**\n   * Allows to get a File representation of an managed object binary. Can be used\n   * to convert this file toBase64 to show it to the end-user.\n   * @param binary The binary managed object\n   * @returns The file representation.\n   */\n  async getFile(binary: IManagedObjectBinary): Promise<File> {\n    const res = await this.inventoryBinaryService.download(binary.id);\n    const arrayBuffer = await res.arrayBuffer();\n    return new File([arrayBuffer], binary.name, { type: binary.contentType });\n  }\n\n  /**\n   * Allows to calculate the hash sum of the provided file.\n   * @param file The file to hash.\n   * @returns The SHA-256 hash of the file.\n   */\n  async getHashSumOfFile(file: File): Promise<string> {\n    const buffer = await file.arrayBuffer();\n    const hashBuffer = await crypto.subtle.digest('SHA-256', buffer);\n    const hashArray = Array.from(new Uint8Array(hashBuffer));\n    const hashHex = hashArray.map(bytes => bytes.toString(16).padStart(2, '0')).join('');\n    return hashHex;\n  }\n\n  /**\n   * Allows to download a file (opens the browser download prompt).\n   * @param binary The binary managed object.\n   */\n  async download(binary: IManagedObjectBinary) {\n    const file = await this.getFile(binary);\n    saveAs(file);\n  }\n\n  /**\n   * Loads the file to JavaScript memory.\n   * Returns an observable that emits progression status object,\n   * and after download is completed, blob property is populated with Blob result object.\n   * Unsubscribing from the returned observable aborts the file fetch request.\n   *\n   * @param binary The binary managed object.\n   */\n  fetchFileWithProgress$(binary: IManagedObjectBinary): Observable<IFetchWithProgress> {\n    const progress: IFetchWithProgress = {\n      totalBytes: +binary.length,\n      bufferedBytes: 0,\n      percentage: 0,\n      bytesPerSecond: 0\n    };\n\n    return of(new AbortController()).pipe(\n      switchMap(async abortController => ({\n        abortController,\n        startTimestamp: new Date().getTime(),\n        response: await this.inventoryBinaryService.download(binary.id, {\n          signal: abortController.signal\n        })\n      })),\n      switchMap(({ response, abortController, startTimestamp }) =>\n        this.processResponse$(abortController, response, startTimestamp, progress, binary.type)\n      ),\n      startWith(progress),\n      share(),\n      catchError(err => throwError(err))\n    );\n  }\n\n  private processResponse$(\n    abortController: AbortController,\n    response: IFetchResponse,\n    startTimestamp: number,\n    progress: IFetchWithProgress,\n    binaryType: string\n  ): Observable<IFetchWithProgress> {\n    return new Observable(sub => {\n      const reader = response.body.getReader();\n      const chunks = [];\n\n      const readStream = async () => {\n        try {\n          while (true) {\n            const { done, value } = await reader.read();\n\n            if (done) {\n              progress.blob = new Blob(chunks, { type: binaryType });\n              sub.next({ ...progress });\n              sub.complete();\n              break;\n            }\n\n            chunks.push(value);\n\n            progress.bufferedBytes += value.length;\n            const currentTimestamp = new Date().getTime();\n            const timestampDiff = currentTimestamp - startTimestamp;\n            progress.bytesPerSecond = Math.round(\n              progress.bufferedBytes / Math.round(timestampDiff / 1000)\n            );\n            progress.percentage = Math.round((progress.bufferedBytes / progress.totalBytes) * 100);\n            sub.next({ ...progress });\n          }\n        } catch (e) {\n          abortController.abort();\n          sub.error(e);\n        }\n      };\n      readStream();\n\n      return {\n        unsubscribe() {\n          abortController.abort();\n          sub.complete();\n        }\n      };\n    });\n  }\n\n  private isGenericType(type: string) {\n    return Object.values(GENERIC_FILE_TYPE).includes(type as GENERIC_FILE_TYPE);\n  }\n}\n"]}
330
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"files.service.js","sourceRoot":"","sources":["../../../../core/common/files.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport {\n  IFetchResponse,\n  IManagedObjectBinary,\n  InventoryBinaryService,\n  SystemOptionsService\n} from '@c8y/client';\nimport { every, first, flatten, get, isNaN, isUndefined, keys, map, uniq, min } from 'lodash-es';\nimport { saveAs } from 'file-saver';\nimport { Observable, of, throwError } from 'rxjs';\nimport { catchError, switchMap, startWith, share } from 'rxjs/operators';\n\nexport interface IFetchWithProgress {\n  totalBytes: number;\n  bufferedBytes: number;\n  percentage: number;\n  bytesPerSecond: number;\n  blob?: Blob;\n}\n\nexport enum GENERIC_FILE_TYPE {\n  ARCHIVE = 'archive',\n  AUDIO = 'audio',\n  CODE = 'code',\n  EXCEL = 'excel',\n  IMAGE = 'image',\n  PDF = 'pdf',\n  POWERPOINT = 'powerpoint',\n  TEXT = 'text',\n  VIDEO = 'video',\n  WORD = 'word',\n  EPL = 'epl'\n}\n@Injectable({ providedIn: 'root' })\nexport class FilesService {\n  readonly DEFAULT_BYTES_LIMIT = 52428800;\n  readonly FILENAME_MAX_LENGTH = 128;\n\n  fileTypeExtensionsMap: { [key: string]: { exts: string[] } } = {\n    [GENERIC_FILE_TYPE.ARCHIVE]: {\n      exts: ['7z', 'apk', 'cab', 'gz', 'iso', 'jar', 'rar', 'tar', 'zip']\n    },\n    [GENERIC_FILE_TYPE.AUDIO]: {\n      exts: ['3gp', 'aiff', 'aac', 'amr', 'm4a', 'm4p', 'mp3', 'oga', 'ogg', 'raw', 'wav', 'wma']\n    },\n    [GENERIC_FILE_TYPE.CODE]: {\n      exts: ['aspx', 'exe', 'htm', 'html', 'jad', 'js', 'json', 'jsp', 'php', 'xml']\n    },\n    [GENERIC_FILE_TYPE.EXCEL]: {\n      exts: ['xls', 'xlsx']\n    },\n    [GENERIC_FILE_TYPE.IMAGE]: {\n      exts: ['bmp', 'gif', 'jpeg', 'jpg', 'png', 'tiff', 'svg', 'ico', 'apng', 'webp']\n    },\n    [GENERIC_FILE_TYPE.PDF]: {\n      exts: ['pdf']\n    },\n    [GENERIC_FILE_TYPE.POWERPOINT]: {\n      exts: ['ppt', 'pptx']\n    },\n    [GENERIC_FILE_TYPE.TEXT]: {\n      exts: ['txt']\n    },\n    [GENERIC_FILE_TYPE.VIDEO]: {\n      exts: ['asf', 'avi', 'flv', 'mov', 'mp4', 'ogv', 'qt', 'rm', 'rmvb', 'wmv', '3gp']\n    },\n    [GENERIC_FILE_TYPE.WORD]: {\n      exts: ['doc', 'docx']\n    },\n    [GENERIC_FILE_TYPE.EPL]: {\n      exts: ['mon']\n    }\n  };\n\n  private fileSizeLimitCfg = {\n    systemOption: {\n      category: 'files',\n      key: 'max.size'\n    },\n    defaultBytesLimit: this.DEFAULT_BYTES_LIMIT,\n    actualBytesLimit: undefined\n  };\n\n  constructor(\n    private systemOptionsService: SystemOptionsService,\n    private inventoryBinaryService: InventoryBinaryService\n  ) {}\n\n  /**\n   * Checks if files have valid size.\n   * @param files Files to check.\n   * @returns Returns true if each file has the correct size.\n   */\n  async haveValidSizes(files: FileList, maxFileSizeInBytes?: number): Promise<boolean> {\n    const limit = min([maxFileSizeInBytes, await this.loadBytesSizeLimit()]);\n    return every(files, (f: File) => {\n      return this.size(f) <= limit;\n    });\n  }\n\n  /**\n   * Checks the system file size limit, if not available returns the default value.\n   * Default limit: [DEFAULT_BYTES_LIMIT]{@link DEFAULT_BYTES_LIMIT}\n   * @returns Returns promise with the limit value.\n   */\n  async loadBytesSizeLimit(): Promise<number> {\n    let bytesLimit = this.DEFAULT_BYTES_LIMIT;\n    if (this.fileSizeLimitCfg.actualBytesLimit) {\n      return this.fileSizeLimitCfg.actualBytesLimit;\n    }\n    const { systemOption } = this.fileSizeLimitCfg;\n\n    try {\n      const {\n        data: { value: actualBytesLimit }\n      } = await this.systemOptionsService.detail(systemOption);\n\n      if (!actualBytesLimit) {\n        return bytesLimit;\n      }\n\n      const parsedActualBytesLimit = parseInt(actualBytesLimit, 10);\n      if (isNaN(parsedActualBytesLimit)) {\n        return bytesLimit;\n      }\n\n      this.fileSizeLimitCfg.actualBytesLimit = parsedActualBytesLimit;\n      bytesLimit = parsedActualBytesLimit;\n    } catch (error) {\n      // do nothing\n    }\n    return bytesLimit;\n  }\n\n  /**\n   * Checks the size of the file\n   * @param file File to check.\n   * @returns Returns size of the file in bytes.\n   */\n  size(file: File | IManagedObjectBinary): number {\n    const fileLength = get(file, 'length') || get(file, 'size');\n    const attachments = get(file, '_attachments');\n    const attachmentsObj = get(attachments, first(keys(attachments)));\n    return isUndefined(fileLength) ? get(attachmentsObj, 'length') : fileLength;\n  }\n\n  /**\n   * Checks whether files have allowed extensions.\n   * If the accept parameter is not specified, all extensions are accepted.\n   * @param files Files to check.\n   * @param accept String of comma separated file extensions and generic types ([GENERIC_FILE_TYPE]{@link GENERIC_FILE_TYPE}), e.g. .zip,.7z,excel.\n   * @returns  Returns true if each file has allowed extension.\n   */\n  haveValidExtensions(files: FileList | File | File[], accept: string): boolean {\n    if (!accept) {\n      return true;\n    }\n    const filesArray = (files as FileList).item\n      ? Array.from(files as FileList)\n      : Array.isArray(files)\n      ? files\n      : [files];\n\n    const filesExts = filesArray.map((file: File) => this.getFileExtension(file)?.toLowerCase());\n    const allowedExts = this.extractFileExtensions(accept);\n    return filesExts.every(ext => allowedExts.includes(ext));\n  }\n\n  /**\n   * Checks if each file has a valid filename length.\n   * @param files Files to check.\n   * @returns Returns true if each file has a valid filename length.\n   */\n  checkMaxLength(files: FileList): boolean {\n    return every(files, (f: File) => {\n      return this.FILENAME_MAX_LENGTH > f.name.length;\n    });\n  }\n\n  /**\n   * Extracts the file extension.\n   * @param file File from which the extension should be extracted.\n   * @returns Returns the file extension or undefined if the file has no extension.\n   */\n  getFileExtension(file: File): string | undefined {\n    const fileNameAndFileExt = file.name.split('.');\n    if (fileNameAndFileExt.length === 1) {\n      // no file ext\n      return undefined;\n    }\n    return fileNameAndFileExt.pop();\n  }\n\n  /**\n   * List of file extensions.\n   * @returns Returns list of file extensions.\n   */\n  getFileExtensions(): string[] {\n    return uniq(flatten(map(this.fileTypeExtensionsMap, ({ exts }) => exts)));\n  }\n\n  /**\n   * The list of generic file types.\n   * @returns Returns the list of generic file types.\n   */\n  getGenericFileTypes(): GENERIC_FILE_TYPE[] {\n    return Object.keys(this.fileTypeExtensionsMap) as GENERIC_FILE_TYPE[];\n  }\n\n  /**\n   * @ignore\n   */\n  mapGenericFileTypesToExtensions(genericFileTypes: GENERIC_FILE_TYPE[] = []): string[] {\n    const fileExts = genericFileTypes.map(gT => {\n      const { exts } = this.fileTypeExtensionsMap[gT];\n      return exts;\n    });\n\n    return uniq(flatten(fileExts));\n  }\n  /**\n   * Extracts a list of file extensions from a string.\n   * Can accept generic file types check: [GENERIC_FILE_TYPE]{@link GENERIC_FILE_TYPE}.\n   *\n   * @param str String from which the file extensions are extracted (comma separated values).\n   * Accepted string format:\n   * * \".zip,.iso\",\n   * * \"zip,ISO\",\n   * * \"archive\".\n   * Important: generic types cannot contain a dot. All values with a dot are treated as a normal extension.\n   * @returns Returns a list of the file extensions.\n   */\n  extractFileExtensions(str: string): string[] {\n    if (!str) {\n      return [];\n    }\n    const types = str.split(',').map(t => t.toLowerCase().trim());\n\n    const genericTypes = types.filter((t: string) => this.isGenericType(t));\n    const defaultFileExts = types.filter((t: string) => !this.isGenericType(t));\n\n    const allTypes = [\n      ...this.mapGenericFileTypesToExtensions(genericTypes as GENERIC_FILE_TYPE[]),\n      ...defaultFileExts\n    ].map(t => t.replace('.', ''));\n\n    return uniq(allTypes);\n  }\n\n  /**\n   * Converts a file to a base64 image string.\n   *\n   * @param file The file to convert to base 64.\n   * @returns The image string in base64 format.\n   */\n  toBase64(file: File): Promise<string> {\n    return new Promise((resolve, reject) => {\n      const reader = new FileReader();\n      reader.readAsDataURL(file);\n      reader.onload = () => resolve(String(reader.result));\n      reader.onerror = error => reject(error);\n    });\n  }\n\n  /**\n   * Allows to get a File representation of an managed object binary. Can be used\n   * to convert this file toBase64 to show it to the end-user.\n   * @param binary The binary managed object\n   * @returns The file representation.\n   */\n  async getFile(binary: IManagedObjectBinary): Promise<File> {\n    const res = await this.inventoryBinaryService.download(binary.id);\n    const arrayBuffer = await res.arrayBuffer();\n    return new File([arrayBuffer], binary.name, { type: binary.contentType });\n  }\n\n  /**\n   * Allows to calculate the hash sum of the provided file.\n   * @param file The file to hash.\n   * @returns The SHA-256 hash of the file.\n   */\n  async getHashSumOfFile(file: File): Promise<string> {\n    const buffer = await file.arrayBuffer();\n    const hashBuffer = await crypto.subtle.digest('SHA-256', buffer);\n    const hashArray = Array.from(new Uint8Array(hashBuffer));\n    const hashHex = hashArray.map(bytes => bytes.toString(16).padStart(2, '0')).join('');\n    return hashHex;\n  }\n\n  /**\n   * Allows to download a file (opens the browser download prompt).\n   * @param binary The binary managed object.\n   */\n  async download(binary: IManagedObjectBinary) {\n    const file = await this.getFile(binary);\n    saveAs(file);\n  }\n\n  /**\n   * Loads the file to JavaScript memory.\n   * Returns an observable that emits progression status object,\n   * and after download is completed, blob property is populated with Blob result object.\n   * Unsubscribing from the returned observable aborts the file fetch request.\n   *\n   * @param binary The binary managed object.\n   */\n  fetchFileWithProgress$(binary: IManagedObjectBinary): Observable<IFetchWithProgress> {\n    const progress: IFetchWithProgress = {\n      totalBytes: +binary.length,\n      bufferedBytes: 0,\n      percentage: 0,\n      bytesPerSecond: 0\n    };\n\n    return of(new AbortController()).pipe(\n      switchMap(async abortController => ({\n        abortController,\n        startTimestamp: new Date().getTime(),\n        response: await this.inventoryBinaryService.download(binary.id, {\n          signal: abortController.signal\n        })\n      })),\n      switchMap(({ response, abortController, startTimestamp }) =>\n        this.processResponse$(abortController, response, startTimestamp, progress, binary.type)\n      ),\n      startWith(progress),\n      share(),\n      catchError(err => throwError(err))\n    );\n  }\n\n  private processResponse$(\n    abortController: AbortController,\n    response: IFetchResponse,\n    startTimestamp: number,\n    progress: IFetchWithProgress,\n    binaryType: string\n  ): Observable<IFetchWithProgress> {\n    return new Observable(sub => {\n      const reader = response.body.getReader();\n      const chunks = [];\n\n      const readStream = async () => {\n        try {\n          while (true) {\n            const { done, value } = await reader.read();\n\n            if (done) {\n              progress.blob = new Blob(chunks, { type: binaryType });\n              sub.next({ ...progress });\n              sub.complete();\n              break;\n            }\n\n            chunks.push(value);\n\n            progress.bufferedBytes += value.length;\n            const currentTimestamp = new Date().getTime();\n            const timestampDiff = currentTimestamp - startTimestamp;\n            progress.bytesPerSecond = Math.round(\n              progress.bufferedBytes / Math.round(timestampDiff / 1000)\n            );\n            progress.percentage = Math.round((progress.bufferedBytes / progress.totalBytes) * 100);\n            sub.next({ ...progress });\n          }\n        } catch (e) {\n          abortController.abort();\n          sub.error(e);\n        }\n      };\n      readStream();\n\n      return {\n        unsubscribe() {\n          abortController.abort();\n          sub.complete();\n        }\n      };\n    });\n  }\n\n  private isGenericType(type: string) {\n    return Object.values(GENERIC_FILE_TYPE).includes(type as GENERIC_FILE_TYPE);\n  }\n}\n"]}
@@ -1,7 +1,7 @@
1
1
  import { ChangeDetectorRef, Component, ElementRef, EventEmitter, HostListener, Input, Output, ViewChild } from '@angular/core';
2
2
  import { NG_VALUE_ACCESSOR, AbstractControl } from '@angular/forms';
3
3
  import { TranslateService } from '@ngx-translate/core';
4
- import { get, map, some } from 'lodash-es';
4
+ import { get, map, some, min } from 'lodash-es';
5
5
  import { BytesPipe } from '../common/bytes.pipe';
6
6
  import { FilesService } from '../common/files.service';
7
7
  import { gettext } from '../i18n/gettext';
@@ -169,7 +169,10 @@ export class DropAreaComponent {
169
169
  this.onFileInvalidType();
170
170
  return;
171
171
  }
172
- const haveValidSizes = await this.filesService.haveValidSizes(files);
172
+ const maxFileSizeInBytes = this.maxFileSizeInMegaBytes
173
+ ? this.convertMegaBytesToBytes(this.maxFileSizeInMegaBytes)
174
+ : null;
175
+ const haveValidSizes = await this.filesService.haveValidSizes(files, maxFileSizeInBytes);
173
176
  if (!haveValidSizes) {
174
177
  await this.onFileInvalidSize();
175
178
  return;
@@ -205,12 +208,18 @@ export class DropAreaComponent {
205
208
  this.errorMessage = gettext('The selected file is not supported.');
206
209
  }
207
210
  async onFileInvalidSize() {
211
+ const maxFileSizeInBytes = this.maxFileSizeInMegaBytes
212
+ ? this.convertMegaBytesToBytes(this.maxFileSizeInMegaBytes)
213
+ : null;
208
214
  const msg = gettext('The selected file is too large. The size limit is {{ limit }}.');
209
- const limit = this.bytes.transform(await this.filesService.loadBytesSizeLimit());
215
+ const limit = this.bytes.transform(min([maxFileSizeInBytes, await this.filesService.loadBytesSizeLimit()]));
210
216
  this.errors = true;
211
217
  this.formControl?.setErrors({ invalidSize: true });
212
218
  this.errorMessage = this.translate.instant(msg, { limit });
213
219
  }
220
+ convertMegaBytesToBytes(maxFileSizeInMegaBytes) {
221
+ return maxFileSizeInMegaBytes * 1048576;
222
+ }
214
223
  getFilesNamesAsString(files) {
215
224
  return map(files, ({ name }) => name).join(', ');
216
225
  }
@@ -288,7 +297,7 @@ export class DropAreaComponent {
288
297
  }
289
298
  }
290
299
  DropAreaComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: DropAreaComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.FilesService }, { token: i2.TranslateService }, { token: i3.BytesPipe }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
291
- DropAreaComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: DropAreaComponent, selector: "c8y-drop-area", inputs: { formControl: "formControl", title: "title", message: "message", icon: "icon", loadingMessage: "loadingMessage", forceHideList: "forceHideList", alwaysShow: "alwaysShow", clickToOpen: "clickToOpen", loading: "loading", progress: "progress", maxAllowedFiles: "maxAllowedFiles", files: "files", accept: "accept" }, outputs: { dropped: "dropped" }, host: { listeners: { "keyup": "onkeyup($event)" } }, providers: [{ provide: NG_VALUE_ACCESSOR, useExisting: DropAreaComponent, multi: true }], viewQueries: [{ propertyName: "area", first: true, predicate: ["area"], descendants: true, static: true }, { propertyName: "zone", first: true, predicate: ["zone"], descendants: true }, { propertyName: "picker", first: true, predicate: ["picker"], descendants: true }], ngImport: i0, template: "<div\n class=\"drop-zone\"\n *ngIf=\"!shouldShowFilesList()\"\n [ngClass]=\"{ 'has-errors': errors }\"\n [style.pointerEvents]=\"loading ? 'none' : 'auto'\"\n #zone\n (dragleave)=\"stopDragging()\"\n (drop)=\"onDrop($event)\"\n (dragover)=\"onOver()\"\n [style.display]=\"isOver || alwaysShow || loading ? 'block' : 'none'\"\n (click)=\"showPicker($event)\"\n tabindex=\"0\"\n>\n <div class=\"file-placeholder\" [ngClass]=\"{ 'drag-over': isOver }\">\n <div *ngIf=\"loading\" class=\"d-flex d-col p-4 flex-center\">\n <div\n class=\"progress progress-striped active m-0\"\n *ngIf=\"progress !== -1\"\n style=\"min-width: 50%\"\n >\n <div\n [attr.aria-label]=\"progress + '%'\"\n class=\"progress-bar\"\n role=\"progressbar\"\n aria-valuenow=\"0\"\n aria-valuemin=\"0\"\n aria-valuemax=\"100\"\n [style.width]=\"progress + '%'\"\n ></div>\n </div>\n <div class=\"spinner-snake\" *ngIf=\"progress === -1\"></div>\n <p *ngIf=\"!hasDropAreaSmallClass\" class=\"m-t-auto m-b-auto m-r-8\">\n {{ loadingMessage | translate }}\n </p>\n </div>\n\n <div *ngIf=\"!loading\" class=\"hint-placeholder pointer\">\n <i class=\"dlt-c8y-icon-{{ icon }}\"></i>\n <p *ngIf=\"!errors\">\n <b>{{ message | translate }}</b>\n <br />\n <span *ngIf=\"alwaysShow && clickToOpen\" translate></span>\n </p>\n <div *ngIf=\"errors\" class=\"has-errors\">\n <p class=\"form-control-feedback-message\">\n {{ errorMessage | translate }}\n </p>\n </div>\n </div>\n </div>\n</div>\n\n<div\n class=\"drop-zone\"\n *ngIf=\"shouldShowFilesList()\"\n [style.display]=\"isOver || alwaysShow || loading ? 'block' : 'none'\"\n tabindex=\"0\"\n>\n <div *ngIf=\"loading\" class=\"p-absolute p-4 fit-w fit-h d-flex d-col j-c-center a-i-center\">\n <div\n class=\"progress progress-striped active m-0\"\n *ngIf=\"progress !== -1\"\n style=\"min-width: 80%\"\n >\n <div\n [attr.aria-label]=\"progress + '%'\"\n class=\"progress-bar\"\n role=\"progressbar\"\n aria-valuenow=\"0\"\n aria-valuemin=\"0\"\n aria-valuemax=\"100\"\n [style.width]=\"progress + '%'\"\n ></div>\n </div>\n <div *ngIf=\"progress === -1\">\n <c8y-loading></c8y-loading>\n </div>\n <p *ngIf=\"!hasDropAreaSmallClass\" class=\"m-b-8\">\n <strong>\n {{ loadingMessage | translate }}\n </strong>\n </p>\n </div>\n <div *ngIf=\"!loading\" class=\"file-placeholder p-4\">\n <div class=\"d-flex p-4 a-i-center\">\n <i c8yIcon=\"file-o\" class=\"icon-20 m-r-8\"></i>\n <span title=\"{{ filesNameString }}\" class=\"text-truncate\">\n {{ filesNameString }}\n </span>\n <button\n title=\"{{ 'Remove' | translate }}\"\n [attr.aria-label]=\"'Remove' | translate\"\n type=\"button\"\n class=\"btn btn-dot btn-dot--danger showOnHover m-l-auto\"\n >\n <i c8yIcon=\"minus-circle\" (click)=\"onDelete()\"></i>\n </button>\n </div>\n </div>\n</div>\n<label for=\"file\" class=\"sr-only\">{{ 'Select file' | translate }}</label>\n<input\n #picker\n *ngIf=\"clickToOpen\"\n (change)=\"onPick($event)\"\n (click)=\"picker.focus()\"\n (blur)=\"onTouched()\"\n [accept]=\"acceptedExts\"\n [multiple]=\"maxAllowedFiles > 1\"\n type=\"file\"\n class=\"hidden\"\n id=\"file\"\n/>\n<div #area [hidden]=\"isOver || loading\" (dragover)=\"toggle()\">\n <ng-content></ng-content>\n</div>\n", dependencies: [{ kind: "directive", type: i4.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: i5.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i6.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i7.LoadingComponent, selector: "c8y-loading" }, { kind: "pipe", type: i8.C8yTranslatePipe, name: "translate" }] });
300
+ DropAreaComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: DropAreaComponent, selector: "c8y-drop-area", inputs: { formControl: "formControl", title: "title", message: "message", icon: "icon", loadingMessage: "loadingMessage", forceHideList: "forceHideList", alwaysShow: "alwaysShow", clickToOpen: "clickToOpen", loading: "loading", progress: "progress", maxAllowedFiles: "maxAllowedFiles", files: "files", maxFileSizeInMegaBytes: "maxFileSizeInMegaBytes", accept: "accept" }, outputs: { dropped: "dropped" }, host: { listeners: { "keyup": "onkeyup($event)" } }, providers: [{ provide: NG_VALUE_ACCESSOR, useExisting: DropAreaComponent, multi: true }], viewQueries: [{ propertyName: "area", first: true, predicate: ["area"], descendants: true, static: true }, { propertyName: "zone", first: true, predicate: ["zone"], descendants: true }, { propertyName: "picker", first: true, predicate: ["picker"], descendants: true }], ngImport: i0, template: "<div\n class=\"drop-zone\"\n *ngIf=\"!shouldShowFilesList()\"\n [ngClass]=\"{ 'has-errors': errors }\"\n [style.pointerEvents]=\"loading ? 'none' : 'auto'\"\n #zone\n (dragleave)=\"stopDragging()\"\n (drop)=\"onDrop($event)\"\n (dragover)=\"onOver()\"\n [style.display]=\"isOver || alwaysShow || loading ? 'block' : 'none'\"\n (click)=\"showPicker($event)\"\n tabindex=\"0\"\n>\n <div class=\"file-placeholder\" [ngClass]=\"{ 'drag-over': isOver }\">\n <div *ngIf=\"loading\" class=\"d-flex d-col p-4 flex-center\">\n <div\n class=\"progress progress-striped active m-0\"\n *ngIf=\"progress !== -1\"\n style=\"min-width: 50%\"\n >\n <div\n [attr.aria-label]=\"progress + '%'\"\n class=\"progress-bar\"\n role=\"progressbar\"\n aria-valuenow=\"0\"\n aria-valuemin=\"0\"\n aria-valuemax=\"100\"\n [style.width]=\"progress + '%'\"\n ></div>\n </div>\n <div class=\"spinner-snake\" *ngIf=\"progress === -1\"></div>\n <p *ngIf=\"!hasDropAreaSmallClass\" class=\"m-t-auto m-b-auto m-r-8\">\n {{ loadingMessage | translate }}\n </p>\n </div>\n\n <div *ngIf=\"!loading\" class=\"hint-placeholder pointer\">\n <i class=\"dlt-c8y-icon-{{ icon }}\"></i>\n <p *ngIf=\"!errors\">\n <b>{{ message | translate }}</b>\n <br />\n <span *ngIf=\"alwaysShow && clickToOpen\" translate></span>\n </p>\n <div *ngIf=\"errors\" class=\"has-errors\">\n <p class=\"form-control-feedback-message\">\n {{ errorMessage | translate }}\n </p>\n </div>\n </div>\n </div>\n</div>\n\n<div\n class=\"drop-zone\"\n *ngIf=\"shouldShowFilesList()\"\n [style.display]=\"isOver || alwaysShow || loading ? 'block' : 'none'\"\n tabindex=\"0\"\n>\n <div *ngIf=\"loading\" class=\"p-absolute p-4 fit-w fit-h d-flex d-col j-c-center a-i-center\">\n <div\n class=\"progress progress-striped active m-0\"\n *ngIf=\"progress !== -1\"\n style=\"min-width: 80%\"\n >\n <div\n [attr.aria-label]=\"progress + '%'\"\n class=\"progress-bar\"\n role=\"progressbar\"\n aria-valuenow=\"0\"\n aria-valuemin=\"0\"\n aria-valuemax=\"100\"\n [style.width]=\"progress + '%'\"\n ></div>\n </div>\n <div *ngIf=\"progress === -1\">\n <c8y-loading></c8y-loading>\n </div>\n <p *ngIf=\"!hasDropAreaSmallClass\" class=\"m-b-8\">\n <strong>\n {{ loadingMessage | translate }}\n </strong>\n </p>\n </div>\n <div *ngIf=\"!loading\" class=\"file-placeholder p-4\">\n <div class=\"d-flex p-4 a-i-center\">\n <i c8yIcon=\"file-o\" class=\"icon-20 m-r-8\"></i>\n <span title=\"{{ filesNameString }}\" class=\"text-truncate\">\n {{ filesNameString }}\n </span>\n <button\n title=\"{{ 'Remove' | translate }}\"\n [attr.aria-label]=\"'Remove' | translate\"\n type=\"button\"\n class=\"btn btn-dot btn-dot--danger showOnHover m-l-auto\"\n >\n <i c8yIcon=\"minus-circle\" (click)=\"onDelete()\"></i>\n </button>\n </div>\n </div>\n</div>\n<label for=\"file\" class=\"sr-only\">{{ 'Select file' | translate }}</label>\n<input\n #picker\n *ngIf=\"clickToOpen\"\n (change)=\"onPick($event)\"\n (click)=\"picker.focus()\"\n (blur)=\"onTouched()\"\n [accept]=\"acceptedExts\"\n [multiple]=\"maxAllowedFiles > 1\"\n type=\"file\"\n class=\"hidden\"\n id=\"file\"\n/>\n<div #area [hidden]=\"isOver || loading\" (dragover)=\"toggle()\">\n <ng-content></ng-content>\n</div>\n", dependencies: [{ kind: "directive", type: i4.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: i5.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i6.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i7.LoadingComponent, selector: "c8y-loading" }, { kind: "pipe", type: i8.C8yTranslatePipe, name: "translate" }] });
292
301
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: DropAreaComponent, decorators: [{
293
302
  type: Component,
294
303
  args: [{ selector: 'c8y-drop-area', providers: [{ provide: NG_VALUE_ACCESSOR, useExisting: DropAreaComponent, multi: true }], template: "<div\n class=\"drop-zone\"\n *ngIf=\"!shouldShowFilesList()\"\n [ngClass]=\"{ 'has-errors': errors }\"\n [style.pointerEvents]=\"loading ? 'none' : 'auto'\"\n #zone\n (dragleave)=\"stopDragging()\"\n (drop)=\"onDrop($event)\"\n (dragover)=\"onOver()\"\n [style.display]=\"isOver || alwaysShow || loading ? 'block' : 'none'\"\n (click)=\"showPicker($event)\"\n tabindex=\"0\"\n>\n <div class=\"file-placeholder\" [ngClass]=\"{ 'drag-over': isOver }\">\n <div *ngIf=\"loading\" class=\"d-flex d-col p-4 flex-center\">\n <div\n class=\"progress progress-striped active m-0\"\n *ngIf=\"progress !== -1\"\n style=\"min-width: 50%\"\n >\n <div\n [attr.aria-label]=\"progress + '%'\"\n class=\"progress-bar\"\n role=\"progressbar\"\n aria-valuenow=\"0\"\n aria-valuemin=\"0\"\n aria-valuemax=\"100\"\n [style.width]=\"progress + '%'\"\n ></div>\n </div>\n <div class=\"spinner-snake\" *ngIf=\"progress === -1\"></div>\n <p *ngIf=\"!hasDropAreaSmallClass\" class=\"m-t-auto m-b-auto m-r-8\">\n {{ loadingMessage | translate }}\n </p>\n </div>\n\n <div *ngIf=\"!loading\" class=\"hint-placeholder pointer\">\n <i class=\"dlt-c8y-icon-{{ icon }}\"></i>\n <p *ngIf=\"!errors\">\n <b>{{ message | translate }}</b>\n <br />\n <span *ngIf=\"alwaysShow && clickToOpen\" translate></span>\n </p>\n <div *ngIf=\"errors\" class=\"has-errors\">\n <p class=\"form-control-feedback-message\">\n {{ errorMessage | translate }}\n </p>\n </div>\n </div>\n </div>\n</div>\n\n<div\n class=\"drop-zone\"\n *ngIf=\"shouldShowFilesList()\"\n [style.display]=\"isOver || alwaysShow || loading ? 'block' : 'none'\"\n tabindex=\"0\"\n>\n <div *ngIf=\"loading\" class=\"p-absolute p-4 fit-w fit-h d-flex d-col j-c-center a-i-center\">\n <div\n class=\"progress progress-striped active m-0\"\n *ngIf=\"progress !== -1\"\n style=\"min-width: 80%\"\n >\n <div\n [attr.aria-label]=\"progress + '%'\"\n class=\"progress-bar\"\n role=\"progressbar\"\n aria-valuenow=\"0\"\n aria-valuemin=\"0\"\n aria-valuemax=\"100\"\n [style.width]=\"progress + '%'\"\n ></div>\n </div>\n <div *ngIf=\"progress === -1\">\n <c8y-loading></c8y-loading>\n </div>\n <p *ngIf=\"!hasDropAreaSmallClass\" class=\"m-b-8\">\n <strong>\n {{ loadingMessage | translate }}\n </strong>\n </p>\n </div>\n <div *ngIf=\"!loading\" class=\"file-placeholder p-4\">\n <div class=\"d-flex p-4 a-i-center\">\n <i c8yIcon=\"file-o\" class=\"icon-20 m-r-8\"></i>\n <span title=\"{{ filesNameString }}\" class=\"text-truncate\">\n {{ filesNameString }}\n </span>\n <button\n title=\"{{ 'Remove' | translate }}\"\n [attr.aria-label]=\"'Remove' | translate\"\n type=\"button\"\n class=\"btn btn-dot btn-dot--danger showOnHover m-l-auto\"\n >\n <i c8yIcon=\"minus-circle\" (click)=\"onDelete()\"></i>\n </button>\n </div>\n </div>\n</div>\n<label for=\"file\" class=\"sr-only\">{{ 'Select file' | translate }}</label>\n<input\n #picker\n *ngIf=\"clickToOpen\"\n (change)=\"onPick($event)\"\n (click)=\"picker.focus()\"\n (blur)=\"onTouched()\"\n [accept]=\"acceptedExts\"\n [multiple]=\"maxAllowedFiles > 1\"\n type=\"file\"\n class=\"hidden\"\n id=\"file\"\n/>\n<div #area [hidden]=\"isOver || loading\" (dragover)=\"toggle()\">\n <ng-content></ng-content>\n</div>\n" }]
@@ -318,6 +327,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImpor
318
327
  type: Input
319
328
  }], files: [{
320
329
  type: Input
330
+ }], maxFileSizeInMegaBytes: [{
331
+ type: Input
321
332
  }], accept: [{
322
333
  type: Input
323
334
  }], area: [{
@@ -340,4 +351,4 @@ var ReadAsType;
340
351
  ReadAsType[ReadAsType["ARRAY_BUFFER"] = 2] = "ARRAY_BUFFER";
341
352
  ReadAsType[ReadAsType["BINARY_STRING"] = 3] = "BINARY_STRING";
342
353
  })(ReadAsType || (ReadAsType = {}));
343
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"drop-area.component.js","sourceRoot":"","sources":["../../../../core/drop-area/drop-area.component.ts","../../../../core/drop-area/drop-area.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  HostListener,\n  Input,\n  OnInit,\n  Output,\n  ViewChild\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, AbstractControl } from '@angular/forms';\n\nimport { TranslateService } from '@ngx-translate/core';\nimport { get, map, some } from 'lodash-es';\nimport { BytesPipe } from '../common/bytes.pipe';\nimport { FilesService } from '../common/files.service';\nimport { gettext } from '../i18n/gettext';\n\n/**\n * A drop-zone which is a file selector allowing users to select file(s) from their file system, either natively or by drag and drop.\n *\n * ## Example:\n *\n * ```html\n *  <div>\n *    <c8y-drop-area\n *      (dropped)=\"uploadFile($event)\"\n *      [icon]=\"'upload'\"\n *      [accept]=\"'.zip,.7z,video'\">\n *    </c8y-drop-area>\n *  </div>\n * ```\n */\n\n@Component({\n  selector: 'c8y-drop-area',\n  templateUrl: './drop-area.component.html',\n  providers: [{ provide: NG_VALUE_ACCESSOR, useExisting: DropAreaComponent, multi: true }]\n})\nexport class DropAreaComponent implements OnInit, ControlValueAccessor {\n  @Input() formControl: AbstractControl<any, any>;\n  @Input() title = gettext('Upload file');\n  @Input() message = gettext('Drop file here');\n  @Input() icon = 'plus-square';\n  @Input() loadingMessage = gettext('Uploading…');\n  @Input() forceHideList = false;\n  /** Affects displaying both the drop zone and the list of dropped files. */\n  @Input() alwaysShow = false;\n  @Input() clickToOpen = true;\n  @Input() loading = false;\n  /**\n   * Current progress of the upload as a percentage. If not given a spinner will be displayed.\n   */\n  @Input() progress = -1; // -1 = spinner\n  @Output() dropped: EventEmitter<DroppedFile[]> = new EventEmitter();\n  @Input() maxAllowedFiles = Infinity;\n  @Input() files: FileList;\n  /** Specifies a filter for what file types the user can pick from the file input dialog box.\n   * ## Example:\n   *\n   * Specify file types by extensions:\n   * ```html\n   *  ...\n   *  [accept]=\"'.zip,.7z'\"\n   *  ...\n   * ```\n   *\n   * Specify file types by extensions and generic types [GENERIC_FILE_TYPE]{@link GENERIC_FILE_TYPE}:\n   * ```html\n   *  ...\n   *  [accept]=\"'.pdf,archive'\"\n   *  ...\n   * ```\n   *\n   * Specify file types by generic types [GENERIC_FILE_TYPE]{@link GENERIC_FILE_TYPE}:\n   *  ```html\n   *  ...\n   *  [accept]=\"'archive,video'\"\n   *  ...\n   *\n   * ```\n   */\n  @Input() accept: string;\n  isOver = false;\n  errors = false;\n  errorMessage: string;\n  filesNameString: string;\n  acceptedExts: string[];\n  hasDropAreaSmallClass: boolean;\n\n  @ViewChild('area', { static: true }) area: ElementRef;\n  @ViewChild('zone', { static: false }) zone: ElementRef;\n  @ViewChild('picker', { static: false }) picker: ElementRef;\n\n  constructor(\n    private cd: ChangeDetectorRef,\n    private filesService: FilesService,\n    private translate: TranslateService,\n    private bytes: BytesPipe,\n    private ref: ElementRef\n  ) {}\n\n  @HostListener('keyup', ['$event'])\n  onkeyup(event: KeyboardEvent) {\n    if (event.key === 'Enter') {\n      this.picker.nativeElement.click();\n    }\n  }\n\n  ngOnInit(): void {\n    this.acceptedExts = this.filesService.extractFileExtensions(this.accept).map(t => `.${t}`);\n    this.alwaysShow = this.alwaysShow || this.area.nativeElement.children.length === 0;\n\n    if (this.files && this.isFilesAnObjectOrArray() && this.files.length > 0) {\n      this.onFilesSelected(this.files);\n    }\n  }\n\n  ngAfterViewChecked() {\n    this.hasDropAreaSmallClass = this.ref.nativeElement.classList.contains('drop-area-sm');\n  }\n\n  /**\n   * Toggles the style of the drop zone element when a file is dragged over the component.\n   */\n  toggle(): void {\n    this.zone.nativeElement.style.height = this.area.nativeElement.offsetHeight + 'px';\n    this.onOver();\n  }\n\n  /**\n   * Shows computer browser with files to drop into drop-area zone.\n   */\n  showPicker($event?): void {\n    this.preventDefault($event);\n    this.picker.nativeElement.value = '';\n    this.picker.nativeElement.click();\n  }\n\n  /**\n   * Triggered when file is on over drop area, but not dropped.\n   */\n  onOver(): void {\n    if (!this.isOver) {\n      this.isOver = true;\n      document.addEventListener('dragover', this.preventDefault);\n      document.addEventListener('drop', this.preventDefault);\n    }\n  }\n\n  /**\n   * Triggered when file is dropped.\n   */\n  onPick($event): void {\n    this.errors = false;\n    this.preventDefault($event);\n    this.onFilesSelected($event.target.files);\n  }\n\n  /**\n   * Handle file when it is dropped into drop-area.\n   */\n  onDrop($event): void {\n    this.preventDefault($event);\n    this.onFilesSelected($event.dataTransfer.files);\n    this.stopDragging();\n  }\n\n  /**\n   * Checks condition what should be displayed: drop-area zone or list of dropped files.\n   */\n  shouldShowFilesList(): boolean {\n    return (\n      this.isFilesAnObjectOrArray() &&\n      !this.forceHideList &&\n      this.alwaysShow &&\n      !this.isFilesArrayEmpty() &&\n      !this.hasEmptyFiles() &&\n      !this.isTooManyFiles()\n    );\n  }\n\n  /**\n   * Triggered when file is picked over web application.\n   */\n  stopDragging(): void {\n    document.removeEventListener('dragover', this.preventDefault);\n    document.removeEventListener('drop', this.preventDefault);\n    this.isOver = false;\n  }\n\n  /**\n   * Delete files already dropped files.\n   */\n  onDelete() {\n    delete this.files;\n    delete this.filesNameString;\n    this.clearErrors();\n    this.dropped.emit(null);\n    this.onChange(null);\n    this.cd.markForCheck();\n  }\n\n  onChange: (value: any) => void = _ => undefined;\n  onTouched: () => void = () => undefined;\n\n  writeValue(value: any) {\n    this.files = value;\n    if (!value) {\n      this.onDelete();\n    } else {\n      this.filesNameString = this.getFilesNamesAsString(value);\n    }\n    this.cd.detectChanges();\n  }\n\n  registerOnChange(fn: any) {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: any) {\n    this.onTouched = fn;\n  }\n\n  private async onFilesSelected(files: FileList) {\n    const hasValidNameLength = this.filesService.checkMaxLength(files);\n    if (!hasValidNameLength) {\n      this.onFileInvalidNameLength();\n      return;\n    }\n\n    const haveValidTypes = this.filesService.haveValidExtensions(files, this.accept);\n    if (!haveValidTypes) {\n      this.onFileInvalidType();\n      return;\n    }\n\n    const haveValidSizes = await this.filesService.haveValidSizes(files);\n    if (!haveValidSizes) {\n      await this.onFileInvalidSize();\n      return;\n    }\n\n    this.files = files;\n    this.filesNameString = this.getFilesNamesAsString(files);\n    this.errors = false;\n\n    if (this.isTooManyFiles()) {\n      this.errors = true;\n      this.formControl?.setErrors({ tooManyFiles: true });\n      this.errorMessage = gettext('Too many files selected.');\n      return;\n    }\n\n    if (this.hasEmptyFiles()) {\n      this.errors = true;\n      this.formControl?.setErrors({ emptyFiles: true });\n      this.errorMessage = gettext('File must not be empty, select another one.');\n      return;\n    }\n\n    const droppedFiles: DroppedFile[] = this.compose(files);\n    this.dropped.emit(droppedFiles);\n    this.onChange(droppedFiles);\n    this.cd.markForCheck();\n  }\n\n  private onFileInvalidNameLength() {\n    this.errors = true;\n    this.formControl?.setErrors({ invalidNameLength: true });\n    this.errorMessage = gettext('The filename is too long.');\n  }\n\n  private onFileInvalidType() {\n    this.errors = true;\n    this.formControl?.setErrors({ invalidType: true });\n    this.errorMessage = gettext('The selected file is not supported.');\n  }\n\n  private async onFileInvalidSize() {\n    const msg = gettext('The selected file is too large. The size limit is {{ limit }}.');\n    const limit = this.bytes.transform(await this.filesService.loadBytesSizeLimit());\n    this.errors = true;\n    this.formControl?.setErrors({ invalidSize: true });\n\n    this.errorMessage = this.translate.instant(msg, { limit });\n  }\n\n  private getFilesNamesAsString(files: FileList): string {\n    return map(files, ({ name }) => name).join(', ');\n  }\n\n  private isFilesArrayEmpty() {\n    return get(this, 'files.length', 0) === 0;\n  }\n\n  private isTooManyFiles() {\n    return get(this, 'files.length', 0) > this.maxAllowedFiles;\n  }\n\n  private isFilesAnObjectOrArray() {\n    return typeof this.files === 'object';\n  }\n\n  private hasEmptyFiles() {\n    let result = true;\n    if (!this.isFilesArrayEmpty()) {\n      result = this.isAnyFileEmpty();\n    }\n    return result;\n  }\n\n  private isAnyFileEmpty(): boolean {\n    return some(Array.from(this.files), ['size', 0]);\n  }\n\n  private clearErrors() {\n    delete this.errorMessage;\n    this.errors = false;\n    this.formControl?.setErrors(null);\n  }\n\n  private preventDefault($event?) {\n    if ($event) {\n      $event.preventDefault();\n    }\n  }\n\n  private compose(files: FileList): DroppedFile[] {\n    return Array.from(files).map(file => ({\n      file,\n      readAsJson: async () => JSON.parse(await this.read(file, ReadAsType.TEXT)),\n      readAsText: async () => this.read(file, ReadAsType.TEXT),\n      readAsArrayBuffer: async () => this.read(file, ReadAsType.ARRAY_BUFFER),\n      readAsBinaryString: async () => this.read(file, ReadAsType.BINARY_STRING),\n      readAsDataURL: async () => this.read(file, ReadAsType.DATA_URL)\n    }));\n  }\n\n  private async read(file, type: ReadAsType): Promise<string> {\n    return new Promise<string>((resolve, reject) => {\n      const reader = new FileReader();\n      switch (type) {\n        case ReadAsType.TEXT: {\n          reader.readAsText(file);\n          break;\n        }\n        case ReadAsType.ARRAY_BUFFER: {\n          reader.readAsArrayBuffer(file);\n          break;\n        }\n        case ReadAsType.BINARY_STRING: {\n          reader.readAsBinaryString(file);\n          break;\n        }\n        case ReadAsType.DATA_URL: {\n          reader.readAsDataURL(file);\n          break;\n        }\n      }\n      reader.onload = () => this.onLoad(reader, resolve, reject);\n    });\n  }\n\n  private onLoad(reader, resolve, reject) {\n    if (reader.readyState !== 2) {\n      return;\n    }\n    if (reader.error) {\n      reject(reader.error);\n    }\n    resolve(reader.result);\n  }\n}\n\nexport interface DroppedFile {\n  file: File;\n  readAsText();\n  readAsArrayBuffer();\n  readAsBinaryString();\n  readAsDataURL();\n  readAsJson();\n}\n\nenum ReadAsType {\n  TEXT,\n  DATA_URL,\n  ARRAY_BUFFER,\n  BINARY_STRING\n}\n","<div\n  class=\"drop-zone\"\n  *ngIf=\"!shouldShowFilesList()\"\n  [ngClass]=\"{ 'has-errors': errors }\"\n  [style.pointerEvents]=\"loading ? 'none' : 'auto'\"\n  #zone\n  (dragleave)=\"stopDragging()\"\n  (drop)=\"onDrop($event)\"\n  (dragover)=\"onOver()\"\n  [style.display]=\"isOver || alwaysShow || loading ? 'block' : 'none'\"\n  (click)=\"showPicker($event)\"\n  tabindex=\"0\"\n>\n  <div class=\"file-placeholder\" [ngClass]=\"{ 'drag-over': isOver }\">\n    <div *ngIf=\"loading\" class=\"d-flex d-col p-4 flex-center\">\n      <div\n        class=\"progress progress-striped active m-0\"\n        *ngIf=\"progress !== -1\"\n        style=\"min-width: 50%\"\n      >\n        <div\n          [attr.aria-label]=\"progress + '%'\"\n          class=\"progress-bar\"\n          role=\"progressbar\"\n          aria-valuenow=\"0\"\n          aria-valuemin=\"0\"\n          aria-valuemax=\"100\"\n          [style.width]=\"progress + '%'\"\n        ></div>\n      </div>\n      <div class=\"spinner-snake\" *ngIf=\"progress === -1\"></div>\n      <p *ngIf=\"!hasDropAreaSmallClass\" class=\"m-t-auto m-b-auto m-r-8\">\n        {{ loadingMessage | translate }}\n      </p>\n    </div>\n\n    <div *ngIf=\"!loading\" class=\"hint-placeholder pointer\">\n      <i class=\"dlt-c8y-icon-{{ icon }}\"></i>\n      <p *ngIf=\"!errors\">\n        <b>{{ message | translate }}</b>\n        <br />\n        <span *ngIf=\"alwaysShow && clickToOpen\" translate></span>\n      </p>\n      <div *ngIf=\"errors\" class=\"has-errors\">\n        <p class=\"form-control-feedback-message\">\n          {{ errorMessage | translate }}\n        </p>\n      </div>\n    </div>\n  </div>\n</div>\n\n<div\n  class=\"drop-zone\"\n  *ngIf=\"shouldShowFilesList()\"\n  [style.display]=\"isOver || alwaysShow || loading ? 'block' : 'none'\"\n  tabindex=\"0\"\n>\n  <div *ngIf=\"loading\" class=\"p-absolute p-4 fit-w fit-h d-flex d-col j-c-center a-i-center\">\n    <div\n      class=\"progress progress-striped active m-0\"\n      *ngIf=\"progress !== -1\"\n      style=\"min-width: 80%\"\n    >\n      <div\n        [attr.aria-label]=\"progress + '%'\"\n        class=\"progress-bar\"\n        role=\"progressbar\"\n        aria-valuenow=\"0\"\n        aria-valuemin=\"0\"\n        aria-valuemax=\"100\"\n        [style.width]=\"progress + '%'\"\n      ></div>\n    </div>\n    <div *ngIf=\"progress === -1\">\n      <c8y-loading></c8y-loading>\n    </div>\n    <p *ngIf=\"!hasDropAreaSmallClass\" class=\"m-b-8\">\n      <strong>\n        {{ loadingMessage | translate }}\n      </strong>\n    </p>\n  </div>\n  <div *ngIf=\"!loading\" class=\"file-placeholder p-4\">\n    <div class=\"d-flex p-4 a-i-center\">\n      <i c8yIcon=\"file-o\" class=\"icon-20 m-r-8\"></i>\n      <span title=\"{{ filesNameString }}\" class=\"text-truncate\">\n        {{ filesNameString }}\n      </span>\n      <button\n        title=\"{{ 'Remove' | translate }}\"\n        [attr.aria-label]=\"'Remove' | translate\"\n        type=\"button\"\n        class=\"btn btn-dot btn-dot--danger showOnHover m-l-auto\"\n      >\n        <i c8yIcon=\"minus-circle\" (click)=\"onDelete()\"></i>\n      </button>\n    </div>\n  </div>\n</div>\n<label for=\"file\" class=\"sr-only\">{{ 'Select file' | translate }}</label>\n<input\n  #picker\n  *ngIf=\"clickToOpen\"\n  (change)=\"onPick($event)\"\n  (click)=\"picker.focus()\"\n  (blur)=\"onTouched()\"\n  [accept]=\"acceptedExts\"\n  [multiple]=\"maxAllowedFiles > 1\"\n  type=\"file\"\n  class=\"hidden\"\n  id=\"file\"\n/>\n<div #area [hidden]=\"isOver || loading\" (dragover)=\"toggle()\">\n  <ng-content></ng-content>\n</div>\n"]}
354
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"drop-area.component.js","sourceRoot":"","sources":["../../../../core/drop-area/drop-area.component.ts","../../../../core/drop-area/drop-area.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  HostListener,\n  Input,\n  OnInit,\n  Output,\n  ViewChild\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, AbstractControl } from '@angular/forms';\n\nimport { TranslateService } from '@ngx-translate/core';\nimport { get, map, some, min } from 'lodash-es';\nimport { BytesPipe } from '../common/bytes.pipe';\nimport { FilesService } from '../common/files.service';\nimport { gettext } from '../i18n/gettext';\n\n/**\n * A drop-zone which is a file selector allowing users to select file(s) from their file system, either natively or by drag and drop.\n *\n * ## Example:\n *\n * ```html\n *  <div>\n *    <c8y-drop-area\n *      (dropped)=\"uploadFile($event)\"\n *      [icon]=\"'upload'\"\n *      [accept]=\"'.zip,.7z,video'\">\n *    </c8y-drop-area>\n *  </div>\n * ```\n */\n\n@Component({\n  selector: 'c8y-drop-area',\n  templateUrl: './drop-area.component.html',\n  providers: [{ provide: NG_VALUE_ACCESSOR, useExisting: DropAreaComponent, multi: true }]\n})\nexport class DropAreaComponent implements OnInit, ControlValueAccessor {\n  @Input() formControl: AbstractControl<any, any>;\n  @Input() title = gettext('Upload file');\n  @Input() message = gettext('Drop file here');\n  @Input() icon = 'plus-square';\n  @Input() loadingMessage = gettext('Uploading…');\n  @Input() forceHideList = false;\n  /** Affects displaying both the drop zone and the list of dropped files. */\n  @Input() alwaysShow = false;\n  @Input() clickToOpen = true;\n  @Input() loading = false;\n  /**\n   * Current progress of the upload as a percentage. If not given a spinner will be displayed.\n   */\n  @Input() progress = -1; // -1 = spinner\n  @Output() dropped: EventEmitter<DroppedFile[]> = new EventEmitter();\n  @Input() maxAllowedFiles = Infinity;\n  @Input() files: FileList;\n  @Input() maxFileSizeInMegaBytes: number;\n  /** Specifies a filter for what file types the user can pick from the file input dialog box.\n   * ## Example:\n   *\n   * Specify file types by extensions:\n   * ```html\n   *  ...\n   *  [accept]=\"'.zip,.7z'\"\n   *  ...\n   * ```\n   *\n   * Specify file types by extensions and generic types [GENERIC_FILE_TYPE]{@link GENERIC_FILE_TYPE}:\n   * ```html\n   *  ...\n   *  [accept]=\"'.pdf,archive'\"\n   *  ...\n   * ```\n   *\n   * Specify file types by generic types [GENERIC_FILE_TYPE]{@link GENERIC_FILE_TYPE}:\n   *  ```html\n   *  ...\n   *  [accept]=\"'archive,video'\"\n   *  ...\n   *\n   * ```\n   */\n  @Input() accept: string;\n  isOver = false;\n  errors = false;\n  errorMessage: string;\n  filesNameString: string;\n  acceptedExts: string[];\n  hasDropAreaSmallClass: boolean;\n\n  @ViewChild('area', { static: true }) area: ElementRef;\n  @ViewChild('zone', { static: false }) zone: ElementRef;\n  @ViewChild('picker', { static: false }) picker: ElementRef;\n\n  constructor(\n    private cd: ChangeDetectorRef,\n    private filesService: FilesService,\n    private translate: TranslateService,\n    private bytes: BytesPipe,\n    private ref: ElementRef\n  ) {}\n\n  @HostListener('keyup', ['$event'])\n  onkeyup(event: KeyboardEvent) {\n    if (event.key === 'Enter') {\n      this.picker.nativeElement.click();\n    }\n  }\n\n  ngOnInit(): void {\n    this.acceptedExts = this.filesService.extractFileExtensions(this.accept).map(t => `.${t}`);\n    this.alwaysShow = this.alwaysShow || this.area.nativeElement.children.length === 0;\n\n    if (this.files && this.isFilesAnObjectOrArray() && this.files.length > 0) {\n      this.onFilesSelected(this.files);\n    }\n  }\n\n  ngAfterViewChecked() {\n    this.hasDropAreaSmallClass = this.ref.nativeElement.classList.contains('drop-area-sm');\n  }\n\n  /**\n   * Toggles the style of the drop zone element when a file is dragged over the component.\n   */\n  toggle(): void {\n    this.zone.nativeElement.style.height = this.area.nativeElement.offsetHeight + 'px';\n    this.onOver();\n  }\n\n  /**\n   * Shows computer browser with files to drop into drop-area zone.\n   */\n  showPicker($event?): void {\n    this.preventDefault($event);\n    this.picker.nativeElement.value = '';\n    this.picker.nativeElement.click();\n  }\n\n  /**\n   * Triggered when file is on over drop area, but not dropped.\n   */\n  onOver(): void {\n    if (!this.isOver) {\n      this.isOver = true;\n      document.addEventListener('dragover', this.preventDefault);\n      document.addEventListener('drop', this.preventDefault);\n    }\n  }\n\n  /**\n   * Triggered when file is dropped.\n   */\n  onPick($event): void {\n    this.errors = false;\n    this.preventDefault($event);\n    this.onFilesSelected($event.target.files);\n  }\n\n  /**\n   * Handle file when it is dropped into drop-area.\n   */\n  onDrop($event): void {\n    this.preventDefault($event);\n    this.onFilesSelected($event.dataTransfer.files);\n    this.stopDragging();\n  }\n\n  /**\n   * Checks condition what should be displayed: drop-area zone or list of dropped files.\n   */\n  shouldShowFilesList(): boolean {\n    return (\n      this.isFilesAnObjectOrArray() &&\n      !this.forceHideList &&\n      this.alwaysShow &&\n      !this.isFilesArrayEmpty() &&\n      !this.hasEmptyFiles() &&\n      !this.isTooManyFiles()\n    );\n  }\n\n  /**\n   * Triggered when file is picked over web application.\n   */\n  stopDragging(): void {\n    document.removeEventListener('dragover', this.preventDefault);\n    document.removeEventListener('drop', this.preventDefault);\n    this.isOver = false;\n  }\n\n  /**\n   * Delete files already dropped files.\n   */\n  onDelete() {\n    delete this.files;\n    delete this.filesNameString;\n    this.clearErrors();\n    this.dropped.emit(null);\n    this.onChange(null);\n    this.cd.markForCheck();\n  }\n\n  onChange: (value: any) => void = _ => undefined;\n  onTouched: () => void = () => undefined;\n\n  writeValue(value: any) {\n    this.files = value;\n    if (!value) {\n      this.onDelete();\n    } else {\n      this.filesNameString = this.getFilesNamesAsString(value);\n    }\n    this.cd.detectChanges();\n  }\n\n  registerOnChange(fn: any) {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: any) {\n    this.onTouched = fn;\n  }\n\n  private async onFilesSelected(files: FileList) {\n    const hasValidNameLength = this.filesService.checkMaxLength(files);\n    if (!hasValidNameLength) {\n      this.onFileInvalidNameLength();\n      return;\n    }\n\n    const haveValidTypes = this.filesService.haveValidExtensions(files, this.accept);\n    if (!haveValidTypes) {\n      this.onFileInvalidType();\n      return;\n    }\n\n    const maxFileSizeInBytes = this.maxFileSizeInMegaBytes\n      ? this.convertMegaBytesToBytes(this.maxFileSizeInMegaBytes)\n      : null;\n    const haveValidSizes = await this.filesService.haveValidSizes(files, maxFileSizeInBytes);\n    if (!haveValidSizes) {\n      await this.onFileInvalidSize();\n      return;\n    }\n\n    this.files = files;\n    this.filesNameString = this.getFilesNamesAsString(files);\n    this.errors = false;\n\n    if (this.isTooManyFiles()) {\n      this.errors = true;\n      this.formControl?.setErrors({ tooManyFiles: true });\n      this.errorMessage = gettext('Too many files selected.');\n      return;\n    }\n\n    if (this.hasEmptyFiles()) {\n      this.errors = true;\n      this.formControl?.setErrors({ emptyFiles: true });\n      this.errorMessage = gettext('File must not be empty, select another one.');\n      return;\n    }\n\n    const droppedFiles: DroppedFile[] = this.compose(files);\n    this.dropped.emit(droppedFiles);\n    this.onChange(droppedFiles);\n    this.cd.markForCheck();\n  }\n\n  private onFileInvalidNameLength() {\n    this.errors = true;\n    this.formControl?.setErrors({ invalidNameLength: true });\n    this.errorMessage = gettext('The filename is too long.');\n  }\n\n  private onFileInvalidType() {\n    this.errors = true;\n    this.formControl?.setErrors({ invalidType: true });\n    this.errorMessage = gettext('The selected file is not supported.');\n  }\n\n  private async onFileInvalidSize() {\n    const maxFileSizeInBytes = this.maxFileSizeInMegaBytes\n      ? this.convertMegaBytesToBytes(this.maxFileSizeInMegaBytes)\n      : null;\n    const msg = gettext('The selected file is too large. The size limit is {{ limit }}.');\n    const limit = this.bytes.transform(\n      min([maxFileSizeInBytes, await this.filesService.loadBytesSizeLimit()])\n    );\n    this.errors = true;\n    this.formControl?.setErrors({ invalidSize: true });\n    this.errorMessage = this.translate.instant(msg, { limit });\n  }\n\n  private convertMegaBytesToBytes(maxFileSizeInMegaBytes: SizeInMegaBytes): SizeInBytes {\n    return maxFileSizeInMegaBytes * 1_048_576;\n  }\n\n  private getFilesNamesAsString(files: FileList): string {\n    return map(files, ({ name }) => name).join(', ');\n  }\n\n  private isFilesArrayEmpty() {\n    return get(this, 'files.length', 0) === 0;\n  }\n\n  private isTooManyFiles() {\n    return get(this, 'files.length', 0) > this.maxAllowedFiles;\n  }\n\n  private isFilesAnObjectOrArray() {\n    return typeof this.files === 'object';\n  }\n\n  private hasEmptyFiles() {\n    let result = true;\n    if (!this.isFilesArrayEmpty()) {\n      result = this.isAnyFileEmpty();\n    }\n    return result;\n  }\n\n  private isAnyFileEmpty(): boolean {\n    return some(Array.from(this.files), ['size', 0]);\n  }\n\n  private clearErrors() {\n    delete this.errorMessage;\n    this.errors = false;\n    this.formControl?.setErrors(null);\n  }\n\n  private preventDefault($event?) {\n    if ($event) {\n      $event.preventDefault();\n    }\n  }\n\n  private compose(files: FileList): DroppedFile[] {\n    return Array.from(files).map(file => ({\n      file,\n      readAsJson: async () => JSON.parse(await this.read(file, ReadAsType.TEXT)),\n      readAsText: async () => this.read(file, ReadAsType.TEXT),\n      readAsArrayBuffer: async () => this.read(file, ReadAsType.ARRAY_BUFFER),\n      readAsBinaryString: async () => this.read(file, ReadAsType.BINARY_STRING),\n      readAsDataURL: async () => this.read(file, ReadAsType.DATA_URL)\n    }));\n  }\n\n  private async read(file, type: ReadAsType): Promise<string> {\n    return new Promise<string>((resolve, reject) => {\n      const reader = new FileReader();\n      switch (type) {\n        case ReadAsType.TEXT: {\n          reader.readAsText(file);\n          break;\n        }\n        case ReadAsType.ARRAY_BUFFER: {\n          reader.readAsArrayBuffer(file);\n          break;\n        }\n        case ReadAsType.BINARY_STRING: {\n          reader.readAsBinaryString(file);\n          break;\n        }\n        case ReadAsType.DATA_URL: {\n          reader.readAsDataURL(file);\n          break;\n        }\n      }\n      reader.onload = () => this.onLoad(reader, resolve, reject);\n    });\n  }\n\n  private onLoad(reader, resolve, reject) {\n    if (reader.readyState !== 2) {\n      return;\n    }\n    if (reader.error) {\n      reject(reader.error);\n    }\n    resolve(reader.result);\n  }\n}\n\nexport interface DroppedFile {\n  file: File;\n  readAsText();\n  readAsArrayBuffer();\n  readAsBinaryString();\n  readAsDataURL();\n  readAsJson();\n}\n\nenum ReadAsType {\n  TEXT,\n  DATA_URL,\n  ARRAY_BUFFER,\n  BINARY_STRING\n}\n\ntype SizeInMegaBytes = number;\ntype SizeInBytes = number;\n","<div\n  class=\"drop-zone\"\n  *ngIf=\"!shouldShowFilesList()\"\n  [ngClass]=\"{ 'has-errors': errors }\"\n  [style.pointerEvents]=\"loading ? 'none' : 'auto'\"\n  #zone\n  (dragleave)=\"stopDragging()\"\n  (drop)=\"onDrop($event)\"\n  (dragover)=\"onOver()\"\n  [style.display]=\"isOver || alwaysShow || loading ? 'block' : 'none'\"\n  (click)=\"showPicker($event)\"\n  tabindex=\"0\"\n>\n  <div class=\"file-placeholder\" [ngClass]=\"{ 'drag-over': isOver }\">\n    <div *ngIf=\"loading\" class=\"d-flex d-col p-4 flex-center\">\n      <div\n        class=\"progress progress-striped active m-0\"\n        *ngIf=\"progress !== -1\"\n        style=\"min-width: 50%\"\n      >\n        <div\n          [attr.aria-label]=\"progress + '%'\"\n          class=\"progress-bar\"\n          role=\"progressbar\"\n          aria-valuenow=\"0\"\n          aria-valuemin=\"0\"\n          aria-valuemax=\"100\"\n          [style.width]=\"progress + '%'\"\n        ></div>\n      </div>\n      <div class=\"spinner-snake\" *ngIf=\"progress === -1\"></div>\n      <p *ngIf=\"!hasDropAreaSmallClass\" class=\"m-t-auto m-b-auto m-r-8\">\n        {{ loadingMessage | translate }}\n      </p>\n    </div>\n\n    <div *ngIf=\"!loading\" class=\"hint-placeholder pointer\">\n      <i class=\"dlt-c8y-icon-{{ icon }}\"></i>\n      <p *ngIf=\"!errors\">\n        <b>{{ message | translate }}</b>\n        <br />\n        <span *ngIf=\"alwaysShow && clickToOpen\" translate></span>\n      </p>\n      <div *ngIf=\"errors\" class=\"has-errors\">\n        <p class=\"form-control-feedback-message\">\n          {{ errorMessage | translate }}\n        </p>\n      </div>\n    </div>\n  </div>\n</div>\n\n<div\n  class=\"drop-zone\"\n  *ngIf=\"shouldShowFilesList()\"\n  [style.display]=\"isOver || alwaysShow || loading ? 'block' : 'none'\"\n  tabindex=\"0\"\n>\n  <div *ngIf=\"loading\" class=\"p-absolute p-4 fit-w fit-h d-flex d-col j-c-center a-i-center\">\n    <div\n      class=\"progress progress-striped active m-0\"\n      *ngIf=\"progress !== -1\"\n      style=\"min-width: 80%\"\n    >\n      <div\n        [attr.aria-label]=\"progress + '%'\"\n        class=\"progress-bar\"\n        role=\"progressbar\"\n        aria-valuenow=\"0\"\n        aria-valuemin=\"0\"\n        aria-valuemax=\"100\"\n        [style.width]=\"progress + '%'\"\n      ></div>\n    </div>\n    <div *ngIf=\"progress === -1\">\n      <c8y-loading></c8y-loading>\n    </div>\n    <p *ngIf=\"!hasDropAreaSmallClass\" class=\"m-b-8\">\n      <strong>\n        {{ loadingMessage | translate }}\n      </strong>\n    </p>\n  </div>\n  <div *ngIf=\"!loading\" class=\"file-placeholder p-4\">\n    <div class=\"d-flex p-4 a-i-center\">\n      <i c8yIcon=\"file-o\" class=\"icon-20 m-r-8\"></i>\n      <span title=\"{{ filesNameString }}\" class=\"text-truncate\">\n        {{ filesNameString }}\n      </span>\n      <button\n        title=\"{{ 'Remove' | translate }}\"\n        [attr.aria-label]=\"'Remove' | translate\"\n        type=\"button\"\n        class=\"btn btn-dot btn-dot--danger showOnHover m-l-auto\"\n      >\n        <i c8yIcon=\"minus-circle\" (click)=\"onDelete()\"></i>\n      </button>\n    </div>\n  </div>\n</div>\n<label for=\"file\" class=\"sr-only\">{{ 'Select file' | translate }}</label>\n<input\n  #picker\n  *ngIf=\"clickToOpen\"\n  (change)=\"onPick($event)\"\n  (click)=\"picker.focus()\"\n  (blur)=\"onTouched()\"\n  [accept]=\"acceptedExts\"\n  [multiple]=\"maxAllowedFiles > 1\"\n  type=\"file\"\n  class=\"hidden\"\n  id=\"file\"\n/>\n<div #area [hidden]=\"isOver || loading\" (dragover)=\"toggle()\">\n  <ng-content></ng-content>\n</div>\n"]}
@@ -23,7 +23,8 @@ export class FieldFile extends FieldType {
23
23
  'progress',
24
24
  'maxAllowedFiles',
25
25
  'files',
26
- 'accept'
26
+ 'accept',
27
+ 'maxFileSizeInMegaBytes'
27
28
  ];
28
29
  }
29
30
  ngOnInit() {
@@ -44,7 +45,7 @@ FieldFile.CONFIG = {
44
45
  ]
45
46
  };
46
47
  FieldFile.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: FieldFile, deps: null, target: i0.ɵɵFactoryTarget.Component });
47
- FieldFile.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: FieldFile, selector: "c8y-field-file", viewQueries: [{ propertyName: "dropArea", first: true, predicate: ["dropArea"], descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: "<div class=\"form-group\" role=\"group\" [attr.aria-labelledby]=\"id + '-fieldset'\">\n <label id=\"{{ id + '-fieldset' }}\" *ngIf=\"to.label || to.description\">\n {{ to.label | humanize }}\n <button\n class=\"btn-help btn-help--sm\"\n type=\"button\"\n [attr.aria-label]=\"'Help' | translate\"\n [popover]=\"to.description\"\n placement=\"right\"\n triggers=\"focus\"\n *ngIf=\"to.description\"\n ></button>\n </label>\n\n <c8y-drop-area #dropArea class=\"drop-area-sm\" [formControl]=\"formControl\"></c8y-drop-area>\n</div>\n", dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.PopoverDirective, selector: "[popover]", inputs: ["adaptivePosition", "boundariesElement", "popover", "popoverContext", "popoverTitle", "placement", "outsideClick", "triggers", "container", "containerClass", "isOpen", "delay"], outputs: ["onShown", "onHidden"], exportAs: ["bs-popover"] }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "component", type: i4.DropAreaComponent, selector: "c8y-drop-area", inputs: ["formControl", "title", "message", "icon", "loadingMessage", "forceHideList", "alwaysShow", "clickToOpen", "loading", "progress", "maxAllowedFiles", "files", "accept"], outputs: ["dropped"] }, { kind: "pipe", type: i5.C8yTranslatePipe, name: "translate" }, { kind: "pipe", type: i6.HumanizePipe, name: "humanize" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
48
+ FieldFile.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: FieldFile, selector: "c8y-field-file", viewQueries: [{ propertyName: "dropArea", first: true, predicate: ["dropArea"], descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: "<div class=\"form-group\" role=\"group\" [attr.aria-labelledby]=\"id + '-fieldset'\">\n <label id=\"{{ id + '-fieldset' }}\" *ngIf=\"to.label || to.description\">\n {{ to.label | humanize }}\n <button\n class=\"btn-help btn-help--sm\"\n type=\"button\"\n [attr.aria-label]=\"'Help' | translate\"\n [popover]=\"to.description\"\n placement=\"right\"\n triggers=\"focus\"\n *ngIf=\"to.description\"\n ></button>\n </label>\n\n <c8y-drop-area #dropArea class=\"drop-area-sm\" [formControl]=\"formControl\"></c8y-drop-area>\n</div>\n", dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.PopoverDirective, selector: "[popover]", inputs: ["adaptivePosition", "boundariesElement", "popover", "popoverContext", "popoverTitle", "placement", "outsideClick", "triggers", "container", "containerClass", "isOpen", "delay"], outputs: ["onShown", "onHidden"], exportAs: ["bs-popover"] }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "component", type: i4.DropAreaComponent, selector: "c8y-drop-area", inputs: ["formControl", "title", "message", "icon", "loadingMessage", "forceHideList", "alwaysShow", "clickToOpen", "loading", "progress", "maxAllowedFiles", "files", "maxFileSizeInMegaBytes", "accept"], outputs: ["dropped"] }, { kind: "pipe", type: i5.C8yTranslatePipe, name: "translate" }, { kind: "pipe", type: i6.HumanizePipe, name: "humanize" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
48
49
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: FieldFile, decorators: [{
49
50
  type: Component,
50
51
  args: [{ selector: 'c8y-field-file', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"form-group\" role=\"group\" [attr.aria-labelledby]=\"id + '-fieldset'\">\n <label id=\"{{ id + '-fieldset' }}\" *ngIf=\"to.label || to.description\">\n {{ to.label | humanize }}\n <button\n class=\"btn-help btn-help--sm\"\n type=\"button\"\n [attr.aria-label]=\"'Help' | translate\"\n [popover]=\"to.description\"\n placement=\"right\"\n triggers=\"focus\"\n *ngIf=\"to.description\"\n ></button>\n </label>\n\n <c8y-drop-area #dropArea class=\"drop-area-sm\" [formControl]=\"formControl\"></c8y-drop-area>\n</div>\n" }]
@@ -52,4 +53,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImpor
52
53
  type: ViewChild,
53
54
  args: ['dropArea', { static: true }]
54
55
  }] } });
55
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmlsZS50eXBlLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL2NvcmUvZHluYW1pYy1mb3Jtcy9maWxlL2ZpbGUudHlwZS5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9jb3JlL2R5bmFtaWMtZm9ybXMvZmlsZS9maWxlLnR5cGUuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLHVCQUF1QixFQUFFLFNBQVMsRUFBVSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDdEYsT0FBTyxFQUFnQixTQUFTLEVBQUUsTUFBTSxrQkFBa0IsQ0FBQztBQUMzRCxPQUFPLEVBQUUsR0FBRyxFQUFFLE1BQU0sUUFBUSxDQUFDO0FBQzdCLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLGlCQUFpQixDQUFDOzs7Ozs7OztBQU9wRCxNQUFNLE9BQU8sU0FBVSxTQUFRLFNBQVM7SUFMeEM7O1FBZVcsbUJBQWMsR0FBYTtZQUNsQyxPQUFPO1lBQ1AsU0FBUztZQUNULE1BQU07WUFDTixnQkFBZ0I7WUFDaEIsWUFBWTtZQUNaLGFBQWE7WUFDYixTQUFTO1lBQ1QsVUFBVTtZQUNWLGlCQUFpQjtZQUNqQixPQUFPO1lBQ1AsUUFBUTtTQUNULENBQUM7S0FhSDtJQVRDLFFBQVE7UUFDTixNQUFNLElBQUksR0FBRyxLQUFLLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7UUFDN0QsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUM1QixJQUFJLENBQUMsUUFBUSxDQUFDLEtBQUssR0FBRyxHQUFHLENBQUMsSUFBSSxDQUFDLEtBQUssRUFBRSxJQUFJLENBQUMsQ0FBQztRQUU1QyxJQUFJLENBQUMsY0FBYzthQUNoQixNQUFNLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLEtBQUssQ0FBQyxLQUFLLFNBQVMsQ0FBQzthQUM3QyxPQUFPLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDLEdBQUcsSUFBSSxDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDL0QsQ0FBQzs7QUFqQ2UsZ0JBQU0sR0FBaUI7SUFDckMsS0FBSyxFQUFFO1FBQ0w7WUFDRSxJQUFJLEVBQUUsTUFBTTtZQUNaLFNBQVMsRUFBRSxTQUFTO1NBQ3JCO0tBQ0Y7Q0FDRCxDQUFBO3NHQVJTLFNBQVM7MEZBQVQsU0FBUyxpTUNWdEIsa2tCQWdCQTsyRkROYSxTQUFTO2tCQUxyQixTQUFTOytCQUNFLGdCQUFnQixtQkFFVCx1QkFBdUIsQ0FBQyxNQUFNOzhCQTBCTixRQUFRO3NCQUFoRCxTQUFTO3VCQUFDLFVBQVUsRUFBRSxFQUFFLE1BQU0sRUFBRSxJQUFJLEVBQUUiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSwgQ29tcG9uZW50LCBPbkluaXQsIFZpZXdDaGlsZCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29uZmlnT3B0aW9uLCBGaWVsZFR5cGUgfSBmcm9tICdAbmd4LWZvcm1seS9jb3JlJztcbmltcG9ydCB7IGdldCB9IGZyb20gJ2xvZGFzaCc7XG5pbXBvcnQgeyBEcm9wQXJlYUNvbXBvbmVudCB9IGZyb20gJy4uLy4uL2Ryb3AtYXJlYSc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2M4eS1maWVsZC1maWxlJyxcbiAgdGVtcGxhdGVVcmw6ICcuL2ZpbGUudHlwZS5jb21wb25lbnQuaHRtbCcsXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoXG59KVxuZXhwb3J0IGNsYXNzIEZpZWxkRmlsZSBleHRlbmRzIEZpZWxkVHlwZSBpbXBsZW1lbnRzIE9uSW5pdCB7XG4gIHN0YXRpYyByZWFkb25seSBDT05GSUc6IENvbmZpZ09wdGlvbiA9IHtcbiAgICB0eXBlczogW1xuICAgICAge1xuICAgICAgICBuYW1lOiAnZmlsZScsXG4gICAgICAgIGNvbXBvbmVudDogRmllbGRGaWxlXG4gICAgICB9XG4gICAgXVxuICB9O1xuXG4gIHJlYWRvbmx5IGRyb3BBcmVhSW5wdXRzOiBzdHJpbmdbXSA9IFtcbiAgICAndGl0bGUnLFxuICAgICdtZXNzYWdlJyxcbiAgICAnaWNvbicsXG4gICAgJ2xvYWRpbmdNZXNzYWdlJyxcbiAgICAnYWx3YXlzU2hvdycsXG4gICAgJ2NsaWNrVG9PcGVuJyxcbiAgICAnbG9hZGluZycsXG4gICAgJ3Byb2dyZXNzJyxcbiAgICAnbWF4QWxsb3dlZEZpbGVzJyxcbiAgICAnZmlsZXMnLFxuICAgICdhY2NlcHQnXG4gIF07XG5cbiAgQFZpZXdDaGlsZCgnZHJvcEFyZWEnLCB7IHN0YXRpYzogdHJ1ZSB9KSBkcm9wQXJlYTogRHJvcEFyZWFDb21wb25lbnQ7XG5cbiAgbmdPbkluaXQoKTogdm9pZCB7XG4gICAgY29uc3Qga2V5cyA9IEFycmF5LmlzQXJyYXkodGhpcy5rZXkpID8gdGhpcy5rZXkgOiBbdGhpcy5rZXldO1xuICAgIGNvbnN0IHBhdGggPSBrZXlzLmpvaW4oJy4nKTtcbiAgICB0aGlzLmRyb3BBcmVhLmZpbGVzID0gZ2V0KHRoaXMubW9kZWwsIHBhdGgpO1xuXG4gICAgdGhpcy5kcm9wQXJlYUlucHV0c1xuICAgICAgLmZpbHRlcihpbnB1dCA9PiB0aGlzLnRvW2lucHV0XSAhPT0gdW5kZWZpbmVkKVxuICAgICAgLmZvckVhY2goaW5wdXQgPT4gKHRoaXMuZHJvcEFyZWFbaW5wdXRdID0gdGhpcy50b1tpbnB1dF0pKTtcbiAgfVxufVxuIiwiPGRpdiBjbGFzcz1cImZvcm0tZ3JvdXBcIiByb2xlPVwiZ3JvdXBcIiBbYXR0ci5hcmlhLWxhYmVsbGVkYnldPVwiaWQgKyAnLWZpZWxkc2V0J1wiPlxuICA8bGFiZWwgaWQ9XCJ7eyBpZCArICctZmllbGRzZXQnIH19XCIgKm5nSWY9XCJ0by5sYWJlbCB8fCB0by5kZXNjcmlwdGlvblwiPlxuICAgIHt7IHRvLmxhYmVsIHwgaHVtYW5pemUgfX1cbiAgICA8YnV0dG9uXG4gICAgICBjbGFzcz1cImJ0bi1oZWxwIGJ0bi1oZWxwLS1zbVwiXG4gICAgICB0eXBlPVwiYnV0dG9uXCJcbiAgICAgIFthdHRyLmFyaWEtbGFiZWxdPVwiJ0hlbHAnIHwgdHJhbnNsYXRlXCJcbiAgICAgIFtwb3BvdmVyXT1cInRvLmRlc2NyaXB0aW9uXCJcbiAgICAgIHBsYWNlbWVudD1cInJpZ2h0XCJcbiAgICAgIHRyaWdnZXJzPVwiZm9jdXNcIlxuICAgICAgKm5nSWY9XCJ0by5kZXNjcmlwdGlvblwiXG4gICAgPjwvYnV0dG9uPlxuICA8L2xhYmVsPlxuXG4gIDxjOHktZHJvcC1hcmVhICNkcm9wQXJlYSBjbGFzcz1cImRyb3AtYXJlYS1zbVwiIFtmb3JtQ29udHJvbF09XCJmb3JtQ29udHJvbFwiPjwvYzh5LWRyb3AtYXJlYT5cbjwvZGl2PlxuIl19
56
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmlsZS50eXBlLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL2NvcmUvZHluYW1pYy1mb3Jtcy9maWxlL2ZpbGUudHlwZS5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9jb3JlL2R5bmFtaWMtZm9ybXMvZmlsZS9maWxlLnR5cGUuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLHVCQUF1QixFQUFFLFNBQVMsRUFBVSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDdEYsT0FBTyxFQUFnQixTQUFTLEVBQUUsTUFBTSxrQkFBa0IsQ0FBQztBQUMzRCxPQUFPLEVBQUUsR0FBRyxFQUFFLE1BQU0sUUFBUSxDQUFDO0FBQzdCLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLGlCQUFpQixDQUFDOzs7Ozs7OztBQU9wRCxNQUFNLE9BQU8sU0FBVSxTQUFRLFNBQVM7SUFMeEM7O1FBZVcsbUJBQWMsR0FBYTtZQUNsQyxPQUFPO1lBQ1AsU0FBUztZQUNULE1BQU07WUFDTixnQkFBZ0I7WUFDaEIsWUFBWTtZQUNaLGFBQWE7WUFDYixTQUFTO1lBQ1QsVUFBVTtZQUNWLGlCQUFpQjtZQUNqQixPQUFPO1lBQ1AsUUFBUTtZQUNSLHdCQUF3QjtTQUN6QixDQUFDO0tBYUg7SUFUQyxRQUFRO1FBQ04sTUFBTSxJQUFJLEdBQUcsS0FBSyxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO1FBQzdELE1BQU0sSUFBSSxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7UUFDNUIsSUFBSSxDQUFDLFFBQVEsQ0FBQyxLQUFLLEdBQUcsR0FBRyxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsSUFBSSxDQUFDLENBQUM7UUFFNUMsSUFBSSxDQUFDLGNBQWM7YUFDaEIsTUFBTSxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsS0FBSyxTQUFTLENBQUM7YUFDN0MsT0FBTyxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUMsQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLEtBQUssQ0FBQyxHQUFHLElBQUksQ0FBQyxFQUFFLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQy9ELENBQUM7O0FBbENlLGdCQUFNLEdBQWlCO0lBQ3JDLEtBQUssRUFBRTtRQUNMO1lBQ0UsSUFBSSxFQUFFLE1BQU07WUFDWixTQUFTLEVBQUUsU0FBUztTQUNyQjtLQUNGO0NBQ0QsQ0FBQTtzR0FSUyxTQUFTOzBGQUFULFNBQVMsaU1DVnRCLGtrQkFnQkE7MkZETmEsU0FBUztrQkFMckIsU0FBUzsrQkFDRSxnQkFBZ0IsbUJBRVQsdUJBQXVCLENBQUMsTUFBTTs4QkEyQk4sUUFBUTtzQkFBaEQsU0FBUzt1QkFBQyxVQUFVLEVBQUUsRUFBRSxNQUFNLEVBQUUsSUFBSSxFQUFFIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENvbXBvbmVudCwgT25Jbml0LCBWaWV3Q2hpbGQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IENvbmZpZ09wdGlvbiwgRmllbGRUeXBlIH0gZnJvbSAnQG5neC1mb3JtbHkvY29yZSc7XG5pbXBvcnQgeyBnZXQgfSBmcm9tICdsb2Rhc2gnO1xuaW1wb3J0IHsgRHJvcEFyZWFDb21wb25lbnQgfSBmcm9tICcuLi8uLi9kcm9wLWFyZWEnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdjOHktZmllbGQtZmlsZScsXG4gIHRlbXBsYXRlVXJsOiAnLi9maWxlLnR5cGUuY29tcG9uZW50Lmh0bWwnLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaFxufSlcbmV4cG9ydCBjbGFzcyBGaWVsZEZpbGUgZXh0ZW5kcyBGaWVsZFR5cGUgaW1wbGVtZW50cyBPbkluaXQge1xuICBzdGF0aWMgcmVhZG9ubHkgQ09ORklHOiBDb25maWdPcHRpb24gPSB7XG4gICAgdHlwZXM6IFtcbiAgICAgIHtcbiAgICAgICAgbmFtZTogJ2ZpbGUnLFxuICAgICAgICBjb21wb25lbnQ6IEZpZWxkRmlsZVxuICAgICAgfVxuICAgIF1cbiAgfTtcblxuICByZWFkb25seSBkcm9wQXJlYUlucHV0czogc3RyaW5nW10gPSBbXG4gICAgJ3RpdGxlJyxcbiAgICAnbWVzc2FnZScsXG4gICAgJ2ljb24nLFxuICAgICdsb2FkaW5nTWVzc2FnZScsXG4gICAgJ2Fsd2F5c1Nob3cnLFxuICAgICdjbGlja1RvT3BlbicsXG4gICAgJ2xvYWRpbmcnLFxuICAgICdwcm9ncmVzcycsXG4gICAgJ21heEFsbG93ZWRGaWxlcycsXG4gICAgJ2ZpbGVzJyxcbiAgICAnYWNjZXB0JyxcbiAgICAnbWF4RmlsZVNpemVJbk1lZ2FCeXRlcydcbiAgXTtcblxuICBAVmlld0NoaWxkKCdkcm9wQXJlYScsIHsgc3RhdGljOiB0cnVlIH0pIGRyb3BBcmVhOiBEcm9wQXJlYUNvbXBvbmVudDtcblxuICBuZ09uSW5pdCgpOiB2b2lkIHtcbiAgICBjb25zdCBrZXlzID0gQXJyYXkuaXNBcnJheSh0aGlzLmtleSkgPyB0aGlzLmtleSA6IFt0aGlzLmtleV07XG4gICAgY29uc3QgcGF0aCA9IGtleXMuam9pbignLicpO1xuICAgIHRoaXMuZHJvcEFyZWEuZmlsZXMgPSBnZXQodGhpcy5tb2RlbCwgcGF0aCk7XG5cbiAgICB0aGlzLmRyb3BBcmVhSW5wdXRzXG4gICAgICAuZmlsdGVyKGlucHV0ID0+IHRoaXMudG9baW5wdXRdICE9PSB1bmRlZmluZWQpXG4gICAgICAuZm9yRWFjaChpbnB1dCA9PiAodGhpcy5kcm9wQXJlYVtpbnB1dF0gPSB0aGlzLnRvW2lucHV0XSkpO1xuICB9XG59XG4iLCI8ZGl2IGNsYXNzPVwiZm9ybS1ncm91cFwiIHJvbGU9XCJncm91cFwiIFthdHRyLmFyaWEtbGFiZWxsZWRieV09XCJpZCArICctZmllbGRzZXQnXCI+XG4gIDxsYWJlbCBpZD1cInt7IGlkICsgJy1maWVsZHNldCcgfX1cIiAqbmdJZj1cInRvLmxhYmVsIHx8IHRvLmRlc2NyaXB0aW9uXCI+XG4gICAge3sgdG8ubGFiZWwgfCBodW1hbml6ZSB9fVxuICAgIDxidXR0b25cbiAgICAgIGNsYXNzPVwiYnRuLWhlbHAgYnRuLWhlbHAtLXNtXCJcbiAgICAgIHR5cGU9XCJidXR0b25cIlxuICAgICAgW2F0dHIuYXJpYS1sYWJlbF09XCInSGVscCcgfCB0cmFuc2xhdGVcIlxuICAgICAgW3BvcG92ZXJdPVwidG8uZGVzY3JpcHRpb25cIlxuICAgICAgcGxhY2VtZW50PVwicmlnaHRcIlxuICAgICAgdHJpZ2dlcnM9XCJmb2N1c1wiXG4gICAgICAqbmdJZj1cInRvLmRlc2NyaXB0aW9uXCJcbiAgICA+PC9idXR0b24+XG4gIDwvbGFiZWw+XG5cbiAgPGM4eS1kcm9wLWFyZWEgI2Ryb3BBcmVhIGNsYXNzPVwiZHJvcC1hcmVhLXNtXCIgW2Zvcm1Db250cm9sXT1cImZvcm1Db250cm9sXCI+PC9jOHktZHJvcC1hcmVhPlxuPC9kaXY+XG4iXX0=
@@ -81,6 +81,17 @@ export class C8yJSONSchema extends FormlyJsonschema {
81
81
  }
82
82
  };
83
83
  }
84
+ /** max size file validation` */
85
+ if (mapSource.maxSize) {
86
+ result = {
87
+ ...result,
88
+ type: 'file',
89
+ props: {
90
+ ...(result.props || {}),
91
+ maxFileSizeInMegaBytes: mapSource.maxSize
92
+ }
93
+ };
94
+ }
84
95
  /** Provides a way to use templateOptions in JSONSchema forms */
85
96
  if (mapSource.templateOptions) {
86
97
  result = {
@@ -104,4 +115,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImpor
104
115
  providedIn: 'root'
105
116
  }]
106
117
  }] });
107
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"c8y-json-schema.service.js","sourceRoot":"","sources":["../../../../../core/dynamic-forms/json-schema/c8y-json-schema.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { FormlyFieldConfig, FormlyTemplateOptions } from '@ngx-formly/core';\nimport { FormlyJsonschema } from '@ngx-formly/core/json-schema';\nimport { JSONSchema7 } from 'json-schema';\nimport { omit } from 'lodash-es';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class C8yJSONSchema extends FormlyJsonschema {\n  toFieldConfig(schema: JSONSchema7, options?: any): FormlyFieldConfig {\n    const config: FormlyFieldConfig = super.toFieldConfig(\n      schema,\n      this.c8yOptions((options || {}).map)\n    );\n    return config;\n  }\n\n  private c8yOptions(\n    map: (mappedField: FormlyFieldConfig, mapSource: JSONSchema7) => FormlyFieldConfig\n  ) {\n    return {\n      map(\n        mappedField: FormlyFieldConfig,\n        mapSource: JSONSchema7 & {\n          allowedFileTypes: string[];\n          templateOptions: FormlyTemplateOptions;\n        }\n      ) {\n        let result: FormlyFieldConfig = mappedField;\n\n        /** `const` type is represented as hidden input */\n        if (mapSource.const) {\n          result = {\n            ...result,\n            defaultValue: mapSource.const,\n            type: 'input',\n            props: {\n              ...result.props,\n              type: 'hidden'\n            }\n          };\n        }\n\n        /** `writeOnly` field represented as password input */\n        if (mapSource.writeOnly) {\n          result = {\n            ...result,\n            props: {\n              ...(result.props || {}),\n              attributes: { ...result.props?.attributes, type: 'password', autocomplete: 'off' }\n            }\n          };\n        }\n\n        /** `examples` values used as placeholder */\n        if (mapSource.examples) {\n          result = {\n            ...result,\n            props: {\n              ...(result.props || {}),\n              placeholder: (mapSource.examples as string[]).join(', ')\n            }\n          };\n        }\n\n        /** `enum` field represented as radio button group */\n        if (mapSource.enum) {\n          result = {\n            ...result,\n            type: 'radio'\n          };\n        }\n\n        /** fields with `contentMediaType` or `contentEncoding` are represented as file inputs */\n        if (mapSource.contentMediaType || mapSource.contentEncoding) {\n          result = {\n            ...result,\n            type: 'file',\n            props: {\n              ...(result.props || {}),\n              accept: mapSource.contentMediaType,\n              maxAllowedFiles: 1,\n              contentEncoding: mapSource.contentEncoding\n            }\n          };\n\n          // TODO: Remove this line when base64 encoding is done by file.type.component\n          // and control's value would pass 'string' type validator.\n          if (mapSource.contentEncoding === 'base64') {\n            result.validators = omit(result.validators, ['type']);\n          }\n        }\n\n        /** file extension validation for fields with `allowedFileTypes` */\n        if ((mapSource as any).allowedFileTypes) {\n          result = {\n            ...result,\n            type: 'file',\n            props: {\n              ...(result.props || {}),\n              ...(mapSource.allowedFileTypes && {\n                accept: mapSource.allowedFileTypes.join(',')\n              })\n            }\n          };\n        }\n\n        /** Provides a way to use templateOptions in JSONSchema forms */\n        if (mapSource.templateOptions) {\n          result = {\n            ...result,\n            props: {\n              ...mapSource.templateOptions,\n              ...result.props\n            }\n          };\n        }\n\n        return map ? map(result, mapSource) : result;\n      }\n    };\n  }\n}\n"]}
118
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"c8y-json-schema.service.js","sourceRoot":"","sources":["../../../../../core/dynamic-forms/json-schema/c8y-json-schema.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { FormlyFieldConfig, FormlyTemplateOptions } from '@ngx-formly/core';\nimport { FormlyJsonschema } from '@ngx-formly/core/json-schema';\nimport { JSONSchema7 } from 'json-schema';\nimport { omit } from 'lodash-es';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class C8yJSONSchema extends FormlyJsonschema {\n  toFieldConfig(schema: JSONSchema7, options?: any): FormlyFieldConfig {\n    const config: FormlyFieldConfig = super.toFieldConfig(\n      schema,\n      this.c8yOptions((options || {}).map)\n    );\n    return config;\n  }\n\n  private c8yOptions(\n    map: (mappedField: FormlyFieldConfig, mapSource: JSONSchema7) => FormlyFieldConfig\n  ) {\n    return {\n      map(\n        mappedField: FormlyFieldConfig,\n        mapSource: JSONSchema7 & {\n          allowedFileTypes: string[];\n          templateOptions: FormlyTemplateOptions;\n        }\n      ) {\n        let result: FormlyFieldConfig = mappedField;\n\n        /** `const` type is represented as hidden input */\n        if (mapSource.const) {\n          result = {\n            ...result,\n            defaultValue: mapSource.const,\n            type: 'input',\n            props: {\n              ...result.props,\n              type: 'hidden'\n            }\n          };\n        }\n\n        /** `writeOnly` field represented as password input */\n        if (mapSource.writeOnly) {\n          result = {\n            ...result,\n            props: {\n              ...(result.props || {}),\n              attributes: { ...result.props?.attributes, type: 'password', autocomplete: 'off' }\n            }\n          };\n        }\n\n        /** `examples` values used as placeholder */\n        if (mapSource.examples) {\n          result = {\n            ...result,\n            props: {\n              ...(result.props || {}),\n              placeholder: (mapSource.examples as string[]).join(', ')\n            }\n          };\n        }\n\n        /** `enum` field represented as radio button group */\n        if (mapSource.enum) {\n          result = {\n            ...result,\n            type: 'radio'\n          };\n        }\n\n        /** fields with `contentMediaType` or `contentEncoding` are represented as file inputs */\n        if (mapSource.contentMediaType || mapSource.contentEncoding) {\n          result = {\n            ...result,\n            type: 'file',\n            props: {\n              ...(result.props || {}),\n              accept: mapSource.contentMediaType,\n              maxAllowedFiles: 1,\n              contentEncoding: mapSource.contentEncoding\n            }\n          };\n\n          // TODO: Remove this line when base64 encoding is done by file.type.component\n          // and control's value would pass 'string' type validator.\n          if (mapSource.contentEncoding === 'base64') {\n            result.validators = omit(result.validators, ['type']);\n          }\n        }\n\n        /** file extension validation for fields with `allowedFileTypes` */\n        if ((mapSource as any).allowedFileTypes) {\n          result = {\n            ...result,\n            type: 'file',\n            props: {\n              ...(result.props || {}),\n              ...(mapSource.allowedFileTypes && {\n                accept: mapSource.allowedFileTypes.join(',')\n              })\n            }\n          };\n        }\n        /** max size file validation` */\n        if ((mapSource as any).maxSize) {\n          result = {\n            ...result,\n            type: 'file',\n            props: {\n              ...(result.props || {}),\n              maxFileSizeInMegaBytes: (mapSource as any).maxSize\n            }\n          };\n        }\n\n        /** Provides a way to use templateOptions in JSONSchema forms */\n        if (mapSource.templateOptions) {\n          result = {\n            ...result,\n            props: {\n              ...mapSource.templateOptions,\n              ...result.props\n            }\n          };\n        }\n\n        return map ? map(result, mapSource) : result;\n      }\n    };\n  }\n}\n"]}