@elderbyte/ngx-starter 13.6.0 → 13.7.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.
Files changed (23) hide show
  1. package/esm2020/lib/common/utils/collection-util.mjs +4 -1
  2. package/esm2020/lib/components/select/auto/elder-auto-select-first.directive.mjs +77 -0
  3. package/esm2020/lib/components/select/auto/elder-select-first-util.mjs +62 -0
  4. package/esm2020/lib/components/select/elder-select/elder-select.component.mjs +14 -9
  5. package/esm2020/lib/components/select/elder-select-on-tab.directive.mjs +104 -0
  6. package/esm2020/lib/components/select/elder-select.module.mjs +11 -11
  7. package/esm2020/lib/components/select/multi/elder-multi-select-base.mjs +103 -39
  8. package/esm2020/lib/components/select/multi/elder-multi-select-chips/elder-multi-select-chips.component.mjs +13 -13
  9. package/fesm2015/elderbyte-ngx-starter.mjs +226 -144
  10. package/fesm2015/elderbyte-ngx-starter.mjs.map +1 -1
  11. package/fesm2020/elderbyte-ngx-starter.mjs +226 -144
  12. package/fesm2020/elderbyte-ngx-starter.mjs.map +1 -1
  13. package/lib/common/utils/collection-util.d.ts +1 -0
  14. package/lib/components/select/{auto-select-first.directive.d.ts → auto/elder-auto-select-first.directive.d.ts} +4 -13
  15. package/lib/components/select/auto/elder-select-first-util.d.ts +22 -0
  16. package/lib/components/select/elder-select/elder-select.component.d.ts +11 -1
  17. package/lib/components/select/{select-on-tab.directive.d.ts → elder-select-on-tab.directive.d.ts} +3 -3
  18. package/lib/components/select/elder-select.module.d.ts +5 -5
  19. package/lib/components/select/multi/elder-multi-select-base.d.ts +39 -15
  20. package/lib/components/select/multi/elder-multi-select-chips/elder-multi-select-chips.component.d.ts +11 -11
  21. package/package.json +1 -1
  22. package/esm2020/lib/components/select/auto-select-first.directive.mjs +0 -126
  23. package/esm2020/lib/components/select/select-on-tab.directive.mjs +0 -104
