@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
@@ -2,11 +2,6 @@ import { Directive, EventEmitter, HostListener, Input, Output } from '@angular/c
2
2
  import { LoggerFactory } from '@elderbyte/ts-logger';
3
3
  import { coerceBooleanProperty } from '@angular/cdk/coercion';
4
4
  import { FileEntry } from './listing/file-entry';
5
- import { ElderFileSelectInput } from './elder-file-select-input';
6
- import { FileSystemApi } from './file-system-api';
7
- import { combineLatest, switchMap } from 'rxjs';
8
- import { FileListingSystemHandle } from './listing/file-listing-system-handle';
9
- import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
10
5
  import * as i0 from "@angular/core";
11
6
  export class ElderFileSelectDirective {
12
7
  /***************************************************************************
@@ -14,10 +9,9 @@ export class ElderFileSelectDirective {
14
9
  * Constructor *
15
10
  * *
16
11
  **************************************************************************/
17
- constructor(renderer, el, destroyRef) {
18
- this.renderer = renderer;
12
+ constructor(el, renderer) {
19
13
  this.el = el;
20
- this.destroyRef = destroyRef;
14
+ this.renderer = renderer;
21
15
  /***************************************************************************
22
16
  * *
23
17
  * Fields *
@@ -26,8 +20,6 @@ export class ElderFileSelectDirective {
26
20
  this.logger = LoggerFactory.getLogger(this.constructor.name);
27
21
  this.elderFileSelectChange = new EventEmitter();
28
22
  this.elderSingleFileSelectChange = new EventEmitter();
29
- this._useDirectoryPicker = false;
30
- this._legacyInput = new ElderFileSelectInput(this.renderer, this.el);
31
23
  }
32
24
  /***************************************************************************
33
25
  * *
@@ -35,16 +27,10 @@ export class ElderFileSelectDirective {
35
27
  * *
36
28
  **************************************************************************/
37
29
  ngOnInit() {
38
- if (!FileSystemApi.isFileSystemSupported() || !FileSystemApi.isDirectoryPickerSupported()) {
39
- this.logger.warn('showOpenFilePicker or showDirectoryPicker is not supported for this browser, falling back to legacy file picker');
40
- this._legacyInput.createFileSelect();
41
- this._legacyInput.filesSelected.pipe(takeUntilDestroyed(this.destroyRef)).subscribe({
42
- next: multiFileChange => this.pushFileChanges(multiFileChange)
43
- });
44
- }
30
+ this.createFileSelect();
45
31
  }
46
32
  ngOnDestroy() {
47
- this._legacyInput.removeFileSelect();
33
+ this.removeFileSelect();
48
34
  }
49
35
  /***************************************************************************
50
36
  * *
@@ -52,49 +38,16 @@ export class ElderFileSelectDirective {
52
38
  * *
53
39
  **************************************************************************/
54
40
  set elderFileSelect(value) {
55
- this._filePickerOptions = {
56
- excludeAcceptAllOptions: true,
57
- id: this._filePickerOptions?.id,
58
- multiple: this._filePickerOptions?.multiple,
59
- startIn: this._filePickerOptions?.startIn,
60
- types: [this.buildTypeOption(value)]
61
- };
62
- this._legacyInput.elderFileSelect = value;
63
- }
64
- buildTypeOption(value) {
65
- if (this.isFileExtension(value)) {
66
- return {
67
- accept: {
68
- '*/*': value.trim()
69
- .replace(/\s+/g, '')
70
- .split(',')
71
- }
72
- };
41
+ this._accept = value;
42
+ if (this._fileInput) {
43
+ this.renderer.setProperty(this._fileInput, 'accept', value);
73
44
  }
74
- else if (this.isMimeType(value)) {
75
- return {
76
- accept: { [value]: [] }
77
- };
78
- }
79
- return null;
80
- }
81
- isFileExtension(input) {
82
- const regex = new RegExp(/^(\.\w+)(, ?\.\w+)*$/);
83
- return regex.test(input);
84
- }
85
- isMimeType(input) {
86
- const regex = new RegExp(/^.+\/.+$/);
87
- return regex.test(input);
88
45
  }
89
46
  set elderFileSelectMultiple(value) {
90
- this._filePickerOptions = {
91
- excludeAcceptAllOptions: this._filePickerOptions?.excludeAcceptAllOptions,
92
- id: this._filePickerOptions?.id,
93
- multiple: coerceBooleanProperty(value),
94
- startIn: this._filePickerOptions?.startIn,
95
- types: this._filePickerOptions?.types
96
- };
97
- this._legacyInput.elderFileSelectMultiple = value;
47
+ this._multiple = coerceBooleanProperty(value);
48
+ if (this._fileInput) {
49
+ this.renderer.setProperty(this._fileInput, 'multiple', value);
50
+ }
98
51
  }
99
52
  /**
100
53
  * Allow the user to select a directory. All files that are recursively contained are selected.
@@ -102,8 +55,11 @@ export class ElderFileSelectDirective {
102
55
  * @param value
103
56
  */
104
57
  set elderFileSelectDirectory(value) {
105
- this._useDirectoryPicker = coerceBooleanProperty(value);
106
- this._legacyInput.elderFileSelectDirectory = value;
58
+ this._directory = coerceBooleanProperty(value);
59
+ if (this._fileInput) {
60
+ this.renderer.setAttribute(this._fileInput, 'webkitdirectory', 'true');
61
+ this.renderer.setAttribute(this._fileInput, 'directory', 'true');
62
+ }
107
63
  }
108
64
  /***************************************************************************
109
65
  * *
@@ -111,39 +67,81 @@ export class ElderFileSelectDirective {
111
67
  * *
112
68
  **************************************************************************/
113
69
  onClick(event) {
114
- if (!this._useDirectoryPicker && FileSystemApi.isFileSystemSupported()) {
115
- this.openFilePicker(this._filePickerOptions).subscribe({
116
- next: multiFileChange => this.pushFileChanges(multiFileChange)
117
- });
118
- }
119
- else if (this._useDirectoryPicker && FileSystemApi.isDirectoryPickerSupported()) {
120
- this.openDirectoryPicker(this._directoryPickerOptions).subscribe({
121
- next: multiFileChange => this.pushFileChanges(multiFileChange)
122
- });
123
- }
124
- else {
125
- this._legacyInput.openFileSelectDialog();
126
- }
70
+ this.openFileSelectDialog();
71
+ }
72
+ openFileSelectDialog() {
73
+ this._fileInput.click();
127
74
  }
128
75
  /***************************************************************************
129
76
  * *
130
77
  * Private methods *
131
78
  * *
132
79
  **************************************************************************/
133
- pushFileChanges(multiFileChange) {
134
- this.elderFileSelectChange.next(multiFileChange);
135
- const file = multiFileChange[0]?.file;
136
- if (file) {
137
- this.elderSingleFileSelectChange.next(file);
80
+ /**
81
+ * <input type="file"
82
+ * hidden #fileInput
83
+ * [multiple]="multiple"
84
+ * [accept]="accept"
85
+ * (change)="fileInputChanged()"
86
+ * />
87
+ */
88
+ createFileSelect() {
89
+ this._fileInput = this.renderer.createElement('input');
90
+ this.renderer.setAttribute(this._fileInput, 'hidden', 'true');
91
+ this.renderer.setAttribute(this._fileInput, 'type', 'file');
92
+ this.renderer.appendChild(this.el.nativeElement, this._fileInput);
93
+ if (this._accept) {
94
+ this.renderer.setProperty(this._fileInput, 'accept', this._accept);
95
+ }
96
+ if (this._multiple) {
97
+ this.renderer.setProperty(this._fileInput, 'multiple', this._multiple);
98
+ }
99
+ if (this._directory) {
100
+ this.renderer.setAttribute(this._fileInput, 'webkitdirectory', 'true');
101
+ this.renderer.setAttribute(this._fileInput, 'directory', 'true');
102
+ }
103
+ this._unlisten = this.renderer.listen(this._fileInput, 'change', event => this.fileInputChanged(event));
104
+ }
105
+ removeFileSelect() {
106
+ if (this._unlisten) {
107
+ this._unlisten();
108
+ this._unlisten = null;
109
+ }
110
+ if (this._fileInput) {
111
+ this.renderer.removeChild(this.el.nativeElement, this._fileInput);
112
+ this._fileInput = null;
138
113
  }
139
114
  }
140
- openFilePicker(pickerOpts) {
141
- return FileSystemApi.openFilePicker(pickerOpts).pipe(switchMap(files => combineLatest(files.map(file => FileEntry.ofFileHandle(file)))));
115
+ fileInputChanged(event) {
116
+ const fileList = this._fileInput.files;
117
+ const files = this.toFileEntries(fileList);
118
+ this.logger.debug('fileInputChanged, files:', files);
119
+ this.emitFileList(files);
120
+ this.clearFileList();
142
121
  }
143
- openDirectoryPicker(pickerOpts) {
144
- return FileSystemApi.openDirectoryPicker(pickerOpts).pipe(switchMap(dir => FileListingSystemHandle.listFiles(dir)));
122
+ clearFileList() {
123
+ // not nice but works
124
+ this._fileInput.value = null;
125
+ }
126
+ emitFileList(files) {
127
+ if (files.length > 0) {
128
+ this.elderFileSelectChange.next(files);
129
+ this.elderSingleFileSelectChange.emit(files[0].file);
130
+ }
131
+ else {
132
+ this.logger.warn('User did not select any File or the Folder was empty.');
133
+ }
134
+ }
135
+ toFileEntries(fileList) {
136
+ const files = [];
137
+ for (const key in fileList) {
138
+ if (!isNaN(parseInt(key, 0))) {
139
+ files.push(FileEntry.ofFile(fileList[key]));
140
+ }
141
+ }
142
+ return files;
145
143
  }
146
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: ElderFileSelectDirective, deps: [{ token: i0.Renderer2 }, { token: i0.ElementRef }, { token: i0.DestroyRef }], target: i0.ɵɵFactoryTarget.Directive }); }
144
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: ElderFileSelectDirective, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Directive }); }
147
145
  static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.0.4", type: ElderFileSelectDirective, isStandalone: true, selector: "[elderFileSelect]", inputs: { elderFileSelect: "elderFileSelect", elderFileSelectMultiple: "elderFileSelectMultiple", elderFileSelectDirectory: "elderFileSelectDirectory" }, outputs: { elderFileSelectChange: "elderFileSelectChange", elderSingleFileSelectChange: "elderSingleFileSelectChange" }, host: { listeners: { "click": "onClick($event)" } }, ngImport: i0 }); }
148
146
  }
149
147
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: ElderFileSelectDirective, decorators: [{
@@ -152,7 +150,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.4", ngImpor
152
150
  selector: '[elderFileSelect]',
153
151
  standalone: true
154
152
  }]
155
- }], ctorParameters: () => [{ type: i0.Renderer2 }, { type: i0.ElementRef }, { type: i0.DestroyRef }], propDecorators: { elderFileSelectChange: [{
153
+ }], ctorParameters: () => [{ type: i0.ElementRef }, { type: i0.Renderer2 }], propDecorators: { elderFileSelectChange: [{
156
154
  type: Output
157
155
  }], elderSingleFileSelectChange: [{
158
156
  type: Output
@@ -166,4 +164,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.4", ngImpor
166
164
  type: HostListener,
167
165
  args: ['click', ['$event']]
168
166
  }] } });
