@elderbyte/ngx-starter 14.5.4 → 14.7.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (87) hide show
  1. package/esm2020/lib/common/data/data-context/data-context-active-page.mjs +1 -1
  2. package/esm2020/lib/common/data/data-context/data-context-base.mjs +2 -2
  3. package/esm2020/lib/common/data/data-context/data-context-continuable-base.mjs +1 -1
  4. package/esm2020/lib/common/data/data-context/data-context.mjs +1 -1
  5. package/esm2020/lib/common/data/datasource/data-source-adapter.mjs +1 -1
  6. package/esm2020/lib/common/data/datasource/data-source-processor.mjs +1 -1
  7. package/esm2020/lib/common/data/datasource/data-source.mjs +1 -1
  8. package/esm2020/lib/common/data/datasource/fetcher/delegate-data-source.mjs +1 -1
  9. package/esm2020/lib/common/data/datasource/local/local-list-data-source.mjs +1 -1
  10. package/esm2020/lib/common/data/datasource/local/local-paged-data-source.mjs +1 -1
  11. package/esm2020/lib/common/data/datasource/rest/rest-client.mjs +2 -2
  12. package/esm2020/lib/common/data/filters/filter-context.mjs +141 -0
  13. package/esm2020/lib/common/data/filters/filter.mjs +109 -0
  14. package/esm2020/lib/common/data/http-params-builder.mjs +2 -2
  15. package/esm2020/lib/common/data/public_api.mjs +3 -3
  16. package/esm2020/lib/common/data/required-filter-evaluator.mjs +1 -1
  17. package/esm2020/lib/common/data/token-chunk-request.mjs +1 -1
  18. package/esm2020/lib/common/utils/filter-util.mjs +1 -1
  19. package/esm2020/lib/components/csv/elder-csv-export-btn/elder-csv-export-btn.component.mjs +1 -1
  20. package/esm2020/lib/components/csv/elder-csv-stream-exporter.mjs +1 -1
  21. package/esm2020/lib/components/data-view/base/data-view-options-provider-binding.mjs +5 -1
  22. package/esm2020/lib/components/data-view/base/elder-data-view-base.mjs +1 -5
  23. package/esm2020/lib/components/data-view/base/elder-data-view-options-provider.mjs +1 -1
  24. package/esm2020/lib/components/data-view/base/elder-data-view-options.mjs +4 -1
  25. package/esm2020/lib/components/forms/search/elder-search-context.directive.mjs +33 -19
  26. package/esm2020/lib/components/forms/search/elder-search-input.directive.mjs +6 -12
  27. package/esm2020/lib/components/http-support/elder-http-client.service.mjs +1 -1
  28. package/esm2020/lib/components/iframes/data-view/data-view-iframe/data-view-iframe.component.mjs +19 -10
  29. package/esm2020/lib/components/iframes/data-view/data-view-iframe-adapter.directive.mjs +3 -4
  30. package/esm2020/lib/components/input/autocomplete/elder-autocomplete/elder-autocomplete-option-selected.mjs +9 -0
  31. package/esm2020/lib/components/input/autocomplete/elder-autocomplete/elder-autocomplete.component.mjs +40 -25
  32. package/esm2020/lib/components/input/autocomplete/elder-autocomplete-many.directive.mjs +197 -0
  33. package/esm2020/lib/components/input/autocomplete/elder-autocomplete.directive.mjs +9 -9
  34. package/esm2020/lib/components/input/autocomplete/elder-autocomplete.module.mjs +19 -8
  35. package/esm2020/lib/components/input/autocomplete/many/word-position-finder.mjs +58 -0
  36. package/esm2020/lib/components/measures/directives/elder-unit-select.directive.mjs +2 -2
  37. package/esm2020/lib/components/public_api.mjs +1 -2
  38. package/esm2020/lib/components/select/elder-select-base.mjs +2 -2
  39. package/esm2020/lib/components/select/popup/selection-model-popup.directive.mjs +1 -1
  40. package/esm2020/lib/components/select/single/elder-select/elder-select.component.mjs +7 -6
  41. package/fesm2015/elderbyte-ngx-starter.mjs +461 -280
  42. package/fesm2015/elderbyte-ngx-starter.mjs.map +1 -1
  43. package/fesm2020/elderbyte-ngx-starter.mjs +458 -280
  44. package/fesm2020/elderbyte-ngx-starter.mjs.map +1 -1
  45. package/lib/common/data/data-context/data-context-active-page.d.ts +1 -1
  46. package/lib/common/data/data-context/data-context-base.d.ts +2 -2
  47. package/lib/common/data/data-context/data-context-continuable-base.d.ts +1 -1
  48. package/lib/common/data/data-context/data-context.d.ts +2 -2
  49. package/lib/common/data/datasource/data-source-adapter.d.ts +1 -1
  50. package/lib/common/data/datasource/data-source-processor.d.ts +1 -1
  51. package/lib/common/data/datasource/data-source.d.ts +1 -1
  52. package/lib/common/data/datasource/fetcher/delegate-data-source.d.ts +1 -1
  53. package/lib/common/data/datasource/local/local-list-data-source.d.ts +1 -1
  54. package/lib/common/data/datasource/local/local-paged-data-source.d.ts +1 -1
  55. package/lib/common/data/datasource/rest/rest-client.d.ts +1 -1
  56. package/lib/common/data/{filter-context.d.ts → filters/filter-context.d.ts} +3 -1
  57. package/lib/common/data/{filter.d.ts → filters/filter.d.ts} +15 -3
  58. package/lib/common/data/http-params-builder.d.ts +1 -1
  59. package/lib/common/data/public_api.d.ts +2 -2
  60. package/lib/common/data/required-filter-evaluator.d.ts +2 -2
  61. package/lib/common/data/token-chunk-request.d.ts +1 -1
  62. package/lib/common/utils/filter-util.d.ts +1 -1
  63. package/lib/components/csv/elder-csv-export-btn/elder-csv-export-btn.component.d.ts +1 -1
  64. package/lib/components/csv/elder-csv-stream-exporter.d.ts +1 -1
  65. package/lib/components/data-view/base/elder-data-view-options.d.ts +2 -1
  66. package/lib/components/forms/search/elder-search-context.directive.d.ts +15 -12
  67. package/lib/components/forms/search/elder-search-input.directive.d.ts +6 -7
  68. package/lib/components/http-support/elder-http-client.service.d.ts +1 -1
  69. package/lib/components/iframes/data-view/data-view-iframe/data-view-iframe.component.d.ts +4 -2
  70. package/lib/components/input/autocomplete/elder-autocomplete/elder-autocomplete-option-selected.d.ts +8 -0
  71. package/lib/components/input/autocomplete/elder-autocomplete/elder-autocomplete.component.d.ts +14 -14
  72. package/lib/components/{multi-autocomplete/elder-multi-autocomplete.component.d.ts → input/autocomplete/elder-autocomplete-many.directive.d.ts} +46 -27
  73. package/lib/components/input/autocomplete/elder-autocomplete.directive.d.ts +2 -2
  74. package/lib/components/input/autocomplete/elder-autocomplete.module.d.ts +11 -7
  75. package/lib/components/{multi-autocomplete/word-position.d.ts → input/autocomplete/many/word-position-finder.d.ts} +0 -0
  76. package/lib/components/public_api.d.ts +0 -1
  77. package/lib/components/select/elder-select-base.d.ts +1 -1
  78. package/lib/components/select/popup/selection-model-popup.directive.d.ts +1 -1
  79. package/lib/components/select/single/elder-select/elder-select.component.d.ts +2 -1
  80. package/package.json +1 -1
  81. package/esm2020/lib/common/data/filter-context.mjs +0 -129
  82. package/esm2020/lib/common/data/filter.mjs +0 -72
  83. package/esm2020/lib/components/multi-autocomplete/elder-multi-auto-complete.module.mjs +0 -30
  84. package/esm2020/lib/components/multi-autocomplete/elder-multi-autocomplete.component.mjs +0 -122
  85. package/esm2020/lib/components/multi-autocomplete/word-position.mjs +0 -38
  86. package/lib/components/multi-autocomplete/elder-multi-auto-complete.module.d.ts +0 -14
  87. package/src/lib/components/multi-autocomplete/elder-multi-autocomplete.component.scss +0 -0
@@ -1,4 +1,4 @@
1
- import { Component, ChangeDetectionStrategy, Input, TemplateRef, ContentChild, Output, ViewChild } from '@angular/core';
1
+ import { ChangeDetectionStrategy, Component, ContentChild, Input, Output, TemplateRef, ViewChild } from '@angular/core';
2
2
  import { LoggerFactory } from '@elderbyte/ts-logger';
3
3
  import { BehaviorSubject, of, Subject } from 'rxjs';
4
4
  import { isContinuableDataContext } from '../../../../common/data/data-context/data-context';
@@ -6,6 +6,7 @@ import { coerceBooleanProperty } from '@angular/cdk/coercion';
6
6
  import { filter, switchMap } from 'rxjs/operators';
7
7
  import { runInZone } from '../../../../common/utils/ng-zone-utils';
8
8
  import { ElderSelectValueDirective } from '../../../select/elder-select-value.directive';
9
+ import { ElderAutocompleteOptionSelected } from './elder-autocomplete-option-selected';
9
10
  import * as i0 from "@angular/core";
10
11
  import * as i1 from "@angular/common";
11
12
  import * as i2 from "@angular/material/autocomplete";