@@ -0,0 +1,104 @@
1
+ import { Directive, HostListener, Inject, SkipSelf } from '@angular/core';
2
+ import { Subject } from 'rxjs/internal/Subject';
3
+ import { delay, map, takeUntil } from 'rxjs/operators';
4
+ import { merge } from 'rxjs';
5
+ import { LoggerFactory } from '@elderbyte/ts-logger';
6
+ import { ELDER_SELECT_BASE } from './elder-select-base';
7
+ import { isElderEntityValueAccessor } from './elder-entity-value-accessor';
8
+ import * as i0 from "@angular/core";
9
+ import * as i1 from "@angular/material/autocomplete";
10
+ import * as i2 from "./elder-select-base";
11
+ export class ElderSelectOnTabDirective {
12
+ /***************************************************************************
13
+ * *
14
+ * Constructor *
15
+ * *
16
+ **************************************************************************/
17
+ constructor(autoTrigger, elderSelectBase) {
18
+ this.autoTrigger = autoTrigger;
19
+ /***************************************************************************
20
+ * *
21
+ * Fields *
22
+ * *
23
+ **************************************************************************/
24
+ this.logger = LoggerFactory.getLogger(this.constructor.name);
25
+ this.destroy$ = new Subject();
26
+ /**
27
+ * Whether the autocomplete panel was open before the event
28
+ */
29
+ this.panelOpen = false;
30
+ /**
31
+ * Whether the user selected an option.
32
+ * (We want to ignore selections if there is already a selection present and the user tabs away)
33
+ */
34
+ this.userInput = false;
35
+ this.controlValueAccessor = elderSelectBase;
36
+ }
37
+ /***************************************************************************
38
+ * *
39
+ * Event Listener *
40
+ * *
41
+ **************************************************************************/
42
+ onBlur() {
43
+ if (!this.panelOpen) {
44
+ return;
45
+ }
46
+ if (this.controlValueAccessor.value) {
47
+ // If there is already a value present, only allow tab based value setting if the user actually modified the value
48
+ if (!this.userInput) {
49
+ this.logger.warn('Discarding TAB since the user did probably not intend to change the value! userInput:', this.userInput);
50
+ return;
51
+ }
52
+ }
53
+ const activeOption = this.autoTrigger.activeOption;
54
+ if (activeOption) {
55
+ const entity = activeOption.value;
56
+ this.updateValueByEntity(entity);
57
+ this.autoTrigger.writeValue(entity);
58
+ this.logger.warn('Wrote option.value to control: ', entity);
59
+ }
60
+ }
61
+ updateValueByEntity(entity) {
62
+ if (isElderEntityValueAccessor(this.controlValueAccessor)) {
63
+ // Since elder-select may use the entity id as value (valueAsId),
64
+ // we can not directly write to the value safely.
65
+ this.controlValueAccessor.updateValueByEntity(entity);
66
+ }
67
+ else {
68
+ // By default, write the selected option to the control value
69
+ this.controlValueAccessor.updateValue(entity);
70
+ }
71
+ }
72
+ /***************************************************************************
73
+ * *
74
+ * Life Cycle *
75
+ * *
76
+ **************************************************************************/
77
+ ngAfterViewInit() {
78
+ const autocomplete = this.autoTrigger.autocomplete;
79
+ merge(autocomplete.opened.pipe(map(() => true)), autocomplete.closed.pipe(map(() => false))).pipe(takeUntil(this.destroy$), delay(0)).subscribe(value => this.panelOpen = value);
80
+ this.autoTrigger.optionSelections.pipe(// TODO https://github.com/angular/components/pull/14813
81
+ takeUntil(this.destroy$), map(opt => opt.isUserInput)).subscribe(isUserInput => this.userInput = isUserInput);
82
+ }
83
+ ngOnDestroy() {
84
+ this.destroy$.next();
85
+ this.destroy$.complete();
86
+ }
87
+ }
88
+ ElderSelectOnTabDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.1", ngImport: i0, type: ElderSelectOnTabDirective, deps: [{ token: i1.MatAutocompleteTrigger }, { token: ELDER_SELECT_BASE, skipSelf: true }], target: i0.ɵɵFactoryTarget.Directive });
89
+ ElderSelectOnTabDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.3.1", type: ElderSelectOnTabDirective, selector: "[elderSelectOnTab]", host: { listeners: { "keydown.tab": "onBlur()" } }, ngImport: i0 });
90
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.1", ngImport: i0, type: ElderSelectOnTabDirective, decorators: [{
91
+ type: Directive,
92
+ args: [{
93
+ selector: '[elderSelectOnTab]'
94
+ }]
95
+ }], ctorParameters: function () { return [{ type: i1.MatAutocompleteTrigger }, { type: i2.ElderSelectBase, decorators: [{
96
+ type: SkipSelf
97
+ }, {
98
+ type: Inject,
99
+ args: [ELDER_SELECT_BASE]
100
+ }] }]; }, propDecorators: { onBlur: [{
101
+ type: HostListener,
102
+ args: ['keydown.tab']
103
+ }] } });
104
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-select-on-tab.directive.js","sourceRoot":"","sources":["../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/select/elder-select-on-tab.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {AfterViewInit, Directive, Host, HostListener, Inject, OnDestroy, SkipSelf} from '@angular/core';\nimport {MatAutocompleteTrigger} from '@angular/material/autocomplete';\nimport {Subject} from 'rxjs/internal/Subject';\nimport {delay, map, takeUntil} from 'rxjs/operators';\nimport {merge} from 'rxjs';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {ValueAccessorBase} from '../../common/forms/value-accessor-base';\nimport {ELDER_SELECT_BASE, ElderSelectBase} from './elder-select-base';\nimport {MatOption} from '@angular/material/core';\nimport {isElderEntityValueAccessor} from './elder-entity-value-accessor';\n\n@Directive({\n  selector: '[elderSelectOnTab]'\n})\nexport class ElderSelectOnTabDirective<TEntity = any, TId = any> implements AfterViewInit, OnDestroy {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly destroy$ = new Subject<never>();\n\n  private readonly controlValueAccessor: ValueAccessorBase<any>;\n\n  /**\n   * Whether the autocomplete panel was open before the event\n   */\n  private panelOpen = false;\n\n  /**\n   * Whether the user selected an option.\n   * (We want to ignore selections if there is already a selection present and the user tabs away)\n   */\n  private userInput = false;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private readonly autoTrigger: MatAutocompleteTrigger,\n    @SkipSelf() @Inject(ELDER_SELECT_BASE) elderSelectBase: ElderSelectBase<any, any, any>\n  ) {\n    this.controlValueAccessor = elderSelectBase;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Event Listener                                                          *\n   *                                                                         *\n   **************************************************************************/\n\n  @HostListener('keydown.tab')\n  public onBlur(): void {\n\n    if (!this.panelOpen) { return; }\n\n    if (this.controlValueAccessor.value) {\n      // If there is already a value present, only allow tab based value setting if the user actually modified the value\n      if (!this.userInput) {\n        this.logger.warn('Discarding TAB since the user did probably not intend to change the value! userInput:', this.userInput);\n        return;\n      }\n    }\n\n    const activeOption = this.autoTrigger.activeOption as MatOption<TEntity>;\n\n    if (activeOption) {\n      const entity = activeOption.value;\n      this.updateValueByEntity(entity);\n      this.autoTrigger.writeValue(entity);\n      this.logger.warn('Wrote option.value to control: ', entity);\n    }\n  }\n\n  private updateValueByEntity(entity: TEntity): void {\n    if (isElderEntityValueAccessor(this.controlValueAccessor)) {\n      // Since elder-select may use the entity id as value (valueAsId),\n      // we can not directly write to the value safely.\n      this.controlValueAccessor.updateValueByEntity(entity);\n    } else {\n      // By default, write the selected option to the control value\n      this.controlValueAccessor.updateValue(entity);\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngAfterViewInit() {\n    const autocomplete = this.autoTrigger.autocomplete;\n    merge(\n      autocomplete.opened.pipe(map(() => true)),\n      autocomplete.closed.pipe(map(() => false))\n    ).pipe(\n      takeUntil(this.destroy$),\n      delay(0)\n    ).subscribe(value => this.panelOpen = value);\n\n    this.autoTrigger.optionSelections.pipe( // TODO https://github.com/angular/components/pull/14813\n      takeUntil(this.destroy$),\n      map(opt => opt.isUserInput)\n    ).subscribe(isUserInput => this.userInput = isUserInput);\n  }\n\n  public ngOnDestroy() {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n}\n"]}
@@ -19,23 +19,23 @@ import { MatProgressBarModule } from '@angular/material/progress-bar';
19
19
  import { ElderTableModule } from '../data-view/table/elder-table.module';
20
20
  import { MatProgressSpinnerModule } from '@angular/material/progress-spinner';
21
21
  import { ElderInfiniteScrollModule } from '../infinitescroll/elder-infinite-scroll.module';
22
- import { SelectOnTabDirective } from './select-on-tab.directive';
22
+ import { ElderSelectOnTabDirective } from './elder-select-on-tab.directive';
23
23
  import { ElderAutocompleteModule } from '../input/autocomplete/elder-autocomplete.module';
24
24
  import { ElderSelectValueDirective } from './elder-select-value.directive';
25
25
  import { ElderMultiSelectChipsComponent } from './multi/elder-multi-select-chips/elder-multi-select-chips.component';
26
26
  import { MatChipsModule } from '@angular/material/chips';
27
27
  import { ElderSelectChipDirective } from './elder-select-chip.directive';
28
- import { AutoSelectFirstDirective } from './auto-select-first.directive';
28
+ import { ElderAutoSelectFirstDirective } from './auto/elder-auto-select-first.directive';
29
29
  import * as i0 from "@angular/core";
30
30
  export { ElderSelectValueDirective } from './elder-select-value.directive';
31
31
  export { ElderSelectChipDirective } from './elder-select-chip.directive';
32
- export { SelectOnTabDirective } from './select-on-tab.directive';
32
+ export { ElderSelectOnTabDirective } from './elder-select-on-tab.directive';
33
33
  export { ElderSelectComponent } from './elder-select/elder-select.component';
34
34
  export { TemplatedSelectionDialogComponent } from './popup/templated-selection-dialog/templated-selection-dialog.component';
35
35
  export { SelectionModelPopupDirective, ElderDataViewSelectionMode } from './popup/selection-model-popup.directive';
36
36
  export { ElderMultiSelectBase } from './multi/elder-multi-select-base';
37
37
  export { ElderMultiSelectChipsComponent } from './multi/elder-multi-select-chips/elder-multi-select-chips.component';
38
- export { AutoSelectFirstDirective } from './auto-select-first.directive';
38
+ export { ElderAutoSelectFirstDirective } from './auto/elder-auto-select-first.directive';
39
39
  export class ElderSelectModule {
40
40
  }
41
41
  ElderSelectModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.1", ngImport: i0, type: ElderSelectModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
@@ -43,10 +43,10 @@ ElderSelectModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "12.0.0", versi
43
43
  ElderSelectValueDirective,
44
44
  TemplatedSelectionDialogComponent,
45
45
  SelectionModelPopupDirective,
46
- SelectOnTabDirective,
46
+ ElderSelectOnTabDirective,
47
47
  ElderMultiSelectChipsComponent,
48
48
  ElderSelectChipDirective,
49
- AutoSelectFirstDirective], imports: [CommonModule, FormsModule, ReactiveFormsModule,
49
+ ElderAutoSelectFirstDirective], imports: [CommonModule, FormsModule, ReactiveFormsModule,
50
50
  MatFormFieldModule,
51
51
  MatIconModule,
52
52
  MatSelectModule,
@@ -61,7 +61,7 @@ ElderSelectModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "12.0.0", versi
61
61
  SelectionModelPopupDirective,
62
62
  ElderMultiSelectChipsComponent,
63
63
  ElderSelectChipDirective,
64
- AutoSelectFirstDirective] });
64
+ ElderAutoSelectFirstDirective] });
65
65
  ElderSelectModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "13.3.1", ngImport: i0, type: ElderSelectModule, imports: [[
66
66
  CommonModule, FormsModule, ReactiveFormsModule,
67
67
  MatFormFieldModule,
@@ -94,10 +94,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.1", ngImpor
94
94
  ElderSelectValueDirective,
95
95
  TemplatedSelectionDialogComponent,
96
96
  SelectionModelPopupDirective,
97
- SelectOnTabDirective,
97
+ ElderSelectOnTabDirective,
98
98
  ElderMultiSelectChipsComponent,
99
99
  ElderSelectChipDirective,
100
- AutoSelectFirstDirective
100
+ ElderAutoSelectFirstDirective
101
101
  ],
102
102
  exports: [
103
103
  ElderSelectComponent,
@@ -106,8 +106,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.1", ngImpor
106
106
  SelectionModelPopupDirective,
107
107
  ElderMultiSelectChipsComponent,
108
108
  ElderSelectChipDirective,
109
- AutoSelectFirstDirective
109
+ ElderAutoSelectFirstDirective
110
110
  ]
111
111
  }]
