@elderbyte/ngx-starter 18.1.0-beta2 → 18.2.0-beta

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 (25) hide show
  1. package/esm2022/lib/common/utils/input-utils.mjs +6 -0
  2. package/esm2022/lib/components/data-view/table/activation/elder-table-activation.directive.mjs +14 -7
  3. package/esm2022/lib/components/files/blob-viewer/elder-blob-viewer.component.mjs +17 -23
  4. package/esm2022/lib/components/files/elder-file-select.directive.mjs +83 -85
  5. package/esm2022/lib/components/files/listing/file-entry.mjs +15 -30
  6. package/esm2022/lib/components/files/listing/file-listing-rx.mjs +70 -30
  7. package/esm2022/lib/components/forms/directives/validation/validators/elder-multiple-of.validator.mjs +17 -13
  8. package/fesm2022/elderbyte-ngx-starter.mjs +154 -425
  9. package/fesm2022/elderbyte-ngx-starter.mjs.map +1 -1
  10. package/lib/common/utils/input-utils.d.ts +18 -0
  11. package/lib/components/data-view/table/activation/elder-table-activation.directive.d.ts +4 -1
  12. package/lib/components/files/blob-viewer/elder-blob-viewer.component.d.ts +3 -3
  13. package/lib/components/files/elder-file-select.directive.d.ts +23 -16
  14. package/lib/components/files/listing/file-entry.d.ts +1 -10
  15. package/lib/components/files/listing/file-listing-rx.d.ts +6 -3
  16. package/lib/components/forms/directives/validation/validators/elder-multiple-of.validator.d.ts +2 -4
  17. package/package.json +1 -1
  18. package/esm2022/lib/components/files/elder-file-select-input.mjs +0 -119
  19. package/esm2022/lib/components/files/file-system-api.mjs +0 -96
  20. package/esm2022/lib/components/files/listing/file-listing-system-handle.mjs +0 -35
  21. package/esm2022/lib/components/files/listing/file-listing-webkit.mjs +0 -77
  22. package/lib/components/files/elder-file-select-input.d.ts +0 -59
  23. package/lib/components/files/file-system-api.d.ts +0 -86
  24. package/lib/components/files/listing/file-listing-system-handle.d.ts +0 -21
  25. package/lib/components/files/listing/file-listing-webkit.d.ts +0 -26
@@ -0,0 +1,18 @@
1
+ /**
2
+ * To support empty input attributes and fall back to a default value.
3
+ * This is centralized in case Angular ever decides to change empty attributes
4
+ * to another value than an empty string (undefined, null etc.)
5
+ *
6
+ * e.g.
7
+ * <some-component directiveA> </some-component>
8
+ *
9
+ * .@Input("directiveA")
10
+ * public set valueA(input: SomeType | InputUtils): SomeType {
11
+ * return input instanceOf InputUtils ? SomeType.Empty() : input;
12
+ * }
13
+ *
14
+ */
15
+ export type NoInputArgs = '';
16
+ export declare class InputUtils {
17
+ static isNoInputArgs(input: any): input is NoInputArgs;
18
+ }
@@ -6,6 +6,7 @@ import { ElderTableActivationOptions } from './elder-table-activation-options';
6
6
  import { ItemActivationOptions } from './item-activation-options';
7
7
  import { ItemActivationEvent } from './item-activation-event';
8
8
  import { ElderActivationContext } from './elder-activation-context';
9
+ import { NoInputArgs } from '../../../../common/utils/input-utils';
9
10
  import * as i0 from "@angular/core";
