@elderbyte/ngx-starter 18.2.0-beta4 → 18.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,119 @@
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"]}
@@ -2,6 +2,11 @@ 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';
5
10
  import * as i0 from "@angular/core";
6
11
  export class ElderFileSelectDirective {
7
12
  /***************************************************************************
@@ -9,9 +14,10 @@ export class ElderFileSelectDirective {
9
14
  * Constructor *
10
15
  * *
11
16
  **************************************************************************/
12
- constructor(el, renderer) {
13
- this.el = el;
17
+ constructor(renderer, el, destroyRef) {
14
18
  this.renderer = renderer;
19
+ this.el = el;
20
+ this.destroyRef = destroyRef;
15
21
  /***************************************************************************
16
22
  * *
17
23
  * Fields *
@@ -20,6 +26,8 @@ export class ElderFileSelectDirective {
20
26
  this.logger = LoggerFactory.getLogger(this.constructor.name);
21
27
  this.elderFileSelectChange = new EventEmitter();
22
28
  this.elderSingleFileSelectChange = new EventEmitter();
29
+ this._useDirectoryPicker = false;
30
+ this._legacyInput = new ElderFileSelectInput(this.renderer, this.el);
23
31
  }
24
32
  /***************************************************************************
25
33
  * *
@@ -27,10 +35,16 @@ export class ElderFileSelectDirective {
27
35
  * *
28
36
  **************************************************************************/
29
37
  ngOnInit() {
30
- this.createFileSelect();
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
+ }
31
45
  }
32
46
  ngOnDestroy() {
33
- this.removeFileSelect();
47
+ this._legacyInput.removeFileSelect();
34
48
  }
35
49
  /***************************************************************************
36
50
  * *
@@ -38,16 +52,49 @@ export class ElderFileSelectDirective {
38
52
  * *
39
53
  **************************************************************************/
40
54
  set elderFileSelect(value) {
41
- this._accept = value;
42
- if (this._fileInput) {
43
- this.renderer.setProperty(this._fileInput, 'accept', 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
+ };
44
73
  }
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);
45
88
  }
46
89
  set elderFileSelectMultiple(value) {
47
- this._multiple = coerceBooleanProperty(value);
48
- if (this._fileInput) {
49
- this.renderer.setProperty(this._fileInput, 'multiple', value);
50
- }
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;
51
98
  }
52
99
  /**
53
100
  * Allow the user to select a directory. All files that are recursively contained are selected.
@@ -55,11 +102,8 @@ export class ElderFileSelectDirective {
55
102
  * @param value
56
103
  */
57
104
  set 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
- }
105
+ this._useDirectoryPicker = coerceBooleanProperty(value);
106
+ this._legacyInput.elderFileSelectDirectory = value;
63
107
  }
64
108
  /***************************************************************************
65
109
  * *
@@ -67,81 +111,39 @@ export class ElderFileSelectDirective {
67
111
  * *
68
112
  **************************************************************************/
69
113
  onClick(event) {
70
- this.openFileSelectDialog();
71
- }
72
- openFileSelectDialog() {
73
- this._fileInput.click();
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
+ }
74
127
  }
75
128
  /***************************************************************************
76
129
  * *
77
130
  * Private methods *
78
131
  * *
79
132
  **************************************************************************/
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;
133
+ pushFileChanges(multiFileChange) {
134
+ this.elderFileSelectChange.next(multiFileChange);
135
+ const file = multiFileChange[0]?.file;
136
+ if (file) {
137
+ this.elderSingleFileSelectChange.next(file);
113
138
  }
114
139
  }
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();
140
+ openFilePicker(pickerOpts) {
141
+ return FileSystemApi.openFilePicker(pickerOpts).pipe(switchMap(files => combineLatest(files.map(file => FileEntry.ofFileHandle(file)))));
121
142
  }
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;
143
+ openDirectoryPicker(pickerOpts) {
144
+ return FileSystemApi.openDirectoryPicker(pickerOpts).pipe(switchMap(dir => FileListingSystemHandle.listFiles(dir)));
143
145
  }
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 }); }
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 }); }
145
147
  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 }); }
146
148
  }
147
149
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.4", ngImport: i0, type: ElderFileSelectDirective, decorators: [{
@@ -150,7 +152,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.4", ngImpor
150
152
  selector: '[elderFileSelect]',
151
153
  standalone: true
152
154
  }]