112
112
  }] });
113
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-select.module.js","sourceRoot":"","sources":["../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/select/elder-select.module.ts"],"names":[],"mappings":"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","sourcesContent":["import {NgModule} from '@angular/core';\nimport {CommonModule} from '@angular/common';\nimport {ElderSelectComponent} from './elder-select/elder-select.component';\nimport {FormsModule, ReactiveFormsModule} from '@angular/forms';\nimport {MatFormFieldModule} from '@angular/material/form-field';\nimport {MatIconModule} from '@angular/material/icon';\nimport {MatSelectModule} from '@angular/material/select';\nimport {TranslateModule} from '@ngx-translate/core';\nimport {TemplatedSelectionDialogComponent} from './popup/templated-selection-dialog/templated-selection-dialog.component';\nimport {FlexModule} from '@angular/flex-layout';\nimport {MatDialogModule} from '@angular/material/dialog';\nimport {MatButtonModule} from '@angular/material/button';\nimport {SelectionModelPopupDirective} from './popup/selection-model-popup.directive';\nimport {ElderFormsDirectivesModule} from '../forms/directives/elder-forms-directives.module';\nimport {MatInputModule} from '@angular/material/input';\nimport {MatAutocompleteModule} from '@angular/material/autocomplete';\nimport {MatToolbarModule} from '@angular/material/toolbar';\nimport {MatProgressBarModule} from '@angular/material/progress-bar';\nimport {ElderTableModule} from '../data-view/table/elder-table.module';\nimport {MatProgressSpinnerModule} from '@angular/material/progress-spinner';\nimport {ElderInfiniteScrollModule} from '../infinitescroll/elder-infinite-scroll.module';\nimport {SelectOnTabDirective} from './select-on-tab.directive';\nimport {ElderAutocompleteModule} from '../input/autocomplete/elder-autocomplete.module';\nimport {ElderSelectValueDirective} from './elder-select-value.directive';\nimport {ElderMultiSelectChipsComponent} from './multi/elder-multi-select-chips/elder-multi-select-chips.component';\nimport {MatChipsModule} from '@angular/material/chips';\nimport {ElderSelectChipDirective} from './elder-select-chip.directive';\nimport {AutoSelectFirstDirective} from './auto-select-first.directive';\n\nexport {ElderSelectValueDirective} from './elder-select-value.directive';\nexport {ElderSelectChipDirective} from './elder-select-chip.directive';\nexport {SelectOnTabDirective} from './select-on-tab.directive';\nexport {ElderSelectComponent} from './elder-select/elder-select.component';\nexport {\n  TemplatedSelectionDialogComponent, ISelectionModelDialogOptions\n} from './popup/templated-selection-dialog/templated-selection-dialog.component';\nexport {SelectionModelPopupDirective, ElderDataViewSelectionMode} from './popup/selection-model-popup.directive';\nexport {ElderMultiSelectBase} from './multi/elder-multi-select-base';\nexport {ElderMultiSelectChipsComponent} from './multi/elder-multi-select-chips/elder-multi-select-chips.component';\nexport {AutoSelectFirstDirective} from './auto-select-first.directive';\n\n@NgModule({\n  imports: [\n    CommonModule, FormsModule, ReactiveFormsModule,\n\n    MatFormFieldModule,\n    MatIconModule,\n    MatSelectModule,\n    MatDialogModule, MatButtonModule,\n    MatInputModule, MatAutocompleteModule,\n    MatToolbarModule, MatProgressBarModule,\n    MatProgressSpinnerModule,\n\n    FlexModule, TranslateModule,\n\n    ElderFormsDirectivesModule, ElderTableModule, ElderInfiniteScrollModule, ElderAutocompleteModule, MatChipsModule,\n  ],\n  declarations: [\n    ElderSelectComponent,\n    ElderSelectValueDirective,\n    TemplatedSelectionDialogComponent,\n    SelectionModelPopupDirective,\n    SelectOnTabDirective,\n    ElderMultiSelectChipsComponent,\n    ElderSelectChipDirective,\n    AutoSelectFirstDirective\n  ],\n  exports: [\n    ElderSelectComponent,\n    ElderSelectValueDirective,\n    TemplatedSelectionDialogComponent,\n    SelectionModelPopupDirective,\n    ElderMultiSelectChipsComponent,\n    ElderSelectChipDirective,\n    AutoSelectFirstDirective\n  ]\n})\nexport class ElderSelectModule {\n}\n"]}
113
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-select.module.js","sourceRoot":"","sources":["../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/select/elder-select.module.ts"],"names":[],"mappings":"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","sourcesContent":["import {NgModule} from '@angular/core';\nimport {CommonModule} from '@angular/common';\nimport {ElderSelectComponent} from './elder-select/elder-select.component';\nimport {FormsModule, ReactiveFormsModule} from '@angular/forms';\nimport {MatFormFieldModule} from '@angular/material/form-field';\nimport {MatIconModule} from '@angular/material/icon';\nimport {MatSelectModule} from '@angular/material/select';\nimport {TranslateModule} from '@ngx-translate/core';\nimport {TemplatedSelectionDialogComponent} from './popup/templated-selection-dialog/templated-selection-dialog.component';\nimport {FlexModule} from '@angular/flex-layout';\nimport {MatDialogModule} from '@angular/material/dialog';\nimport {MatButtonModule} from '@angular/material/button';\nimport {SelectionModelPopupDirective} from './popup/selection-model-popup.directive';\nimport {ElderFormsDirectivesModule} from '../forms/directives/elder-forms-directives.module';\nimport {MatInputModule} from '@angular/material/input';\nimport {MatAutocompleteModule} from '@angular/material/autocomplete';\nimport {MatToolbarModule} from '@angular/material/toolbar';\nimport {MatProgressBarModule} from '@angular/material/progress-bar';\nimport {ElderTableModule} from '../data-view/table/elder-table.module';\nimport {MatProgressSpinnerModule} from '@angular/material/progress-spinner';\nimport {ElderInfiniteScrollModule} from '../infinitescroll/elder-infinite-scroll.module';\nimport {ElderSelectOnTabDirective} from './elder-select-on-tab.directive';\nimport {ElderAutocompleteModule} from '../input/autocomplete/elder-autocomplete.module';\nimport {ElderSelectValueDirective} from './elder-select-value.directive';\nimport {ElderMultiSelectChipsComponent} from './multi/elder-multi-select-chips/elder-multi-select-chips.component';\nimport {MatChipsModule} from '@angular/material/chips';\nimport {ElderSelectChipDirective} from './elder-select-chip.directive';\nimport {ElderAutoSelectFirstDirective} from './auto/elder-auto-select-first.directive';\n\nexport {ElderSelectValueDirective} from './elder-select-value.directive';\nexport {ElderSelectChipDirective} from './elder-select-chip.directive';\nexport {ElderSelectOnTabDirective} from './elder-select-on-tab.directive';\nexport {ElderSelectComponent} from './elder-select/elder-select.component';\nexport {\n  TemplatedSelectionDialogComponent, ISelectionModelDialogOptions\n} from './popup/templated-selection-dialog/templated-selection-dialog.component';\nexport {SelectionModelPopupDirective, ElderDataViewSelectionMode} from './popup/selection-model-popup.directive';\nexport {ElderMultiSelectBase} from './multi/elder-multi-select-base';\nexport {ElderMultiSelectChipsComponent} from './multi/elder-multi-select-chips/elder-multi-select-chips.component';\nexport {ElderAutoSelectFirstDirective} from './auto/elder-auto-select-first.directive';\n\n@NgModule({\n  imports: [\n    CommonModule, FormsModule, ReactiveFormsModule,\n\n    MatFormFieldModule,\n    MatIconModule,\n    MatSelectModule,\n    MatDialogModule, MatButtonModule,\n    MatInputModule, MatAutocompleteModule,\n    MatToolbarModule, MatProgressBarModule,\n    MatProgressSpinnerModule,\n\n    FlexModule, TranslateModule,\n\n    ElderFormsDirectivesModule, ElderTableModule, ElderInfiniteScrollModule, ElderAutocompleteModule, MatChipsModule,\n  ],\n  declarations: [\n    ElderSelectComponent,\n    ElderSelectValueDirective,\n    TemplatedSelectionDialogComponent,\n    SelectionModelPopupDirective,\n    ElderSelectOnTabDirective,\n    ElderMultiSelectChipsComponent,\n    ElderSelectChipDirective,\n    ElderAutoSelectFirstDirective\n  ],\n  exports: [\n    ElderSelectComponent,\n    ElderSelectValueDirective,\n    TemplatedSelectionDialogComponent,\n    SelectionModelPopupDirective,\n    ElderMultiSelectChipsComponent,\n    ElderSelectChipDirective,\n    ElderAutoSelectFirstDirective\n  ]\n})\nexport class ElderSelectModule {\n}\n"]}
@@ -1,9 +1,10 @@
1
1
  import { Directive, Input, Output } from '@angular/core';
