@elderbyte/ngx-starter 18.2.0-beta3 → 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/data-view/table/activation/elder-table-activation.directive.mjs +2 -2
- 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 +388 -108
- 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,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"]}
|
|
@@ -1,3 +1,6 @@
|
|
|
1
|
+
import { from } from 'rxjs';
|
|
2
|
+
import { combineLatestWith, map } from 'rxjs/operators';
|
|
3
|
+
import { FileSystemApi } from '../file-system-api';
|
|
1
4
|
/**
|
|
2
5
|
* Represents a file and the relative path where the user has selected it.
|
|
3
6
|
* This is useful if the user has selected a folder and files
|
|
@@ -12,26 +15,24 @@ export class FileEntry {
|
|
|
12
15
|
* which is also supported by this method.
|
|
13
16
|
*/
|
|
14
17
|
static ofFile(file) {
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
}
|
|
25
|
-
return FileEntry.relativeFile(file, relativeParent);
|
|
18
|
+
return FileEntry.relativeFile(file, this.buildRelativeParent(file));
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* Creates an Observable of a File Entry by resolving its FileSystemFileHandle.
|
|
22
|
+
* @param fileHandle
|
|
23
|
+
* @param rootDirectory is needed to build a relative parent
|
|
24
|
+
*/
|
|
25
|
+
static ofFileHandle(fileHandle, rootDirectory) {
|
|
26
|
+
return from(fileHandle.getFile()).pipe(combineLatestWith(FileSystemApi.buildRelativeParent(fileHandle, rootDirectory)), map(([file, relativeParent]) => FileEntry.relativeFile(file, relativeParent, fileHandle)));
|
|
26
27
|
}
|
|
27
28
|
/**
|
|
28
29
|
* Creates a file Entry with a relative parent path
|
|
29
30
|
*/
|
|
30
|
-
static relativeFile(file, relativeParent) {
|
|
31
|
+
static relativeFile(file, relativeParent, fileHandle) {
|
|
31
32
|
if (relativeParent && relativeParent.endsWith('/')) {
|
|
32
33
|
relativeParent = relativeParent.substring(0, relativeParent.length - 1);
|
|
33
34
|
}
|
|
34
|
-
return new FileEntry(file, relativeParent);
|
|
35
|
+
return new FileEntry(file, relativeParent, fileHandle);
|
|
35
36
|
}
|
|
36
37
|
static toFileMap(entries) {
|
|
37
38
|
const map = new Map();
|
|
@@ -51,9 +52,10 @@ export class FileEntry {
|
|
|
51
52
|
* Contains the relative path from the selected folder
|
|
52
53
|
* to this file. Only relevant if the user has selected a folder.
|
|
53
54
|
*/
|
|
54
|
-
relativeParent) {
|
|
55
|
+
relativeParent, fileHandle) {
|
|
55
56
|
this.file = file;
|
|
56
57
|
this.relativeParent = relativeParent;
|
|
58
|
+
this.fileHandle = fileHandle;
|
|
57
59
|
this.relativePath = FileEntry.buildRelativePath(relativeParent, file);
|
|
58
60
|
}
|
|
59
61
|
/***************************************************************************
|
|
@@ -72,5 +74,18 @@ export class FileEntry {
|
|
|
72
74
|
return file.name;
|
|
73
75
|
}
|
|
74
76
|
}
|
|
77
|
+
static buildRelativeParent(file) {
|
|
78
|
+
let relativeParent = null;
|
|
79
|
+
if (file.webkitRelativePath) {
|
|
80
|
+
if (file.webkitRelativePath.endsWith(file.name)) {
|
|
81
|
+
const nameStart = file.webkitRelativePath.length - file.name.length;
|
|
82
|
+
relativeParent = file.webkitRelativePath.substring(0, nameStart);
|
|
83
|
+
}
|
|
84
|
+
else {
|
|
85
|
+
relativeParent = file.webkitRelativePath;
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
return relativeParent;
|
|
89
|
+
}
|
|
75
90
|
}
|
|
76
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
91
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-entry.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/files/listing/file-entry.ts"],"names":[],"mappings":"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","sourcesContent":["import {from, Observable} from 'rxjs';\nimport {combineLatestWith, map} from 'rxjs/operators';\nimport {FileSystemApi} from '../file-system-api';\n\n/**\n * Represents a file and the relative path where the user has selected it.\n * This is useful if the user has selected a folder and files\n * are listed recursively.\n */\nexport class FileEntry {\n\n  /**\n   * Creates a file Entry without an explicit relative parent.\n   *\n   * However, depending on how the file was selected, some browsers\n   * encode the relative parent path in the property 'webkitRelativePath'\n   * which is also supported by this method.\n   */\n  public static ofFile(file: File): FileEntry {\n\n    return FileEntry.relativeFile(\n      file,\n      this.buildRelativeParent(file)\n    );\n  }\n\n\n  /**\n   * Creates an Observable of a File Entry by resolving its FileSystemFileHandle.\n   * @param fileHandle\n   * @param rootDirectory is needed to build a relative parent\n   */\n  public static ofFileHandle(\n    fileHandle: FileSystemFileHandle,\n    rootDirectory?: FileSystemDirectoryHandle\n  ): Observable<FileEntry> {\n\n      return from(fileHandle.getFile()).pipe(\n        combineLatestWith(FileSystemApi.buildRelativeParent(fileHandle, rootDirectory)),\n        map(([file, relativeParent]) => FileEntry.relativeFile(\n          file,\n          relativeParent,\n          fileHandle))\n      );\n  }\n\n\n\n  /**\n   * Creates a file Entry with a relative parent path\n   */\n  public static relativeFile(\n    file: File,\n    relativeParent: string | null,\n    fileHandle?: FileSystemFileHandle\n  ): FileEntry {\n    if (relativeParent && relativeParent.endsWith('/')) {\n      relativeParent = relativeParent.substring(0, relativeParent.length - 1);\n    }\n    return new FileEntry(file, relativeParent, fileHandle);\n  }\n\n  public static toFileMap(entries: FileEntry[]): Map<File, FileEntry> {\n    const map = new Map<File, FileEntry>();\n    entries.forEach(e => map.set(e.file, e));\n    return map;\n  }\n\n  public static toFileArray(entries: FileEntry[]): File[] {\n    return entries.map(e => e.file);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Returns a string which specifies the file's path relative to the directory selected by the user.\n   */\n  public readonly relativePath: string;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  private constructor(\n    public readonly file: File,\n    /**\n     * Contains the relative path from the selected folder\n     * to this file. Only relevant if the user has selected a folder.\n     */\n    public readonly relativeParent: string | null,\n    public readonly fileHandle?: FileSystemFileHandle\n  ) {\n    this.relativePath = FileEntry.buildRelativePath(relativeParent, file);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Returns a string which specifies the file's path relative to the directory selected by the user.\n   */\n  private static buildRelativePath(relativeParent: string, file: File): string {\n    if (relativeParent != null) {\n      return relativeParent + '/' + file.name;\n    } else {\n      return file.name;\n    }\n  }\n\n\n  private static buildRelativeParent(file: File): string {\n    let relativeParent: string = null;\n\n    if (file.webkitRelativePath) {\n      if (file.webkitRelativePath.endsWith(file.name)) {\n        const nameStart = file.webkitRelativePath.length - file.name.length;\n        relativeParent = file.webkitRelativePath.substring(0, nameStart);\n      } else {\n        relativeParent = file.webkitRelativePath;\n      }\n    }\n    return relativeParent;\n  }\n}\n"]}
|
|
@@ -1,7 +1,10 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { of, switchMap, zip } from 'rxjs';
|
|
2
2
|
import { FileEntry } from './file-entry';
|
|
3
3
|
import { map } from 'rxjs/operators';
|
|
4
4
|
import { LoggerFactory } from '@elderbyte/ts-logger';
|
|
5
|
+
import { FileListingWebkit } from './file-listing-webkit';
|
|
6
|
+
import { FileListingSystemHandle } from './file-listing-system-handle';
|
|
7
|
+
import { FileSystemApi } from '../file-system-api';
|
|
5
8
|
export class FileListingRx {
|
|
6
9
|
constructor() {
|
|
7
10
|
this.log = LoggerFactory.getLogger(this.constructor.name);
|
|
@@ -16,89 +19,46 @@ export class FileListingRx {
|
|
|
16
19
|
const obs = [];
|
|
17
20
|
for (let i = 0; i < transferItemList.length; i++) {
|
|
18
21
|
const transferItem = transferItemList[i];
|
|
19
|
-
|
|
20
|
-
if (entry) {
|
|
21
|
-
obs.push(this.listFilesRecursive(entry));
|
|
22
|
-
}
|
|
23
|
-
else {
|
|
24
|
-
const itemAsFile = transferItem.getAsFile();
|
|
25
|
-
if (itemAsFile) {
|
|
26
|
-
obs.push(of([FileEntry.ofFile(itemAsFile)]));
|
|
27
|
-
}
|
|
28
|
-
else {
|
|
29
|
-
if (transferItem.kind == 'file') {
|
|
30
|
-
obs.push(of([
|
|
31
|
-
FileEntry.ofFile(new File([], '', {
|
|
32
|
-
type: transferItem.type
|
|
33
|
-
}))
|
|
34
|
-
]));
|
|
35
|
-
}
|
|
36
|
-
else {
|
|
37
|
-
this.log.warn('Could not handle DataTransferItem!', transferItem);
|
|
38
|
-
}
|
|
39
|
-
}
|
|
40
|
-
}
|
|
22
|
+
obs.push(this.resolveTransferItem(transferItem));
|
|
41
23
|
}
|
|
42
24
|
return zip(obs).pipe(map(files => files.flat()));
|
|
43
25
|
}
|
|
44
|
-
listFilesRecursive(entry) {
|
|
45
|
-
return this.listFilesRecursiveAsync(entry, null);
|
|
46
|
-
}
|
|
47
26
|
/***************************************************************************
|
|
48
27
|
* *
|
|
49
28
|
* Private methods *
|
|
50
29
|
* *
|
|
51
30
|
**************************************************************************/
|
|
52
|
-
|
|
53
|
-
if (
|
|
54
|
-
|
|
55
|
-
return this.readEntries(directoryEntry).pipe(mergeMap(entries => zip(entries.map(entry => this.listFilesRecursiveAsync(entry, directoryEntry)))), map(files => files.flat()));
|
|
56
|
-
}
|
|
57
|
-
else if (entry.isFile) {
|
|
58
|
-
const fileEntry = entry;
|
|
59
|
-
return this.observableFile(fileEntry, parent).pipe(map(file => [file]));
|
|
60
|
-
}
|
|
61
|
-
else {
|
|
62
|
-
return EMPTY;
|
|
31
|
+
resolveTransferItem(transferItem) {
|
|
32
|
+
if (FileSystemApi.isGetAsFileSystemHandleSupported(transferItem)) {
|
|
33
|
+
return FileSystemApi.getAsFileSystemHandle(transferItem).pipe(switchMap(handle => handle ? FileListingSystemHandle.listFiles(handle) : this.legacyFileListing(transferItem)));
|
|
63
34
|
}
|
|
35
|
+
return this.legacyFileListing(transferItem);
|
|
64
36
|
}
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
observer.complete();
|
|
70
|
-
}, err => {
|
|
71
|
-
observer.error(err);
|
|
72
|
-
observer.complete();
|
|
73
|
-
});
|
|
74
|
-
});
|
|
75
|
-
}
|
|
76
|
-
trimStaringSlash(path) {
|
|
77
|
-
if (path) {
|
|
78
|
-
if (path.startsWith('/')) {
|
|
79
|
-
path = path.substring(1);
|
|
80
|
-
}
|
|
37
|
+
legacyFileListing(transferItem) {
|
|
38
|
+
const entry = transferItem.webkitGetAsEntry();
|
|
39
|
+
if (entry) {
|
|
40
|
+
return FileListingWebkit.listFilesRecursive(entry);
|
|
81
41
|
}
|
|
82
|
-
return
|
|
83
|
-
}
|
|
84
|
-
readEntries(directoryEntry) {
|
|
85
|
-
return new Observable(observer => {
|
|
86
|
-
this.consumeReaderToCompletion(observer, directoryEntry.createReader());
|
|
87
|
-
});
|
|
42
|
+
return this.fallbackFileListing(transferItem);
|
|
88
43
|
}
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
44
|
+
fallbackFileListing(transferItem) {
|
|
45
|
+
const itemAsFile = transferItem.getAsFile();
|
|
46
|
+
if (itemAsFile) {
|
|
47
|
+
return of([FileEntry.ofFile(itemAsFile)]);
|
|
48
|
+
}
|
|
49
|
+
else {
|
|
50
|
+
if (transferItem.kind == 'file') {
|
|
51
|
+
return of([
|
|
52
|
+
FileEntry.ofFile(new File([], '', {
|
|
53
|
+
type: transferItem.type
|
|
54
|
+
}))
|
|
55
|
+
]);
|
|
94
56
|
}
|
|
95
57
|
else {
|
|
96
|
-
|
|
58
|
+
this.log.warn('Could not handle DataTransferItem!', transferItem);
|
|
59
|
+
return of([]);
|
|
97
60
|
}
|
|
98
|
-
}
|
|
99
|
-
observer.error(err);
|
|
100
|
-
observer.complete();
|
|
101
|
-
});
|
|
61
|
+
}
|
|
102
62
|
}
|
|
103
63
|
}
|
|
104
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-listing-rx.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/files/listing/file-listing-rx.ts"],"names":[],"mappings":"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","sourcesContent":["import {EMPTY, mergeMap, Observable, of, Subscriber, zip} from 'rxjs';\nimport {FileEntry} from './file-entry';\nimport {map} from 'rxjs/operators';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\n\nexport class FileListingRx {\n\n  public static readonly INSTANCE = new FileListingRx();\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public toFileList(transferItemList: DataTransferItemList): Observable<FileEntry[]> {\n    const obs: Observable<FileEntry[]>[] = [];\n    for (let i = 0; i < transferItemList.length; i++) {\n      const transferItem = transferItemList[i];\n      const entry = transferItem.webkitGetAsEntry();\n      if (entry) {\n        obs.push(this.listFilesRecursive(entry));\n      } else {\n        const itemAsFile = transferItem.getAsFile();\n        if (itemAsFile) {\n          obs.push(of([FileEntry.ofFile(itemAsFile)]));\n        } else {\n          if (transferItem.kind == 'file') {\n            obs.push(of([\n              FileEntry.ofFile(\n                new File([], '', {\n                  type: transferItem.type\n                })\n              )\n            ]));\n          } else {\n            this.log.warn('Could not handle DataTransferItem!', transferItem);\n          }\n        }\n      }\n    }\n    return zip(obs).pipe(\n      map(files => files.flat()),\n    );\n  }\n\n  public listFilesRecursive(\n    entry: FileSystemEntry\n  ): Observable<FileEntry[]> {\n    return this.listFilesRecursiveAsync(entry, null);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private listFilesRecursiveAsync(\n    entry: FileSystemEntry,\n    parent: FileSystemDirectoryEntry\n  ): Observable<FileEntry[]> {\n    if (entry.isDirectory) {\n      const directoryEntry = entry as FileSystemDirectoryEntry;\n      return this.readEntries(directoryEntry).pipe(\n        mergeMap(entries => zip(\n          entries.map(entry => this.listFilesRecursiveAsync(entry, directoryEntry))\n        )),\n        map(files => files.flat()),\n      );\n    } else if (entry.isFile) {\n      const fileEntry = entry as FileSystemFileEntry;\n      return this.observableFile(fileEntry, parent).pipe(\n        map(file => [file])\n      );\n    } else {\n      return EMPTY;\n    }\n  }\n\n  private observableFile(\n    fileEntry: FileSystemFileEntry,\n    parent: FileSystemDirectoryEntry | null\n  ): Observable<FileEntry> {\n    return new Observable<FileEntry>(\n      observer => {\n        fileEntry.file(file => {\n          observer.next(FileEntry.relativeFile(file, this.trimStaringSlash(parent?.fullPath)));\n          observer.complete();\n        }, err => {\n          observer.error(err);\n          observer.complete();\n        });\n      }\n    );\n  }\n\n  private trimStaringSlash(path: string | null): string {\n    if (path) {\n      if (path.startsWith('/')) {\n        path = path.substring(1);\n      }\n    }\n    return path;\n  }\n\n  private readEntries(\n    directoryEntry: FileSystemDirectoryEntry\n  ): Observable<FileSystemEntry[]> {\n    return new Observable<FileSystemEntry[]>(\n      observer => {\n        this.consumeReaderToCompletion(observer, directoryEntry.createReader());\n      }\n    );\n  }\n\n  private consumeReaderToCompletion(\n    observer: Subscriber<FileSystemEntry[]>,\n    directoryReader: FileSystemDirectoryReader\n  ): void {\n    directoryReader.readEntries(entries => {\n      if (entries && entries.length > 0) {\n        observer.next(entries);\n        this.consumeReaderToCompletion(observer, directoryReader);\n      } else {\n        observer.complete();\n      }\n    }, err => {\n      observer.error(err);\n      observer.complete();\n    });\n  }\n\n}\n"]}
|
|
64
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmlsZS1saXN0aW5nLXJ4LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvZWxkZXJieXRlL25neC1zdGFydGVyL3NyYy9saWIvY29tcG9uZW50cy9maWxlcy9saXN0aW5nL2ZpbGUtbGlzdGluZy1yeC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQWEsRUFBRSxFQUFFLFNBQVMsRUFBRSxHQUFHLEVBQUMsTUFBTSxNQUFNLENBQUM7QUFDcEQsT0FBTyxFQUFDLFNBQVMsRUFBQyxNQUFNLGNBQWMsQ0FBQztBQUN2QyxPQUFPLEVBQUMsR0FBRyxFQUFDLE1BQU0sZ0JBQWdCLENBQUM7QUFDbkMsT0FBTyxFQUFDLGFBQWEsRUFBQyxNQUFNLHNCQUFzQixDQUFDO0FBQ25ELE9BQU8sRUFBQyxpQkFBaUIsRUFBQyxNQUFNLHVCQUF1QixDQUFDO0FBQ3hELE9BQU8sRUFBQyx1QkFBdUIsRUFBQyxNQUFNLDhCQUE4QixDQUFDO0FBQ3JFLE9BQU8sRUFBQyxhQUFhLEVBQUMsTUFBTSxvQkFBb0IsQ0FBQztBQUVqRCxNQUFNLE9BQU8sYUFBYTtJQUExQjtRQUltQixRQUFHLEdBQUcsYUFBYSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxDQUFDO0lBK0R4RSxDQUFDO2FBakV3QixhQUFRLEdBQUcsSUFBSSxhQUFhLEVBQUUsQUFBdEIsQ0FBdUI7SUFJdEQ7Ozs7Z0ZBSTRFO0lBRXJFLFVBQVUsQ0FBQyxnQkFBc0M7UUFDdEQsTUFBTSxHQUFHLEdBQThCLEVBQUUsQ0FBQztRQUMxQyxLQUFLLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsZ0JBQWdCLENBQUMsTUFBTSxFQUFFLENBQUMsRUFBRSxFQUFFLENBQUM7WUFDakQsTUFBTSxZQUFZLEdBQUcsZ0JBQWdCLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDekMsR0FBRyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsbUJBQW1CLENBQUMsWUFBWSxDQUFDLENBQUMsQ0FBQztRQUNuRCxDQUFDO1FBQ0QsT0FBTyxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUMsSUFBSSxDQUNsQixHQUFHLENBQUMsS0FBSyxDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FDM0IsQ0FBQztJQUNKLENBQUM7SUFFRDs7OztnRkFJNEU7SUFFcEUsbUJBQW1CLENBQUMsWUFBOEI7UUFFeEQsSUFBSSxhQUFhLENBQUMsZ0NBQWdDLENBQUMsWUFBWSxDQUFDLEVBQUUsQ0FBQztZQUNqRSxPQUFPLGFBQWEsQ0FBQyxxQkFBcUIsQ0FBQyxZQUFZLENBQUMsQ0FBQyxJQUFJLENBQzNELFNBQVMsQ0FBQyxNQUFNLENBQUMsRUFBRSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsdUJBQXVCLENBQUMsU0FBUyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsaUJBQWlCLENBQUMsWUFBWSxDQUFDLENBQUMsQ0FDL0csQ0FBQztRQUNKLENBQUM7UUFDRCxPQUFPLElBQUksQ0FBQyxpQkFBaUIsQ0FBQyxZQUFZLENBQUMsQ0FBQztJQUM5QyxDQUFDO0lBR08saUJBQWlCLENBQUMsWUFBOEI7UUFDdEQsTUFBTSxLQUFLLEdBQUcsWUFBWSxDQUFDLGdCQUFnQixFQUFFLENBQUM7UUFDOUMsSUFBSSxLQUFLLEVBQUUsQ0FBQztZQUNWLE9BQU8saUJBQWlCLENBQUMsa0JBQWtCLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDckQsQ0FBQztRQUNELE9BQU8sSUFBSSxDQUFDLG1CQUFtQixDQUFDLFlBQVksQ0FBQyxDQUFDO0lBQ2hELENBQUM7SUFFTyxtQkFBbUIsQ0FBQyxZQUE4QjtRQUN4RCxNQUFNLFVBQVUsR0FBRyxZQUFZLENBQUMsU0FBUyxFQUFFLENBQUM7UUFDNUMsSUFBSSxVQUFVLEVBQUUsQ0FBQztZQUNmLE9BQU8sRUFBRSxDQUFDLENBQUMsU0FBUyxDQUFDLE1BQU0sQ0FBQyxVQUFVLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDNUMsQ0FBQzthQUFNLENBQUM7WUFDTixJQUFJLFlBQVksQ0FBQyxJQUFJLElBQUksTUFBTSxFQUFFLENBQUM7Z0JBQ2hDLE9BQU8sRUFBRSxDQUFDO29CQUNSLFNBQVMsQ0FBQyxNQUFNLENBQ2QsSUFBSSxJQUFJLENBQUMsRUFBRSxFQUFFLEVBQUUsRUFBRTt3QkFDZixJQUFJLEVBQUUsWUFBWSxDQUFDLElBQUk7cUJBQ3hCLENBQUMsQ0FDSDtpQkFDRixDQUFDLENBQUM7WUFDTCxDQUFDO2lCQUFNLENBQUM7Z0JBQ04sSUFBSSxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsb0NBQW9DLEVBQUUsWUFBWSxDQUFDLENBQUM7Z0JBQ2xFLE9BQU8sRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDO1lBQ2hCLENBQUM7UUFDSCxDQUFDO0lBQ0gsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7T2JzZXJ2YWJsZSwgb2YsIHN3aXRjaE1hcCwgemlwfSBmcm9tICdyeGpzJztcbmltcG9ydCB7RmlsZUVudHJ5fSBmcm9tICcuL2ZpbGUtZW50cnknO1xuaW1wb3J0IHttYXB9IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcbmltcG9ydCB7TG9nZ2VyRmFjdG9yeX0gZnJvbSAnQGVsZGVyYnl0ZS90cy1sb2dnZXInO1xuaW1wb3J0IHtGaWxlTGlzdGluZ1dlYmtpdH0gZnJvbSAnLi9maWxlLWxpc3Rpbmctd2Via2l0JztcbmltcG9ydCB7RmlsZUxpc3RpbmdTeXN0ZW1IYW5kbGV9IGZyb20gJy4vZmlsZS1saXN0aW5nLXN5c3RlbS1oYW5kbGUnO1xuaW1wb3J0IHtGaWxlU3lzdGVtQXBpfSBmcm9tICcuLi9maWxlLXN5c3RlbS1hcGknO1xuXG5leHBvcnQgY2xhc3MgRmlsZUxpc3RpbmdSeCB7XG5cbiAgcHVibGljIHN0YXRpYyByZWFkb25seSBJTlNUQU5DRSA9IG5ldyBGaWxlTGlzdGluZ1J4KCk7XG5cbiAgcHJpdmF0ZSByZWFkb25seSBsb2cgPSBMb2dnZXJGYWN0b3J5LmdldExvZ2dlcih0aGlzLmNvbnN0cnVjdG9yLm5hbWUpO1xuXG4gIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKipcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiBQdWJsaWMgQVBJICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqL1xuXG4gIHB1YmxpYyB0b0ZpbGVMaXN0KHRyYW5zZmVySXRlbUxpc3Q6IERhdGFUcmFuc2Zlckl0ZW1MaXN0KTogT2JzZXJ2YWJsZTxGaWxlRW50cnlbXT4ge1xuICAgIGNvbnN0IG9iczogT2JzZXJ2YWJsZTxGaWxlRW50cnlbXT5bXSA9IFtdO1xuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgdHJhbnNmZXJJdGVtTGlzdC5sZW5ndGg7IGkrKykge1xuICAgICAgY29uc3QgdHJhbnNmZXJJdGVtID0gdHJhbnNmZXJJdGVtTGlzdFtpXTtcbiAgICAgIG9icy5wdXNoKHRoaXMucmVzb2x2ZVRyYW5zZmVySXRlbSh0cmFuc2Zlckl0ZW0pKTtcbiAgICB9XG4gICAgcmV0dXJuIHppcChvYnMpLnBpcGUoXG4gICAgICBtYXAoZmlsZXMgPT4gZmlsZXMuZmxhdCgpKVxuICAgICk7XG4gIH1cblxuICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogUHJpdmF0ZSBtZXRob2RzICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi9cblxuICBwcml2YXRlIHJlc29sdmVUcmFuc2Zlckl0ZW0odHJhbnNmZXJJdGVtOiBEYXRhVHJhbnNmZXJJdGVtKTogT2JzZXJ2YWJsZTxGaWxlRW50cnlbXT4ge1xuXG4gICAgaWYgKEZpbGVTeXN0ZW1BcGkuaXNHZXRBc0ZpbGVTeXN0ZW1IYW5kbGVTdXBwb3J0ZWQodHJhbnNmZXJJdGVtKSkge1xuICAgICAgcmV0dXJuIEZpbGVTeXN0ZW1BcGkuZ2V0QXNGaWxlU3lzdGVtSGFuZGxlKHRyYW5zZmVySXRlbSkucGlwZShcbiAgICAgICAgc3dpdGNoTWFwKGhhbmRsZSA9PiBoYW5kbGUgPyBGaWxlTGlzdGluZ1N5c3RlbUhhbmRsZS5saXN0RmlsZXMoaGFuZGxlKSA6IHRoaXMubGVnYWN5RmlsZUxpc3RpbmcodHJhbnNmZXJJdGVtKSlcbiAgICAgICk7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLmxlZ2FjeUZpbGVMaXN0aW5nKHRyYW5zZmVySXRlbSk7XG4gIH1cblxuXG4gIHByaXZhdGUgbGVnYWN5RmlsZUxpc3RpbmcodHJhbnNmZXJJdGVtOiBEYXRhVHJhbnNmZXJJdGVtKTogT2JzZXJ2YWJsZTxGaWxlRW50cnlbXT4ge1xuICAgIGNvbnN0IGVudHJ5ID0gdHJhbnNmZXJJdGVtLndlYmtpdEdldEFzRW50cnkoKTtcbiAgICBpZiAoZW50cnkpIHtcbiAgICAgIHJldHVybiBGaWxlTGlzdGluZ1dlYmtpdC5saXN0RmlsZXNSZWN1cnNpdmUoZW50cnkpO1xuICAgIH1cbiAgICByZXR1cm4gdGhpcy5mYWxsYmFja0ZpbGVMaXN0aW5nKHRyYW5zZmVySXRlbSk7XG4gIH1cblxuICBwcml2YXRlIGZhbGxiYWNrRmlsZUxpc3RpbmcodHJhbnNmZXJJdGVtOiBEYXRhVHJhbnNmZXJJdGVtKTogT2JzZXJ2YWJsZTxGaWxlRW50cnlbXT4ge1xuICAgIGNvbnN0IGl0ZW1Bc0ZpbGUgPSB0cmFuc2Zlckl0ZW0uZ2V0QXNGaWxlKCk7XG4gICAgaWYgKGl0ZW1Bc0ZpbGUpIHtcbiAgICAgIHJldHVybiBvZihbRmlsZUVudHJ5Lm9mRmlsZShpdGVtQXNGaWxlKV0pO1xuICAgIH0gZWxzZSB7XG4gICAgICBpZiAodHJhbnNmZXJJdGVtLmtpbmQgPT0gJ2ZpbGUnKSB7XG4gICAgICAgIHJldHVybiBvZihbXG4gICAgICAgICAgRmlsZUVudHJ5Lm9mRmlsZShcbiAgICAgICAgICAgIG5ldyBGaWxlKFtdLCAnJywge1xuICAgICAgICAgICAgICB0eXBlOiB0cmFuc2Zlckl0ZW0udHlwZVxuICAgICAgICAgICAgfSlcbiAgICAgICAgICApXG4gICAgICAgIF0pO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgdGhpcy5sb2cud2FybignQ291bGQgbm90IGhhbmRsZSBEYXRhVHJhbnNmZXJJdGVtIScsIHRyYW5zZmVySXRlbSk7XG4gICAgICAgIHJldHVybiBvZihbXSk7XG4gICAgICB9XG4gICAgfVxuICB9XG59XG4iXX0=
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import { switchMap, toArray, zip } from 'rxjs';
|
|
2
|
+
import { FileEntry } from './file-entry';
|
|
3
|
+
import { FileSystemApi } from '../file-system-api';
|
|
4
|
+
import { map } from 'rxjs/operators';
|
|
5
|
+
export class FileListingSystemHandle {
|
|
6
|
+
/***************************************************************************
|
|
7
|
+
* *
|
|
8
|
+
* Public methods *
|
|
9
|
+
* *
|
|
10
|
+
**************************************************************************/
|
|
11
|
+
/**
|
|
12
|
+
* Creates a File Entry Array Observable by resolving its FileSystemHandle.
|
|
13
|
+
* @returns An `Observable<FileEntry[]>` with the FileSystemFileHandle included
|
|
14
|
+
*/
|
|
15
|
+
static listFiles(handle, rootDirectory) {
|
|
16
|
+
switch (handle.kind) {
|
|
17
|
+
case 'file':
|
|
18
|
+
return FileEntry.ofFileHandle(handle, rootDirectory).pipe(toArray());
|
|
19
|
+
case 'directory':
|
|
20
|
+
return FileListingSystemHandle.listFilesRecursive(handle, rootDirectory);
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
/***************************************************************************
|
|
24
|
+
* *
|
|
25
|
+
* Private methods *
|
|
26
|
+
* *
|
|
27
|
+
**************************************************************************/
|
|
28
|
+
static listFilesRecursive(dirHandle, rootDirectory) {
|
|
29
|
+
return FileSystemApi.getHandlesFromDirectory(dirHandle).pipe(map(handles => FileListingSystemHandle.listFilesOfEach(handles, rootDirectory ? rootDirectory : dirHandle)), switchMap(fileEntriesArray$ => zip(fileEntriesArray$)), map(fileEntriesArray => fileEntriesArray.flat()));
|
|
30
|
+
}
|
|
31
|
+
static listFilesOfEach(handles, rootDirectory) {
|
|
32
|
+
return handles.map(fileOrDirHandle => FileListingSystemHandle.listFiles(fileOrDirHandle, rootDirectory));
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmlsZS1saXN0aW5nLXN5c3RlbS1oYW5kbGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9lbGRlcmJ5dGUvbmd4LXN0YXJ0ZXIvc3JjL2xpYi9jb21wb25lbnRzL2ZpbGVzL2xpc3RpbmcvZmlsZS1saXN0aW5nLXN5c3RlbS1oYW5kbGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFhLFNBQVMsRUFBRSxPQUFPLEVBQUUsR0FBRyxFQUFDLE1BQU0sTUFBTSxDQUFDO0FBQ3pELE9BQU8sRUFBQyxTQUFTLEVBQUMsTUFBTSxjQUFjLENBQUM7QUFDdkMsT0FBTyxFQUFDLGFBQWEsRUFBQyxNQUFNLG9CQUFvQixDQUFDO0FBQ2pELE9BQU8sRUFBQyxHQUFHLEVBQUMsTUFBTSxnQkFBZ0IsQ0FBQztBQUVuQyxNQUFNLE9BQU8sdUJBQXVCO0lBRWxDOzs7O2dGQUk0RTtJQUU1RTs7O09BR0c7SUFDSSxNQUFNLENBQUMsU0FBUyxDQUFDLE1BQXdCLEVBQUUsYUFBeUM7UUFDekYsUUFBUSxNQUFNLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDcEIsS0FBSyxNQUFNO2dCQUNULE9BQU8sU0FBUyxDQUFDLFlBQVksQ0FBQyxNQUE4QixFQUFFLGFBQWEsQ0FBQyxDQUFDLElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQyxDQUFDO1lBQy9GLEtBQUssV0FBVztnQkFDZCxPQUFPLHVCQUF1QixDQUFDLGtCQUFrQixDQUFDLE1BQW1DLEVBQUUsYUFBYSxDQUFDLENBQUM7UUFDMUcsQ0FBQztJQUNILENBQUM7SUFFRDs7OztnRkFJNEU7SUFHcEUsTUFBTSxDQUFDLGtCQUFrQixDQUFDLFNBQW9DLEVBQUUsYUFBeUM7UUFDL0csT0FBTyxhQUFhLENBQUMsdUJBQXVCLENBQUMsU0FBUyxDQUFDLENBQUMsSUFBSSxDQUMxRCxHQUFHLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQyx1QkFBdUIsQ0FBQyxlQUFlLENBQUMsT0FBTyxFQUFFLGFBQWEsQ0FBQyxDQUFDLENBQUMsYUFBYSxDQUFDLENBQUMsQ0FBQyxTQUFTLENBQUMsQ0FBQyxFQUMzRyxTQUFTLENBQUMsaUJBQWlCLENBQUMsRUFBRSxDQUFDLEdBQUcsQ0FBQyxpQkFBaUIsQ0FBQyxDQUFDLEVBQ3RELEdBQUcsQ0FBQyxnQkFBZ0IsQ0FBQyxFQUFFLENBQUMsZ0JBQWdCLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FDakQsQ0FBQztJQUNKLENBQUM7SUFFTyxNQUFNLENBQUMsZUFBZSxDQUM1QixPQUEyQixFQUMzQixhQUF5QztRQUV6QyxPQUFPLE9BQU8sQ0FBQyxHQUFHLENBQUMsZUFBZSxDQUFDLEVBQUUsQ0FBQyx1QkFBdUIsQ0FBQyxTQUFTLENBQUMsZUFBZSxFQUFFLGFBQWEsQ0FBQyxDQUFDLENBQUE7SUFDMUcsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtPYnNlcnZhYmxlLCBzd2l0Y2hNYXAsIHRvQXJyYXksIHppcH0gZnJvbSAncnhqcyc7XG5pbXBvcnQge0ZpbGVFbnRyeX0gZnJvbSAnLi9maWxlLWVudHJ5JztcbmltcG9ydCB7RmlsZVN5c3RlbUFwaX0gZnJvbSAnLi4vZmlsZS1zeXN0ZW0tYXBpJztcbmltcG9ydCB7bWFwfSBmcm9tICdyeGpzL29wZXJhdG9ycyc7XG5cbmV4cG9ydCBjbGFzcyBGaWxlTGlzdGluZ1N5c3RlbUhhbmRsZSB7XG5cbiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqIFB1YmxpYyBtZXRob2RzICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgLyoqXG4gICAqIENyZWF0ZXMgYSBGaWxlIEVudHJ5IEFycmF5IE9ic2VydmFibGUgYnkgcmVzb2x2aW5nIGl0cyBGaWxlU3lzdGVtSGFuZGxlLlxuICAgKiBAcmV0dXJucyBBbiBgT2JzZXJ2YWJsZTxGaWxlRW50cnlbXT5gIHdpdGggdGhlIEZpbGVTeXN0ZW1GaWxlSGFuZGxlIGluY2x1ZGVkXG4gICAqL1xuICBwdWJsaWMgc3RhdGljIGxpc3RGaWxlcyhoYW5kbGU6IEZpbGVTeXN0ZW1IYW5kbGUsIHJvb3REaXJlY3Rvcnk/OiBGaWxlU3lzdGVtRGlyZWN0b3J5SGFuZGxlKTogT2JzZXJ2YWJsZTxGaWxlRW50cnlbXT4ge1xuICAgIHN3aXRjaCAoaGFuZGxlLmtpbmQpIHtcbiAgICAgIGNhc2UgJ2ZpbGUnOlxuICAgICAgICByZXR1cm4gRmlsZUVudHJ5Lm9mRmlsZUhhbmRsZShoYW5kbGUgYXMgRmlsZVN5c3RlbUZpbGVIYW5kbGUsIHJvb3REaXJlY3RvcnkpLnBpcGUodG9BcnJheSgpKTtcbiAgICAgIGNhc2UgJ2RpcmVjdG9yeSc6XG4gICAgICAgIHJldHVybiBGaWxlTGlzdGluZ1N5c3RlbUhhbmRsZS5saXN0RmlsZXNSZWN1cnNpdmUoaGFuZGxlIGFzIEZpbGVTeXN0ZW1EaXJlY3RvcnlIYW5kbGUsIHJvb3REaXJlY3RvcnkpO1xuICAgIH1cbiAgfVxuXG4gIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKipcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiBQcml2YXRlIG1ldGhvZHMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqL1xuXG5cbiAgcHJpdmF0ZSBzdGF0aWMgbGlzdEZpbGVzUmVjdXJzaXZlKGRpckhhbmRsZTogRmlsZVN5c3RlbURpcmVjdG9yeUhhbmRsZSwgcm9vdERpcmVjdG9yeT86IEZpbGVTeXN0ZW1EaXJlY3RvcnlIYW5kbGUpOiBPYnNlcnZhYmxlPEZpbGVFbnRyeVtdPiB7XG4gICAgcmV0dXJuIEZpbGVTeXN0ZW1BcGkuZ2V0SGFuZGxlc0Zyb21EaXJlY3RvcnkoZGlySGFuZGxlKS5waXBlKFxuICAgICAgbWFwKGhhbmRsZXMgPT4gRmlsZUxpc3RpbmdTeXN0ZW1IYW5kbGUubGlzdEZpbGVzT2ZFYWNoKGhhbmRsZXMsIHJvb3REaXJlY3RvcnkgPyByb290RGlyZWN0b3J5IDogZGlySGFuZGxlKSksXG4gICAgICBzd2l0Y2hNYXAoZmlsZUVudHJpZXNBcnJheSQgPT4gemlwKGZpbGVFbnRyaWVzQXJyYXkkKSksXG4gICAgICBtYXAoZmlsZUVudHJpZXNBcnJheSA9PiBmaWxlRW50cmllc0FycmF5LmZsYXQoKSlcbiAgICApO1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgbGlzdEZpbGVzT2ZFYWNoKFxuICAgIGhhbmRsZXM6IEZpbGVTeXN0ZW1IYW5kbGVbXSxcbiAgICByb290RGlyZWN0b3J5PzogRmlsZVN5c3RlbURpcmVjdG9yeUhhbmRsZVxuICApOiBPYnNlcnZhYmxlPEZpbGVFbnRyeVtdPltdIHtcbiAgICByZXR1cm4gaGFuZGxlcy5tYXAoZmlsZU9yRGlySGFuZGxlID0+IEZpbGVMaXN0aW5nU3lzdGVtSGFuZGxlLmxpc3RGaWxlcyhmaWxlT3JEaXJIYW5kbGUsIHJvb3REaXJlY3RvcnkpKVxuICB9XG59XG4iXX0=
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
import { EMPTY, mergeMap, Observable, zip } from 'rxjs';
|
|
2
|
+
import { FileEntry } from './file-entry';
|
|
3
|
+
import { map } from 'rxjs/operators';
|
|
4
|
+
export class FileListingWebkit {
|
|
5
|
+
/***************************************************************************
|
|
6
|
+
* *
|
|
7
|
+
* Constructor *
|
|
8
|
+
* *
|
|
9
|
+
**************************************************************************/
|
|
10
|
+
constructor() {
|
|
11
|
+
}
|
|
12
|
+
/***************************************************************************
|
|
13
|
+
* *
|
|
14
|
+
* Public methods *
|
|
15
|
+
* *
|
|
16
|
+
**************************************************************************/
|
|
17
|
+
static listFilesRecursive(entry) {
|
|
18
|
+
return FileListingWebkit.listFilesRecursiveAsync(entry, null);
|
|
19
|
+
}
|
|
20
|
+
/***************************************************************************
|
|
21
|
+
* *
|
|
22
|
+
* Private methods *
|
|
23
|
+
* *
|
|
24
|
+
**************************************************************************/
|
|
25
|
+
static listFilesRecursiveAsync(entry, parent) {
|
|
26
|
+
if (entry.isDirectory) {
|
|
27
|
+
const directoryEntry = entry;
|
|
28
|
+
return FileListingWebkit.readEntries(directoryEntry).pipe(mergeMap(entries => zip(entries.map(entry => this.listFilesRecursiveAsync(entry, directoryEntry)))), map(files => files.flat()));
|
|
29
|
+
}
|
|
30
|
+
else if (entry.isFile) {
|
|
31
|
+
const fileEntry = entry;
|
|
32
|
+
return FileListingWebkit.observableFile(fileEntry, parent).pipe(map(file => [file]));
|
|
33
|
+
}
|
|
34
|
+
else {
|
|
35
|
+
return EMPTY;
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
static observableFile(fileEntry, parent) {
|
|
39
|
+
return new Observable(observer => {
|
|
40
|
+
fileEntry.file(file => {
|
|
41
|
+
observer.next(FileEntry.relativeFile(file, FileListingWebkit.trimStaringSlash(parent?.fullPath)));
|
|
42
|
+
observer.complete();
|
|
43
|
+
}, err => {
|
|
44
|
+
observer.error(err);
|
|
45
|
+
observer.complete();
|
|
46
|
+
});
|
|
47
|
+
});
|
|
48
|
+
}
|
|
49
|
+
static trimStaringSlash(path) {
|
|
50
|
+
if (path) {
|
|
51
|
+
if (path.startsWith('/')) {
|
|
52
|
+
path = path.substring(1);
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
return path;
|
|
56
|
+
}
|
|
57
|
+
static readEntries(directoryEntry) {
|
|
58
|
+
return new Observable(observer => {
|
|
59
|
+
FileListingWebkit.consumeReaderToCompletion(observer, directoryEntry.createReader());
|
|
60
|
+
});
|
|
61
|
+
}
|
|
62
|
+
static consumeReaderToCompletion(observer, directoryReader) {
|
|
63
|
+
directoryReader.readEntries(entries => {
|
|
64
|
+
if (entries && entries.length > 0) {
|
|
65
|
+
observer.next(entries);
|
|
66
|
+
this.consumeReaderToCompletion(observer, directoryReader);
|
|
67
|
+
}
|
|
68
|
+
else {
|
|
69
|
+
observer.complete();
|
|
70
|
+
}
|
|
71
|
+
}, err => {
|
|
72
|
+
observer.error(err);
|
|
73
|
+
observer.complete();
|
|
74
|
+
});
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file-listing-webkit.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/files/listing/file-listing-webkit.ts"],"names":[],"mappings":"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","sourcesContent":["import {EMPTY, mergeMap, Observable, Subscriber, zip} from 'rxjs';\nimport {FileEntry} from './file-entry';\nimport {map} from 'rxjs/operators';\n\nexport class FileListingWebkit {\n\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor() {\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public methods                                                          *\n   *                                                                         *\n   **************************************************************************/\n\n\n  public static listFilesRecursive(\n    entry: FileSystemEntry\n  ): Observable<FileEntry[]> {\n    return FileListingWebkit.listFilesRecursiveAsync(entry, null);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private static listFilesRecursiveAsync(\n    entry: FileSystemEntry,\n    parent: FileSystemDirectoryEntry\n  ): Observable<FileEntry[]> {\n    if (entry.isDirectory) {\n      const directoryEntry = entry as FileSystemDirectoryEntry;\n      return FileListingWebkit.readEntries(directoryEntry).pipe(\n        mergeMap(entries => zip(\n          entries.map(entry => this.listFilesRecursiveAsync(entry, directoryEntry))\n        )),\n        map(files => files.flat())\n      );\n    } else if (entry.isFile) {\n      const fileEntry = entry as FileSystemFileEntry;\n      return FileListingWebkit.observableFile(fileEntry, parent).pipe(\n        map(file => [file])\n      );\n    } else {\n      return EMPTY;\n    }\n  }\n\n  private static observableFile(\n    fileEntry: FileSystemFileEntry,\n    parent: FileSystemDirectoryEntry | null\n  ): Observable<FileEntry> {\n    return new Observable<FileEntry>(\n      observer => {\n        fileEntry.file(file => {\n          observer.next(FileEntry.relativeFile(file, FileListingWebkit.trimStaringSlash(parent?.fullPath)));\n          observer.complete();\n        }, err => {\n          observer.error(err);\n          observer.complete();\n        });\n      }\n    );\n  }\n\n  private static trimStaringSlash(path: string | null): string {\n    if (path) {\n      if (path.startsWith('/')) {\n        path = path.substring(1);\n      }\n    }\n    return path;\n  }\n\n  private static readEntries(\n    directoryEntry: FileSystemDirectoryEntry\n  ): Observable<FileSystemEntry[]> {\n    return new Observable<FileSystemEntry[]>(\n      observer => {\n        FileListingWebkit.consumeReaderToCompletion(observer, directoryEntry.createReader());\n      }\n    );\n  }\n\n  private static consumeReaderToCompletion(\n    observer: Subscriber<FileSystemEntry[]>,\n    directoryReader: FileSystemDirectoryReader\n  ): void {\n    directoryReader.readEntries(entries => {\n      if (entries && entries.length > 0) {\n        observer.next(entries);\n        this.consumeReaderToCompletion(observer, directoryReader);\n      } else {\n        observer.complete();\n      }\n    }, err => {\n      observer.error(err);\n      observer.complete();\n    });\n  }\n\n}\n"]}
|