@elderbyte/ngx-starter 18.2.0-beta4 → 18.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/components/files/elder-file-select-input.mjs +119 -0
- package/esm2022/lib/components/files/elder-file-select.directive.mjs +85 -83
- package/esm2022/lib/components/files/file-system-api.mjs +96 -0
- package/esm2022/lib/components/files/listing/file-entry.mjs +30 -15
- package/esm2022/lib/components/files/listing/file-listing-rx.mjs +30 -70
- package/esm2022/lib/components/files/listing/file-listing-system-handle.mjs +35 -0
- package/esm2022/lib/components/files/listing/file-listing-webkit.mjs +77 -0
- package/fesm2022/elderbyte-ngx-starter.mjs +387 -107
- package/fesm2022/elderbyte-ngx-starter.mjs.map +1 -1
- package/lib/components/files/elder-file-select-input.d.ts +59 -0
- package/lib/components/files/elder-file-select.directive.d.ts +16 -23
- package/lib/components/files/file-system-api.d.ts +86 -0
- package/lib/components/files/listing/file-entry.d.ts +10 -1
- package/lib/components/files/listing/file-listing-rx.d.ts +3 -6
- package/lib/components/files/listing/file-listing-system-handle.d.ts +21 -0
- package/lib/components/files/listing/file-listing-webkit.d.ts +26 -0
- package/package.json +1 -1
|
@@ -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,
|
|
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
|
-
|
|
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.
|
|
42
|
-
|
|
43
|
-
this.
|
|
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.
|
|
48
|
-
|
|
49
|
-
this.
|
|
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.
|
|
59
|
-
|
|
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.
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
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
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
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
|
-
|
|
116
|
-
|
|
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
|
-
|
|
123
|
-
|
|
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.
|
|
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.
|
|
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"]}
|