2
2
  import { ElderSelectBase, ElderSelectComponentState } from '../elder-select-base';
3
- import { of } from 'rxjs';
3
+ import { BehaviorSubject, combineLatest, of } from 'rxjs';
4
4
  import { LoggerFactory } from '@elderbyte/ts-logger';
5
- import { catchError, map } from 'rxjs/operators';
5
+ import { catchError, filter, map } from 'rxjs/operators';
6
6
  import { Sets } from '../../../common/sets';
7
+ import { CollectionUtil } from '../../../common/utils/collection-util';
7
8
  import * as i0 from "@angular/core";
8
9
  /**
9
10
  * Base component implementation of elder-multi-select.
@@ -29,67 +30,123 @@ export class ElderMultiSelectBase extends ElderSelectBase {
29
30
  * *
30
31
  **************************************************************************/
31
32
  this.logger = LoggerFactory.getLogger(this.constructor.name);
32
- this._valueIds = null;
33
- this.valueIdsChange = this.valueChange.pipe(map(values => {
34
- if (values) {
35
- return values.map(v => this.getEntityId(v));
36
- }
37
- else {
38
- return null;
39
- }
40
- }));
33
+ this.entities$ = new BehaviorSubject([]);
34
+ this.entityIdsChange = this.valueChange.pipe(map(values => this.entityIdsFromValues(values)));
35
+ this.entityIdsUpdated = this.valueUpdated.pipe(map(values => this.entityIdsFromValues(values)));
36
+ this.entitiesUpdated = combineLatest([this.entityIdsUpdated, this.entities$]).pipe(filter(([updatedIds, entities]) => this.equalIds(this.getEntityIds(entities), updatedIds)), map(([updatedIds, entities]) => entities));
41
37
  }