@@ -37,24 +38,19 @@ export class ElderAutocompleteComponent {
37
38
  * The underlying Material Autocomplete
38
39
  */
39
40
  this.matAutocomplete$ = new BehaviorSubject(null);
40
- /** Underlying data context. */
41
- this.suggestionsDc$ = new BehaviorSubject(null);
41
+ /**
42
+ * The DataContext which holds the autocomplete suggestions.
43
+ */
44
+ this.dataContext$ = new BehaviorSubject(null);
42
45
  this.triggerReload$ = new Subject();
43
46
  this.displayPropertyResolver$ = new BehaviorSubject((value) => this.propertyStringValue(value, null));
44
47
  this.optionSelected = new Subject();
45
- this.dataState$ = this.suggestionsDc$.pipe(filter(dc => !!dc), switchMap(dc => dc.status), runInZone(this.zone));
46
- this.availableSuggestions$ = this.suggestionsDc$.pipe(switchMap(dc => dc ? dc.data : of([])), runInZone(this.zone));
48
+ this.dataState$ = this.dataContext$.pipe(filter(dc => !!dc), switchMap(dc => dc.status), runInZone(this.zone));
49
+ this.availableSuggestions$ = this.dataContext$.pipe(switchMap(dc => dc ? dc.data : of([])), runInZone(this.zone));
47
50
  }
48
51
  set matAutocomplete(auto) {
49
52
  this.matAutocomplete$.next(auto);
50
53
  }
51
- /***************************************************************************
52
- * *
53
- * Life Cycle *
54
- * *
55
- **************************************************************************/
56
- ngOnInit() {
57
- }
58
54
  /***************************************************************************
59
55
  * *
60
56
  * Properties *
@@ -72,11 +68,11 @@ export class ElderAutocompleteComponent {
72
68
  get valueTemplate() {
73
69
  return this._valueTemplate || this.valueTemplateQuery;
74
70
  }
75
- set suggestionsDc(data) {
76
- this.suggestionsDc$.next(data);
71
+ set dataContext(data) {
72
+ this.dataContext$.next(data);
77
73
  }
78
- get suggestionsDc() {
79
- return this.suggestionsDc$.getValue();
74
+ get dataContext() {
75
+ return this.dataContext$.getValue();
80
76
  }
81
77
  set displayPropertyResolver(resolver) {
82
78
  this.displayPropertyResolver$.next(resolver);
@@ -87,7 +83,16 @@ export class ElderAutocompleteComponent {
87
83
  * *
88
84
  **************************************************************************/
89
85
  onOptionSelected(event) {
90
- this.optionSelected.next(event.option.value);
86
+ const option = event.option;
87
+ const id = option.id;
88
+ this.optionSelected.next(new ElderAutocompleteOptionSelected(this, id, this.getEntityById(id), option.value));
89
+ }
90
+ getEntityById(id) {
91
+ const dc = this.dataContext;
92
+ if (dc) {
93
+ return dc.snapshot.data.find(e => this.getId(e) == id);
94
+ }
95
+ return undefined;
91
96
  }