153
- }], ctorParameters: () => [{ type: i0.ElementRef }, { type: i0.Renderer2 }], propDecorators: { elderFileSelectChange: [{
155
+ }], ctorParameters: () => [{ type: i0.Renderer2 }, { type: i0.ElementRef }, { type: i0.DestroyRef }], propDecorators: { elderFileSelectChange: [{
154
156
  type: Output
155
157
  }], elderSingleFileSelectChange: [{
156
158
  type: Output
@@ -164,4 +166,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.4", ngImpor
164
166
  type: HostListener,
165
167
  args: ['click', ['$event']]
166
168
  }] } });
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":"AAAA,OAAO,EACL,SAAS,EAET,YAAY,EACZ,YAAY,EACZ,KAAK,EAGL,MAAM,EAEP,MAAM,eAAe,CAAC;AACvB,OAAO,EAAC,aAAa,EAAC,MAAM,sBAAsB,CAAC;AACnD,OAAO,EAAe,qBAAqB,EAAC,MAAM,uBAAuB,CAAC;AAC1E,OAAO,EAAC,SAAS,EAAC,MAAM,sBAAsB,CAAC;;AAM/C,MAAM,OAAO,wBAAwB;IAuBnC;;;;gFAI4E;IAE5E,YACU,EAAc,EACd,QAAmB;QADnB,OAAE,GAAF,EAAE,CAAY;QACd,aAAQ,GAAR,QAAQ,CAAW;QA7B7B;;;;oFAI4E;QAE3D,WAAM,GAAG,aAAa,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;QAMzD,0BAAqB,GAAG,IAAI,YAAY,EAAe,CAAC;QAGxD,gCAA2B,GAAG,IAAI,YAAY,EAAQ,CAAC;IAgBvE,CAAC;IAED;;;;gFAI4E;IAErE,QAAQ;QACb,IAAI,CAAC,gBAAgB,EAAE,CAAC;IAC1B,CAAC;IAEM,WAAW;QAChB,IAAI,CAAC,gBAAgB,EAAE,CAAC;IAC1B,CAAC;IAED;;;;gFAI4E;IAE5E,IACW,eAAe,CAAC,KAAa;QACtC,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;QACrB,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YACpB,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,UAAU,EAAE,QAAQ,EAAE,KAAK,CAAC,CAAC;QAC9D,CAAC;IACH,CAAC;IAED,IACW,uBAAuB,CAAC,KAAmB;QACpD,IAAI,CAAC,SAAS,GAAG,qBAAqB,CAAC,KAAK,CAAC,CAAC;QAC9C,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YACpB,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,UAAU,EAAE,UAAU,EAAE,KAAK,CAAC,CAAC;QAChE,CAAC;IACH,CAAC;IAED;;;;OAIG;IACH,IACW,wBAAwB,CAAC,KAAmB;QACrD,IAAI,CAAC,UAAU,GAAG,qBAAqB,CAAC,KAAK,CAAC,CAAC;QAC/C,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YACpB,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,CAAC,UAAU,EAAE,iBAAiB,EAAE,MAAM,CAAC,CAAC;YACvE,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,CAAC,UAAU,EAAE,WAAW,EAAE,MAAM,CAAC,CAAC;QACnE,CAAC;IACH,CAAC;IAED;;;;gFAI4E;IAGrE,OAAO,CAAC,KAAU;QACvB,IAAI,CAAC,oBAAoB,EAAE,CAAC;IAC9B,CAAC;IAEM,oBAAoB;QACzB,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;IAC1B,CAAC;IAED;;;;gFAI4E;IAE5E;;;;;;;OAOG;IACK,gBAAgB;QAEtB,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC;QACvD,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,CAAC,UAAU,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC;QAC9D,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,CAAC,UAAU,EAAE,MAAM,EAAE,MAAM,CAAC,CAAC;QAC5D,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,EAAE,CAAC,aAAa,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC;QAElE,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;YACjB,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,UAAU,EAAE,QAAQ,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;QACrE,CAAC;QACD,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACnB,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,UAAU,EAAE,UAAU,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;QACzE,CAAC;QACD,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YACpB,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,CAAC,UAAU,EAAE,iBAAiB,EAAE,MAAM,CAAC,CAAC;YACvE,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,CAAC,UAAU,EAAE,WAAW,EAAE,MAAM,CAAC,CAAC;QACnE,CAAC;QAED,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CACnC,IAAI,CAAC,UAAU,EACf,QAAQ,EACR,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,CACtC,CAAC;IACJ,CAAC;IAEO,gBAAgB;QACtB,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACnB,IAAI,CAAC,SAAS,EAAE,CAAC;YACjB,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACxB,CAAC;QACD,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YACpB,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,EAAE,CAAC,aAAa,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC;YAClE,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QACzB,CAAC;IACH,CAAC;IAEO,gBAAgB,CAAC,KAAU;QACjC,MAAM,QAAQ,GAAG,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC;QACvC,MAAM,KAAK,GAAG,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;QAC3C,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,0BAA0B,EAAE,KAAK,CAAC,CAAC;QACrD,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;QACzB,IAAI,CAAC,aAAa,EAAE,CAAC;IACvB,CAAC;IAEO,aAAa;QACnB,qBAAqB;QACrB,IAAI,CAAC,UAAU,CAAC,KAAK,GAAG,IAAI,CAAC;IAC/B,CAAC;IAEO,YAAY,CAAC,KAAkB;QACrC,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACrB,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YACvC,IAAI,CAAC,2BAA2B,CAAC,IAAI,CACnC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CACd,CAAC;QACJ,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,uDAAuD,CAAC,CAAC;QAC5E,CAAC;IACH,CAAC;IAEO,aAAa,CAAC,QAAkB;QACtC,MAAM,KAAK,GAAgB,EAAE,CAAC;QAC9B,KAAK,MAAM,GAAG,IAAI,QAAQ,EAAE,CAAC;YAC3B,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;gBAC7B,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;YAC9C,CAAC;QACH,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;8GAtLU,wBAAwB;kGAAxB,wBAAwB;;2FAAxB,wBAAwB;kBAJpC,SAAS;mBAAC;oBACP,QAAQ,EAAE,mBAAmB;oBAC7B,UAAU,EAAE,IAAI;iBACnB;uGAeiB,qBAAqB;sBADpC,MAAM;gBAIS,2BAA2B;sBAD1C,MAAM;gBAwCI,eAAe;sBADzB,KAAK;gBASK,uBAAuB;sBADjC,KAAK;gBAcK,wBAAwB;sBADlC,KAAK;gBAgBC,OAAO;sBADb,YAAY;uBAAC,OAAO,EAAE,CAAC,QAAQ,CAAC","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"]}
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"]}
@@ -0,0 +1,96 @@
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"]}