@elderbyte/ngx-starter 15.10.1 → 15.10.2

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.
@@ -25,7 +25,7 @@ export class ElderDropZoneComponent {
25
25
  }
26
26
  }
27
27
  ElderDropZoneComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.7", ngImport: i0, type: ElderDropZoneComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
28
- ElderDropZoneComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.7", type: ElderDropZoneComponent, selector: "elder-drop-zone", inputs: { icon: "icon", prompt: "prompt" }, outputs: { filesDropped: "filesDropped", dragOver: "dragOver", dragEnd: "dragEnd" }, ngImport: i0, template: "<div class=\"full p-lg\"\n>\n <div class=\"layout-col full elder-drop-zone\"\n elderFileDropZone\n (dropFiles)=\"filesDropped.emit($event)\"\n (dragOver)=\"dragOver.emit($event)\"\n (dragEnd)=\"dragEnd.emit($event)\"\n >\n <div class=\"layout-col flex\">\n <ng-content></ng-content>\n <!-- Default content -->\n <div class=\"layout-col flex place-center-center gap-sm elder-default-content\">\n <mat-icon class=\"elder-drop-zone-icon\">{{icon}}</mat-icon>\n <span class=\"elder-drop-zone-prompt\">{{prompt | translate}}</span>\n </div>\n </div>\n </div>\n</div>\n\n\n", styles: [""], dependencies: [{ kind: "component", type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: i2.ElderFileDropZoneDirective, selector: "[elderFileDropZone]", inputs: ["dragOverClass"], outputs: ["dragOver", "dragEnd", "dragOverFiles", "dropFiles", "elderFileDropZoneChange"] }, { kind: "pipe", type: i3.TranslatePipe, name: "translate" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
28
+ ElderDropZoneComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.7", type: ElderDropZoneComponent, selector: "elder-drop-zone", inputs: { icon: "icon", prompt: "prompt" }, outputs: { filesDropped: "filesDropped", dragOver: "dragOver", dragEnd: "dragEnd" }, ngImport: i0, template: "<div class=\"full p-lg\"\n>\n <div class=\"layout-col full elder-drop-zone\"\n elderFileDropZone\n (dropFiles)=\"filesDropped.emit($event)\"\n (dragOver)=\"dragOver.emit($event)\"\n (dragEnd)=\"dragEnd.emit($event)\"\n >\n <div class=\"layout-col flex\">\n <ng-content></ng-content>\n <!-- Default content -->\n <div class=\"layout-col flex place-center-center gap-sm elder-default-content\">\n <mat-icon class=\"elder-drop-zone-icon\">{{icon}}</mat-icon>\n <span class=\"elder-drop-zone-prompt\">{{prompt | translate}}</span>\n </div>\n </div>\n </div>\n</div>\n\n\n", styles: [""], dependencies: [{ kind: "component", type: i1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: i2.ElderFileDropZoneDirective, selector: "[elderFileDropZone]", inputs: ["dragOverClass"], outputs: ["dragEnter", "dragOver", "dragEnd", "dragEnterFiles", "dropFiles", "elderFileDropZoneChange"] }, { kind: "pipe", type: i3.TranslatePipe, name: "translate" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
29
29
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.7", ngImport: i0, type: ElderDropZoneComponent, decorators: [{
30
30
  type: Component,
31
31
  args: [{ selector: 'elder-drop-zone', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"full p-lg\"\n>\n <div class=\"layout-col full elder-drop-zone\"\n elderFileDropZone\n (dropFiles)=\"filesDropped.emit($event)\"\n (dragOver)=\"dragOver.emit($event)\"\n (dragEnd)=\"dragEnd.emit($event)\"\n >\n <div class=\"layout-col flex\">\n <ng-content></ng-content>\n <!-- Default content -->\n <div class=\"layout-col flex place-center-center gap-sm elder-default-content\">\n <mat-icon class=\"elder-drop-zone-icon\">{{icon}}</mat-icon>\n <span class=\"elder-drop-zone-prompt\">{{prompt | translate}}</span>\n </div>\n </div>\n </div>\n</div>\n\n\n" }]
@@ -20,8 +20,15 @@ export class ElderFileDropZoneDirective {
20
20
  this.logger = LoggerFactory.getLogger(this.constructor.name);
21
21
  this.dropFilesEnabled = false;
22
22
  this._dropFiles = new EventEmitter();
23
- this.dragOverFilesEnabled = false;
24
- this._dragOverFiles = new EventEmitter();
23
+ this.dragEnterFilesEnabled = false;
24
+ this._dragEnterFiles = new EventEmitter();
25
+ /**
26
+ * Fired once when drag zone is entered.
27
+ */
28
+ this.dragEnter = new EventEmitter();
29
+ /**
30
+ * Fired continuously inside drag-zone
31
+ */
25
32
  this.dragOver = new EventEmitter();
26
33
  this.dragEnd = new EventEmitter();
27
34
  this.dragOverClass = 'is-dragover';
@@ -31,9 +38,9 @@ export class ElderFileDropZoneDirective {
31
38
  * Properties *
32
39
  * *
33
40
  **************************************************************************/
34
- get dragOverFiles() {
35
- this.dragOverFilesEnabled = true;
36
- return this._dragOverFiles;
41
+ get dragEnterFiles() {
42
+ this.dragEnterFilesEnabled = true;
43
+ return this._dragEnterFiles;
37
44
  }
38
45
  get dropFiles() {
39
46
  this.dropFilesEnabled = true;
@@ -47,13 +54,17 @@ export class ElderFileDropZoneDirective {
47
54
  * Public API *
48
55
  * *
49
56
  **************************************************************************/
57
+ onDragEnter(evt) {
58
+ evt.preventDefault(); // Prevent default browser action
59
+ this.dragEnter.emit(evt);
60
+ if (this.dragEnterFilesEnabled) {
61
+ this.collectFilesAsync(evt, files => this.dragEnterFiles.emit(files));
62
+ }
63
+ }
50
64
  onDragOver(evt) {
51
65
  evt.preventDefault(); // Prevent default browser action
52
66
  this.renderer.addClass(this.hostRef.nativeElement, this.dragOverClass);
53
67
  this.dragOver.emit(evt);
54
- if (this.dragOverFilesEnabled) {
55
- this.collectFilesAsync(evt, files => this.dragOverFiles.emit(files));
56
- }
57
68
  }
58
69
  onDragEnd(evt) {
59
70
  evt.preventDefault(); // Prevent default browser action
@@ -94,24 +105,29 @@ export class ElderFileDropZoneDirective {
94
105
  }
95
106
  }
96
107
  ElderFileDropZoneDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.7", ngImport: i0, type: ElderFileDropZoneDirective, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Directive });
97
- ElderFileDropZoneDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.7", type: ElderFileDropZoneDirective, selector: "[elderFileDropZone]", inputs: { dragOverClass: "dragOverClass" }, outputs: { dragOver: "dragOver", dragEnd: "dragEnd", dragOverFiles: "dragOverFiles", dropFiles: "dropFiles", elderFileDropZoneChange: "elderFileDropZoneChange" }, host: { listeners: { "dragover": "onDragOver($event)", "dragend": "onDragEnd($event)", "dragleave": "onDragEnd($event)", "drop": "onDrop($event)" } }, ngImport: i0 });
108
+ ElderFileDropZoneDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.7", type: ElderFileDropZoneDirective, selector: "[elderFileDropZone]", inputs: { dragOverClass: "dragOverClass" }, outputs: { dragEnter: "dragEnter", dragOver: "dragOver", dragEnd: "dragEnd", dragEnterFiles: "dragEnterFiles", dropFiles: "dropFiles", elderFileDropZoneChange: "elderFileDropZoneChange" }, host: { listeners: { "dragenter": "onDragEnter($event)", "dragover": "onDragOver($event)", "dragend": "onDragEnd($event)", "dragleave": "onDragEnd($event)", "drop": "onDrop($event)" } }, ngImport: i0 });
98
109
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.7", ngImport: i0, type: ElderFileDropZoneDirective, decorators: [{
99
110
  type: Directive,
100
111
  args: [{
101
112
  selector: '[elderFileDropZone]'
102
113
  }]
103
- }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i0.NgZone }]; }, propDecorators: { dragOver: [{
114
+ }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i0.NgZone }]; }, propDecorators: { dragEnter: [{
115
+ type: Output
116
+ }], dragOver: [{
104
117
  type: Output
105
118
  }], dragEnd: [{
106
119
  type: Output
107
120
  }], dragOverClass: [{
108
121
  type: Input
109
- }], dragOverFiles: [{
122
+ }], dragEnterFiles: [{
110
123
  type: Output
111
124
  }], dropFiles: [{
112
125
  type: Output
113
126
  }], elderFileDropZoneChange: [{
114
127
  type: Output
128
+ }], onDragEnter: [{
129
+ type: HostListener,
130
+ args: ['dragenter', ['$event']]
115
131
  }], onDragOver: [{
116
132
  type: HostListener,
117
133
  args: ['dragover', ['$event']]
@@ -125,4 +141,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.7", ngImpor
125
141
  type: HostListener,
126
142
  args: ['drop', ['$event']]
127
143
  }] } });
128
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-file-drop-zone.directive.js","sourceRoot":"","sources":["../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/files/elder-file-drop-zone.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {Directive, ElementRef, EventEmitter, HostListener, Input, NgZone, OnDestroy, OnInit, Output, Renderer2} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {FileEntry} from './listing/file-entry';\nimport {FileListingRx} from './listing/file-listing-rx';\n\n@Directive({\n  selector: '[elderFileDropZone]'\n})\nexport class ElderFileDropZoneDirective {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private dropFilesEnabled = false;\n  private readonly _dropFiles = new EventEmitter<FileEntry[]>();\n\n  private dragOverFilesEnabled = false;\n  private readonly _dragOverFiles = new EventEmitter<FileEntry[]>();\n\n\n  @Output()\n  public readonly dragOver = new EventEmitter<DragEvent>();\n\n  @Output()\n  public readonly dragEnd = new EventEmitter<DragEvent>();\n\n  @Input()\n  public dragOverClass = 'is-dragover';\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private hostRef: ElementRef,\n    private renderer: Renderer2,\n    private ngZone: NgZone\n  ) {\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Output()\n  public get dragOverFiles(): EventEmitter<FileEntry[]> {\n    this.dragOverFilesEnabled = true;\n    return this._dragOverFiles;\n  }\n\n  @Output()\n  public get dropFiles(): EventEmitter<FileEntry[]> {\n    this.dropFilesEnabled = true;\n    return this._dropFiles;\n  }\n\n  @Output()\n  public get elderFileDropZoneChange(): EventEmitter<FileEntry[]> {\n    return this.dropFiles;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @HostListener('dragover', ['$event'])\n  public onDragOver(evt: DragEvent): void {\n    evt.preventDefault(); // Prevent default browser action\n    this.renderer.addClass(this.hostRef.nativeElement, this.dragOverClass);\n    this.dragOver.emit(evt);\n\n    if (this.dragOverFilesEnabled) {\n      this.collectFilesAsync(evt, files => this.dragOverFiles.emit(files));\n    }\n  }\n\n  @HostListener('dragend', ['$event'])\n  @HostListener('dragleave', ['$event'])\n  public onDragEnd(evt: DragEvent): void {\n    evt.preventDefault(); // Prevent default browser action\n    this.handleDragEnd(evt);\n    this.dragEnd.emit(evt);\n  }\n\n  @HostListener('drop', ['$event'])\n  public onDrop(evt: DragEvent): void {\n\n    this.logger.debug('Drop event!', evt);\n\n    evt.preventDefault();\n    evt.stopPropagation();\n    this.handleDragEnd(evt); // TODO Do this later\n\n    if (this.dropFilesEnabled) {\n      this.collectFilesAsync(evt, files => this._dropFiles.emit(files));\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private handleDragEnd(evt: DragEvent): void {\n    this.renderer.removeClass(this.hostRef.nativeElement, this.dragOverClass);\n  }\n\n  private collectFilesAsync(event: DragEvent, filesFn: ((entries: FileEntry[]) => void)): void {\n    this.ngZone.runOutsideAngular(() => {\n      try {\n        const items = event.dataTransfer.items;\n        FileListingRx.INSTANCE.toFileList(items).subscribe({\n            next: files => this.ngZone.run(() => filesFn(files)),\n            error: err => this.logger.error('Failed to list Files from DataTransferItemList', err)\n          }\n        );\n      } catch (error) {\n        this.logger.error('Failed to handle DragEvent!', error);\n        // this.ngZone.run(() => this.error.emit('Folder upload is unsupported by this browser'));\n      }\n    });\n  }\n}\n"]}
144
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-file-drop-zone.directive.js","sourceRoot":"","sources":["../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/files/elder-file-drop-zone.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {Directive, ElementRef, EventEmitter, HostListener, Input, NgZone, OnDestroy, OnInit, Output, Renderer2} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {FileEntry} from './listing/file-entry';\nimport {FileListingRx} from './listing/file-listing-rx';\n\n@Directive({\n  selector: '[elderFileDropZone]'\n})\nexport class ElderFileDropZoneDirective {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private dropFilesEnabled = false;\n  private readonly _dropFiles = new EventEmitter<FileEntry[]>();\n\n  private dragEnterFilesEnabled = false;\n  private readonly _dragEnterFiles = new EventEmitter<FileEntry[]>();\n\n  /**\n   * Fired once when drag zone is entered.\n   */\n  @Output()\n  public readonly dragEnter = new EventEmitter<DragEvent>();\n\n  /**\n   * Fired continuously inside drag-zone\n   */\n  @Output()\n  public readonly dragOver = new EventEmitter<DragEvent>();\n\n\n  @Output()\n  public readonly dragEnd = new EventEmitter<DragEvent>();\n\n  @Input()\n  public dragOverClass = 'is-dragover';\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private hostRef: ElementRef,\n    private renderer: Renderer2,\n    private ngZone: NgZone\n  ) {\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Output()\n  public get dragEnterFiles(): EventEmitter<FileEntry[]> {\n    this.dragEnterFilesEnabled = true;\n    return this._dragEnterFiles;\n  }\n\n  @Output()\n  public get dropFiles(): EventEmitter<FileEntry[]> {\n    this.dropFilesEnabled = true;\n    return this._dropFiles;\n  }\n\n  @Output()\n  public get elderFileDropZoneChange(): EventEmitter<FileEntry[]> {\n    return this.dropFiles;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @HostListener('dragenter', ['$event'])\n  public onDragEnter(evt: DragEvent): void {\n    evt.preventDefault(); // Prevent default browser action\n    this.dragEnter.emit(evt);\n    if (this.dragEnterFilesEnabled) {\n      this.collectFilesAsync(evt, files => this.dragEnterFiles.emit(files));\n    }\n  }\n\n  @HostListener('dragover', ['$event'])\n  public onDragOver(evt: DragEvent): void {\n    evt.preventDefault(); // Prevent default browser action\n    this.renderer.addClass(this.hostRef.nativeElement, this.dragOverClass);\n    this.dragOver.emit(evt);\n  }\n\n  @HostListener('dragend', ['$event'])\n  @HostListener('dragleave', ['$event'])\n  public onDragEnd(evt: DragEvent): void {\n    evt.preventDefault(); // Prevent default browser action\n    this.handleDragEnd(evt);\n    this.dragEnd.emit(evt);\n  }\n\n  @HostListener('drop', ['$event'])\n  public onDrop(evt: DragEvent): void {\n\n    this.logger.debug('Drop event!', evt);\n\n    evt.preventDefault();\n    evt.stopPropagation();\n    this.handleDragEnd(evt); // TODO Do this later\n\n    if (this.dropFilesEnabled) {\n      this.collectFilesAsync(evt, files => this._dropFiles.emit(files));\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private handleDragEnd(evt: DragEvent): void {\n    this.renderer.removeClass(this.hostRef.nativeElement, this.dragOverClass);\n  }\n\n  private collectFilesAsync(event: DragEvent, filesFn: ((entries: FileEntry[]) => void)): void {\n    this.ngZone.runOutsideAngular(() => {\n      try {\n        const items = event.dataTransfer.items;\n        FileListingRx.INSTANCE.toFileList(items).subscribe({\n            next: files => this.ngZone.run(() => filesFn(files)),\n            error: err => this.logger.error('Failed to list Files from DataTransferItemList', err)\n          }\n        );\n      } catch (error) {\n        this.logger.error('Failed to handle DragEvent!', error);\n        // this.ngZone.run(() => this.error.emit('Folder upload is unsupported by this browser'));\n      }\n    });\n  }\n}\n"]}
@@ -1,7 +1,11 @@
1
1
  import { EMPTY, mergeMap, Observable, of, zip } from 'rxjs';
2
2
  import { FileEntry } from './file-entry';
3
3
  import { map } from 'rxjs/operators';
4
+ import { LoggerFactory } from '@elderbyte/ts-logger';
4
5
  export class FileListingRx {
6
+ constructor() {
7
+ this.log = LoggerFactory.getLogger(this.constructor.name);
8
+ }
5
9
  /***************************************************************************
6
10
  * *
7
11
  * Public API *
@@ -16,7 +20,22 @@ export class FileListingRx {
16
20
  obs.push(this.listFilesRecursive(entry));
17
21
  }
18
22
  else {
19
- obs.push(of([FileEntry.ofFile(transferItem.getAsFile())]));
23
+ const itemAsFile = transferItem.getAsFile();
24
+ if (itemAsFile) {
25
+ obs.push(of([FileEntry.ofFile(itemAsFile)]));
26
+ }
27
+ else {
28
+ if (transferItem.kind == 'file') {
29
+ obs.push(of([
30
+ FileEntry.ofFile(new File([], '', {
31
+ type: transferItem.type
32
+ }))
33
+ ]));
34
+ }
35
+ else {
36
+ this.log.warn('Could not handle DataTransferItem!', transferItem);
37
+ }
38
+ }
20
39
  }
21
40
  }
22
41
  return zip(obs).pipe(map(files => files.flat()));
@@ -82,4 +101,4 @@ export class FileListingRx {
82
101
  }
83
102
  }
84
103
  FileListingRx.INSTANCE = new FileListingRx();
85
- //# 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';\n\nexport class FileListingRx {\n\n  public static readonly INSTANCE = new FileListingRx();\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        obs.push(of([FileEntry.ofFile(transferItem.getAsFile())]));\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"]}
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"]}
@@ -9946,6 +9946,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.7", ngImpor
9946
9946
  }] } });
9947
9947
 
9948
9948
  class FileListingRx {
9949
+ constructor() {
9950
+ this.log = LoggerFactory.getLogger(this.constructor.name);
9951
+ }
9949
9952
  /***************************************************************************
9950
9953
  * *
9951
9954
  * Public API *
@@ -9960,7 +9963,22 @@ class FileListingRx {
9960
9963
  obs.push(this.listFilesRecursive(entry));
9961
9964
  }
9962
9965
  else {
9963
- obs.push(of([FileEntry.ofFile(transferItem.getAsFile())]));
9966
+ const itemAsFile = transferItem.getAsFile();
9967
+ if (itemAsFile) {
9968
+ obs.push(of([FileEntry.ofFile(itemAsFile)]));
9969
+ }
9970
+ else {
9971
+ if (transferItem.kind == 'file') {
9972
+ obs.push(of([
9973
+ FileEntry.ofFile(new File([], '', {
9974
+ type: transferItem.type
9975
+ }))
9976
+ ]));
9977
+ }
9978
+ else {
9979
+ this.log.warn('Could not handle DataTransferItem!', transferItem);
9980
+ }
9981
+ }
9964
9982
  }
9965
9983
  }
9966
9984
  return zip(obs).pipe(map(files => files.flat()));
@@ -10045,8 +10063,15 @@ class ElderFileDropZoneDirective {
10045
10063
  this.logger = LoggerFactory.getLogger(this.constructor.name);
10046
10064
  this.dropFilesEnabled = false;
10047
10065
  this._dropFiles = new EventEmitter();
10048
- this.dragOverFilesEnabled = false;
10049
- this._dragOverFiles = new EventEmitter();
10066
+ this.dragEnterFilesEnabled = false;
10067
+ this._dragEnterFiles = new EventEmitter();
10068
+ /**
10069
+ * Fired once when drag zone is entered.
10070
+ */
10071
+ this.dragEnter = new EventEmitter();
10072
+ /**
10073
+ * Fired continuously inside drag-zone
10074
+ */
10050
10075
  this.dragOver = new EventEmitter();
10051
10076
  this.dragEnd = new EventEmitter();
10052
10077
  this.dragOverClass = 'is-dragover';
@@ -10056,9 +10081,9 @@ class ElderFileDropZoneDirective {
10056
10081
  * Properties *
10057
10082
  * *
10058
10083
  **************************************************************************/
10059
- get dragOverFiles() {
10060
- this.dragOverFilesEnabled = true;
10061
- return this._dragOverFiles;
10084
+ get dragEnterFiles() {
10085
+ this.dragEnterFilesEnabled = true;
10086
+ return this._dragEnterFiles;
10062
10087
  }
10063
10088
  get dropFiles() {
10064
10089
  this.dropFilesEnabled = true;
@@ -10072,13 +10097,17 @@ class ElderFileDropZoneDirective {
10072
10097
  * Public API *
10073
10098
  * *
10074
10099
  **************************************************************************/
10100
+ onDragEnter(evt) {
10101
+ evt.preventDefault(); // Prevent default browser action
10102
+ this.dragEnter.emit(evt);
10103
+ if (this.dragEnterFilesEnabled) {
10104
+ this.collectFilesAsync(evt, files => this.dragEnterFiles.emit(files));
10105
+ }
10106
+ }
10075
10107
  onDragOver(evt) {
10076
10108
  evt.preventDefault(); // Prevent default browser action
10077
10109
  this.renderer.addClass(this.hostRef.nativeElement, this.dragOverClass);
10078
10110
  this.dragOver.emit(evt);
10079
- if (this.dragOverFilesEnabled) {
10080
- this.collectFilesAsync(evt, files => this.dragOverFiles.emit(files));
10081
- }
10082
10111
  }
10083
10112
  onDragEnd(evt) {
10084
10113
  evt.preventDefault(); // Prevent default browser action
@@ -10119,24 +10148,29 @@ class ElderFileDropZoneDirective {
10119
10148
  }
10120
10149
  }
10121
10150
  ElderFileDropZoneDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.7", ngImport: i0, type: ElderFileDropZoneDirective, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Directive });
10122
- ElderFileDropZoneDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.7", type: ElderFileDropZoneDirective, selector: "[elderFileDropZone]", inputs: { dragOverClass: "dragOverClass" }, outputs: { dragOver: "dragOver", dragEnd: "dragEnd", dragOverFiles: "dragOverFiles", dropFiles: "dropFiles", elderFileDropZoneChange: "elderFileDropZoneChange" }, host: { listeners: { "dragover": "onDragOver($event)", "dragend": "onDragEnd($event)", "dragleave": "onDragEnd($event)", "drop": "onDrop($event)" } }, ngImport: i0 });
10151
+ ElderFileDropZoneDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.7", type: ElderFileDropZoneDirective, selector: "[elderFileDropZone]", inputs: { dragOverClass: "dragOverClass" }, outputs: { dragEnter: "dragEnter", dragOver: "dragOver", dragEnd: "dragEnd", dragEnterFiles: "dragEnterFiles", dropFiles: "dropFiles", elderFileDropZoneChange: "elderFileDropZoneChange" }, host: { listeners: { "dragenter": "onDragEnter($event)", "dragover": "onDragOver($event)", "dragend": "onDragEnd($event)", "dragleave": "onDragEnd($event)", "drop": "onDrop($event)" } }, ngImport: i0 });
10123
10152
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.7", ngImport: i0, type: ElderFileDropZoneDirective, decorators: [{
10124
10153
  type: Directive,
10125
10154
  args: [{
10126
10155
  selector: '[elderFileDropZone]'
10127
10156
  }]
10128
- }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i0.NgZone }]; }, propDecorators: { dragOver: [{
10157
+ }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i0.NgZone }]; }, propDecorators: { dragEnter: [{
10158
+ type: Output
10159
+ }], dragOver: [{
10129
10160
  type: Output
10130
10161
  }], dragEnd: [{
10131
10162
  type: Output
10132
10163
  }], dragOverClass: [{
10133
10164
  type: Input
10134
- }], dragOverFiles: [{
10165
+ }], dragEnterFiles: [{
10135
10166
  type: Output
10136
10167
  }], dropFiles: [{
10137
10168
  type: Output
10138
10169
  }], elderFileDropZoneChange: [{
10139
10170
  type: Output
10171
+ }], onDragEnter: [{
10172
+ type: HostListener,
10173
+ args: ['dragenter', ['$event']]
10140
10174
  }], onDragOver: [{
10141
10175
  type: HostListener,
10142
10176
  args: ['dragover', ['$event']]
@@ -10723,7 +10757,7 @@ class ElderDropZoneComponent {
10723
10757
  }
10724
10758
  }
10725
10759
  ElderDropZoneComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.7", ngImport: i0, type: ElderDropZoneComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
10726
- ElderDropZoneComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.7", type: ElderDropZoneComponent, selector: "elder-drop-zone", inputs: { icon: "icon", prompt: "prompt" }, outputs: { filesDropped: "filesDropped", dragOver: "dragOver", dragEnd: "dragEnd" }, ngImport: i0, template: "<div class=\"full p-lg\"\n>\n <div class=\"layout-col full elder-drop-zone\"\n elderFileDropZone\n (dropFiles)=\"filesDropped.emit($event)\"\n (dragOver)=\"dragOver.emit($event)\"\n (dragEnd)=\"dragEnd.emit($event)\"\n >\n <div class=\"layout-col flex\">\n <ng-content></ng-content>\n <!-- Default content -->\n <div class=\"layout-col flex place-center-center gap-sm elder-default-content\">\n <mat-icon class=\"elder-drop-zone-icon\">{{icon}}</mat-icon>\n <span class=\"elder-drop-zone-prompt\">{{prompt | translate}}</span>\n </div>\n </div>\n </div>\n</div>\n\n\n", styles: [""], dependencies: [{ kind: "component", type: i3$1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: ElderFileDropZoneDirective, selector: "[elderFileDropZone]", inputs: ["dragOverClass"], outputs: ["dragOver", "dragEnd", "dragOverFiles", "dropFiles", "elderFileDropZoneChange"] }, { kind: "pipe", type: i2.TranslatePipe, name: "translate" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
10760
+ ElderDropZoneComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.7", type: ElderDropZoneComponent, selector: "elder-drop-zone", inputs: { icon: "icon", prompt: "prompt" }, outputs: { filesDropped: "filesDropped", dragOver: "dragOver", dragEnd: "dragEnd" }, ngImport: i0, template: "<div class=\"full p-lg\"\n>\n <div class=\"layout-col full elder-drop-zone\"\n elderFileDropZone\n (dropFiles)=\"filesDropped.emit($event)\"\n (dragOver)=\"dragOver.emit($event)\"\n (dragEnd)=\"dragEnd.emit($event)\"\n >\n <div class=\"layout-col flex\">\n <ng-content></ng-content>\n <!-- Default content -->\n <div class=\"layout-col flex place-center-center gap-sm elder-default-content\">\n <mat-icon class=\"elder-drop-zone-icon\">{{icon}}</mat-icon>\n <span class=\"elder-drop-zone-prompt\">{{prompt | translate}}</span>\n </div>\n </div>\n </div>\n</div>\n\n\n", styles: [""], dependencies: [{ kind: "component", type: i3$1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: ElderFileDropZoneDirective, selector: "[elderFileDropZone]", inputs: ["dragOverClass"], outputs: ["dragEnter", "dragOver", "dragEnd", "dragEnterFiles", "dropFiles", "elderFileDropZoneChange"] }, { kind: "pipe", type: i2.TranslatePipe, name: "translate" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
10727
10761
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.7", ngImport: i0, type: ElderDropZoneComponent, decorators: [{
10728
10762
  type: Component,
10729
10763
  args: [{ selector: 'elder-drop-zone', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"full p-lg\"\n>\n <div class=\"layout-col full elder-drop-zone\"\n elderFileDropZone\n (dropFiles)=\"filesDropped.emit($event)\"\n (dragOver)=\"dragOver.emit($event)\"\n (dragEnd)=\"dragEnd.emit($event)\"\n >\n <div class=\"layout-col flex\">\n <ng-content></ng-content>\n <!-- Default content -->\n <div class=\"layout-col flex place-center-center gap-sm elder-default-content\">\n <mat-icon class=\"elder-drop-zone-icon\">{{icon}}</mat-icon>\n <span class=\"elder-drop-zone-prompt\">{{prompt | translate}}</span>\n </div>\n </div>\n </div>\n</div>\n\n\n" }]