92
97
  onAutocompleteOpened(event) {
93
98
  this.logger.debug('Autocomplete opened:', event);
@@ -97,9 +102,9 @@ export class ElderAutocompleteComponent {
97
102
  }
98
103
  onAutoCompleteCloseToEnd(event) {
99
104
  // this.logger.debug('Autocomplete is close to end!');
100
- if (isContinuableDataContext(this.suggestionsDc)) {
101
- if (!this.suggestionsDc.snapshot.isLoading) {
102
- this.suggestionsDc.loadMore();
105
+ if (isContinuableDataContext(this.dataContext)) {
106
+ if (!this.dataContext.snapshot.isLoading) {
107
+ this.dataContext.loadMore();
103
108
  }
104
109
  }
105
110
  }
@@ -107,7 +112,7 @@ export class ElderAutocompleteComponent {
107
112
  return String(this.getId(value));
108
113
  }
109
114
  getId(value) {
110
- return this.suggestionsDc?.dataSource?.getId(value) || value;
115
+ return this.dataContext?.dataSource?.getId(value) || value;
111
116
  }
112
117
  isOptionAvailable(option) {
113
118
  if (this.isOptionDisabledFn) {
@@ -125,6 +130,14 @@ export class ElderAutocompleteComponent {
125
130
  return true;
126
131
  }
127
132
  }
133
+ toOptionValue(option) {
134
+ if (this.optionValueConverterFn) {
135
+ return this.optionValueConverterFn(option);
136
+ }
137
+ else {
138
+ return option;
139
+ }
140
+ }
128
141
  /***************************************************************************
129
142
  * *
130
143
  * Private methods *
@@ -144,10 +157,10 @@ export class ElderAutocompleteComponent {
144
157
  }
145
158
  }
146
159
  ElderAutocompleteComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.1.0", ngImport: i0, type: ElderAutocompleteComponent, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component });
147
- ElderAutocompleteComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.1.0", type: ElderAutocompleteComponent, selector: "elder-autocomplete", inputs: { isOptionDisabledFn: "isOptionDisabledFn", isOptionHiddenFn: "isOptionHiddenFn", enabled: "enabled", valueTemplate: "valueTemplate", suggestionsDc: "suggestionsDc", displayPropertyResolver: "displayPropertyResolver" }, outputs: { optionSelected: "optionSelected" }, queries: [{ propertyName: "valueTemplateQuery", first: true, predicate: ElderSelectValueDirective, descendants: true, read: TemplateRef, static: true }], viewQueries: [{ propertyName: "matAutocomplete", first: true, predicate: ["auto"], descendants: true }], exportAs: ["elderAutocomplete"], ngImport: i0, template: "<mat-autocomplete\n #auto=\"matAutocomplete\"\n [autoActiveFirstOption]=\"true\"\n (opened)=\"onAutocompleteOpened($event)\"\n (optionSelected)=\"onOptionSelected($event)\"\n elderInfiniteScroll elderElderInfiniteAutocomplete (closeToEnd)=\"onAutoCompleteCloseToEnd($event)\"\n>\n <!--\n <mat-option *ngIf=\"allowNull\" [value]=\"null\">{{nullDisplay | translate}}</mat-option>\n -->\n\n <mat-option disabled *ngIf=\"!(suggestionsDc$ | async)\">\n <span class=\"mat-caption\">No DataSource / DataContext!</span>\n </mat-option>\n\n <ng-container *ngIf=\"suggestionsDc$ | async as dc\">\n\n <mat-option disabled *ngIf=\"dc.isClosed\">\n <div fxLayout=\"row\" fxLayoutAlign=\"start center\" fxLayoutGap=\"8px\">\n <mat-icon color=\"warn\">warning</mat-icon>\n <span class=\"mat-caption\">DataContext Closed!</span>\n </div>\n </mat-option>\n\n <ng-container *ngIf=\"availableSuggestions$ | async as suggestions\">\n\n <mat-option *ngIf=\"suggestions.length === 0\" disabled>\n No Data.\n </mat-option>\n\n <ng-container *ngFor=\"let suggestion of suggestions\">\n <mat-option\n *ngIf=\"isOptionVisible(suggestion)\"\n [value]=\"suggestion\"\n [id]=\"getIdAsString(suggestion)\"\n [disabled]=\"!isOptionAvailable(suggestion)\"\n >\n <ng-container\n *ngTemplateOutlet=\"valueTemplate || simpleValueTemplate; context: {$implicit: suggestion}\">\n </ng-container>\n </mat-option>\n </ng-container>\n\n\n\n </ng-container>\n </ng-container>\n\n <ng-container *ngIf=\"dataState$ | async as state\">\n\n <mat-option disabled *ngIf=\"!state.idle || state.loading\">\n <mat-progress-bar\n [value]=\"100\"\n [mode]=\"state.loading ? 'query' : 'determinate'\"\n [color]=\"state.error ? 'warn' : 'primary'\"\n ></mat-progress-bar>\n </mat-option>\n\n </ng-container>\n</mat-autocomplete>\n\n<ng-template #simpleValueTemplate let-value>\n <ng-container *ngIf=\"displayPropertyResolver$ | async as propertyResolver\">\n <p class=\"noselect\">{{propertyResolver(value)}}</p>\n </ng-container>\n</ng-template>\n", styles: [""], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.MatAutocomplete, selector: "mat-autocomplete", inputs: ["disableRipple"], exportAs: ["matAutocomplete"] }, { kind: "component", type: i3.MatOption, selector: "mat-option", exportAs: ["matOption"] }, { kind: "directive", type: i4.ElderInfiniteScrollDirective, selector: "[elderInfiniteScroll]", inputs: ["listenToHost", "eventThrottle", "offsetFactor", "ignoreScrollEvent", "containerId", "scrollContainer"], outputs: ["closeToEnd", "scrolling"] }, { kind: "directive", type: i5.ElderInfiniteAutocompleteDirective, selector: "mat-autocomplete[elderElderInfiniteAutocomplete]" }, { kind: "component", type: i6.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: i7.DefaultLayoutDirective, selector: " [fxLayout], [fxLayout.xs], [fxLayout.sm], [fxLayout.md], [fxLayout.lg], [fxLayout.xl], [fxLayout.lt-sm], [fxLayout.lt-md], [fxLayout.lt-lg], [fxLayout.lt-xl], [fxLayout.gt-xs], [fxLayout.gt-sm], [fxLayout.gt-md], [fxLayout.gt-lg]", inputs: ["fxLayout", "fxLayout.xs", "fxLayout.sm", "fxLayout.md", "fxLayout.lg", "fxLayout.xl", "fxLayout.lt-sm", "fxLayout.lt-md", "fxLayout.lt-lg", "fxLayout.lt-xl", "fxLayout.gt-xs", "fxLayout.gt-sm", "fxLayout.gt-md", "fxLayout.gt-lg"] }, { kind: "directive", type: i7.DefaultLayoutGapDirective, selector: " [fxLayoutGap], [fxLayoutGap.xs], [fxLayoutGap.sm], [fxLayoutGap.md], [fxLayoutGap.lg], [fxLayoutGap.xl], [fxLayoutGap.lt-sm], [fxLayoutGap.lt-md], [fxLayoutGap.lt-lg], [fxLayoutGap.lt-xl], [fxLayoutGap.gt-xs], [fxLayoutGap.gt-sm], [fxLayoutGap.gt-md], [fxLayoutGap.gt-lg]", inputs: ["fxLayoutGap", "fxLayoutGap.xs", "fxLayoutGap.sm", "fxLayoutGap.md", "fxLayoutGap.lg", "fxLayoutGap.xl", "fxLayoutGap.lt-sm", "fxLayoutGap.lt-md", "fxLayoutGap.lt-lg", "fxLayoutGap.lt-xl", "fxLayoutGap.gt-xs", "fxLayoutGap.gt-sm", "fxLayoutGap.gt-md", "fxLayoutGap.gt-lg"] }, { kind: "directive", type: i7.DefaultLayoutAlignDirective, selector: " [fxLayoutAlign], [fxLayoutAlign.xs], [fxLayoutAlign.sm], [fxLayoutAlign.md], [fxLayoutAlign.lg], [fxLayoutAlign.xl], [fxLayoutAlign.lt-sm], [fxLayoutAlign.lt-md], [fxLayoutAlign.lt-lg], [fxLayoutAlign.lt-xl], [fxLayoutAlign.gt-xs], [fxLayoutAlign.gt-sm], [fxLayoutAlign.gt-md], [fxLayoutAlign.gt-lg]", inputs: ["fxLayoutAlign", "fxLayoutAlign.xs", "fxLayoutAlign.sm", "fxLayoutAlign.md", "fxLayoutAlign.lg", "fxLayoutAlign.xl", "fxLayoutAlign.lt-sm", "fxLayoutAlign.lt-md", "fxLayoutAlign.lt-lg", "fxLayoutAlign.lt-xl", "fxLayoutAlign.gt-xs", "fxLayoutAlign.gt-sm", "fxLayoutAlign.gt-md", "fxLayoutAlign.gt-lg"] }, { kind: "component", type: i8.MatProgressBar, selector: "mat-progress-bar", inputs: ["color", "value", "bufferValue", "mode"], outputs: ["animationEnd"], exportAs: ["matProgressBar"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
160
+ ElderAutocompleteComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.1.0", type: ElderAutocompleteComponent, selector: "elder-autocomplete", inputs: { isOptionDisabledFn: "isOptionDisabledFn", isOptionHiddenFn: "isOptionHiddenFn", optionValueConverterFn: "optionValueConverterFn", enabled: "enabled", valueTemplate: "valueTemplate", dataContext: "dataContext", displayPropertyResolver: "displayPropertyResolver" }, outputs: { optionSelected: "optionSelected" }, queries: [{ propertyName: "valueTemplateQuery", first: true, predicate: ElderSelectValueDirective, descendants: true, read: TemplateRef, static: true }], viewQueries: [{ propertyName: "matAutocomplete", first: true, predicate: ["auto"], descendants: true }], exportAs: ["elderAutocomplete"], ngImport: i0, template: "<mat-autocomplete\n #auto=\"matAutocomplete\"\n [autoActiveFirstOption]=\"true\"\n (opened)=\"onAutocompleteOpened($event)\"\n (optionSelected)=\"onOptionSelected($event)\"\n elderInfiniteScroll elderElderInfiniteAutocomplete (closeToEnd)=\"onAutoCompleteCloseToEnd($event)\"\n>\n <!--\n <mat-option *ngIf=\"allowNull\" [value]=\"null\">{{nullDisplay | translate}}</mat-option>\n -->\n\n <mat-option disabled *ngIf=\"!(dataContext$ | async)\">\n <span class=\"mat-caption\">No DataSource / DataContext!</span>\n </mat-option>\n\n <ng-container *ngIf=\"dataContext$ | async as dc\">\n\n <mat-option disabled *ngIf=\"dc.isClosed\">\n <div fxLayout=\"row\" fxLayoutAlign=\"start center\" fxLayoutGap=\"8px\">\n <mat-icon color=\"warn\">warning</mat-icon>\n <span class=\"mat-caption\">DataContext Closed!</span>\n </div>\n </mat-option>\n\n <ng-container *ngIf=\"availableSuggestions$ | async as suggestions\">\n\n <mat-option *ngIf=\"suggestions.length === 0\" disabled>\n No Data.\n </mat-option>\n\n <ng-container *ngFor=\"let suggestion of suggestions\">\n <mat-option\n *ngIf=\"isOptionVisible(suggestion)\"\n [value]=\"toOptionValue(suggestion)\"\n [id]=\"getIdAsString(suggestion)\"\n [disabled]=\"!isOptionAvailable(suggestion)\"\n >\n <ng-container\n *ngTemplateOutlet=\"valueTemplate || simpleValueTemplate; context: {$implicit: suggestion}\">\n </ng-container>\n <!--\n <span class=\"mat-caption\">value: {{toOptionValue(suggestion)}}</span>\n -->\n </mat-option>\n </ng-container>\n </ng-container>\n </ng-container>\n\n <ng-container *ngIf=\"dataState$ | async as state\">\n\n <mat-option disabled *ngIf=\"!state.idle || state.loading\">\n <mat-progress-bar\n [value]=\"100\"\n [mode]=\"state.loading ? 'query' : 'determinate'\"\n [color]=\"state.error ? 'warn' : 'primary'\"\n ></mat-progress-bar>\n </mat-option>\n\n </ng-container>\n</mat-autocomplete>\n\n<ng-template #simpleValueTemplate let-value>\n <ng-container *ngIf=\"displayPropertyResolver$ | async as propertyResolver\">\n <span class=\"noselect\">{{propertyResolver(value)}}</span>\n </ng-container>\n</ng-template>\n", styles: [""], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.MatAutocomplete, selector: "mat-autocomplete", inputs: ["disableRipple"], exportAs: ["matAutocomplete"] }, { kind: "component", type: i3.MatOption, selector: "mat-option", exportAs: ["matOption"] }, { kind: "directive", type: i4.ElderInfiniteScrollDirective, selector: "[elderInfiniteScroll]", inputs: ["listenToHost", "eventThrottle", "offsetFactor", "ignoreScrollEvent", "containerId", "scrollContainer"], outputs: ["closeToEnd", "scrolling"] }, { kind: "directive", type: i5.ElderInfiniteAutocompleteDirective, selector: "mat-autocomplete[elderElderInfiniteAutocomplete]" }, { kind: "component", type: i6.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: i7.DefaultLayoutDirective, selector: " [fxLayout], [fxLayout.xs], [fxLayout.sm], [fxLayout.md], [fxLayout.lg], [fxLayout.xl], [fxLayout.lt-sm], [fxLayout.lt-md], [fxLayout.lt-lg], [fxLayout.lt-xl], [fxLayout.gt-xs], [fxLayout.gt-sm], [fxLayout.gt-md], [fxLayout.gt-lg]", inputs: ["fxLayout", "fxLayout.xs", "fxLayout.sm", "fxLayout.md", "fxLayout.lg", "fxLayout.xl", "fxLayout.lt-sm", "fxLayout.lt-md", "fxLayout.lt-lg", "fxLayout.lt-xl", "fxLayout.gt-xs", "fxLayout.gt-sm", "fxLayout.gt-md", "fxLayout.gt-lg"] }, { kind: "directive", type: i7.DefaultLayoutGapDirective, selector: " [fxLayoutGap], [fxLayoutGap.xs], [fxLayoutGap.sm], [fxLayoutGap.md], [fxLayoutGap.lg], [fxLayoutGap.xl], [fxLayoutGap.lt-sm], [fxLayoutGap.lt-md], [fxLayoutGap.lt-lg], [fxLayoutGap.lt-xl], [fxLayoutGap.gt-xs], [fxLayoutGap.gt-sm], [fxLayoutGap.gt-md], [fxLayoutGap.gt-lg]", inputs: ["fxLayoutGap", "fxLayoutGap.xs", "fxLayoutGap.sm", "fxLayoutGap.md", "fxLayoutGap.lg", "fxLayoutGap.xl", "fxLayoutGap.lt-sm", "fxLayoutGap.lt-md", "fxLayoutGap.lt-lg", "fxLayoutGap.lt-xl", "fxLayoutGap.gt-xs", "fxLayoutGap.gt-sm", "fxLayoutGap.gt-md", "fxLayoutGap.gt-lg"] }, { kind: "directive", type: i7.DefaultLayoutAlignDirective, selector: " [fxLayoutAlign], [fxLayoutAlign.xs], [fxLayoutAlign.sm], [fxLayoutAlign.md], [fxLayoutAlign.lg], [fxLayoutAlign.xl], [fxLayoutAlign.lt-sm], [fxLayoutAlign.lt-md], [fxLayoutAlign.lt-lg], [fxLayoutAlign.lt-xl], [fxLayoutAlign.gt-xs], [fxLayoutAlign.gt-sm], [fxLayoutAlign.gt-md], [fxLayoutAlign.gt-lg]", inputs: ["fxLayoutAlign", "fxLayoutAlign.xs", "fxLayoutAlign.sm", "fxLayoutAlign.md", "fxLayoutAlign.lg", "fxLayoutAlign.xl", "fxLayoutAlign.lt-sm", "fxLayoutAlign.lt-md", "fxLayoutAlign.lt-lg", "fxLayoutAlign.lt-xl", "fxLayoutAlign.gt-xs", "fxLayoutAlign.gt-sm", "fxLayoutAlign.gt-md", "fxLayoutAlign.gt-lg"] }, { kind: "component", type: i8.MatProgressBar, selector: "mat-progress-bar", inputs: ["color", "value", "bufferValue", "mode"], outputs: ["animationEnd"], exportAs: ["matProgressBar"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
148
161
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.1.0", ngImport: i0, type: ElderAutocompleteComponent, decorators: [{
149
162
  type: Component,
150
- args: [{ selector: 'elder-autocomplete', exportAs: 'elderAutocomplete', changeDetection: ChangeDetectionStrategy.OnPush, template: "<mat-autocomplete\n #auto=\"matAutocomplete\"\n [autoActiveFirstOption]=\"true\"\n (opened)=\"onAutocompleteOpened($event)\"\n (optionSelected)=\"onOptionSelected($event)\"\n elderInfiniteScroll elderElderInfiniteAutocomplete (closeToEnd)=\"onAutoCompleteCloseToEnd($event)\"\n>\n <!--\n <mat-option *ngIf=\"allowNull\" [value]=\"null\">{{nullDisplay | translate}}</mat-option>\n -->\n\n <mat-option disabled *ngIf=\"!(suggestionsDc$ | async)\">\n <span class=\"mat-caption\">No DataSource / DataContext!</span>\n </mat-option>\n\n <ng-container *ngIf=\"suggestionsDc$ | async as dc\">\n\n <mat-option disabled *ngIf=\"dc.isClosed\">\n <div fxLayout=\"row\" fxLayoutAlign=\"start center\" fxLayoutGap=\"8px\">\n <mat-icon color=\"warn\">warning</mat-icon>\n <span class=\"mat-caption\">DataContext Closed!</span>\n </div>\n </mat-option>\n\n <ng-container *ngIf=\"availableSuggestions$ | async as suggestions\">\n\n <mat-option *ngIf=\"suggestions.length === 0\" disabled>\n No Data.\n </mat-option>\n\n <ng-container *ngFor=\"let suggestion of suggestions\">\n <mat-option\n *ngIf=\"isOptionVisible(suggestion)\"\n [value]=\"suggestion\"\n [id]=\"getIdAsString(suggestion)\"\n [disabled]=\"!isOptionAvailable(suggestion)\"\n >\n <ng-container\n *ngTemplateOutlet=\"valueTemplate || simpleValueTemplate; context: {$implicit: suggestion}\">\n </ng-container>\n </mat-option>\n </ng-container>\n\n\n\n </ng-container>\n </ng-container>\n\n <ng-container *ngIf=\"dataState$ | async as state\">\n\n <mat-option disabled *ngIf=\"!state.idle || state.loading\">\n <mat-progress-bar\n [value]=\"100\"\n [mode]=\"state.loading ? 'query' : 'determinate'\"\n [color]=\"state.error ? 'warn' : 'primary'\"\n ></mat-progress-bar>\n </mat-option>\n\n </ng-container>\n</mat-autocomplete>\n\n<ng-template #simpleValueTemplate let-value>\n <ng-container *ngIf=\"displayPropertyResolver$ | async as propertyResolver\">\n <p class=\"noselect\">{{propertyResolver(value)}}</p>\n </ng-container>\n</ng-template>\n" }]
163
+ args: [{ selector: 'elder-autocomplete', exportAs: 'elderAutocomplete', changeDetection: ChangeDetectionStrategy.OnPush, template: "<mat-autocomplete\n #auto=\"matAutocomplete\"\n [autoActiveFirstOption]=\"true\"\n (opened)=\"onAutocompleteOpened($event)\"\n (optionSelected)=\"onOptionSelected($event)\"\n elderInfiniteScroll elderElderInfiniteAutocomplete (closeToEnd)=\"onAutoCompleteCloseToEnd($event)\"\n>\n <!--\n <mat-option *ngIf=\"allowNull\" [value]=\"null\">{{nullDisplay | translate}}</mat-option>\n -->\n\n <mat-option disabled *ngIf=\"!(dataContext$ | async)\">\n <span class=\"mat-caption\">No DataSource / DataContext!</span>\n </mat-option>\n\n <ng-container *ngIf=\"dataContext$ | async as dc\">\n\n <mat-option disabled *ngIf=\"dc.isClosed\">\n <div fxLayout=\"row\" fxLayoutAlign=\"start center\" fxLayoutGap=\"8px\">\n <mat-icon color=\"warn\">warning</mat-icon>\n <span class=\"mat-caption\">DataContext Closed!</span>\n </div>\n </mat-option>\n\n <ng-container *ngIf=\"availableSuggestions$ | async as suggestions\">\n\n <mat-option *ngIf=\"suggestions.length === 0\" disabled>\n No Data.\n </mat-option>\n\n <ng-container *ngFor=\"let suggestion of suggestions\">\n <mat-option\n *ngIf=\"isOptionVisible(suggestion)\"\n [value]=\"toOptionValue(suggestion)\"\n [id]=\"getIdAsString(suggestion)\"\n [disabled]=\"!isOptionAvailable(suggestion)\"\n >\n <ng-container\n *ngTemplateOutlet=\"valueTemplate || simpleValueTemplate; context: {$implicit: suggestion}\">\n </ng-container>\n <!--\n <span class=\"mat-caption\">value: {{toOptionValue(suggestion)}}</span>\n -->\n </mat-option>\n </ng-container>\n </ng-container>\n </ng-container>\n\n <ng-container *ngIf=\"dataState$ | async as state\">\n\n <mat-option disabled *ngIf=\"!state.idle || state.loading\">\n <mat-progress-bar\n [value]=\"100\"\n [mode]=\"state.loading ? 'query' : 'determinate'\"\n [color]=\"state.error ? 'warn' : 'primary'\"\n ></mat-progress-bar>\n </mat-option>\n\n </ng-container>\n</mat-autocomplete>\n\n<ng-template #simpleValueTemplate let-value>\n <ng-container *ngIf=\"displayPropertyResolver$ | async as propertyResolver\">\n <span class=\"noselect\">{{propertyResolver(value)}}</span>\n </ng-container>\n</ng-template>\n" }]
151
164
  }], ctorParameters: function () { return [{ type: i0.NgZone }]; }, propDecorators: { valueTemplateQuery: [{
152
165
  type: ContentChild,
153
166
  args: [ElderSelectValueDirective, { read: TemplateRef, static: true }]
@@ -158,15 +171,17 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.1.0", ngImpor
158
171
  type: Input
159
172
  }], isOptionHiddenFn: [{
160
173
  type: Input
174
+ }], optionValueConverterFn: [{
175
+ type: Input
161
176
  }], optionSelected: [{
162
177
  type: Output
163
178
  }], enabled: [{
164
179
  type: Input
165
180
  }], valueTemplate: [{
166
181
  type: Input
167
- }], suggestionsDc: [{
182
+ }], dataContext: [{
168
183
  type: Input
169
184
  }], displayPropertyResolver: [{
170
185
  type: Input
171
186
  }] } });
172
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-autocomplete.component.js","sourceRoot":"","sources":["../../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/input/autocomplete/elder-autocomplete/elder-autocomplete.component.ts","../../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/input/autocomplete/elder-autocomplete/elder-autocomplete.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  OnInit,\n  ChangeDetectionStrategy,\n  Input,\n  TemplateRef,\n  ContentChild,\n  NgZone,\n  Output,\n  ViewChild,\n  AfterViewInit\n} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {BehaviorSubject, Observable, of, Subject} from 'rxjs';\nimport {IDataContext, isContinuableDataContext} from '../../../../common/data/data-context/data-context';\nimport {coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {filter, switchMap} from 'rxjs/operators';\nimport {runInZone} from '../../../../common/utils/ng-zone-utils';\nimport {DataContextStatus} from '../../../../common/data/data-context/data-context-status';\nimport {MatAutocomplete, MatAutocompleteSelectedEvent} from '@angular/material/autocomplete';\nimport {ElderSelectValueDirective} from '../../../select/elder-select-value.directive';\n\ntype TextResolverFn = (value: any) => string;\n\n@Component({\n  selector: 'elder-autocomplete',\n  exportAs: 'elderAutocomplete',\n  templateUrl: './elder-autocomplete.component.html',\n  styleUrls: ['./elder-autocomplete.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ElderAutocompleteComponent<T = any, TId = any> implements OnInit {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  /**\n   * Is this auto-complete enabled?\n   */\n  private _enabled = true;\n\n  /**\n   * The value template to customize the ui look.\n   */\n  @ContentChild(ElderSelectValueDirective, {read: TemplateRef, static: true})\n  public valueTemplateQuery: TemplateRef<any>;\n  private _valueTemplate: TemplateRef<any>;\n\n  /**\n   * The underlying Material Autocomplete\n   */\n  public readonly matAutocomplete$ = new BehaviorSubject<MatAutocomplete>(null);\n\n  @ViewChild('auto')\n  public set matAutocomplete(auto: MatAutocomplete) {\n    this.matAutocomplete$.next(auto);\n  }\n\n  @Input()\n  public isOptionDisabledFn: (option: T) => boolean;\n\n  @Input()\n  public isOptionHiddenFn: (option: T) => boolean;\n\n  /** Underlying data context. */\n  public readonly suggestionsDc$ = new BehaviorSubject<IDataContext<T>>(null);\n\n  public readonly triggerReload$ = new Subject<any>();\n\n  public readonly displayPropertyResolver$ = new BehaviorSubject<TextResolverFn>((value: T) => this.propertyStringValue(value, null));\n\n  public readonly dataState$: Observable<DataContextStatus>;\n\n  public readonly availableSuggestions$: Observable<T[]>;\n\n  @Output()\n  public readonly optionSelected = new Subject();\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private readonly zone: NgZone,\n  ) {\n    this.dataState$ = this.suggestionsDc$.pipe(\n      filter(dc => !!dc),\n      switchMap(dc => dc.status),\n      runInZone(this.zone)\n    );\n\n    this.availableSuggestions$ = this.suggestionsDc$.pipe(\n      switchMap(dc => dc ? dc.data : of([])),\n      runInZone(this.zone)\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\n  public set enabled(value: boolean) {\n    this._enabled = coerceBooleanProperty(value);\n  }\n\n  public get enabled(): boolean {\n    return this._enabled;\n  }\n\n  @Input()\n  public set valueTemplate(template: TemplateRef<any>) {\n    this._valueTemplate = template;\n  }\n\n  public get valueTemplate(): TemplateRef<any> {\n    return this._valueTemplate || this.valueTemplateQuery;\n  }\n\n  @Input()\n  public set suggestionsDc(data: IDataContext<T>) {\n    this.suggestionsDc$.next(data);\n  }\n\n  public get suggestionsDc(): IDataContext<T> {\n    return this.suggestionsDc$.getValue();\n  }\n\n  @Input()\n  public set displayPropertyResolver(resolver: TextResolverFn) {\n    this.displayPropertyResolver$.next(resolver);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public onOptionSelected(event: MatAutocompleteSelectedEvent): void {\n    this.optionSelected.next(event.option.value);\n  }\n\n  public onAutocompleteOpened(event: any): void {\n    this.logger.debug('Autocomplete opened:', event);\n    if (this.enabled) {\n      this.triggerReload$.next(null);\n    }\n  }\n\n  public onAutoCompleteCloseToEnd(event: UIEvent): void {\n    // this.logger.debug('Autocomplete is close to end!');\n    if (isContinuableDataContext(this.suggestionsDc)) {\n      if (!this.suggestionsDc.snapshot.isLoading) {\n        this.suggestionsDc.loadMore();\n      }\n    }\n  }\n\n  public getIdAsString(value: T): string {\n    return String(this.getId(value));\n  }\n\n  public getId(value: T): TId {\n    return this.suggestionsDc?.dataSource?.getId(value) || value;\n  }\n\n  public isOptionAvailable(option: T): boolean {\n    if (this.isOptionDisabledFn) {\n      return !this.isOptionDisabledFn(option);\n    } else {\n      return true;\n    }\n  }\n\n  public isOptionVisible(option: T): boolean {\n    if (this.isOptionHiddenFn) {\n      return !this.isOptionHiddenFn(option);\n    } else {\n      return true;\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * TODO Duplicate in elder-select\n   */\n  private propertyStringValue(value: T, property: string): string | null {\n    if (value === null || value === undefined) {\n      return null;\n    }\n\n    if (property) {\n      return (<any>value)[property];\n    }\n\n    return String(value);\n  }\n}\n","<mat-autocomplete\n  #auto=\"matAutocomplete\"\n  [autoActiveFirstOption]=\"true\"\n  (opened)=\"onAutocompleteOpened($event)\"\n  (optionSelected)=\"onOptionSelected($event)\"\n  elderInfiniteScroll elderElderInfiniteAutocomplete (closeToEnd)=\"onAutoCompleteCloseToEnd($event)\"\n>\n  <!--\n  <mat-option *ngIf=\"allowNull\" [value]=\"null\">{{nullDisplay | translate}}</mat-option>\n  -->\n\n  <mat-option disabled  *ngIf=\"!(suggestionsDc$ | async)\">\n    <span class=\"mat-caption\">No DataSource / DataContext!</span>\n  </mat-option>\n\n  <ng-container *ngIf=\"suggestionsDc$ | async as dc\">\n\n    <mat-option disabled  *ngIf=\"dc.isClosed\">\n      <div fxLayout=\"row\" fxLayoutAlign=\"start center\" fxLayoutGap=\"8px\">\n        <mat-icon color=\"warn\">warning</mat-icon>\n        <span class=\"mat-caption\">DataContext Closed!</span>\n      </div>\n    </mat-option>\n\n    <ng-container *ngIf=\"availableSuggestions$ | async as suggestions\">\n\n      <mat-option *ngIf=\"suggestions.length === 0\" disabled>\n        No Data.\n      </mat-option>\n\n      <ng-container *ngFor=\"let suggestion of suggestions\">\n        <mat-option\n          *ngIf=\"isOptionVisible(suggestion)\"\n          [value]=\"suggestion\"\n          [id]=\"getIdAsString(suggestion)\"\n          [disabled]=\"!isOptionAvailable(suggestion)\"\n        >\n          <ng-container\n            *ngTemplateOutlet=\"valueTemplate || simpleValueTemplate; context: {$implicit: suggestion}\">\n          </ng-container>\n        </mat-option>\n      </ng-container>\n\n\n\n    </ng-container>\n  </ng-container>\n\n  <ng-container *ngIf=\"dataState$ | async as state\">\n\n    <mat-option disabled *ngIf=\"!state.idle || state.loading\">\n      <mat-progress-bar\n        [value]=\"100\"\n        [mode]=\"state.loading ? 'query' : 'determinate'\"\n        [color]=\"state.error ? 'warn' : 'primary'\"\n      ></mat-progress-bar>\n    </mat-option>\n\n  </ng-container>\n</mat-autocomplete>\n\n<ng-template #simpleValueTemplate let-value>\n  <ng-container *ngIf=\"displayPropertyResolver$ | async as propertyResolver\">\n    <p class=\"noselect\">{{propertyResolver(value)}}</p>\n  </ng-container>\n</ng-template>\n"]}
187
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-autocomplete.component.js","sourceRoot":"","sources":["../../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/input/autocomplete/elder-autocomplete/elder-autocomplete.component.ts","../../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/input/autocomplete/elder-autocomplete/elder-autocomplete.component.html"],"names":[],"mappings":"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","sourcesContent":["import {ChangeDetectionStrategy, Component, ContentChild, Input, NgZone, OnInit, Output, TemplateRef, ViewChild} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {BehaviorSubject, Observable, of, Subject} from 'rxjs';\nimport {IDataContext, isContinuableDataContext} from '../../../../common/data/data-context/data-context';\nimport {coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {filter, switchMap} from 'rxjs/operators';\nimport {runInZone} from '../../../../common/utils/ng-zone-utils';\nimport {DataContextStatus} from '../../../../common/data/data-context/data-context-status';\nimport {MatAutocomplete, MatAutocompleteSelectedEvent} from '@angular/material/autocomplete';\nimport {ElderSelectValueDirective} from '../../../select/elder-select-value.directive';\nimport {ElderAutocompleteOptionSelected} from './elder-autocomplete-option-selected';\n\ntype TextResolverFn = (value: any) => string;\n\n@Component({\n  selector: 'elder-autocomplete',\n  exportAs: 'elderAutocomplete',\n  templateUrl: './elder-autocomplete.component.html',\n  styleUrls: ['./elder-autocomplete.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ElderAutocompleteComponent<T = any, TId = any> {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  /**\n   * Is this auto-complete enabled?\n   */\n  private _enabled = true;\n\n  /**\n   * The value template to customize the ui look.\n   */\n  @ContentChild(ElderSelectValueDirective, {read: TemplateRef, static: true})\n  public valueTemplateQuery: TemplateRef<any>;\n  private _valueTemplate: TemplateRef<any>;\n\n  /**\n   * The underlying Material Autocomplete\n   */\n  public readonly matAutocomplete$ = new BehaviorSubject<MatAutocomplete>(null);\n\n  @ViewChild('auto')\n  public set matAutocomplete(auto: MatAutocomplete) {\n    this.matAutocomplete$.next(auto);\n  }\n\n  @Input()\n  public isOptionDisabledFn: (option: T) => boolean;\n\n  @Input()\n  public isOptionHiddenFn: (option: T) => boolean;\n\n  @Input()\n  public optionValueConverterFn: (option: T) => any;\n\n  /**\n   * The DataContext which holds the autocomplete suggestions.\n   */\n  public readonly dataContext$ = new BehaviorSubject<IDataContext<T>>(null);\n\n  public readonly triggerReload$ = new Subject<any>();\n\n  public readonly displayPropertyResolver$ = new BehaviorSubject<TextResolverFn>((value: T) => this.propertyStringValue(value, null));\n\n  public readonly dataState$: Observable<DataContextStatus>;\n\n  public readonly availableSuggestions$: Observable<T[]>;\n\n  @Output()\n  public readonly optionSelected = new Subject<ElderAutocompleteOptionSelected>();\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private readonly zone: NgZone,\n  ) {\n    this.dataState$ = this.dataContext$.pipe(\n      filter(dc => !!dc),\n      switchMap(dc => dc.status),\n      runInZone(this.zone)\n    );\n\n    this.availableSuggestions$ = this.dataContext$.pipe(\n      switchMap(dc => dc ? dc.data : of([])),\n      runInZone(this.zone)\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\n  public set enabled(value: boolean) {\n    this._enabled = coerceBooleanProperty(value);\n  }\n\n  public get enabled(): boolean {\n    return this._enabled;\n  }\n\n  @Input()\n  public set valueTemplate(template: TemplateRef<any>) {\n    this._valueTemplate = template;\n  }\n\n  public get valueTemplate(): TemplateRef<any> {\n    return this._valueTemplate || this.valueTemplateQuery;\n  }\n\n  @Input()\n  public set dataContext(data: IDataContext<T>) {\n    this.dataContext$.next(data);\n  }\n\n  public get dataContext(): IDataContext<T> {\n    return this.dataContext$.getValue();\n  }\n\n  @Input()\n  public set displayPropertyResolver(resolver: TextResolverFn) {\n    this.displayPropertyResolver$.next(resolver);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public onOptionSelected(event: MatAutocompleteSelectedEvent): void {\n    const option = event.option;\n    const id = option.id as unknown as TId;\n    this.optionSelected.next(\n      new ElderAutocompleteOptionSelected(\n        this,\n        id,\n        this.getEntityById(id),\n        option.value\n      )\n    );\n  }\n\n  private getEntityById(id: TId): T | undefined {\n    const dc = this.dataContext;\n    if(dc){\n      return dc.snapshot.data.find(e => this.getId(e) == id);\n    }\n    return undefined;\n  }\n\n  public onAutocompleteOpened(event: any): void {\n    this.logger.debug('Autocomplete opened:', event);\n    if (this.enabled) {\n      this.triggerReload$.next(null);\n    }\n  }\n\n  public onAutoCompleteCloseToEnd(event: UIEvent): void {\n    // this.logger.debug('Autocomplete is close to end!');\n    if (isContinuableDataContext(this.dataContext)) {\n      if (!this.dataContext.snapshot.isLoading) {\n        this.dataContext.loadMore();\n      }\n    }\n  }\n\n  public getIdAsString(value: T): string {\n    return String(this.getId(value));\n  }\n\n  public getId(value: T): TId {\n    return this.dataContext?.dataSource?.getId(value) || value;\n  }\n\n  public isOptionAvailable(option: T): boolean {\n    if (this.isOptionDisabledFn) {\n      return !this.isOptionDisabledFn(option);\n    } else {\n      return true;\n    }\n  }\n\n  public isOptionVisible(option: T): boolean {\n    if (this.isOptionHiddenFn) {\n      return !this.isOptionHiddenFn(option);\n    } else {\n      return true;\n    }\n  }\n\n  public toOptionValue(option: T): any {\n    if (this.optionValueConverterFn) {\n      return this.optionValueConverterFn(option);\n    } else {\n      return option;\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * TODO Duplicate in elder-select\n   */\n  private propertyStringValue(value: T, property: string): string | null {\n    if (value === null || value === undefined) {\n      return null;\n    }\n\n    if (property) {\n      return (<any>value)[property];\n    }\n\n    return String(value);\n  }\n}\n","<mat-autocomplete\n  #auto=\"matAutocomplete\"\n  [autoActiveFirstOption]=\"true\"\n  (opened)=\"onAutocompleteOpened($event)\"\n  (optionSelected)=\"onOptionSelected($event)\"\n  elderInfiniteScroll elderElderInfiniteAutocomplete (closeToEnd)=\"onAutoCompleteCloseToEnd($event)\"\n>\n  <!--\n  <mat-option *ngIf=\"allowNull\" [value]=\"null\">{{nullDisplay | translate}}</mat-option>\n  -->\n\n  <mat-option disabled  *ngIf=\"!(dataContext$ | async)\">\n    <span class=\"mat-caption\">No DataSource / DataContext!</span>\n  </mat-option>\n\n  <ng-container *ngIf=\"dataContext$ | async as dc\">\n\n    <mat-option disabled  *ngIf=\"dc.isClosed\">\n      <div fxLayout=\"row\" fxLayoutAlign=\"start center\" fxLayoutGap=\"8px\">\n        <mat-icon color=\"warn\">warning</mat-icon>\n        <span class=\"mat-caption\">DataContext Closed!</span>\n      </div>\n    </mat-option>\n\n    <ng-container *ngIf=\"availableSuggestions$ | async as suggestions\">\n\n      <mat-option *ngIf=\"suggestions.length === 0\" disabled>\n        No Data.\n      </mat-option>\n\n      <ng-container *ngFor=\"let suggestion of suggestions\">\n        <mat-option\n          *ngIf=\"isOptionVisible(suggestion)\"\n          [value]=\"toOptionValue(suggestion)\"\n          [id]=\"getIdAsString(suggestion)\"\n          [disabled]=\"!isOptionAvailable(suggestion)\"\n        >\n          <ng-container\n            *ngTemplateOutlet=\"valueTemplate || simpleValueTemplate; context: {$implicit: suggestion}\">\n          </ng-container>\n          <!--\n          <span class=\"mat-caption\">value: {{toOptionValue(suggestion)}}</span>\n          -->\n        </mat-option>\n      </ng-container>\n    </ng-container>\n  </ng-container>\n\n  <ng-container *ngIf=\"dataState$ | async as state\">\n\n    <mat-option disabled *ngIf=\"!state.idle || state.loading\">\n      <mat-progress-bar\n        [value]=\"100\"\n        [mode]=\"state.loading ? 'query' : 'determinate'\"\n        [color]=\"state.error ? 'warn' : 'primary'\"\n      ></mat-progress-bar>\n    </mat-option>\n\n  </ng-container>\n</mat-autocomplete>\n\n<ng-template #simpleValueTemplate let-value>\n  <ng-container *ngIf=\"displayPropertyResolver$ | async as propertyResolver\">\n    <span class=\"noselect\">{{propertyResolver(value)}}</span>\n  </ng-container>\n</ng-template>\n"]}
@@ -0,0 +1,197 @@
1
+ import { Directive, HostListener, Input } from '@angular/core';
2
+ import { LoggerFactory } from '@elderbyte/ts-logger';
3
+ import { merge, Subject } from 'rxjs';
4
+ import { debounceTime, filter, map, skipWhile, takeUntil } from 'rxjs/operators';
5
+ import { Filter } from '../../../common/data/filters/filter';
6
+ import { WordPositionFinder } from './many/word-position-finder';
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "@angular/material/autocomplete";
9
+ export class ElderAutocompleteManyDirective {
10
+ /***************************************************************************
11
+ * *
12
+ * Constructor *
13
+ * *
14
+ **************************************************************************/
15
+ constructor(autoCompleteTrigger, inputRef) {
16
+ this.autoCompleteTrigger = autoCompleteTrigger;
17
+ this.inputRef = inputRef;
18
+ /***************************************************************************
19
+ * *
20
+ * Fields *
21
+ * *
22
+ **************************************************************************/
23
+ this.queryFilter = 'query';
24
+ this.filters = [];
25
+ this.sorts = [];
26
+ this.textPropertyResolverFn = (option) => {
27
+ if (!option) {
28
+ return option;
29
+ }
30
+ else if (typeof option === 'string') {
31
+ return option;
32
+ }
33
+ else if (this.textProperty) {
34
+ return option[this.textProperty];
35
+ }
36
+ else {
37
+ return undefined;
38
+ }
39
+ };
40
+ this.inputKeyup$ = new Subject();
41
+ this.ignoreKeys = ['ArrowRight', 'ArrowUp', 'ArrowDown', 'ArrowLeft'];
42
+ this.log = LoggerFactory.getLogger(this.constructor.name);
43
+ this.destroy$ = new Subject();
44
+ this.destroyAutocompleteBinding$ = new Subject();
45
+ this.wordFinder = new WordPositionFinder();
46
+ }
47
+ /***************************************************************************
48
+ * *
49
+ * Properties *
50
+ * *
51
+ **************************************************************************/
52
+ set autocomplete(auto) {
53
+ this._elderAutocomplete = auto;
54
+ this.destroyAutocompleteBinding$.next();
55
+ if (auto) {
56
+ auto.matAutocomplete$
57
+ .pipe(takeUntil(this.destroyAutocompleteBinding$), filter(matAuto => !!matAuto))
58
+ .subscribe(matAuto => this.autoCompleteTrigger.autocomplete = matAuto);
59
+ auto.optionSelected.pipe(takeUntil(this.destroyAutocompleteBinding$)).subscribe(event => this.onOptionSelected(event));
60
+ }
61
+ }
62
+ get autocomplete() {
63
+ return this._elderAutocomplete;
64
+ }
65
+ get cursorPosition() {
66
+ if (this.inputRef?.nativeElement) {
67
+ return this.inputRef.nativeElement.selectionStart;
68
+ }
69
+ return null;
70
+ }
71
+ set cursorPosition(position) {
72
+ if (this.inputRef?.nativeElement) {
73
+ this.inputRef.nativeElement.selectionStart = position;
74
+ this.inputRef.nativeElement.selectionEnd = position;
75
+ }
76
+ }
77
+ get dataContext() {
78
+ return this._elderAutocomplete.dataContext;
79
+ }
80
+ /***************************************************************************
81
+ * *
82
+ * Host Listener *
83
+ * *
84
+ **************************************************************************/
85
+ onKeyUp(event) {
86
+ if (this.ignoreKeys.some(k => k === event.key)) {
87
+ return;
88
+ }
89
+ this.inputKeyup$.next(event);
90
+ }
91
+ /***************************************************************************
92
+ * *
93
+ * Life Cycle *
94
+ * *
95
+ **************************************************************************/
96
+ ngOnInit() {
97
+ this._elderAutocomplete.displayPropertyResolver = this.textPropertyResolverFn;
98
+ this._elderAutocomplete.optionValueConverterFn = (option) => {
99
+ const suggestion = this.resolveSuggestionText(option);
100
+ return this.insertSuggestion(suggestion);
101
+ };
102
+ merge(this.inputKeyup$.pipe(skipWhile(e => !this._elderAutocomplete.enabled), debounceTime(150), map(e => e?.target?.value), map(value => this.extractWordAtCursor(value))), this.autocomplete.triggerReload$).pipe(takeUntil(this.destroy$), filter(value => !value || typeof (value) === 'string' || typeof (value) === 'number')).subscribe(value => this.updateSuggestions(value));
103
+ }
104
+ ngOnDestroy() {
105
+ this.destroy$.next();
106
+ this.destroy$.complete();
107
+ this.destroyAutocompleteBinding$.next();
108
+ this.destroyAutocompleteBinding$.complete();
109
+ }
110
+ /***************************************************************************
111
+ * *
112
+ * Private methods *
113
+ * *
114
+ **************************************************************************/
115
+ extractWordAtCursor(value) {
116
+ if (this.cursorPosition) {
117
+ const cursor = this.cursorPosition;
118
+ return this.extractWordFrom(value, cursor);
119
+ }
120
+ else {
121
+ return null;
122
+ }
123
+ }
124
+ rememberCursorPosition() {
125
+ this.cursorBeforeAutocomplete = this.cursorPosition;
126
+ }
127
+ extractWordFrom(text, position) {
128
+ const word = this.wordFinder.findWord(text, position);
129
+ return word.value;
130
+ }
131
+ updateSuggestions(value) {
132
+ const filters = [...this.filters, new Filter(this.queryFilter, value)];
133
+ const sorts = this.sorts;
134
+ if (this.dataContext) {
135
+ try {
136
+ this.rememberCursorPosition();
137
+ this.dataContext.start(sorts, filters);
138
+ }
139
+ catch (err) {
140
+ this.log.error('Failed to start DataContext!', err);
141
+ }
142
+ }
143
+ }
144
+ insertSuggestion(suggestion) {
145
+ const currentValue = this.currentInputValue();
146
+ if (!suggestion) {
147
+ return currentValue;
148
+ }
149
+ if (this.cursorPosition !== undefined) {
150
+ const cursor = this.cursorPosition;
151
+ const replacement = this.wordFinder.replaceWord(currentValue, cursor, suggestion);
152
+ return replacement.newText;
153
+ }
154
+ else {
155
+ this.log.warn('Insert not possible as no cursor-position was available!');
156
+ return currentValue;
157
+ }
158
+ }
159
+ currentInputValue() {
160
+ const input = this.inputRef?.nativeElement;
161
+ return input?.value ?? '';
162
+ }
163
+ resolveSuggestionText(option) {
164
+ return this.textPropertyResolverFn(option);
165
+ }
166
+ onOptionSelected(event) {
167
+ const wordAtCursor = this.wordFinder.findWord(this.currentInputValue(), this.cursorBeforeAutocomplete);
168
+ const autocompletedPhrase = this.resolveSuggestionText(event.entity);
169
+ // The wordAtCursor might not be the full autocompleted phrase (if there are multiple words)
170
+ this.cursorPosition = wordAtCursor.start + autocompletedPhrase.length;
171
+ }
172
+ }
173
+ ElderAutocompleteManyDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.1.0", ngImport: i0, type: ElderAutocompleteManyDirective, deps: [{ token: i1.MatAutocompleteTrigger }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Directive });
174
+ ElderAutocompleteManyDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "14.1.0", type: ElderAutocompleteManyDirective, selector: "[elderAutocompleteMany]", inputs: { queryFilter: "queryFilter", filters: "filters", sorts: "sorts", textProperty: "textProperty", textPropertyResolverFn: "textPropertyResolverFn", autocomplete: ["elderAutocompleteMany", "autocomplete"] }, host: { listeners: { "keyup": "onKeyUp($event)" } }, ngImport: i0 });
175
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.1.0", ngImport: i0, type: ElderAutocompleteManyDirective, decorators: [{
176
+ type: Directive,
177
+ args: [{
178
+ selector: '[elderAutocompleteMany]'
179
+ }]
180
+ }], ctorParameters: function () { return [{ type: i1.MatAutocompleteTrigger }, { type: i0.ElementRef }]; }, propDecorators: { queryFilter: [{
181
+ type: Input
182
+ }], filters: [{
183
+ type: Input
184
+ }], sorts: [{
185
+ type: Input
186
+ }], textProperty: [{
187
+ type: Input
188
+ }], textPropertyResolverFn: [{
189
+ type: Input
190
+ }], autocomplete: [{
191
+ type: Input,
192
+ args: ['elderAutocompleteMany']
193
+ }], onKeyUp: [{
194
+ type: HostListener,
195
+ args: ['keyup', ['$event']]
196
+ }] } });
197
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-autocomplete-many.directive.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/input/autocomplete/elder-autocomplete-many.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {Directive, ElementRef, HostListener, Input, OnDestroy, OnInit} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {EMPTY, merge, Subject} from 'rxjs';\nimport {debounceTime, filter, map, skipWhile, takeUntil} from 'rxjs/operators';\nimport {ElderAutocompleteComponent} from './elder-autocomplete/elder-autocomplete.component';\nimport {MatAutocompleteTrigger} from '@angular/material/autocomplete';\nimport {Filter} from '../../../common/data/filters/filter';\nimport {Sort} from '../../../common/data/sort';\nimport {IDataContext} from '../../../common/data/data-context/data-context';\nimport {TextResolverFn} from '../../select/elder-select-base';\nimport {WordPositionFinder} from './many/word-position-finder';\nimport {ElderAutocompleteOptionSelected} from './elder-autocomplete/elder-autocomplete-option-selected';\n\n@Directive({\n  selector: '[elderAutocompleteMany]'\n})\nexport class ElderAutocompleteManyDirective<T> implements OnInit, OnDestroy {\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\n  public queryFilter = 'query';\n\n  @Input()\n  public filters: Filter[] = [];\n\n  @Input()\n  public sorts: Sort[] = [];\n\n  /**\n   * The property on a suggestion object which is used\n   * as the display and insert value.\n   */\n  @Input()\n  public textProperty: string;\n\n  @Input()\n  public textPropertyResolverFn: TextResolverFn = (option: any) => {\n    if (!option) {\n      return option;\n    } else if (typeof option === 'string') {\n      return option as string;\n    } else if (this.textProperty) {\n      return option[this.textProperty] as string;\n    } else {\n      return undefined;\n    }\n  };\n\n  public readonly inputKeyup$ = new Subject<KeyboardEvent>();\n\n  private readonly ignoreKeys = ['ArrowRight', 'ArrowUp', 'ArrowDown', 'ArrowLeft'];\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n  private readonly wordFinder: WordPositionFinder;\n\n  private _elderAutocomplete: ElderAutocompleteComponent;\n\n  private readonly destroy$ = new Subject<void>();\n  private readonly destroyAutocompleteBinding$ = new Subject<void>();\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private readonly autoCompleteTrigger: MatAutocompleteTrigger,\n    private readonly inputRef: ElementRef<HTMLInputElement>\n  ) {\n    this.wordFinder = new WordPositionFinder();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input('elderAutocompleteMany')\n  public set autocomplete(auto: ElderAutocompleteComponent) {\n    this._elderAutocomplete = auto;\n    this.destroyAutocompleteBinding$.next();\n    if (auto) {\n      auto.matAutocomplete$\n        .pipe(\n          takeUntil(this.destroyAutocompleteBinding$),\n          filter(matAuto => !!matAuto)\n        )\n        .subscribe(matAuto => this.autoCompleteTrigger.autocomplete = matAuto);\n\n      auto.optionSelected.pipe(\n        takeUntil(this.destroyAutocompleteBinding$)\n      ).subscribe(event => this.onOptionSelected(event))\n    }\n  }\n\n  public get autocomplete(): ElderAutocompleteComponent {\n    return this._elderAutocomplete;\n  }\n\n  private get cursorPosition(): number | null {\n    if (this.inputRef?.nativeElement) {\n      return this.inputRef.nativeElement.selectionStart;\n    }\n    return null;\n  }\n\n  private set cursorPosition(position: number) {\n    if (this.inputRef?.nativeElement) {\n      this.inputRef.nativeElement.selectionStart = position;\n      this.inputRef.nativeElement.selectionEnd = position;\n    }\n  }\n\n  private get dataContext(): IDataContext<any> {\n    return this._elderAutocomplete.dataContext;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Host Listener                                                           *\n   *                                                                         *\n   **************************************************************************/\n\n  @HostListener('keyup', ['$event'])\n  public onKeyUp(event: KeyboardEvent) {\n    if (this.ignoreKeys.some(k => k === event.key)) {\n      return;\n    }\n    this.inputKeyup$.next(event);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n    this._elderAutocomplete.displayPropertyResolver = this.textPropertyResolverFn;\n    this._elderAutocomplete.optionValueConverterFn = (option: T) => {\n      const suggestion = this.resolveSuggestionText(option);\n      return this.insertSuggestion(suggestion);\n    };\n\n    merge(\n      this.inputKeyup$.pipe(\n        skipWhile(e => !this._elderAutocomplete.enabled),\n        debounceTime(150),\n        map(e => (e?.target as HTMLInputElement)?.value),\n        map(value => this.extractWordAtCursor(value))\n      ),\n      this.autocomplete.triggerReload$\n    ).pipe(\n      takeUntil(this.destroy$),\n      filter(value => !value || typeof (value) === 'string' || typeof (value) === 'number')\n    ).subscribe(\n      value => this.updateSuggestions(value)\n    );\n  }\n\n  public ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n    this.destroyAutocompleteBinding$.next();\n    this.destroyAutocompleteBinding$.complete();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n\n  private extractWordAtCursor(value: string): string {\n    if (this.cursorPosition) {\n      const cursor = this.cursorPosition;\n      return this.extractWordFrom(value, cursor);\n    } else {\n      return null;\n    }\n  }\n\n  private cursorBeforeAutocomplete: number;\n  private rememberCursorPosition(): void {\n    this.cursorBeforeAutocomplete = this.cursorPosition;\n  }\n\n  private extractWordFrom(text: string, position: number): string {\n    const word = this.wordFinder.findWord(text, position);\n    return word.value;\n  }\n\n  private updateSuggestions(value: string | number | undefined | null): void {\n    const filters = [...this.filters, new Filter(this.queryFilter, <any>value)];\n    const sorts = this.sorts;\n    if (this.dataContext) {\n      try {\n        this.rememberCursorPosition();\n        this.dataContext.start(sorts, filters);\n      } catch (err) {\n        this.log.error('Failed to start DataContext!', err);\n      }\n    }\n  }\n\n  private insertSuggestion(suggestion: string): string {\n    const currentValue = this.currentInputValue();\n    if(!suggestion){\n      return currentValue;\n    }\n    if (this.cursorPosition !== undefined) {\n      const cursor = this.cursorPosition;\n      const replacement = this.wordFinder.replaceWord(currentValue, cursor, suggestion);\n      return replacement.newText;\n    } else {\n      this.log.warn('Insert not possible as no cursor-position was available!');\n      return currentValue;\n    }\n  }\n\n  private currentInputValue(): string {\n    const input = this.inputRef?.nativeElement;\n    return input?.value ?? '';\n  }\n\n  private resolveSuggestionText(option: T): string {\n    return this.textPropertyResolverFn(option);\n  }\n\n  private onOptionSelected(\n    event: ElderAutocompleteOptionSelected\n  ): void {\n    const wordAtCursor = this.wordFinder.findWord(this.currentInputValue(), this.cursorBeforeAutocomplete);\n    const autocompletedPhrase = this.resolveSuggestionText(event.entity);\n    // The wordAtCursor might not be the full autocompleted phrase (if there are multiple words)\n    this.cursorPosition = wordAtCursor.start + autocompletedPhrase.length;\n  }\n}\n"]}
@@ -2,7 +2,7 @@ import { Directive, HostListener, Input } from '@angular/core';
2
2
  import { LoggerFactory } from '@elderbyte/ts-logger';
3
3
  import { merge, Subject } from 'rxjs';
4
4
  import { debounceTime, filter, map, skipWhile, takeUntil } from 'rxjs/operators';
5
- import { Filter } from '../../../common/data/filter';
5
+ import { Filter } from '../../../common/data/filters/filter';
6
6
  import * as i0 from "@angular/core";
7
7
  import * as i1 from "@angular/material/autocomplete";
8
8
  export class ElderAutocompleteDirective {
@@ -27,12 +27,6 @@ export class ElderAutocompleteDirective {
27
27
  this.destroyMatAutoBinding$ = new Subject();
28
28
  this.destroy$ = new Subject();
29
29
  }
30
- ngOnDestroy() {
31
- this.destroy$.next();
32
- this.destroy$.complete();
33
- this.destroyMatAutoBinding$.next();
34
- this.destroyMatAutoBinding$.complete();
35
- }
36
30
  /***************************************************************************
37
31
  * *
38
32
  * Host Listener *
@@ -52,6 +46,12 @@ export class ElderAutocompleteDirective {
52
46
  ngOnInit() {
53
47
  merge(this.inputKeyup$.pipe(skipWhile(e => !this._elderAutocomplete.enabled), debounceTime(150), map(e => e?.target?.value)), this.autocomplete.triggerReload$).pipe(takeUntil(this.destroy$), filter(value => !value || typeof (value) === 'string' || typeof (value) === 'number')).subscribe(value => this.updateSuggestions(value));
54
48
  }
49
+ ngOnDestroy() {
50
+ this.destroy$.next();
51
+ this.destroy$.complete();
52
+ this.destroyMatAutoBinding$.next();
53
+ this.destroyMatAutoBinding$.complete();
54
+ }
55
55
  /***************************************************************************
56
56
  * *
57
57
  * Properties *
@@ -70,7 +70,7 @@ export class ElderAutocompleteDirective {
70
70
  return this._elderAutocomplete;
71
71
  }
72
72
  get dataContext() {
73
- return this._elderAutocomplete.suggestionsDc;
73
+ return this._elderAutocomplete.dataContext;
74
74
  }
75
75
  /***************************************************************************
76
76
  * *
@@ -111,4 +111,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.1.0", ngImpor
111
111
  type: Input,
112
112
  args: ['elderAutocomplete']
113
113
  }] } });
114
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-autocomplete.directive.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/input/autocomplete/elder-autocomplete.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {AfterViewInit, Directive, HostListener, Input, OnDestroy, OnInit} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {MatAutocompleteTrigger} from '@angular/material/autocomplete';\nimport {ElderAutocompleteComponent} from './elder-autocomplete/elder-autocomplete.component';\nimport {merge, Subject} from 'rxjs';\nimport {debounceTime, filter, map, skipWhile, takeUntil} from 'rxjs/operators';\nimport {Filter} from '../../../common/data/filter';\nimport {IDataContext} from '../../../common/data/data-context/data-context';\nimport {Sort} from '../../../common/data/sort';\n\n@Directive({\n  selector: '[elderAutocomplete]'\n})\nexport class ElderAutocompleteDirective implements OnInit, OnDestroy {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly ignoreKeys = ['ArrowRight', 'ArrowUp', 'ArrowDown', 'ArrowLeft'];\n  public readonly inputKeyup$ = new Subject<KeyboardEvent>();\n\n  @Input()\n  public queryFilter = 'query';\n\n  @Input()\n  public filters: Filter[] = [];\n\n  @Input()\n  public sorts: Sort[] = [];\n\n  private _elderAutocomplete: ElderAutocompleteComponent;\n\n  private readonly destroyMatAutoBinding$ = new Subject<void>();\n  private readonly destroy$ = new Subject<void>();\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private readonly autoCompleteTrigger: MatAutocompleteTrigger\n  ) {\n  }\n\n  public ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n    this.destroyMatAutoBinding$.next();\n    this.destroyMatAutoBinding$.complete();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Host Listener                                                           *\n   *                                                                         *\n   **************************************************************************/\n\n  @HostListener('keyup', ['$event'])\n  public onKeyUp(event: KeyboardEvent) {\n    if (this.ignoreKeys.some(k => k === event.key)) {\n      return;\n    }\n    this.inputKeyup$.next(event);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n    merge(\n      this.inputKeyup$.pipe(\n        skipWhile(e => !this._elderAutocomplete.enabled),\n        debounceTime(150),\n        map(e => (e?.target as HTMLInputElement)?.value),\n      ),\n      this.autocomplete.triggerReload$\n    ).pipe(\n      takeUntil(this.destroy$),\n      filter(value => !value || typeof (value) === 'string' || typeof (value) === 'number')\n    ).subscribe(\n      value => this.updateSuggestions(value)\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input('elderAutocomplete')\n  public set autocomplete(auto: ElderAutocompleteComponent) {\n    this._elderAutocomplete = auto;\n    this.destroyMatAutoBinding$.next();\n    if (auto) {\n      auto.matAutocomplete$\n        .pipe(\n          takeUntil(this.destroyMatAutoBinding$),\n          filter(matAuto => !!matAuto)\n        )\n        .subscribe(\n          matAuto => this.autoCompleteTrigger.autocomplete = matAuto\n        );\n    }\n  }\n\n  public get autocomplete(): ElderAutocompleteComponent {\n    return this._elderAutocomplete;\n  }\n\n  public get dataContext(): IDataContext<any> {\n    return this._elderAutocomplete.suggestionsDc;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private updateSuggestions(value: string | number | undefined | null): void {\n    const filters = [...this.filters, new Filter(this.queryFilter, <any>value)];\n    const sorts = this.sorts;\n    if (this.dataContext) {\n      try {\n        this.dataContext.start(sorts, filters);\n      } catch (err) {\n        this.logger.error('Failed to start DC!', err);\n        // this.updateState(ElderSelectComponentState.error(err));\n      }\n    }\n  }\n}\n"]}
114
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-autocomplete.directive.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/input/autocomplete/elder-autocomplete.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {AfterViewInit, Directive, HostListener, Input, OnDestroy, OnInit} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {MatAutocompleteTrigger} from '@angular/material/autocomplete';\nimport {ElderAutocompleteComponent} from './elder-autocomplete/elder-autocomplete.component';\nimport {merge, Subject} from 'rxjs';\nimport {debounceTime, filter, map, skipWhile, takeUntil} from 'rxjs/operators';\nimport {Filter} from '../../../common/data/filters/filter';\nimport {IDataContext} from '../../../common/data/data-context/data-context';\nimport {Sort} from '../../../common/data/sort';\n\n@Directive({\n  selector: '[elderAutocomplete]'\n})\nexport class ElderAutocompleteDirective implements OnInit, OnDestroy {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly ignoreKeys = ['ArrowRight', 'ArrowUp', 'ArrowDown', 'ArrowLeft'];\n  public readonly inputKeyup$ = new Subject<KeyboardEvent>();\n\n  @Input()\n  public queryFilter = 'query';\n\n  @Input()\n  public filters: Filter[] = [];\n\n  @Input()\n  public sorts: Sort[] = [];\n\n  private _elderAutocomplete: ElderAutocompleteComponent;\n\n  private readonly destroyMatAutoBinding$ = new Subject<void>();\n  private readonly destroy$ = new Subject<void>();\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private readonly autoCompleteTrigger: MatAutocompleteTrigger\n  ) {\n  }\n\n\n\n  /***************************************************************************\n   *                                                                         *\n   * Host Listener                                                           *\n   *                                                                         *\n   **************************************************************************/\n\n  @HostListener('keyup', ['$event'])\n  public onKeyUp(event: KeyboardEvent) {\n    if (this.ignoreKeys.some(k => k === event.key)) {\n      return;\n    }\n    this.inputKeyup$.next(event);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n    merge(\n      this.inputKeyup$.pipe(\n        skipWhile(e => !this._elderAutocomplete.enabled),\n        debounceTime(150),\n        map(e => (e?.target as HTMLInputElement)?.value),\n      ),\n      this.autocomplete.triggerReload$\n    ).pipe(\n      takeUntil(this.destroy$),\n      filter(value => !value || typeof (value) === 'string' || typeof (value) === 'number')\n    ).subscribe(\n      value => this.updateSuggestions(value)\n    );\n  }\n\n  public ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n    this.destroyMatAutoBinding$.next();\n    this.destroyMatAutoBinding$.complete();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input('elderAutocomplete')\n  public set autocomplete(auto: ElderAutocompleteComponent) {\n    this._elderAutocomplete = auto;\n    this.destroyMatAutoBinding$.next();\n    if (auto) {\n      auto.matAutocomplete$\n        .pipe(\n          takeUntil(this.destroyMatAutoBinding$),\n          filter(matAuto => !!matAuto)\n        )\n        .subscribe(\n          matAuto => this.autoCompleteTrigger.autocomplete = matAuto\n        );\n    }\n  }\n\n  public get autocomplete(): ElderAutocompleteComponent {\n    return this._elderAutocomplete;\n  }\n\n  public get dataContext(): IDataContext<any> {\n    return this._elderAutocomplete.dataContext;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private updateSuggestions(value: string | number | undefined | null): void {\n    const filters = [...this.filters, new Filter(this.queryFilter, <any>value)];\n    const sorts = this.sorts;\n    if (this.dataContext) {\n      try {\n        this.dataContext.start(sorts, filters);\n      } catch (err) {\n        this.logger.error('Failed to start DC!', err);\n        // this.updateState(ElderSelectComponentState.error(err));\n      }\n    }\n  }\n}\n"]}