10
11
  export declare class ElderTableActivationDirective<T> implements ElderActivationContext<T>, AfterViewInit, OnDestroy {
11
12
  private readonly elderTable;
@@ -20,7 +21,7 @@ export declare class ElderTableActivationDirective<T> implements ElderActivation
20
21
  private _data;
21
22
  readonly activeItem$: BehaviorSubject<T>;
22
23
  readonly rows$: BehaviorSubject<ElderTableRowDirective<T>[]>;
23
- activationOptions: ElderTableActivationOptions;
24
+ private _activationOptions;
24
25
  /***************************************************************************
25
26
  * *
26
27
  * Constructor *
@@ -45,6 +46,8 @@ export declare class ElderTableActivationDirective<T> implements ElderActivation
45
46
  * Properties *
46
47
  * *
47
48
  **************************************************************************/
49
+ set activationOptions(options: ElderTableActivationOptions | NoInputArgs);
50
+ get activationOptions(): ElderTableActivationOptions;
48
51
  get activeItemEventChange(): Observable<ItemActivationEvent<T>>;
49
52
  get activeItemChange(): Observable<T>;
50
53
  get activeItem(): T;
@@ -4,6 +4,7 @@ import { BlobUrl } from '../blob-url';
4
4
  import { BehaviorSubject } from 'rxjs';
5
5
  import { HttpClientPristine } from '../../../common/http/http-client-pristine';
6
6
  import { ElderToastService } from '../../toasts/elder-toast.service';
7
+ import { BooleanInput } from '@angular/cdk/coercion';
7
8
  import * as i0 from "@angular/core";
8
9
  declare class BlobContext {
9
10
  readonly safeUrl: SafeUrl;
@@ -43,7 +44,6 @@ export declare class ElderBlobViewerComponent implements OnInit, OnDestroy {
43
44
  private readonly _url$;
44
45
  private readonly _mimeType$;
45
46
  private readonly destroy$;
46
- private _displayUnknownAsText;
47
47
  /***************************************************************************
48
48
  * *
49
49
  * Constructor *
@@ -66,7 +66,7 @@ export declare class ElderBlobViewerComponent implements OnInit, OnDestroy {
66
66
  * Attempt to display unknown blobs as text. (i.e. mime-type is octet-stream)
67
67
  * @param unknownAsText
68
68
  */
69
- set displayUnknownAsText(unknownAsText: boolean);
69
+ displayUnknownAsText: import("@angular/core").InputSignalWithTransform<boolean, BooleanInput>;
70
70
  /**
71
71
  * Set a in memory blob to display.
72
72
  * @param blob
@@ -102,6 +102,6 @@ export declare class ElderBlobViewerComponent implements OnInit, OnDestroy {
102
102
  private canDisplayAsText;
103
103
  private downloadAsText;
104
104
  static ɵfac: i0.ɵɵFactoryDeclaration<ElderBlobViewerComponent, never>;
105
- static ɵcmp: i0.ɵɵComponentDeclaration<ElderBlobViewerComponent, "elder-blob-viewer", never, { "displayUnknownAsText": { "alias": "displayUnknownAsText"; "required": false; }; "blob": { "alias": "blob"; "required": false; }; "blobUrl": { "alias": "blobUrl"; "required": false; }; "url": { "alias": "url"; "required": false; }; "mimeType": { "alias": "mimeType"; "required": false; }; }, {}, never, never, true, never>;
105
+ static ɵcmp: i0.ɵɵComponentDeclaration<ElderBlobViewerComponent, "elder-blob-viewer", never, { "displayUnknownAsText": { "alias": "displayUnknownAsText"; "required": false; "isSignal": true; }; "blob": { "alias": "blob"; "required": false; }; "blobUrl": { "alias": "blobUrl"; "required": false; }; "url": { "alias": "url"; "required": false; }; "mimeType": { "alias": "mimeType"; "required": false; }; }, {}, never, never, true, never>;
106
106
  }
107
107
  export {};
@@ -1,31 +1,29 @@
1
- import { DestroyRef, ElementRef, EventEmitter, OnDestroy, OnInit, Renderer2 } from '@angular/core';
1
+ import { ElementRef, EventEmitter, OnDestroy, OnInit, Renderer2 } from '@angular/core';
2
2
  import { BooleanInput } from '@angular/cdk/coercion';
3
3
  import { FileEntry } from './listing/file-entry';
4
- import { FileSystemDirectoryPickerOptions, FileSystemFilePickerOptions } from './file-system-api';
5
- import { Observable } from 'rxjs';
6
4
  import * as i0 from "@angular/core";
7
5
  export declare class ElderFileSelectDirective implements OnInit, OnDestroy {
8
- private renderer;
9
6
  private el;
10
- private destroyRef;
7
+ private renderer;
11
8
  /***************************************************************************
12
9
  * *
13
10
  * Fields *
14
11
  * *
15
12
  **************************************************************************/
16
13
  private readonly logger;
14
+ private _fileInput;
15
+ private _unlisten;
17
16
  readonly elderFileSelectChange: EventEmitter<FileEntry[]>;
18
17
  readonly elderSingleFileSelectChange: EventEmitter<File>;
19
- private _legacyInput;
20
- private _filePickerOptions;
21
- private _directoryPickerOptions;
22
- private _useDirectoryPicker;
18
+ private _multiple;
19
+ private _directory;
20
+ private _accept;
23
21
  /***************************************************************************
24
22
  * *
25
23
  * Constructor *
26
24
  * *
27
25
  **************************************************************************/
28
- constructor(renderer: Renderer2, el: ElementRef, destroyRef: DestroyRef);
26
+ constructor(el: ElementRef, renderer: Renderer2);
29
27
  /***************************************************************************
30
28
  * *
31
29
  * Life Cycle *
@@ -39,9 +37,6 @@ export declare class ElderFileSelectDirective implements OnInit, OnDestroy {
39
37
  * *
40
38
  **************************************************************************/
41
39
  set elderFileSelect(value: string);
42
- private buildTypeOption;
43
- private isFileExtension;
44
- private isMimeType;
45
40
  set elderFileSelectMultiple(value: BooleanInput);
46
41
  /**
47
42
  * Allow the user to select a directory. All files that are recursively contained are selected.
@@ -55,14 +50,26 @@ export declare class ElderFileSelectDirective implements OnInit, OnDestroy {
55
50
  * *
56
51
  **************************************************************************/
57
52
  onClick(event: any): void;
53
+ openFileSelectDialog(): void;
58
54
  /***************************************************************************
59
55
  * *
60
56
  * Private methods *
61
57
  * *
62
58
  **************************************************************************/
63
- private pushFileChanges;
64
- openFilePicker(pickerOpts: FileSystemFilePickerOptions): Observable<FileEntry[]>;
65
- openDirectoryPicker(pickerOpts: FileSystemDirectoryPickerOptions): Observable<FileEntry[]>;
59
+ /**
60
+ * <input type="file"
61
+ * hidden #fileInput
62
+ * [multiple]="multiple"
63
+ * [accept]="accept"
64
+ * (change)="fileInputChanged()"
65
+ * />
66
+ */
67
+ private createFileSelect;
68
+ private removeFileSelect;
69
+ private fileInputChanged;
70
+ private clearFileList;
71
+ private emitFileList;
72
+ private toFileEntries;
66
73
  static ɵfac: i0.ɵɵFactoryDeclaration<ElderFileSelectDirective, never>;
67
74
  static ɵdir: i0.ɵɵDirectiveDeclaration<ElderFileSelectDirective, "[elderFileSelect]", never, { "elderFileSelect": { "alias": "elderFileSelect"; "required": false; }; "elderFileSelectMultiple": { "alias": "elderFileSelectMultiple"; "required": false; }; "elderFileSelectDirectory": { "alias": "elderFileSelectDirectory"; "required": false; }; }, { "elderFileSelectChange": "elderFileSelectChange"; "elderSingleFileSelectChange": "elderSingleFileSelectChange"; }, never, never, true, never>;
68
75
  }
@@ -1,4 +1,3 @@
1
- import { Observable } from 'rxjs';
2
1
  /**
3
2
  * Represents a file and the relative path where the user has selected it.
4
3
  * This is useful if the user has selected a folder and files
@@ -11,7 +10,6 @@ export declare class FileEntry {
11
10
  * to this file. Only relevant if the user has selected a folder.
12
11
  */
13
12
  readonly relativeParent: string | null;
14
- readonly fileHandle?: FileSystemFileHandle;
15
13
  /**
16
14
  * Creates a file Entry without an explicit relative parent.
17
15
  *
@@ -20,16 +18,10 @@ export declare class FileEntry {
20
18
  * which is also supported by this method.
21
19
  */
22
20
  static ofFile(file: File): FileEntry;
23
- /**
24
- * Creates an Observable of a File Entry by resolving its FileSystemFileHandle.
25
- * @param fileHandle
26
- * @param rootDirectory is needed to build a relative parent
27
- */
28
- static ofFileHandle(fileHandle: FileSystemFileHandle, rootDirectory?: FileSystemDirectoryHandle): Observable<FileEntry>;
29
21
  /**
30
22
  * Creates a file Entry with a relative parent path
31
23
  */
32
- static relativeFile(file: File, relativeParent: string | null, fileHandle?: FileSystemFileHandle): FileEntry;
24
+ static relativeFile(file: File, relativeParent: string | null): FileEntry;
33
25
  static toFileMap(entries: FileEntry[]): Map<File, FileEntry>;
34
26
  static toFileArray(entries: FileEntry[]): File[];
35
27
  /***************************************************************************
@@ -56,5 +48,4 @@ export declare class FileEntry {
56
48
  * Returns a string which specifies the file's path relative to the directory selected by the user.
57
49
  */
58
50
  private static buildRelativePath;
59
- private static buildRelativeParent;
60
51
  }
@@ -9,12 +9,15 @@ export declare class FileListingRx {
9
9
  * *
10
10
  **************************************************************************/
11
11
  toFileList(transferItemList: DataTransferItemList): Observable<FileEntry[]>;
12
+ listFilesRecursive(entry: FileSystemEntry): Observable<FileEntry[]>;
12
13
  /***************************************************************************
13
14
  * *
14
15
  * Private methods *
15
16
  * *
16
17
  **************************************************************************/
17
- private resolveTransferItem;
18
- private legacyFileListing;
19
- private fallbackFileListing;
18
+ private listFilesRecursiveAsync;
19
+ private observableFile;
20
+ private trimStaringSlash;
21
+ private readEntries;
22
+ private consumeReaderToCompletion;
20
23
  }
@@ -21,9 +21,7 @@ export declare class ElderMultipleOfValidator implements Validator {
21
21
  * Fields *
22
22
  * *
23
23
  **************************************************************************/
24
- private _factor;
25
- set elderMultipleOf(value: number | number[] | null);
26
- get factor(): number | number[] | null;
24
+ factor: import("@angular/core").InputSignalWithTransform<number | number[], number | "" | number[]>;
27
25
  /***************************************************************************
28
26
  * *
29
27
  * Public API *
@@ -31,5 +29,5 @@ export declare class ElderMultipleOfValidator implements Validator {
31
29
  **************************************************************************/
32
30
  validate(control: AbstractControl): ValidationErrors | null;
33
31
  static ɵfac: i0.ɵɵFactoryDeclaration<ElderMultipleOfValidator, never>;
34
- static ɵdir: i0.ɵɵDirectiveDeclaration<ElderMultipleOfValidator, "[elderMultipleOf][formControlName],[elderMultipleOf][formControl],[elderMultipleOf][ngModel]", never, { "elderMultipleOf": { "alias": "elderMultipleOf"; "required": false; }; }, {}, never, never, true, never>;
32
+ static ɵdir: i0.ɵɵDirectiveDeclaration<ElderMultipleOfValidator, "[elderMultipleOf][formControlName],[elderMultipleOf][formControl],[elderMultipleOf][ngModel]", never, { "factor": { "alias": "elderMultipleOf"; "required": false; "isSignal": true; }; }, {}, never, never, true, never>;
35
33
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@elderbyte/ngx-starter",
3
- "version": "18.1.0-beta2",
3
+ "version": "18.2.0-beta",
4
4
  "peerDependencies": {
5
5
  "@angular/core": "^17.0.0 || ^18.0.0",
6
6
  "@angular/common": "^17.0.0 || ^18.0.0",
@@ -1,119 +0,0 @@
1
- import { FileEntry } from './listing/file-entry';
2
- import { coerceBooleanProperty } from '@angular/cdk/coercion';
3
- import { LoggerFactory } from '@elderbyte/ts-logger';
4
- import { Subject } from 'rxjs';
5
- export class ElderFileSelectInput {
6
- /***************************************************************************
7
- * *
8
- * Constructor *
9
- * *
10
- **************************************************************************/
11
- constructor(renderer, el) {
12
- this.renderer = renderer;
13
- this.el = el;
14
- /***************************************************************************
15
- * *
16
- * Fields *
17
- * *
18
- **************************************************************************/
19
- this.logger = LoggerFactory.getLogger(this.constructor.name);
20
- this._filesSelected = new Subject;
21
- }
22
- /***************************************************************************
23
- * *
24
- * Properties *
25
- * *
26
- **************************************************************************/
27
- get filesSelected() {
28
- return this._filesSelected;
29
- }
30
- set elderFileSelect(value) {
31
- this._accept = value;
32
- if (this._fileInput) {
33
- this.renderer.setProperty(this._fileInput, 'accept', value);
34
- }
35
- }
36
- set elderFileSelectMultiple(value) {
37
- this._multiple = coerceBooleanProperty(value);
38
- if (this._fileInput) {
39
- this.renderer.setProperty(this._fileInput, 'multiple', value);
40
- }
41
- }
42
- set elderFileSelectDirectory(value) {
43
- this._directory = coerceBooleanProperty(value);
44
- if (this._fileInput) {
45
- this.renderer.setAttribute(this._fileInput, 'webkitdirectory', 'true');
46
- this.renderer.setAttribute(this._fileInput, 'directory', 'true');
47
- }
48
- }
49
- /***************************************************************************
50
- * *
51
- * Public API *
52
- * *
53
- **************************************************************************/
54
- /**
55
- * <input type="file"
56
- * hidden #fileInput
57
- * [multiple]="multiple"
58
- * [accept]="accept"
59
- * (change)="fileInputChanged()"
60
- * />
61
- */
62
- createFileSelect() {
63
- this._fileInput = this.renderer.createElement('input');
64
- this.renderer.setAttribute(this._fileInput, 'hidden', 'true');
65
- this.renderer.setAttribute(this._fileInput, 'type', 'file');
66
- this.renderer.appendChild(this.el.nativeElement, this._fileInput);
67
- if (this._accept) {
68
- this.renderer.setProperty(this._fileInput, 'accept', this._accept);
69
- }
70
- if (this._multiple) {
71
- this.renderer.setProperty(this._fileInput, 'multiple', this._multiple);
72
- }
73
- if (this._directory) {
74
- this.renderer.setAttribute(this._fileInput, 'webkitdirectory', 'true');
75
- this.renderer.setAttribute(this._fileInput, 'directory', 'true');
76
- }
77
- this._unlisten = this.renderer.listen(this._fileInput, 'change', event => {
78
- this.fileInputChanged(this._fileInput.files);
79
- });
80
- }
81
- removeFileSelect() {
82
- if (this._unlisten) {
83
- this._unlisten();
84
- this._unlisten = null;
85
- }
86
- if (this._fileInput) {
87
- this.renderer.removeChild(this.el.nativeElement, this._fileInput);
88
- this._fileInput = null;
89
- }
90
- }
91
- openFileSelectDialog() {
92
- this._fileInput.click();
93
- }
94
- /***************************************************************************
95
- * *
96
- * Private methods *
97
- * *
98
- **************************************************************************/
99
- fileInputChanged(fileList) {
100
- const files = this.toFileEntries(fileList);
101
- this.logger.debug('fileInputChanged, files:', files);
102
- this.clearFileList();
103
- this._filesSelected.next(files);
104
- }
105
- clearFileList() {
106
- // not nice but works
107
- this._fileInput.value = null;
108
- }
109
- toFileEntries(fileList) {
110
- const files = [];
111
- for (const key in fileList) {
112
- if (!isNaN(parseInt(key, 0))) {
113
- files.push(FileEntry.ofFile(fileList[key]));
114
- }
115
- }
116
- return files;
117
- }
118
- }
119
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-file-select-input.js","sourceRoot":"","sources":["../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/files/elder-file-select-input.ts"],"names":[],"mappings":"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","sourcesContent":["import {FileEntry} from './listing/file-entry';\nimport {ElementRef, Renderer2} from '@angular/core';\nimport {BooleanInput, coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {Observable, Subject} from 'rxjs';\n\n\nexport class ElderFileSelectInput {\n\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private _fileInput: HTMLInputElement;\n  private _unlisten: () => void;\n\n  private _multiple: boolean;\n  private _directory: boolean;\n  private _accept: string;\n\n  private readonly _filesSelected = new Subject<FileEntry[]>;\n\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private renderer: Renderer2,\n    private el: ElementRef\n  ) {\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public get filesSelected(): Observable<FileEntry[]> {\n    return this._filesSelected;\n  }\n\n  public set elderFileSelect(value: string) {\n    this._accept = value;\n    if (this._fileInput) {\n      this.renderer.setProperty(this._fileInput, 'accept', value);\n    }\n  }\n\n  public set elderFileSelectMultiple(value: BooleanInput) {\n    this._multiple = coerceBooleanProperty(value);\n    if (this._fileInput) {\n      this.renderer.setProperty(this._fileInput, 'multiple', value);\n    }\n  }\n\n  public set elderFileSelectDirectory(value: BooleanInput) {\n    this._directory = coerceBooleanProperty(value);\n    if (this._fileInput) {\n      this.renderer.setAttribute(this._fileInput, 'webkitdirectory', 'true');\n      this.renderer.setAttribute(this._fileInput, 'directory', 'true');\n    }\n  }\n\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * <input type=\"file\"\n   *     hidden #fileInput\n   *     [multiple]=\"multiple\"\n   *     [accept]=\"accept\"\n   *     (change)=\"fileInputChanged()\"\n   * />\n   */\n  public createFileSelect(): void {\n\n    this._fileInput = this.renderer.createElement('input');\n    this.renderer.setAttribute(this._fileInput, 'hidden', 'true');\n    this.renderer.setAttribute(this._fileInput, 'type', 'file');\n    this.renderer.appendChild(this.el.nativeElement, this._fileInput);\n\n    if (this._accept) {\n      this.renderer.setProperty(this._fileInput, 'accept', this._accept);\n    }\n    if (this._multiple) {\n      this.renderer.setProperty(this._fileInput, 'multiple', this._multiple);\n    }\n    if (this._directory) {\n      this.renderer.setAttribute(this._fileInput, 'webkitdirectory', 'true');\n      this.renderer.setAttribute(this._fileInput, 'directory', 'true');\n    }\n\n    this._unlisten = this.renderer.listen(\n      this._fileInput,\n      'change',\n      event => {\n        this.fileInputChanged(this._fileInput.files);\n      }\n    );\n  }\n\n  public removeFileSelect(): void {\n    if (this._unlisten) {\n      this._unlisten();\n      this._unlisten = null;\n    }\n    if (this._fileInput) {\n      this.renderer.removeChild(this.el.nativeElement, this._fileInput);\n      this._fileInput = null;\n    }\n  }\n\n  public openFileSelectDialog(): void {\n    this._fileInput.click();\n  }\n\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n\n  private fileInputChanged(fileList: FileList): void {\n    const files = this.toFileEntries(fileList);\n    this.logger.debug('fileInputChanged, files:', files);\n    this.clearFileList();\n    this._filesSelected.next(files);\n  }\n\n  private clearFileList() {\n    // not nice but works\n    this._fileInput.value = null;\n  }\n\n  private toFileEntries(fileList: FileList): FileEntry[] {\n    const files: FileEntry[] = [];\n    for (const key in fileList) {\n      if (!isNaN(parseInt(key, 0))) {\n        files.push(FileEntry.ofFile(fileList[key]));\n      }\n    }\n    return files;\n  }\n\n\n\n}\n"]}
@@ -1,96 +0,0 @@
1
- import { from, of, throwError } from 'rxjs';
2
- import { map } from 'rxjs/operators';
3
- /***************************************************************************
4
- * *
5
- * FileSystem API *
6
- * *
7
- **************************************************************************/
8
- /**
9
- * Holds experimental features of the browser `File System API`. No typescript lint checks
10
- * are performed here!
11
- */
12
- export class FileSystemApi {
13
- /***************************************************************************
14
- * *
15
- * DataTransferItem *
16
- * *
17
- **************************************************************************/
18
- static isGetAsFileSystemHandleSupported(transferItem) {
19
- // @ts-ignore
20
- return transferItem.getAsFileSystemHandle !== undefined;
21
- }
22
- /**
23
- * Returns an FileSystemHandle Observable of the given DataTransferItem. The FileSystemHandle can only be accessed in the `dragstart`
24
- * or `drop` event. Otherwise, it will return null!
25
- */
26
- static getAsFileSystemHandle(transferItem) {
27
- // @ts-ignore
28
- return from(transferItem.getAsFileSystemHandle());
29
- }
30
- /***************************************************************************
31
- * *
32
- * File Picker *
33
- * *
34
- **************************************************************************/
35
- static isFileSystemSupported() {
36
- return 'showOpenFilePicker' in window;
37
- }
38
- static openFilePicker(pickerOpts) {
39
- try {
40
- // @ts-ignore
41
- return from(window.showOpenFilePicker(pickerOpts));
42
- }
43
- catch (e) {
44
- return throwError(() => new Error(e));
45
- }
46
- }
47
- /***************************************************************************
48
- * *
49
- * Directory Picker *
50
- * *
51
- **************************************************************************/
52
- static isDirectoryPickerSupported() {
53
- return 'showDirectoryPicker' in window;
54
- }
55
- static openDirectoryPicker(pickerOpts) {
56
- try {
57
- // @ts-ignore
58
- return from(window.showDirectoryPicker(pickerOpts));
59
- }
60
- catch (e) {
61
- return throwError(() => new Error(e));
62
- }
63
- }
64
- /***************************************************************************
65
- * *
66
- * FileSystemDirectoryHandle *
67
- * *
68
- **************************************************************************/
69
- static getHandlesFromDirectory(dirHandle) {
70
- return from(FileSystemApi.getHandlesFromDirectoryAsync(dirHandle));
71
- }
72
- static buildRelativeParent(fileOrDirHandle, rootDirectory) {
73
- if (rootDirectory == null) {
74
- return of(undefined);
75
- }
76
- const relativeParents$ = from(rootDirectory.resolve(fileOrDirHandle));
77
- return relativeParents$.pipe(map(parents => {
78
- // path/to/folder/file.jpg
79
- const path = parents;
80
- // root/path/to/folder/file.jpg
81
- path.unshift(rootDirectory.name);
82
- // root/path/to/folder
83
- path.pop();
84
- return path.join('/');
85
- }));
86
- }
87
- static async getHandlesFromDirectoryAsync(dirHandle) {
88
- const handles = [];
89
- // @ts-ignore
90
- for await (const handle of dirHandle.values()) {
91
- handles.push(handle);
92
- }
93
- return handles;
94
- }
95
- }
96
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-system-api.js","sourceRoot":"","sources":["../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/files/file-system-api.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,IAAI,EAAc,EAAE,EAAE,UAAU,EAAC,MAAM,MAAM,CAAC;AACtD,OAAO,EAAC,GAAG,EAAC,MAAM,gBAAgB,CAAC;AAiDnC;;;;4EAI4E;AAE5E;;;GAGG;AACH,MAAM,OAAO,aAAa;IAExB;;;;gFAI4E;IAErE,MAAM,CAAC,gCAAgC,CAAC,YAA8B;QAC3E,aAAa;QACb,OAAO,YAAY,CAAC,qBAAqB,KAAK,SAAS,CAAC;IAC1D,CAAC;IAED;;;OAGG;IACI,MAAM,CAAC,qBAAqB,CAAC,YAA8B;QAChE,aAAa;QACb,OAAO,IAAI,CAAC,YAAY,CAAC,qBAAqB,EAAE,CAAC,CAAC;IACpD,CAAC;IAED;;;;gFAI4E;IAErE,MAAM,CAAC,qBAAqB;QACjC,OAAO,oBAAoB,IAAI,MAAM,CAAC;IACxC,CAAC;IAEM,MAAM,CAAC,cAAc,CAAC,UAAuC;QAClE,IAAI,CAAC;YACH,aAAa;YACb,OAAO,IAAI,CAAC,MAAM,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC,CAAC;QACrD,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QACxC,CAAC;IACH,CAAC;IAED;;;;gFAI4E;IAErE,MAAM,CAAC,0BAA0B;QACtC,OAAO,qBAAqB,IAAI,MAAM,CAAC;IACzC,CAAC;IAEM,MAAM,CAAC,mBAAmB,CAAC,UAA4C;QAC5E,IAAI,CAAC;YACH,aAAa;YACb,OAAO,IAAI,CAAC,MAAM,CAAC,mBAAmB,CAAC,UAAU,CAAC,CAAC,CAAC;QACtD,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QACxC,CAAC;IACH,CAAC;IAED;;;;gFAI4E;IAErE,MAAM,CAAC,uBAAuB,CAAC,SAAoC;QACxE,OAAO,IAAI,CAAC,aAAa,CAAC,4BAA4B,CAAC,SAAS,CAAC,CAAC,CAAC;IACrE,CAAC;IAEM,MAAM,CAAC,mBAAmB,CAC/B,eAAiC,EACjC,aAAyC;QAGzC,IAAI,aAAa,IAAI,IAAI,EAAE,CAAC;YACxB,OAAO,EAAE,CAAC,SAAS,CAAC,CAAC;QACzB,CAAC;QAED,MAAM,gBAAgB,GAAG,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,eAAe,CAAC,CAAC,CAAC;QACtE,OAAO,gBAAgB,CAAC,IAAI,CAC1B,GAAG,CAAC,OAAO,CAAC,EAAE;YACZ,0BAA0B;YAC1B,MAAM,IAAI,GAAG,OAAO,CAAC;YAErB,+BAA+B;YAC/B,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;YAEjC,sBAAsB;YACtB,IAAI,CAAC,GAAG,EAAE,CAAC;YACX,OAAO,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACxB,CAAC,CACF,CAAC,CAAA;IACJ,CAAC;IAEO,MAAM,CAAC,KAAK,CAAC,4BAA4B,CAAC,SAAoC;QACpF,MAAM,OAAO,GAAuB,EAAE,CAAC;QACvC,aAAa;QACb,IAAI,KAAK,EAAE,MAAM,MAAM,IAAI,SAAS,CAAC,MAAM,EAAE,EAAE,CAAC;YAC9C,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACvB,CAAC;QACD,OAAO,OAAO,CAAC;IACjB,CAAC;CACF","sourcesContent":["import {from, Observable, of, throwError} from 'rxjs';\nimport {map} from 'rxjs/operators';\n\n/***************************************************************************\n *                                                                         *\n * File Picker Options                                                     *\n *                                                                         *\n **************************************************************************/\n\nexport interface FileSystemFilePickerOptions {\n  /**\n   *\n   * ```\n   *   types: [\n   *     {\n   *       description: \"Images\",\n   *       accept: {\n   *         \"image/*\": [\".png\", \".gif\", \".jpeg\", \".jpg\"],\n   *       },\n   *     },\n   *   ],\n   *   excludeAcceptAllOption: true,\n   *   multiple: false,\n   * ```\n   */\n  excludeAcceptAllOptions?: boolean,\n  id?: string,\n  multiple?: boolean,\n  startIn?: FileSystemHandle | 'desktop' | 'documents' | 'downloads' | 'music' | 'pictures' | 'videos',\n  types?: FileSystemFilePickerTypeOptions[]\n}\n\nexport interface FileSystemFilePickerTypeOptions {\n  accept: Record<string,string[]>,\n  description?: string\n}\n\n/***************************************************************************\n *                                                                         *\n * Directory Picker Options                                                *\n *                                                                         *\n **************************************************************************/\n\nexport interface FileSystemDirectoryPickerOptions {\n  id?: string,\n  mode?: string,\n  startIn?: FileSystemHandle | 'desktop' | 'documents' | 'downloads' | 'music' | 'pictures' | 'videos'\n}\n\n\n/***************************************************************************\n *                                                                         *\n * FileSystem API                                                          *\n *                                                                         *\n **************************************************************************/\n\n/**\n * Holds experimental features of the browser `File System API`. No typescript lint checks\n * are performed here!\n */\nexport class FileSystemApi {\n\n  /***************************************************************************\n   *                                                                         *\n   * DataTransferItem                                                        *\n   *                                                                         *\n   **************************************************************************/\n\n  public static isGetAsFileSystemHandleSupported(transferItem: DataTransferItem): boolean {\n    // @ts-ignore\n    return transferItem.getAsFileSystemHandle !== undefined;\n  }\n\n  /**\n   * Returns an FileSystemHandle Observable of the given DataTransferItem. The FileSystemHandle can only be accessed in the `dragstart`\n   * or `drop` event. Otherwise, it will return null!\n   */\n  public static getAsFileSystemHandle(transferItem: DataTransferItem): Observable<FileSystemHandle | null> {\n    // @ts-ignore\n    return from(transferItem.getAsFileSystemHandle());\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * File Picker                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  public static isFileSystemSupported(): boolean {\n    return 'showOpenFilePicker' in window;\n  }\n\n  public static openFilePicker(pickerOpts: FileSystemFilePickerOptions): Observable<FileSystemFileHandle[]> {\n    try {\n      // @ts-ignore\n      return from(window.showOpenFilePicker(pickerOpts));\n    } catch (e) {\n      return throwError(() => new Error(e));\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Directory Picker                                                        *\n   *                                                                         *\n   **************************************************************************/\n\n  public static isDirectoryPickerSupported(): boolean {\n    return 'showDirectoryPicker' in window;\n  }\n\n  public static openDirectoryPicker(pickerOpts: FileSystemDirectoryPickerOptions): Observable<FileSystemDirectoryHandle> {\n    try {\n      // @ts-ignore\n      return from(window.showDirectoryPicker(pickerOpts));\n    } catch (e) {\n      return throwError(() => new Error(e));\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * FileSystemDirectoryHandle                                               *\n   *                                                                         *\n   **************************************************************************/\n\n  public static getHandlesFromDirectory(dirHandle: FileSystemDirectoryHandle): Observable<FileSystemHandle[]> {\n    return from(FileSystemApi.getHandlesFromDirectoryAsync(dirHandle));\n  }\n\n  public static buildRelativeParent(\n    fileOrDirHandle: FileSystemHandle,\n    rootDirectory?: FileSystemDirectoryHandle\n  ): Observable<string | undefined> {\n\n    if (rootDirectory == null) {\n        return of(undefined);\n    }\n\n    const relativeParents$ = from(rootDirectory.resolve(fileOrDirHandle));\n    return relativeParents$.pipe(\n      map(parents => {\n        // path/to/folder/file.jpg\n        const path = parents;\n\n        // root/path/to/folder/file.jpg\n        path.unshift(rootDirectory.name);\n\n        // root/path/to/folder\n        path.pop();\n        return path.join('/');\n      }\n    ))\n  }\n\n  private static async getHandlesFromDirectoryAsync(dirHandle: FileSystemDirectoryHandle): Promise<FileSystemHandle[]> {\n    const handles: FileSystemHandle[] = [];\n    // @ts-ignore\n    for await (const handle of dirHandle.values()) {\n      handles.push(handle);\n    }\n    return handles;\n  }\n}\n"]}
@@ -1,35 +0,0 @@
1
- import { switchMap, toArray, zip } from 'rxjs';
2
- import { FileEntry } from './file-entry';
3
- import { FileSystemApi } from '../file-system-api';
4
- import { map } from 'rxjs/operators';
5
- export class FileListingSystemHandle {
6
- /***************************************************************************
7
- * *
8
- * Public methods *
9
- * *
10
- **************************************************************************/
11
- /**
12
- * Creates a File Entry Array Observable by resolving its FileSystemHandle.
13
- * @returns An `Observable<FileEntry[]>` with the FileSystemFileHandle included
14
- */
15
- static listFiles(handle, rootDirectory) {
16
- switch (handle.kind) {
17
- case 'file':
18
- return FileEntry.ofFileHandle(handle, rootDirectory).pipe(toArray());
19
- case 'directory':
20
- return FileListingSystemHandle.listFilesRecursive(handle, rootDirectory);
21
- }
22
- }
23
- /***************************************************************************
24
- * *
25
- * Private methods *
26
- * *
27
- **************************************************************************/
28
- static listFilesRecursive(dirHandle, rootDirectory) {
29
- return FileSystemApi.getHandlesFromDirectory(dirHandle).pipe(map(handles => FileListingSystemHandle.listFilesOfEach(handles, rootDirectory ? rootDirectory : dirHandle)), switchMap(fileEntriesArray$ => zip(fileEntriesArray$)), map(fileEntriesArray => fileEntriesArray.flat()));
30
- }
31
- static listFilesOfEach(handles, rootDirectory) {
32
- return handles.map(fileOrDirHandle => FileListingSystemHandle.listFiles(fileOrDirHandle, rootDirectory));
33
- }
34
- }
35
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmlsZS1saXN0aW5nLXN5c3RlbS1oYW5kbGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9lbGRlcmJ5dGUvbmd4LXN0YXJ0ZXIvc3JjL2xpYi9jb21wb25lbnRzL2ZpbGVzL2xpc3RpbmcvZmlsZS1saXN0aW5nLXN5c3RlbS1oYW5kbGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFhLFNBQVMsRUFBRSxPQUFPLEVBQUUsR0FBRyxFQUFDLE1BQU0sTUFBTSxDQUFDO0FBQ3pELE9BQU8sRUFBQyxTQUFTLEVBQUMsTUFBTSxjQUFjLENBQUM7QUFDdkMsT0FBTyxFQUFDLGFBQWEsRUFBQyxNQUFNLG9CQUFvQixDQUFDO0FBQ2pELE9BQU8sRUFBQyxHQUFHLEVBQUMsTUFBTSxnQkFBZ0IsQ0FBQztBQUVuQyxNQUFNLE9BQU8sdUJBQXVCO0lBRWxDOzs7O2dGQUk0RTtJQUU1RTs7O09BR0c7SUFDSSxNQUFNLENBQUMsU0FBUyxDQUFDLE1BQXdCLEVBQUUsYUFBeUM7UUFDekYsUUFBUSxNQUFNLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDcEIsS0FBSyxNQUFNO2dCQUNULE9BQU8sU0FBUyxDQUFDLFlBQVksQ0FBQyxNQUE4QixFQUFFLGFBQWEsQ0FBQyxDQUFDLElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQyxDQUFDO1lBQy9GLEtBQUssV0FBVztnQkFDZCxPQUFPLHVCQUF1QixDQUFDLGtCQUFrQixDQUFDLE1BQW1DLEVBQUUsYUFBYSxDQUFDLENBQUM7UUFDMUcsQ0FBQztJQUNILENBQUM7SUFFRDs7OztnRkFJNEU7SUFHcEUsTUFBTSxDQUFDLGtCQUFrQixDQUFDLFNBQW9DLEVBQUUsYUFBeUM7UUFDL0csT0FBTyxhQUFhLENBQUMsdUJBQXVCLENBQUMsU0FBUyxDQUFDLENBQUMsSUFBSSxDQUMxRCxHQUFHLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQyx1QkFBdUIsQ0FBQyxlQUFlLENBQUMsT0FBTyxFQUFFLGFBQWEsQ0FBQyxDQUFDLENBQUMsYUFBYSxDQUFDLENBQUMsQ0FBQyxTQUFTLENBQUMsQ0FBQyxFQUMzRyxTQUFTLENBQUMsaUJBQWlCLENBQUMsRUFBRSxDQUFDLEdBQUcsQ0FBQyxpQkFBaUIsQ0FBQyxDQUFDLEVBQ3RELEdBQUcsQ0FBQyxnQkFBZ0IsQ0FBQyxFQUFFLENBQUMsZ0JBQWdCLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FDakQsQ0FBQztJQUNKLENBQUM7SUFFTyxNQUFNLENBQUMsZUFBZSxDQUM1QixPQUEyQixFQUMzQixhQUF5QztRQUV6QyxPQUFPLE9BQU8sQ0FBQyxHQUFHLENBQUMsZUFBZSxDQUFDLEVBQUUsQ0FBQyx1QkFBdUIsQ0FBQyxTQUFTLENBQUMsZUFBZSxFQUFFLGFBQWEsQ0FBQyxDQUFDLENBQUE7SUFDMUcsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtPYnNlcnZhYmxlLCBzd2l0Y2hNYXAsIHRvQXJyYXksIHppcH0gZnJvbSAncnhqcyc7XG5pbXBvcnQge0ZpbGVFbnRyeX0gZnJvbSAnLi9maWxlLWVudHJ5JztcbmltcG9ydCB7RmlsZVN5c3RlbUFwaX0gZnJvbSAnLi4vZmlsZS1zeXN0ZW0tYXBpJztcbmltcG9ydCB7bWFwfSBmcm9tICdyeGpzL29wZXJhdG9ycyc7XG5cbmV4cG9ydCBjbGFzcyBGaWxlTGlzdGluZ1N5c3RlbUhhbmRsZSB7XG5cbiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqIFB1YmxpYyBtZXRob2RzICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgLyoqXG4gICAqIENyZWF0ZXMgYSBGaWxlIEVudHJ5IEFycmF5IE9ic2VydmFibGUgYnkgcmVzb2x2aW5nIGl0cyBGaWxlU3lzdGVtSGFuZGxlLlxuICAgKiBAcmV0dXJucyBBbiBgT2JzZXJ2YWJsZTxGaWxlRW50cnlbXT5gIHdpdGggdGhlIEZpbGVTeXN0ZW1GaWxlSGFuZGxlIGluY2x1ZGVkXG4gICAqL1xuICBwdWJsaWMgc3RhdGljIGxpc3RGaWxlcyhoYW5kbGU6IEZpbGVTeXN0ZW1IYW5kbGUsIHJvb3REaXJlY3Rvcnk/OiBGaWxlU3lzdGVtRGlyZWN0b3J5SGFuZGxlKTogT2JzZXJ2YWJsZTxGaWxlRW50cnlbXT4ge1xuICAgIHN3aXRjaCAoaGFuZGxlLmtpbmQpIHtcbiAgICAgIGNhc2UgJ2ZpbGUnOlxuICAgICAgICByZXR1cm4gRmlsZUVudHJ5Lm9mRmlsZUhhbmRsZShoYW5kbGUgYXMgRmlsZVN5c3RlbUZpbGVIYW5kbGUsIHJvb3REaXJlY3RvcnkpLnBpcGUodG9BcnJheSgpKTtcbiAgICAgIGNhc2UgJ2RpcmVjdG9yeSc6XG4gICAgICAgIHJldHVybiBGaWxlTGlzdGluZ1N5c3RlbUhhbmRsZS5saXN0RmlsZXNSZWN1cnNpdmUoaGFuZGxlIGFzIEZpbGVTeXN0ZW1EaXJlY3RvcnlIYW5kbGUsIHJvb3REaXJlY3RvcnkpO1xuICAgIH1cbiAgfVxuXG4gIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKipcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiBQcml2YXRlIG1ldGhvZHMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqL1xuXG5cbiAgcHJpdmF0ZSBzdGF0aWMgbGlzdEZpbGVzUmVjdXJzaXZlKGRpckhhbmRsZTogRmlsZVN5c3RlbURpcmVjdG9yeUhhbmRsZSwgcm9vdERpcmVjdG9yeT86IEZpbGVTeXN0ZW1EaXJlY3RvcnlIYW5kbGUpOiBPYnNlcnZhYmxlPEZpbGVFbnRyeVtdPiB7XG4gICAgcmV0dXJuIEZpbGVTeXN0ZW1BcGkuZ2V0SGFuZGxlc0Zyb21EaXJlY3RvcnkoZGlySGFuZGxlKS5waXBlKFxuICAgICAgbWFwKGhhbmRsZXMgPT4gRmlsZUxpc3RpbmdTeXN0ZW1IYW5kbGUubGlzdEZpbGVzT2ZFYWNoKGhhbmRsZXMsIHJvb3REaXJlY3RvcnkgPyByb290RGlyZWN0b3J5IDogZGlySGFuZGxlKSksXG4gICAgICBzd2l0Y2hNYXAoZmlsZUVudHJpZXNBcnJheSQgPT4gemlwKGZpbGVFbnRyaWVzQXJyYXkkKSksXG4gICAgICBtYXAoZmlsZUVudHJpZXNBcnJheSA9PiBmaWxlRW50cmllc0FycmF5LmZsYXQoKSlcbiAgICApO1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgbGlzdEZpbGVzT2ZFYWNoKFxuICAgIGhhbmRsZXM6IEZpbGVTeXN0ZW1IYW5kbGVbXSxcbiAgICByb290RGlyZWN0b3J5PzogRmlsZVN5c3RlbURpcmVjdG9yeUhhbmRsZVxuICApOiBPYnNlcnZhYmxlPEZpbGVFbnRyeVtdPltdIHtcbiAgICByZXR1cm4gaGFuZGxlcy5tYXAoZmlsZU9yRGlySGFuZGxlID0+IEZpbGVMaXN0aW5nU3lzdGVtSGFuZGxlLmxpc3RGaWxlcyhmaWxlT3JEaXJIYW5kbGUsIHJvb3REaXJlY3RvcnkpKVxuICB9XG59XG4iXX0=