169
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-file-select.directive.js","sourceRoot":"","sources":["../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/files/elder-file-select.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  DestroyRef,\n  Directive, ElementRef,\n  EventEmitter,\n  HostListener,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output, Renderer2\n} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {BooleanInput, coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {FileEntry} from './listing/file-entry';\nimport {ElderFileSelectInput} from './elder-file-select-input';\nimport {\n  FileSystemApi,\n  FileSystemDirectoryPickerOptions,\n  FileSystemFilePickerOptions,\n  FileSystemFilePickerTypeOptions\n} from './file-system-api';\nimport {combineLatest, Observable, switchMap} from 'rxjs';\nimport {FileListingSystemHandle} from './listing/file-listing-system-handle';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\n\n\n@Directive({\n  selector: '[elderFileSelect]',\n  standalone: true\n})\nexport class ElderFileSelectDirective implements OnInit, OnDestroy {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  @Output()\n  public readonly elderFileSelectChange = new EventEmitter<FileEntry[]>();\n\n  @Output()\n  public readonly elderSingleFileSelectChange = new EventEmitter<File>();\n\n  private _legacyInput: ElderFileSelectInput;\n  private _filePickerOptions: FileSystemFilePickerOptions;\n  private _directoryPickerOptions: FileSystemDirectoryPickerOptions;\n  private _useDirectoryPicker: boolean = false;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private renderer: Renderer2,\n    private el: ElementRef,\n    private destroyRef: DestroyRef\n  ) {\n    this._legacyInput = new ElderFileSelectInput(this.renderer, this.el);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n    if (!FileSystemApi.isFileSystemSupported() || !FileSystemApi.isDirectoryPickerSupported()) {\n      this.logger.warn('showOpenFilePicker or showDirectoryPicker is not supported for this browser, falling back to legacy file picker');\n      this._legacyInput.createFileSelect();\n      this._legacyInput.filesSelected.pipe(\n        takeUntilDestroyed(this.destroyRef)\n      ).subscribe({\n        next: multiFileChange => this.pushFileChanges(multiFileChange)\n      });\n    }\n  }\n\n  public ngOnDestroy(): void {\n    this._legacyInput.removeFileSelect();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\n  public set elderFileSelect(value: string) {\n\n    this._filePickerOptions = {\n      excludeAcceptAllOptions: true,\n      id: this._filePickerOptions?.id,\n      multiple: this._filePickerOptions?.multiple,\n      startIn: this._filePickerOptions?.startIn,\n      types: [this.buildTypeOption(value)]\n    };\n\n    this._legacyInput.elderFileSelect = value;\n  }\n\n  private buildTypeOption(value: string): FileSystemFilePickerTypeOptions {\n    if (this.isFileExtension(value)) {\n      return {\n        accept: {\n          '*/*': value.trim()\n            .replace(/\\s+/g, '')\n            .split(',')\n        }\n      };\n    } else if (this.isMimeType(value)) {\n      return {\n        accept: {[value]: []}\n      };\n    }\n    return null;\n  }\n\n  private isFileExtension(input: string): boolean {\n    const regex = new RegExp(/^(\\.\\w+)(, ?\\.\\w+)*$/);\n    return regex.test(input);\n  }\n\n  private isMimeType(input: string): boolean {\n    const regex = new RegExp(/^.+\\/.+$/);\n    return regex.test(input);\n  }\n\n\n  @Input()\n  public set elderFileSelectMultiple(value: BooleanInput) {\n\n    this._filePickerOptions = {\n      excludeAcceptAllOptions: this._filePickerOptions?.excludeAcceptAllOptions,\n      id: this._filePickerOptions?.id,\n      multiple: coerceBooleanProperty(value),\n      startIn: this._filePickerOptions?.startIn,\n      types: this._filePickerOptions?.types\n    };\n\n    this._legacyInput.elderFileSelectMultiple = value;\n  }\n\n  /**\n   * Allow the user to select a directory. All files that are recursively contained are selected.\n   * However, the user will no longer be able to select individual files if this is enabled.\n   * @param value\n   */\n  @Input()\n  public set elderFileSelectDirectory(value: BooleanInput) {\n    this._useDirectoryPicker = coerceBooleanProperty(value);\n    this._legacyInput.elderFileSelectDirectory = value;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @HostListener('click', ['$event'])\n  public onClick(event: any): void {\n    if (!this._useDirectoryPicker && FileSystemApi.isFileSystemSupported()) {\n      this.openFilePicker(this._filePickerOptions).subscribe({\n        next: multiFileChange => this.pushFileChanges(multiFileChange)\n      });\n    } else if (this._useDirectoryPicker && FileSystemApi.isDirectoryPickerSupported()) {\n      this.openDirectoryPicker(this._directoryPickerOptions).subscribe({\n        next: multiFileChange => this.pushFileChanges(multiFileChange)\n      });\n    } else {\n      this._legacyInput.openFileSelectDialog();\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private pushFileChanges(multiFileChange: FileEntry[]): void {\n    this.elderFileSelectChange.next(multiFileChange);\n    const file = multiFileChange[0]?.file;\n    if (file) {\n      this.elderSingleFileSelectChange.next(file);\n    }\n  }\n\n  public openFilePicker(pickerOpts: FileSystemFilePickerOptions): Observable<FileEntry[]> {\n    return FileSystemApi.openFilePicker(pickerOpts).pipe(\n      switchMap(files =>\n        combineLatest(files.map(file => FileEntry.ofFileHandle(file)))\n      )\n    );\n  }\n\n  public openDirectoryPicker(pickerOpts: FileSystemDirectoryPickerOptions): Observable<FileEntry[]> {\n    return FileSystemApi.openDirectoryPicker(pickerOpts).pipe(\n      switchMap(dir => FileListingSystemHandle.listFiles(dir))\n    );\n  }\n\n}\n\n\n\n"]}
167
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-file-select.directive.js","sourceRoot":"","sources":["../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/files/elder-file-select.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Directive,\n  ElementRef,\n  EventEmitter,\n  HostListener,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output,\n  Renderer2\n} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {BooleanInput, coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {FileEntry} from './listing/file-entry';\n\n@Directive({\n    selector: '[elderFileSelect]',\n    standalone: true\n})\nexport class ElderFileSelectDirective implements OnInit, OnDestroy {\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  @Output()\n  public readonly elderFileSelectChange = new EventEmitter<FileEntry[]>();\n\n  @Output()\n  public readonly elderSingleFileSelectChange = new EventEmitter<File>();\n\n  private _multiple: boolean;\n  private _directory: boolean;\n  private _accept: string;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private el: ElementRef,\n    private renderer: Renderer2\n  ) {\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n    this.createFileSelect();\n  }\n\n  public ngOnDestroy(): void {\n    this.removeFileSelect();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\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  @Input()\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  /**\n   * Allow the user to select a directory. All files that are recursively contained are selected.\n   * However, the user will no longer be able to select individual files if this is enabled.\n   * @param value\n   */\n  @Input()\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   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @HostListener('click', ['$event'])\n  public onClick(event: any): void {\n    this.openFileSelectDialog();\n  }\n\n  public openFileSelectDialog(): void {\n    this._fileInput.click();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * <input type=\"file\"\n   *     hidden #fileInput\n   *     [multiple]=\"multiple\"\n   *     [accept]=\"accept\"\n   *     (change)=\"fileInputChanged()\"\n   * />\n   */\n  private 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 => this.fileInputChanged(event)\n    );\n  }\n\n  private 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  private fileInputChanged(event: any): void {\n    const fileList = this._fileInput.files;\n    const files = this.toFileEntries(fileList);\n    this.logger.debug('fileInputChanged, files:', files);\n    this.emitFileList(files);\n    this.clearFileList();\n  }\n\n  private clearFileList() {\n    // not nice but works\n    this._fileInput.value = null;\n  }\n\n  private emitFileList(files: FileEntry[]): void {\n    if (files.length > 0) {\n      this.elderFileSelectChange.next(files);\n      this.elderSingleFileSelectChange.emit(\n        files[0].file\n      );\n    } else {\n      this.logger.warn('User did not select any File or the Folder was empty.');\n    }\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"]}
@@ -1,6 +1,3 @@
1
- import { from } from 'rxjs';
2
- import { combineLatestWith, map } from 'rxjs/operators';
3
- import { FileSystemApi } from '../file-system-api';
4
1
  /**
5
2
  * Represents a file and the relative path where the user has selected it.
6
3
  * This is useful if the user has selected a folder and files
@@ -15,24 +12,26 @@ export class FileEntry {
15
12
  * which is also supported by this method.
16
13
  */
17
14
  static ofFile(file) {
18
- return FileEntry.relativeFile(file, this.buildRelativeParent(file));
19
- }
20
- /**
21
- * Creates an Observable of a File Entry by resolving its FileSystemFileHandle.
22
- * @param fileHandle
23
- * @param rootDirectory is needed to build a relative parent
24
- */
25
- static ofFileHandle(fileHandle, rootDirectory) {
26
- return from(fileHandle.getFile()).pipe(combineLatestWith(FileSystemApi.buildRelativeParent(fileHandle, rootDirectory)), map(([file, relativeParent]) => FileEntry.relativeFile(file, relativeParent, fileHandle)));
15
+ let relativeParent = null;
16
+ if (file.webkitRelativePath) {
17
+ if (file.webkitRelativePath.endsWith(file.name)) {
18
+ const nameStart = file.webkitRelativePath.length - file.name.length;
19
+ relativeParent = file.webkitRelativePath.substring(0, nameStart);
20
+ }
21
+ else {
22
+ relativeParent = file.webkitRelativePath;
23
+ }
24
+ }
25
+ return FileEntry.relativeFile(file, relativeParent);
27
26
  }
28
27
  /**
29
28
  * Creates a file Entry with a relative parent path
30
29
  */
31
- static relativeFile(file, relativeParent, fileHandle) {
30
+ static relativeFile(file, relativeParent) {
32
31
  if (relativeParent && relativeParent.endsWith('/')) {
33
32
  relativeParent = relativeParent.substring(0, relativeParent.length - 1);
34
33
  }
35
- return new FileEntry(file, relativeParent, fileHandle);
34
+ return new FileEntry(file, relativeParent);
36
35
  }
37
36
  static toFileMap(entries) {
38
37
  const map = new Map();
@@ -52,10 +51,9 @@ export class FileEntry {
52
51
  * Contains the relative path from the selected folder
53
52
  * to this file. Only relevant if the user has selected a folder.
54
53
  */
55
- relativeParent, fileHandle) {
54
+ relativeParent) {
56
55
  this.file = file;
57
56
  this.relativeParent = relativeParent;
58
- this.fileHandle = fileHandle;
59
57
  this.relativePath = FileEntry.buildRelativePath(relativeParent, file);
60
58
  }
61
59
  /***************************************************************************
@@ -74,18 +72,5 @@ export class FileEntry {
74
72
  return file.name;
75
73
  }
76
74
  }
77
- static buildRelativeParent(file) {
78
- let relativeParent = null;
79
- if (file.webkitRelativePath) {
80
- if (file.webkitRelativePath.endsWith(file.name)) {
81
- const nameStart = file.webkitRelativePath.length - file.name.length;
82
- relativeParent = file.webkitRelativePath.substring(0, nameStart);
83
- }
84
- else {
85
- relativeParent = file.webkitRelativePath;
86
- }
87
- }
88
- return relativeParent;
89
- }
90
75
  }
91
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-entry.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/files/listing/file-entry.ts"],"names":[],"mappings":"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","sourcesContent":["import {from, Observable} from 'rxjs';\nimport {combineLatestWith, map} from 'rxjs/operators';\nimport {FileSystemApi} from '../file-system-api';\n\n/**\n * Represents a file and the relative path where the user has selected it.\n * This is useful if the user has selected a folder and files\n * are listed recursively.\n */\nexport class FileEntry {\n\n  /**\n   * Creates a file Entry without an explicit relative parent.\n   *\n   * However, depending on how the file was selected, some browsers\n   * encode the relative parent path in the property 'webkitRelativePath'\n   * which is also supported by this method.\n   */\n  public static ofFile(file: File): FileEntry {\n\n    return FileEntry.relativeFile(\n      file,\n      this.buildRelativeParent(file)\n    );\n  }\n\n\n  /**\n   * Creates an Observable of a File Entry by resolving its FileSystemFileHandle.\n   * @param fileHandle\n   * @param rootDirectory is needed to build a relative parent\n   */\n  public static ofFileHandle(\n    fileHandle: FileSystemFileHandle,\n    rootDirectory?: FileSystemDirectoryHandle\n  ): Observable<FileEntry> {\n\n      return from(fileHandle.getFile()).pipe(\n        combineLatestWith(FileSystemApi.buildRelativeParent(fileHandle, rootDirectory)),\n        map(([file, relativeParent]) => FileEntry.relativeFile(\n          file,\n          relativeParent,\n          fileHandle))\n      );\n  }\n\n\n\n  /**\n   * Creates a file Entry with a relative parent path\n   */\n  public static relativeFile(\n    file: File,\n    relativeParent: string | null,\n    fileHandle?: FileSystemFileHandle\n  ): FileEntry {\n    if (relativeParent && relativeParent.endsWith('/')) {\n      relativeParent = relativeParent.substring(0, relativeParent.length - 1);\n    }\n    return new FileEntry(file, relativeParent, fileHandle);\n  }\n\n  public static toFileMap(entries: FileEntry[]): Map<File, FileEntry> {\n    const map = new Map<File, FileEntry>();\n    entries.forEach(e => map.set(e.file, e));\n    return map;\n  }\n\n  public static toFileArray(entries: FileEntry[]): File[] {\n    return entries.map(e => e.file);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Returns a string which specifies the file's path relative to the directory selected by the user.\n   */\n  public readonly relativePath: string;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  private constructor(\n    public readonly file: File,\n    /**\n     * Contains the relative path from the selected folder\n     * to this file. Only relevant if the user has selected a folder.\n     */\n    public readonly relativeParent: string | null,\n    public readonly fileHandle?: FileSystemFileHandle\n  ) {\n    this.relativePath = FileEntry.buildRelativePath(relativeParent, file);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Returns a string which specifies the file's path relative to the directory selected by the user.\n   */\n  private static buildRelativePath(relativeParent: string, file: File): string {\n    if (relativeParent != null) {\n      return relativeParent + '/' + file.name;\n    } else {\n      return file.name;\n    }\n  }\n\n\n  private static buildRelativeParent(file: File): string {\n    let relativeParent: string = null;\n\n    if (file.webkitRelativePath) {\n      if (file.webkitRelativePath.endsWith(file.name)) {\n        const nameStart = file.webkitRelativePath.length - file.name.length;\n        relativeParent = file.webkitRelativePath.substring(0, nameStart);\n      } else {\n        relativeParent = file.webkitRelativePath;\n      }\n    }\n    return relativeParent;\n  }\n}\n"]}
76
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmlsZS1lbnRyeS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2VsZGVyYnl0ZS9uZ3gtc3RhcnRlci9zcmMvbGliL2NvbXBvbmVudHMvZmlsZXMvbGlzdGluZy9maWxlLWVudHJ5LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7O0dBSUc7QUFDSCxNQUFNLE9BQU8sU0FBUztJQUVwQjs7Ozs7O09BTUc7SUFDSSxNQUFNLENBQUMsTUFBTSxDQUFDLElBQVU7UUFFN0IsSUFBSSxjQUFjLEdBQVcsSUFBSSxDQUFDO1FBRWxDLElBQUksSUFBSSxDQUFDLGtCQUFrQixFQUFFLENBQUM7WUFDNUIsSUFBSSxJQUFJLENBQUMsa0JBQWtCLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDO2dCQUNoRCxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsa0JBQWtCLENBQUMsTUFBTSxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDO2dCQUNwRSxjQUFjLEdBQUcsSUFBSSxDQUFDLGtCQUFrQixDQUFDLFNBQVMsQ0FBQyxDQUFDLEVBQUUsU0FBUyxDQUFDLENBQUM7WUFDbkUsQ0FBQztpQkFBTSxDQUFDO2dCQUNOLGNBQWMsR0FBRyxJQUFJLENBQUMsa0JBQWtCLENBQUM7WUFDM0MsQ0FBQztRQUNILENBQUM7UUFFRCxPQUFPLFNBQVMsQ0FBQyxZQUFZLENBQzNCLElBQUksRUFDSixjQUFjLENBQ2YsQ0FBQztJQUNKLENBQUM7SUFFRDs7T0FFRztJQUNJLE1BQU0sQ0FBQyxZQUFZLENBQUMsSUFBVSxFQUFFLGNBQTZCO1FBQ2xFLElBQUksY0FBYyxJQUFJLGNBQWMsQ0FBQyxRQUFRLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQztZQUNuRCxjQUFjLEdBQUcsY0FBYyxDQUFDLFNBQVMsQ0FBQyxDQUFDLEVBQUUsY0FBYyxDQUFDLE1BQU0sR0FBRyxDQUFDLENBQUMsQ0FBQztRQUMxRSxDQUFDO1FBQ0QsT0FBTyxJQUFJLFNBQVMsQ0FBQyxJQUFJLEVBQUUsY0FBYyxDQUFDLENBQUM7SUFDN0MsQ0FBQztJQUVNLE1BQU0sQ0FBQyxTQUFTLENBQUMsT0FBb0I7UUFDMUMsTUFBTSxHQUFHLEdBQUcsSUFBSSxHQUFHLEVBQW1CLENBQUM7UUFDdkMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLEdBQUcsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLElBQUksRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ3pDLE9BQU8sR0FBRyxDQUFDO0lBQ2IsQ0FBQztJQUVNLE1BQU0sQ0FBQyxXQUFXLENBQUMsT0FBb0I7UUFDNUMsT0FBTyxPQUFPLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQ2xDLENBQUM7SUFhRDs7OztnRkFJNEU7SUFFNUUsWUFDa0IsSUFBVTtJQUMxQjs7O09BR0c7SUFDYSxjQUE2QjtRQUw3QixTQUFJLEdBQUosSUFBSSxDQUFNO1FBS1YsbUJBQWMsR0FBZCxjQUFjLENBQWU7UUFFN0MsSUFBSSxDQUFDLFlBQVksR0FBRyxTQUFTLENBQUMsaUJBQWlCLENBQUMsY0FBYyxFQUFFLElBQUksQ0FBQyxDQUFDO0lBQ3hFLENBQUM7SUFFRDs7OztnRkFJNEU7SUFFNUU7O09BRUc7SUFDSyxNQUFNLENBQUMsaUJBQWlCLENBQUMsY0FBc0IsRUFBRSxJQUFVO1FBQ2pFLElBQUksY0FBYyxJQUFJLElBQUksRUFBRSxDQUFDO1lBQzNCLE9BQU8sY0FBYyxHQUFHLEdBQUcsR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDO1FBQzFDLENBQUM7YUFBTSxDQUFDO1lBQ04sT0FBTyxJQUFJLENBQUMsSUFBSSxDQUFDO1FBQ25CLENBQUM7SUFDSCxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFJlcHJlc2VudHMgYSBmaWxlIGFuZCB0aGUgcmVsYXRpdmUgcGF0aCB3aGVyZSB0aGUgdXNlciBoYXMgc2VsZWN0ZWQgaXQuXG4gKiBUaGlzIGlzIHVzZWZ1bCBpZiB0aGUgdXNlciBoYXMgc2VsZWN0ZWQgYSBmb2xkZXIgYW5kIGZpbGVzXG4gKiBhcmUgbGlzdGVkIHJlY3Vyc2l2ZWx5LlxuICovXG5leHBvcnQgY2xhc3MgRmlsZUVudHJ5IHtcblxuICAvKipcbiAgICogQ3JlYXRlcyBhIGZpbGUgRW50cnkgd2l0aG91dCBhbiBleHBsaWNpdCByZWxhdGl2ZSBwYXJlbnQuXG4gICAqXG4gICAqIEhvd2V2ZXIsIGRlcGVuZGluZyBvbiBob3cgdGhlIGZpbGUgd2FzIHNlbGVjdGVkLCBzb21lIGJyb3dzZXJzXG4gICAqIGVuY29kZSB0aGUgcmVsYXRpdmUgcGFyZW50IHBhdGggaW4gdGhlIHByb3BlcnR5ICd3ZWJraXRSZWxhdGl2ZVBhdGgnXG4gICAqIHdoaWNoIGlzIGFsc28gc3VwcG9ydGVkIGJ5IHRoaXMgbWV0aG9kLlxuICAgKi9cbiAgcHVibGljIHN0YXRpYyBvZkZpbGUoZmlsZTogRmlsZSk6IEZpbGVFbnRyeSB7XG5cbiAgICBsZXQgcmVsYXRpdmVQYXJlbnQ6IHN0cmluZyA9IG51bGw7XG5cbiAgICBpZiAoZmlsZS53ZWJraXRSZWxhdGl2ZVBhdGgpIHtcbiAgICAgIGlmIChmaWxlLndlYmtpdFJlbGF0aXZlUGF0aC5lbmRzV2l0aChmaWxlLm5hbWUpKSB7XG4gICAgICAgIGNvbnN0IG5hbWVTdGFydCA9IGZpbGUud2Via2l0UmVsYXRpdmVQYXRoLmxlbmd0aCAtIGZpbGUubmFtZS5sZW5ndGg7XG4gICAgICAgIHJlbGF0aXZlUGFyZW50ID0gZmlsZS53ZWJraXRSZWxhdGl2ZVBhdGguc3Vic3RyaW5nKDAsIG5hbWVTdGFydCk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICByZWxhdGl2ZVBhcmVudCA9IGZpbGUud2Via2l0UmVsYXRpdmVQYXRoO1xuICAgICAgfVxuICAgIH1cblxuICAgIHJldHVybiBGaWxlRW50cnkucmVsYXRpdmVGaWxlKFxuICAgICAgZmlsZSxcbiAgICAgIHJlbGF0aXZlUGFyZW50XG4gICAgKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDcmVhdGVzIGEgZmlsZSBFbnRyeSB3aXRoIGEgcmVsYXRpdmUgcGFyZW50IHBhdGhcbiAgICovXG4gIHB1YmxpYyBzdGF0aWMgcmVsYXRpdmVGaWxlKGZpbGU6IEZpbGUsIHJlbGF0aXZlUGFyZW50OiBzdHJpbmcgfCBudWxsKSB7XG4gICAgaWYgKHJlbGF0aXZlUGFyZW50ICYmIHJlbGF0aXZlUGFyZW50LmVuZHNXaXRoKCcvJykpIHtcbiAgICAgIHJlbGF0aXZlUGFyZW50ID0gcmVsYXRpdmVQYXJlbnQuc3Vic3RyaW5nKDAsIHJlbGF0aXZlUGFyZW50Lmxlbmd0aCAtIDEpO1xuICAgIH1cbiAgICByZXR1cm4gbmV3IEZpbGVFbnRyeShmaWxlLCByZWxhdGl2ZVBhcmVudCk7XG4gIH1cblxuICBwdWJsaWMgc3RhdGljIHRvRmlsZU1hcChlbnRyaWVzOiBGaWxlRW50cnlbXSk6IE1hcDxGaWxlLCBGaWxlRW50cnk+IHtcbiAgICBjb25zdCBtYXAgPSBuZXcgTWFwPEZpbGUsIEZpbGVFbnRyeT4oKTtcbiAgICBlbnRyaWVzLmZvckVhY2goZSA9PiBtYXAuc2V0KGUuZmlsZSwgZSkpO1xuICAgIHJldHVybiBtYXA7XG4gIH1cblxuICBwdWJsaWMgc3RhdGljIHRvRmlsZUFycmF5KGVudHJpZXM6IEZpbGVFbnRyeVtdKTogRmlsZVtdIHtcbiAgICByZXR1cm4gZW50cmllcy5tYXAoZSA9PiBlLmZpbGUpO1xuICB9XG5cbiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqIEZpZWxkcyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgLyoqXG4gICAqIFJldHVybnMgYSBzdHJpbmcgd2hpY2ggc3BlY2lmaWVzIHRoZSBmaWxlJ3MgcGF0aCByZWxhdGl2ZSB0byB0aGUgZGlyZWN0b3J5IHNlbGVjdGVkIGJ5IHRoZSB1c2VyLlxuICAgKi9cbiAgcHVibGljIHJlYWRvbmx5IHJlbGF0aXZlUGF0aDogc3RyaW5nO1xuXG4gIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKipcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiBDb25zdHJ1Y3RvciAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqL1xuXG4gIHByaXZhdGUgY29uc3RydWN0b3IoXG4gICAgcHVibGljIHJlYWRvbmx5IGZpbGU6IEZpbGUsXG4gICAgLyoqXG4gICAgICogQ29udGFpbnMgdGhlIHJlbGF0aXZlIHBhdGggZnJvbSB0aGUgc2VsZWN0ZWQgZm9sZGVyXG4gICAgICogdG8gdGhpcyBmaWxlLiBPbmx5IHJlbGV2YW50IGlmIHRoZSB1c2VyIGhhcyBzZWxlY3RlZCBhIGZvbGRlci5cbiAgICAgKi9cbiAgICBwdWJsaWMgcmVhZG9ubHkgcmVsYXRpdmVQYXJlbnQ6IHN0cmluZyB8IG51bGxcbiAgKSB7XG4gICAgdGhpcy5yZWxhdGl2ZVBhdGggPSBGaWxlRW50cnkuYnVpbGRSZWxhdGl2ZVBhdGgocmVsYXRpdmVQYXJlbnQsIGZpbGUpO1xuICB9XG5cbiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqIFByaXZhdGUgbWV0aG9kcyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgLyoqXG4gICAqIFJldHVybnMgYSBzdHJpbmcgd2hpY2ggc3BlY2lmaWVzIHRoZSBmaWxlJ3MgcGF0aCByZWxhdGl2ZSB0byB0aGUgZGlyZWN0b3J5IHNlbGVjdGVkIGJ5IHRoZSB1c2VyLlxuICAgKi9cbiAgcHJpdmF0ZSBzdGF0aWMgYnVpbGRSZWxhdGl2ZVBhdGgocmVsYXRpdmVQYXJlbnQ6IHN0cmluZywgZmlsZTogRmlsZSk6IHN0cmluZyB7XG4gICAgaWYgKHJlbGF0aXZlUGFyZW50ICE9IG51bGwpIHtcbiAgICAgIHJldHVybiByZWxhdGl2ZVBhcmVudCArICcvJyArIGZpbGUubmFtZTtcbiAgICB9IGVsc2Uge1xuICAgICAgcmV0dXJuIGZpbGUubmFtZTtcbiAgICB9XG4gIH1cbn1cbiJdfQ==
@@ -1,10 +1,7 @@
1
- import { of, switchMap, zip } from 'rxjs';
1
+ import { EMPTY, mergeMap, Observable, of, zip } from 'rxjs';
2
2
  import { FileEntry } from './file-entry';
3
3
  import { map } from 'rxjs/operators';
4
4
  import { LoggerFactory } from '@elderbyte/ts-logger';
5
- import { FileListingWebkit } from './file-listing-webkit';
6
- import { FileListingSystemHandle } from './file-listing-system-handle';
7
- import { FileSystemApi } from '../file-system-api';
8
5
  export class FileListingRx {
9
6
  constructor() {
10
7
  this.log = LoggerFactory.getLogger(this.constructor.name);
@@ -19,46 +16,89 @@ export class FileListingRx {
19
16
  const obs = [];
20
17
  for (let i = 0; i < transferItemList.length; i++) {
21
18
  const transferItem = transferItemList[i];
22
- obs.push(this.resolveTransferItem(transferItem));
19
+ const entry = transferItem.webkitGetAsEntry();
20
+ if (entry) {
21
+ obs.push(this.listFilesRecursive(entry));
22
+ }
23
+ else {
24
+ const itemAsFile = transferItem.getAsFile();
25
+ if (itemAsFile) {
26
+ obs.push(of([FileEntry.ofFile(itemAsFile)]));
27
+ }
28
+ else {
29
+ if (transferItem.kind == 'file') {
30
+ obs.push(of([
31
+ FileEntry.ofFile(new File([], '', {
32
+ type: transferItem.type
33
+ }))
34
+ ]));
35
+ }
36
+ else {
37
+ this.log.warn('Could not handle DataTransferItem!', transferItem);
38
+ }
39
+ }
40
+ }
23
41
  }
24
42
  return zip(obs).pipe(map(files => files.flat()));
25
43
  }
44
+ listFilesRecursive(entry) {
45
+ return this.listFilesRecursiveAsync(entry, null);
46
+ }
26
47
  /***************************************************************************
27
48
  * *
28
49
  * Private methods *
29
50
  * *
30
51
  **************************************************************************/
31
- resolveTransferItem(transferItem) {
32
- if (FileSystemApi.isGetAsFileSystemHandleSupported(transferItem)) {
33
- return FileSystemApi.getAsFileSystemHandle(transferItem).pipe(switchMap(handle => handle ? FileListingSystemHandle.listFiles(handle) : this.legacyFileListing(transferItem)));
52
+ listFilesRecursiveAsync(entry, parent) {
53
+ if (entry.isDirectory) {
54
+ const directoryEntry = entry;
55
+ return this.readEntries(directoryEntry).pipe(mergeMap(entries => zip(entries.map(entry => this.listFilesRecursiveAsync(entry, directoryEntry)))), map(files => files.flat()));
34
56
  }
35
- return this.legacyFileListing(transferItem);
36
- }
37
- legacyFileListing(transferItem) {
38
- const entry = transferItem.webkitGetAsEntry();
39
- if (entry) {
40
- return FileListingWebkit.listFilesRecursive(entry);
57
+ else if (entry.isFile) {
58
+ const fileEntry = entry;
59
+ return this.observableFile(fileEntry, parent).pipe(map(file => [file]));
60
+ }
61
+ else {
62
+ return EMPTY;
41
63
  }
42
- return this.fallbackFileListing(transferItem);
43
64
  }
44
- fallbackFileListing(transferItem) {
45
- const itemAsFile = transferItem.getAsFile();
46
- if (itemAsFile) {
47
- return of([FileEntry.ofFile(itemAsFile)]);
65
+ observableFile(fileEntry, parent) {
66
+ return new Observable(observer => {
67
+ fileEntry.file(file => {
68
+ observer.next(FileEntry.relativeFile(file, this.trimStaringSlash(parent?.fullPath)));
69
+ observer.complete();
70
+ }, err => {
71
+ observer.error(err);
72
+ observer.complete();
73
+ });
74
+ });
75
+ }
76
+ trimStaringSlash(path) {
77
+ if (path) {
78
+ if (path.startsWith('/')) {
79
+ path = path.substring(1);
80
+ }
48
81
  }
49
- else {
50
- if (transferItem.kind == 'file') {
51
- return of([
52
- FileEntry.ofFile(new File([], '', {
53
- type: transferItem.type
54
- }))
55
- ]);
82
+ return path;
83
+ }
84
+ readEntries(directoryEntry) {
85
+ return new Observable(observer => {
86
+ this.consumeReaderToCompletion(observer, directoryEntry.createReader());
87
+ });
88
+ }
89
+ consumeReaderToCompletion(observer, directoryReader) {
90
+ directoryReader.readEntries(entries => {
91
+ if (entries && entries.length > 0) {
92
+ observer.next(entries);
93
+ this.consumeReaderToCompletion(observer, directoryReader);
56
94
  }
57
95
  else {
58
- this.log.warn('Could not handle DataTransferItem!', transferItem);
59
- return of([]);
96
+ observer.complete();
60
97
  }
61
- }
98
+ }, err => {
99
+ observer.error(err);
100
+ observer.complete();
101
+ });
62
102
  }
63
103
  }
64
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmlsZS1saXN0aW5nLXJ4LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvZWxkZXJieXRlL25neC1zdGFydGVyL3NyYy9saWIvY29tcG9uZW50cy9maWxlcy9saXN0aW5nL2ZpbGUtbGlzdGluZy1yeC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQWEsRUFBRSxFQUFFLFNBQVMsRUFBRSxHQUFHLEVBQUMsTUFBTSxNQUFNLENBQUM7QUFDcEQsT0FBTyxFQUFDLFNBQVMsRUFBQyxNQUFNLGNBQWMsQ0FBQztBQUN2QyxPQUFPLEVBQUMsR0FBRyxFQUFDLE1BQU0sZ0JBQWdCLENBQUM7QUFDbkMsT0FBTyxFQUFDLGFBQWEsRUFBQyxNQUFNLHNCQUFzQixDQUFDO0FBQ25ELE9BQU8sRUFBQyxpQkFBaUIsRUFBQyxNQUFNLHVCQUF1QixDQUFDO0FBQ3hELE9BQU8sRUFBQyx1QkFBdUIsRUFBQyxNQUFNLDhCQUE4QixDQUFDO0FBQ3JFLE9BQU8sRUFBQyxhQUFhLEVBQUMsTUFBTSxvQkFBb0IsQ0FBQztBQUVqRCxNQUFNLE9BQU8sYUFBYTtJQUExQjtRQUltQixRQUFHLEdBQUcsYUFBYSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxDQUFDO0lBK0R4RSxDQUFDO2FBakV3QixhQUFRLEdBQUcsSUFBSSxhQUFhLEVBQUUsQUFBdEIsQ0FBdUI7SUFJdEQ7Ozs7Z0ZBSTRFO0lBRXJFLFVBQVUsQ0FBQyxnQkFBc0M7UUFDdEQsTUFBTSxHQUFHLEdBQThCLEVBQUUsQ0FBQztRQUMxQyxLQUFLLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsZ0JBQWdCLENBQUMsTUFBTSxFQUFFLENBQUMsRUFBRSxFQUFFLENBQUM7WUFDakQsTUFBTSxZQUFZLEdBQUcsZ0JBQWdCLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDekMsR0FBRyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsbUJBQW1CLENBQUMsWUFBWSxDQUFDLENBQUMsQ0FBQztRQUNuRCxDQUFDO1FBQ0QsT0FBTyxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUMsSUFBSSxDQUNsQixHQUFHLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FDM0IsQ0FBQztJQUNKLENBQUM7SUFFRDs7OztnRkFJNEU7SUFFcEUsbUJBQW1CLENBQUMsWUFBOEI7UUFFeEQsSUFBSSxhQUFhLENBQUMsZ0NBQWdDLENBQUMsWUFBWSxDQUFDLEVBQUUsQ0FBQztZQUNqRSxPQUFPLGFBQWEsQ0FBQyxxQkFBcUIsQ0FBQyxZQUFZLENBQUMsQ0FBQyxJQUFJLENBQzNELFNBQVMsQ0FBQyxNQUFNLENBQUMsRUFBRSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsdUJBQXVCLENBQUMsU0FBUyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsaUJBQWlCLENBQUMsWUFBWSxDQUFDLENBQUMsQ0FDL0csQ0FBQztRQUNKLENBQUM7UUFDRCxPQUFPLElBQUksQ0FBQyxpQkFBaUIsQ0FBQyxZQUFZLENBQUMsQ0FBQztJQUM5QyxDQUFDO0lBR08saUJBQWlCLENBQUMsWUFBOEI7UUFDdEQsTUFBTSxLQUFLLEdBQUcsWUFBWSxDQUFDLGdCQUFnQixFQUFFLENBQUM7UUFDOUMsSUFBSSxLQUFLLEVBQUUsQ0FBQztZQUNWLE9BQU8saUJBQWlCLENBQUMsa0JBQWtCLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDckQsQ0FBQztRQUNELE9BQU8sSUFBSSxDQUFDLG1CQUFtQixDQUFDLFlBQVksQ0FBQyxDQUFDO0lBQ2hELENBQUM7SUFFTyxtQkFBbUIsQ0FBQyxZQUE4QjtRQUN4RCxNQUFNLFVBQVUsR0FBRyxZQUFZLENBQUMsU0FBUyxFQUFFLENBQUM7UUFDNUMsSUFBSSxVQUFVLEVBQUUsQ0FBQztZQUNmLE9BQU8sRUFBRSxDQUFDLENBQUMsU0FBUyxDQUFDLE1BQU0sQ0FBQyxVQUFVLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDNUMsQ0FBQzthQUFNLENBQUM7WUFDTixJQUFJLFlBQVksQ0FBQyxJQUFJLElBQUksTUFBTSxFQUFFLENBQUM7Z0JBQ2hDLE9BQU8sRUFBRSxDQUFDO29CQUNSLFNBQVMsQ0FBQyxNQUFNLENBQ2QsSUFBSSxJQUFJLENBQUMsRUFBRSxFQUFFLEVBQUUsRUFBRTt3QkFDZixJQUFJLEVBQUUsWUFBWSxDQUFDLElBQUk7cUJBQ3hCLENBQUMsQ0FDSDtpQkFDRixDQUFDLENBQUM7WUFDTCxDQUFDO2lCQUFNLENBQUM7Z0JBQ04sSUFBSSxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsb0NBQW9DLEVBQUUsWUFBWSxDQUFDLENBQUM7Z0JBQ2xFLE9BQU8sRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDO1lBQ2hCLENBQUM7UUFDSCxDQUFDO0lBQ0gsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7T2JzZXJ2YWJsZSwgb2YsIHN3aXRjaE1hcCwgemlwfSBmcm9tICdyeGpzJztcbmltcG9ydCB7RmlsZUVudHJ5fSBmcm9tICcuL2ZpbGUtZW50cnknO1xuaW1wb3J0IHttYXB9IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcbmltcG9ydCB7TG9nZ2VyRmFjdG9yeX0gZnJvbSAnQGVsZGVyYnl0ZS90cy1sb2dnZXInO1xuaW1wb3J0IHtGaWxlTGlzdGluZ1dlYmtpdH0gZnJvbSAnLi9maWxlLWxpc3Rpbmctd2Via2l0JztcbmltcG9ydCB7RmlsZUxpc3RpbmdTeXN0ZW1IYW5kbGV9IGZyb20gJy4vZmlsZS1saXN0aW5nLXN5c3RlbS1oYW5kbGUnO1xuaW1wb3J0IHtGaWxlU3lzdGVtQXBpfSBmcm9tICcuLi9maWxlLXN5c3RlbS1hcGknO1xuXG5leHBvcnQgY2xhc3MgRmlsZUxpc3RpbmdSeCB7XG5cbiAgcHVibGljIHN0YXRpYyByZWFkb25seSBJTlNUQU5DRSA9IG5ldyBGaWxlTGlzdGluZ1J4KCk7XG5cbiAgcHJpdmF0ZSByZWFkb25seSBsb2cgPSBMb2dnZXJGYWN0b3J5LmdldExvZ2dlcih0aGlzLmNvbnN0cnVjdG9yLm5hbWUpO1xuXG4gIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKipcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiBQdWJsaWMgQVBJICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqL1xuXG4gIHB1YmxpYyB0b0ZpbGVMaXN0KHRyYW5zZmVySXRlbUxpc3Q6IERhdGFUcmFuc2Zlckl0ZW1MaXN0KTogT2JzZXJ2YWJsZTxGaWxlRW50cnlbXT4ge1xuICAgIGNvbnN0IG9iczogT2JzZXJ2YWJsZTxGaWxlRW50cnlbXT5bXSA9IFtdO1xuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgdHJhbnNmZXJJdGVtTGlzdC5sZW5ndGg7IGkrKykge1xuICAgICAgY29uc3QgdHJhbnNmZXJJdGVtID0gdHJhbnNmZXJJdGVtTGlzdFtpXTtcbiAgICAgIG9icy5wdXNoKHRoaXMucmVzb2x2ZVRyYW5zZmVySXRlbSh0cmFuc2Zlckl0ZW0pKTtcbiAgICB9XG4gICAgcmV0dXJuIHppcChvYnMpLnBpcGUoXG4gICAgICBtYXAoZmlsZXMgPT4gZmlsZXMuZmxhdCgpKVxuICAgICk7XG4gIH1cblxuICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogUHJpdmF0ZSBtZXRob2RzICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi9cblxuICBwcml2YXRlIHJlc29sdmVUcmFuc2Zlckl0ZW0odHJhbnNmZXJJdGVtOiBEYXRhVHJhbnNmZXJJdGVtKTogT2JzZXJ2YWJsZTxGaWxlRW50cnlbXT4ge1xuXG4gICAgaWYgKEZpbGVTeXN0ZW1BcGkuaXNHZXRBc0ZpbGVTeXN0ZW1IYW5kbGVTdXBwb3J0ZWQodHJhbnNmZXJJdGVtKSkge1xuICAgICAgcmV0dXJuIEZpbGVTeXN0ZW1BcGkuZ2V0QXNGaWxlU3lzdGVtSGFuZGxlKHRyYW5zZmVySXRlbSkucGlwZShcbiAgICAgICAgc3dpdGNoTWFwKGhhbmRsZSA9PiBoYW5kbGUgPyBGaWxlTGlzdGluZ1N5c3RlbUhhbmRsZS5saXN0RmlsZXMoaGFuZGxlKSA6IHRoaXMubGVnYWN5RmlsZUxpc3RpbmcodHJhbnNmZXJJdGVtKSlcbiAgICAgICk7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLmxlZ2FjeUZpbGVMaXN0aW5nKHRyYW5zZmVySXRlbSk7XG4gIH1cblxuXG4gIHByaXZhdGUgbGVnYWN5RmlsZUxpc3RpbmcodHJhbnNmZXJJdGVtOiBEYXRhVHJhbnNmZXJJdGVtKTogT2JzZXJ2YWJsZTxGaWxlRW50cnlbXT4ge1xuICAgIGNvbnN0IGVudHJ5ID0gdHJhbnNmZXJJdGVtLndlYmtpdEdldEFzRW50cnkoKTtcbiAgICBpZiAoZW50cnkpIHtcbiAgICAgIHJldHVybiBGaWxlTGlzdGluZ1dlYmtpdC5saXN0RmlsZXNSZWN1cnNpdmUoZW50cnkpO1xuICAgIH1cbiAgICByZXR1cm4gdGhpcy5mYWxsYmFja0ZpbGVMaXN0aW5nKHRyYW5zZmVySXRlbSk7XG4gIH1cblxuICBwcml2YXRlIGZhbGxiYWNrRmlsZUxpc3RpbmcodHJhbnNmZXJJdGVtOiBEYXRhVHJhbnNmZXJJdGVtKTogT2JzZXJ2YWJsZTxGaWxlRW50cnlbXT4ge1xuICAgIGNvbnN0IGl0ZW1Bc0ZpbGUgPSB0cmFuc2Zlckl0ZW0uZ2V0QXNGaWxlKCk7XG4gICAgaWYgKGl0ZW1Bc0ZpbGUpIHtcbiAgICAgIHJldHVybiBvZihbRmlsZUVudHJ5Lm9mRmlsZShpdGVtQXNGaWxlKV0pO1xuICAgIH0gZWxzZSB7XG4gICAgICBpZiAodHJhbnNmZXJJdGVtLmtpbmQgPT0gJ2ZpbGUnKSB7XG4gICAgICAgIHJldHVybiBvZihbXG4gICAgICAgICAgRmlsZUVudHJ5Lm9mRmlsZShcbiAgICAgICAgICAgIG5ldyBGaWxlKFtdLCAnJywge1xuICAgICAgICAgICAgICB0eXBlOiB0cmFuc2Zlckl0ZW0udHlwZVxuICAgICAgICAgICAgfSlcbiAgICAgICAgICApXG4gICAgICAgIF0pO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgdGhpcy5sb2cud2FybignQ291bGQgbm90IGhhbmRsZSBEYXRhVHJhbnNmZXJJdGVtIScsIHRyYW5zZmVySXRlbSk7XG4gICAgICAgIHJldHVybiBvZihbXSk7XG4gICAgICB9XG4gICAgfVxuICB9XG59XG4iXX0=
104
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-listing-rx.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/files/listing/file-listing-rx.ts"],"names":[],"mappings":"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","sourcesContent":["import {EMPTY, mergeMap, Observable, of, Subscriber, zip} from 'rxjs';\nimport {FileEntry} from './file-entry';\nimport {map} from 'rxjs/operators';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\n\nexport class FileListingRx {\n\n  public static readonly INSTANCE = new FileListingRx();\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public toFileList(transferItemList: DataTransferItemList): Observable<FileEntry[]> {\n    const obs: Observable<FileEntry[]>[] = [];\n    for (let i = 0; i < transferItemList.length; i++) {\n      const transferItem = transferItemList[i];\n      const entry = transferItem.webkitGetAsEntry();\n      if (entry) {\n        obs.push(this.listFilesRecursive(entry));\n      } else {\n        const itemAsFile = transferItem.getAsFile();\n        if (itemAsFile) {\n          obs.push(of([FileEntry.ofFile(itemAsFile)]));\n        } else {\n          if (transferItem.kind == 'file') {\n            obs.push(of([\n              FileEntry.ofFile(\n                new File([], '', {\n                  type: transferItem.type\n                })\n              )\n            ]));\n          } else {\n            this.log.warn('Could not handle DataTransferItem!', transferItem);\n          }\n        }\n      }\n    }\n    return zip(obs).pipe(\n      map(files => files.flat()),\n    );\n  }\n\n  public listFilesRecursive(\n    entry: FileSystemEntry\n  ): Observable<FileEntry[]> {\n    return this.listFilesRecursiveAsync(entry, null);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private listFilesRecursiveAsync(\n    entry: FileSystemEntry,\n    parent: FileSystemDirectoryEntry\n  ): Observable<FileEntry[]> {\n    if (entry.isDirectory) {\n      const directoryEntry = entry as FileSystemDirectoryEntry;\n      return this.readEntries(directoryEntry).pipe(\n        mergeMap(entries => zip(\n          entries.map(entry => this.listFilesRecursiveAsync(entry, directoryEntry))\n        )),\n        map(files => files.flat()),\n      );\n    } else if (entry.isFile) {\n      const fileEntry = entry as FileSystemFileEntry;\n      return this.observableFile(fileEntry, parent).pipe(\n        map(file => [file])\n      );\n    } else {\n      return EMPTY;\n    }\n  }\n\n  private observableFile(\n    fileEntry: FileSystemFileEntry,\n    parent: FileSystemDirectoryEntry | null\n  ): Observable<FileEntry> {\n    return new Observable<FileEntry>(\n      observer => {\n        fileEntry.file(file => {\n          observer.next(FileEntry.relativeFile(file, this.trimStaringSlash(parent?.fullPath)));\n          observer.complete();\n        }, err => {\n          observer.error(err);\n          observer.complete();\n        });\n      }\n    );\n  }\n\n  private trimStaringSlash(path: string | null): string {\n    if (path) {\n      if (path.startsWith('/')) {\n        path = path.substring(1);\n      }\n    }\n    return path;\n  }\n\n  private readEntries(\n    directoryEntry: FileSystemDirectoryEntry\n  ): Observable<FileSystemEntry[]> {\n    return new Observable<FileSystemEntry[]>(\n      observer => {\n        this.consumeReaderToCompletion(observer, directoryEntry.createReader());\n      }\n    );\n  }\n\n  private consumeReaderToCompletion(\n    observer: Subscriber<FileSystemEntry[]>,\n    directoryReader: FileSystemDirectoryReader\n  ): void {\n    directoryReader.readEntries(entries => {\n      if (entries && entries.length > 0) {\n        observer.next(entries);\n        this.consumeReaderToCompletion(observer, directoryReader);\n      } else {\n        observer.complete();\n      }\n    }, err => {\n      observer.error(err);\n      observer.complete();\n    });\n  }\n\n}\n"]}