42
38
  /***************************************************************************
43
39
  * *
44
40
  * Properties *
45
41
  * *
46
42
  **************************************************************************/
47
- /**
48
- * Select the value with the given id.
49
- * @param ids
50
- */
51
- set valueIds(ids) {
52
- if (!this.equalIds(ids, this._valueIds)) {
53
- this._valueIds = ids;
54
- this.selectValuesByIds(ids);
55
- }
43
+ set entities(entities) {
44
+ this.entities$.next(entities);
45
+ this.writeValueInternal(this.entitiesToValues(entities));
46
+ }
47
+ get entities() {
48
+ return this.entities$.getValue();
56
49
  }
57
- get valueIds() {
58
- if (this.value) {
59
- return this.getValueIds(this.value);
50
+ set entityIds(ids) {
51
+ if (this.valueAsId) {
52
+ this.writeValueInternal(ids);
60
53
  }
61
54
  else {
62
- return this._valueIds;
55
+ this.selectEntitiesByIds(ids);
63
56
  }
64
57
  }
58
+ get entityIds() {
59
+ return this.entityIdsFromValues(this.value);
60
+ }
61
+ /***************************************************************************
62
+ * *
63
+ * Public API *
64
+ * *
65
+ **************************************************************************/
66
+ updateValueByEntities(entities) {
67
+ this.updateValue(this.entitiesToValues(entities));
68
+ }
65
69
  /***************************************************************************
66
70
  * *
67
71
  * Internal Methods *
68
72
  * *
69
73
  **************************************************************************/
70
74
  onSuggestionsDcChanged(data) {
71
- if (this._valueIds) {
72
- this.selectValuesByIds(this._valueIds);
75
+ if (this.valueAsId) {
76
+ this.selectEntitiesByIds(this.entityIds);
73
77
  }
74
78
  }
79
+ /**
80
+ * This method is invoked after a value has been written to this control.
81
+ *
82
+ */
83
+ writeToControl(value) {
84
+ if (this.valueAsId) {
85
+ const entityIds = value;
86
+ // Value was written as entity ids, ensure we select entity by id
87
+ this.logger.debug('writeToControl: value was written as ids: ' + entityIds, value);
88
+ this.selectEntitiesByIds(entityIds);
89
+ }
90
+ else {
91
+ // Value was written as entity, ensure entity is updated
92
+ const currentEntities = this.entities;
93
+ const newEntities = value;
94
+ this.logger.debug('writeToControl: value was written as entities (size): ' + newEntities?.length, value);
95
+ if (!this.equalEntities(currentEntities, newEntities)) {
96
+ this.entities$.next(newEntities);
97
+ }
98
+ else {
99
+ this.logger.warn('Ignored written entities as they are already set to entities$!');
100
+ }
101
+ }
102
+ super.writeToControl(value);
103
+ }
75
104
  /***************************************************************************
76
105
  * *
77
106
  * Private Methods *
78
107
  * *
79
108
  **************************************************************************/
80
- getValueIds(values) {
109
+ entityIdsFromValues(values) {
110
+ return values?.map(v => this.entityIdFromValue(v));
111
+ }
112
+ entityIdFromValue(value) {
113
+ if (this.valueAsId) {
114
+ return value;
115
+ }
116
+ else {
117
+ const entity = value;
118
+ return this.getEntityId(entity);
119
+ }
120
+ }
121
+ entitiesToValues(entities) {
122
+ return entities?.map(e => this.entityToValue(e));
123
+ }
124
+ entityToValue(entity) {
125
+ if (this.valueAsId) {
126
+ const id = this.getEntityId(entity);
127
+ return id;
128
+ }
129
+ else {
130
+ return entity;
131
+ }
132
+ }
133
+ getEntityIds(values) {
81
134
  return values.map(v => this.getEntityId(v));
82
135
  }
83
136
  equalIds(idsA, idsB) {
84
137
  return Sets.equalContent(idsA, idsB);
85
138
  }
86
- selectValuesByIds(ids) {
87
- if (this.value && this.equalIds(this.getValueIds(this.value), ids)) {
88
- return; // Already selected
139
+ equalEntities(entitiesA, entitiesB) {
140
+ return this.equalIds(this.getEntityIds(entitiesA), this.getEntityIds(entitiesB));
141
+ }
142
+ selectEntitiesByIds(ids) {
143
+ const currentEntities = this.entities;
144
+ if (currentEntities && this.equalIds(this.getEntityIds(currentEntities), ids)) {
145
+ return; // Entities already loaded
89
146
  }
90
- if (ids === null || ids === undefined || ids.length === 0) {
91
- if (this.value !== null && this.value !== undefined && this.value.length !== 0) {
92
- this.writeValueInternal([]); // TODO Or write empty list?? []
147
+ if (!CollectionUtil.hasElements(ids)) {
148
+ if (CollectionUtil.hasElements(currentEntities)) {
149
+ this.writeValueInternal([]);
93
150
  }
94
151
  }
95
152
  else {
@@ -101,8 +158,8 @@ export class ElderMultiSelectBase extends ElderSelectBase {
101
158
  loaded.forEach((v, k) => existing.set(k, v));
102
159
  return existing;
103
160
  }), map(allValuesMap => Array.from(allValuesMap.values())) // TODO Sort?
104
- ).subscribe(values => {
105
- this.writeValueInternal(values);
161
+ ).subscribe(entities => {
162
+ this.entities = entities;
106
163
  });
107
164
  }
108
165
  else {
@@ -116,6 +173,7 @@ export class ElderMultiSelectBase extends ElderSelectBase {
116
173
  this.updateState(ElderSelectComponentState.error(err));
117
174
  return of([]);
118
175
  }), map(values => {
176
+ this.logger.debug('Loaded entities by id:', values);
119
177
  const valueMap = new Map();
120
178
  values.forEach(v => valueMap.set(this.getEntityId(v), v));
121
179
  return valueMap;
@@ -134,12 +192,18 @@ export class ElderMultiSelectBase extends ElderSelectBase {
134
192
  }
135
193
  }
136
194
  ElderMultiSelectBase.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.1", ngImport: i0, type: ElderMultiSelectBase, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Directive });
137
- ElderMultiSelectBase.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.3.1", type: ElderMultiSelectBase, inputs: { valueIds: "valueIds" }, outputs: { valueIdsChange: "valueIdsChange" }, usesInheritance: true, ngImport: i0 });
195
+ ElderMultiSelectBase.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.3.1", type: ElderMultiSelectBase, inputs: { entities: "entities", entityIds: "entityIds" }, outputs: { entityIdsChange: "entityIdsChange", entityIdsUpdated: "entityIdsUpdated", entitiesUpdated: "entitiesUpdated" }, usesInheritance: true, ngImport: i0 });
138
196
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.1", ngImport: i0, type: ElderMultiSelectBase, decorators: [{
139
197
  type: Directive
140
- }], ctorParameters: function () { return [{ type: i0.NgZone }]; }, propDecorators: { valueIdsChange: [{
198
+ }], ctorParameters: function () { return [{ type: i0.NgZone }]; }, propDecorators: { entityIdsChange: [{
199
+ type: Output
200
+ }], entityIdsUpdated: [{
141
201
  type: Output
142
- }], valueIds: [{
202
+ }], entitiesUpdated: [{
203
+ type: Output
204
+ }], entities: [{
205
+ type: Input
206
+ }], entityIds: [{
143
207
  type: Input
144
208
  }] } });
145
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-multi-select-base.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/select/multi/elder-multi-select-base.ts"],"names":[],"mappings":"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","sourcesContent":["import {Directive, Input, NgZone, Output} from '@angular/core';\nimport {ElderSelectBase, ElderSelectComponentState} from '../elder-select-base';\nimport {IDataContext} from '../../../common/data/data-context/data-context';\nimport {Observable, of} from 'rxjs';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {catchError, map} from 'rxjs/operators';\nimport {Sets} from '../../../common/sets';\n\n\n/**\n * Base component implementation of elder-multi-select.\n *\n * Multi Select has the following abstract concept:\n *\n * - Suggestion-Model: A DataSource and derived DataContext for the (auto-complete) suggestions.\n * - The backing Value is a list of the current selected entities. T[]\n *\n */\n@Directive()\n// tslint:disable-next-line:directive-class-suffix\nexport abstract class ElderMultiSelectBase<TId, T> extends ElderSelectBase<TId, T, T[]> {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private _valueIds: TId[] = null;\n\n  @Output()\n  public readonly valueIdsChange: Observable<TId[]>;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  protected constructor(\n    zone: NgZone\n  ) {\n    super(zone);\n\n    this.valueIdsChange = this.valueChange.pipe(\n      map(values => {\n        if (values) {\n          return values.map(v => this.getEntityId(v));\n        } else {\n          return null;\n        }\n      })\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Select the value with the given id.\n   * @param ids\n   */\n  @Input()\n  public set valueIds(ids: TId[]) {\n    if (!this.equalIds(ids, this._valueIds)) {\n      this._valueIds = ids;\n      this.selectValuesByIds(ids);\n    }\n  }\n\n  public get valueIds(): TId[] {\n    if (this.value) {\n      return this.getValueIds(this.value);\n    } else {\n      return this._valueIds;\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Internal Methods                                                        *\n   *                                                                         *\n   **************************************************************************/\n\n  protected onSuggestionsDcChanged(data: IDataContext<T>): void {\n    if (this._valueIds) {\n      this.selectValuesByIds(this._valueIds);\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private Methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private getValueIds(values: T[]) {\n    return values.map(v => this.getEntityId(v));\n  }\n\n  private equalIds(idsA: TId[], idsB: TId[]): boolean {\n    return Sets.equalContent(idsA, idsB);\n  }\n\n  private selectValuesByIds(ids: TId[]): void {\n\n    if (this.value && this.equalIds(this.getValueIds(this.value), ids)) {\n      return; // Already selected\n    }\n\n    if (ids === null || ids === undefined || ids.length === 0) {\n      if (this.value !== null && this.value !== undefined && this.value.length !== 0) {\n        this.writeValueInternal([]); // TODO Or write empty list?? []\n      }\n    } else {\n      if (this.suggestionsDc) {\n        const existing = this.findInDataContext(ids);\n        const missingIds = ids.filter(id => !existing.has(id));\n        const valueLoadRequest$ = missingIds.length > 0 ? this.loadEntityByIds(missingIds) : of(new Map());\n\n        valueLoadRequest$.pipe(\n          map(loaded => {\n            loaded.forEach((v, k) => existing.set(k, v));\n            return existing;\n          }),\n          map(allValuesMap => Array.from(allValuesMap.values())) // TODO Sort?\n        ).subscribe(\n          values => {\n            this.writeValueInternal(values);\n          }\n        );\n      } else {\n        this.logger.warn('Failed to select value by Ids: ' + ids + ' - DataContext not available.');\n      }\n    }\n  }\n\n  private loadEntityByIds(ids: TId[]): Observable<Map<TId, T>> {\n    return this.suggestionsDc.dataSource.findByIds(ids).pipe(\n      catchError(err => {\n        this.logger.error('Failed to load values by ids ' + ids, err);\n        this.updateState(ElderSelectComponentState.error(err));\n        return of(<T[]>[]);\n      }),\n      map(values => {\n        const valueMap = new Map<TId, T>();\n        values.forEach(v => valueMap.set(this.getEntityId(v), v));\n        return valueMap;\n      })\n    );\n  }\n\n  private findInDataContext(ids: TId[]): Map<TId, T> {\n    const existingValues = new Map<TId, T>();\n    if (ids?.length > 0) {\n      const data = this.suggestionsDc.snapshot.data;\n      const requiredIds = new Set(ids);\n      data\n        .filter(d => requiredIds.has(this.getEntityId(d)))\n        .forEach(d => existingValues.set(this.getEntityId(d), d));\n    }\n    return existingValues;\n  }\n}\n"]}
209
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-multi-select-base.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/select/multi/elder-multi-select-base.ts"],"names":[],"mappings":"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","sourcesContent":["import {Directive, Input, NgZone, Output} from '@angular/core';\nimport {ElderSelectBase, ElderSelectComponentState} from '../elder-select-base';\nimport {IDataContext} from '../../../common/data/data-context/data-context';\nimport {BehaviorSubject, combineLatest, Observable, of} from 'rxjs';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {catchError, filter, map} from 'rxjs/operators';\nimport {Sets} from '../../../common/sets';\nimport {CollectionUtil} from '../../../common/utils/collection-util';\n\n\n/**\n * Base component implementation of elder-multi-select.\n *\n * Multi Select has the following abstract concept:\n *\n * - Suggestion-Model: A DataSource and derived DataContext for the (auto-complete) suggestions.\n * - The backing Value is a list of the current selected entities. T[]\n *\n */\n@Directive()\n// tslint:disable-next-line:directive-class-suffix\nexport abstract class ElderMultiSelectBase<TId, TEntity, TValue> extends ElderSelectBase<TId, TEntity, TValue[]> {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  protected readonly entities$ = new BehaviorSubject<TEntity[]>([]);\n\n  @Output()\n  public readonly entityIdsChange: Observable<TId[]>;\n\n  /**\n   * Similar to entity-id change, but emits only when the user\n   * has updated the value.\n   */\n  @Output()\n  public readonly entityIdsUpdated: Observable<TId[]>;\n\n  /**\n   * Similar to entity change, but emits only when the user\n   * has updated the value.\n   */\n  @Output()\n  public readonly entitiesUpdated: Observable<TEntity[]>;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  protected constructor(\n    zone: NgZone\n  ) {\n    super(zone);\n\n    this.entityIdsChange = this.valueChange.pipe(\n      map(values => this.entityIdsFromValues(values))\n    );\n\n    this.entityIdsUpdated = this.valueUpdated.pipe(\n      map(values => this.entityIdsFromValues(values))\n    );\n\n    this.entitiesUpdated = combineLatest(\n      [this.entityIdsUpdated, this.entities$]\n    ).pipe(\n      filter(([updatedIds, entities]) => this.equalIds(this.getEntityIds(entities), updatedIds)),\n      map(([updatedIds, entities]) => entities)\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\n  public set entities(entities: TEntity[]) {\n    this.entities$.next(entities);\n    this.writeValueInternal(\n      this.entitiesToValues(entities)\n    );\n  }\n\n  public get entities(): TEntity[] {\n    return this.entities$.getValue();\n  }\n\n  @Input()\n  public set entityIds(ids: TId[]) {\n    if (this.valueAsId) {\n      this.writeValueInternal(<any>ids as TValue[]);\n    } else {\n      this.selectEntitiesByIds(ids);\n    }\n  }\n\n  public get entityIds(): TId[] {\n    return this.entityIdsFromValues(this.value);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public updateValueByEntities(entities: TEntity[]): void {\n    this.updateValue(\n      this.entitiesToValues(entities)\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Internal Methods                                                        *\n   *                                                                         *\n   **************************************************************************/\n\n  protected onSuggestionsDcChanged(data: IDataContext<TEntity>): void {\n    if (this.valueAsId) {\n      this.selectEntitiesByIds(this.entityIds);\n    }\n  }\n\n  /**\n   * This method is invoked after a value has been written to this control.\n   *\n   */\n  protected writeToControl(value: TValue[]): void {\n    if (this.valueAsId) {\n      const entityIds = <any>value as TId[];\n      // Value was written as entity ids, ensure we select entity by id\n      this.logger.debug('writeToControl: value was written as ids: ' + entityIds, value);\n      this.selectEntitiesByIds(entityIds);\n    } else {\n      // Value was written as entity, ensure entity is updated\n      const currentEntities = this.entities;\n      const newEntities = <any>value as TEntity[];\n      this.logger.debug('writeToControl: value was written as entities (size): ' + newEntities?.length, value);\n      if (!this.equalEntities(currentEntities, newEntities)) {\n        this.entities$.next(newEntities);\n      } else {\n        this.logger.warn('Ignored written entities as they are already set to entities$!');\n      }\n    }\n    super.writeToControl(value);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private Methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private entityIdsFromValues(values: TValue[]): TId[] {\n    return values?.map(v => this.entityIdFromValue(v));\n  }\n\n  private entityIdFromValue(value: TValue): TId {\n    if (this.valueAsId) {\n      return <any>value as TId;\n    } else {\n      const entity = (<any>value) as TEntity;\n      return this.getEntityId(entity);\n    }\n  }\n\n  private entitiesToValues(entities: TEntity[]): TValue[] {\n    return entities?.map(e => this.entityToValue(e));\n  }\n\n  private entityToValue(entity: TEntity): TValue {\n    if (this.valueAsId) {\n      const id = this.getEntityId(entity);\n      return <any>id as TValue;\n    } else {\n      return <any>entity as TValue;\n    }\n  }\n\n  private getEntityIds(values: TEntity[]) {\n    return values.map(v => this.getEntityId(v));\n  }\n\n  private equalIds(idsA: TId[], idsB: TId[]): boolean {\n    return Sets.equalContent(idsA, idsB);\n  }\n\n  private equalEntities(entitiesA: TEntity[], entitiesB: TEntity[]) {\n    return this.equalIds(this.getEntityIds(entitiesA), this.getEntityIds(entitiesB));\n  }\n\n  private selectEntitiesByIds(ids: TId[]): void {\n\n    const currentEntities = this.entities;\n    if (currentEntities && this.equalIds(this.getEntityIds(currentEntities), ids)) {\n      return; // Entities already loaded\n    }\n\n    if (!CollectionUtil.hasElements(ids)) {\n      if (CollectionUtil.hasElements(currentEntities)) {\n        this.writeValueInternal([]);\n      }\n    } else {\n      if (this.suggestionsDc) {\n        const existing = this.findInDataContext(ids);\n        const missingIds = ids.filter(id => !existing.has(id));\n        const valueLoadRequest$ = missingIds.length > 0 ? this.loadEntityByIds(missingIds) : of(new Map());\n\n        valueLoadRequest$.pipe(\n          map(loaded => {\n            loaded.forEach((v, k) => existing.set(k, v));\n            return existing;\n          }),\n          map(allValuesMap => Array.from(allValuesMap.values())) // TODO Sort?\n        ).subscribe(\n          entities => {\n            this.entities = entities;\n          }\n        );\n      } else {\n        this.logger.warn('Failed to select value by Ids: ' + ids + ' - DataContext not available.');\n      }\n    }\n  }\n\n  private loadEntityByIds(ids: TId[]): Observable<Map<TId, TEntity>> {\n    return this.suggestionsDc.dataSource.findByIds(ids).pipe(\n      catchError(err => {\n        this.logger.error('Failed to load values by ids ' + ids, err);\n        this.updateState(ElderSelectComponentState.error(err));\n        return of(<TEntity[]>[]);\n      }),\n      map(values => {\n        this.logger.debug('Loaded entities by id:', values);\n        const valueMap = new Map<TId, TEntity>();\n        values.forEach(v => valueMap.set(this.getEntityId(v), v));\n        return valueMap;\n      })\n    );\n  }\n\n  private findInDataContext(ids: TId[]): Map<TId, TEntity> {\n    const existingValues = new Map<TId, TEntity>();\n    if (ids?.length > 0) {\n      const data = this.suggestionsDc.snapshot.data;\n      const requiredIds = new Set(ids);\n      data\n        .filter(d => requiredIds.has(this.getEntityId(d)))\n        .forEach(d => existingValues.set(this.getEntityId(d), d));\n    }\n    return existingValues;\n  }\n\n\n}\n"]}