@elderbyte/ngx-starter 14.3.2 → 14.4.0-beta.3

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 (58) hide show
  1. package/_index.scss +4 -0
  2. package/esm2020/lib/common/forms/elder-entity-value-accessor.mjs +22 -0
  3. package/esm2020/lib/common/forms/elder-form-field-control-base.directive.mjs +169 -0
  4. package/esm2020/lib/common/forms/elder-from-field-base.mjs +94 -0
  5. package/esm2020/lib/common/forms/elder-from-field-entity-base.mjs +39 -0
  6. package/esm2020/lib/common/forms/elder-from-field-multi-entity-base.mjs +39 -0
  7. package/esm2020/lib/common/forms/elder-multi-entity-value-accessor.mjs +22 -0
  8. package/esm2020/lib/common/forms/public_api.mjs +7 -1
  9. package/esm2020/lib/components/forms/directives/elder-forms-directives.module.mjs +7 -1
  10. package/esm2020/lib/components/input/autocomplete/elder-autocomplete.directive.mjs +4 -4
  11. package/esm2020/lib/components/measures/directives/elder-unit-select.directive.mjs +2 -2
  12. package/esm2020/lib/components/select/auto/elder-auto-select-first.directive.mjs +2 -2
  13. package/esm2020/lib/components/select/auto/elder-select-first-util.mjs +3 -3
  14. package/esm2020/lib/components/select/elder-select-base.mjs +38 -27
  15. package/esm2020/lib/components/select/elder-select-chip.directive.mjs +19 -5
  16. package/esm2020/lib/components/select/elder-select-on-tab.directive.mjs +2 -2
  17. package/esm2020/lib/components/select/elder-select.module.mjs +23 -12
  18. package/esm2020/lib/components/select/multi/elder-multi-select-base.mjs +78 -39
  19. package/esm2020/lib/components/select/multi/elder-multi-select-chips/elder-multi-select-chips.component.mjs +115 -83
  20. package/esm2020/lib/components/select/multi/elder-multi-select-form-field.mjs +61 -0
  21. package/esm2020/lib/components/select/popup/selection-model-popup.directive.mjs +9 -4
  22. package/esm2020/lib/components/select/public_api.mjs +4 -1
  23. package/esm2020/lib/components/select/single/elder-clear-select.directive.mjs +57 -0
  24. package/esm2020/lib/components/select/single/elder-select/elder-select.component.mjs +455 -0
  25. package/esm2020/lib/components/select/single/elder-select-form-field.mjs +45 -0
  26. package/fesm2015/elderbyte-ngx-starter.mjs +1019 -377
  27. package/fesm2015/elderbyte-ngx-starter.mjs.map +1 -1
  28. package/fesm2020/elderbyte-ngx-starter.mjs +1012 -375
  29. package/fesm2020/elderbyte-ngx-starter.mjs.map +1 -1
  30. package/lib/{components/select → common/forms}/elder-entity-value-accessor.d.ts +10 -0
  31. package/lib/common/forms/elder-form-field-control-base.directive.d.ts +101 -0
  32. package/lib/common/forms/elder-from-field-base.d.ts +52 -0
  33. package/lib/common/forms/elder-from-field-entity-base.d.ts +31 -0
  34. package/lib/common/forms/elder-from-field-multi-entity-base.d.ts +31 -0
  35. package/lib/common/forms/elder-multi-entity-value-accessor.d.ts +42 -0
  36. package/lib/common/forms/public_api.d.ts +6 -0
  37. package/lib/components/input/autocomplete/elder-autocomplete.directive.d.ts +1 -1
  38. package/lib/components/measures/directives/elder-unit-select.directive.d.ts +1 -1
  39. package/lib/components/select/auto/elder-auto-select-first.directive.d.ts +1 -1
  40. package/lib/components/select/auto/elder-select-first-util.d.ts +1 -1
  41. package/lib/components/select/elder-select-base.d.ts +21 -15
  42. package/lib/components/select/elder-select-chip.directive.d.ts +9 -2
  43. package/lib/components/select/elder-select.module.d.ts +27 -24
  44. package/lib/components/select/multi/elder-multi-select-base.d.ts +19 -14
  45. package/lib/components/select/multi/elder-multi-select-chips/elder-multi-select-chips.component.d.ts +27 -12
  46. package/lib/components/select/multi/elder-multi-select-form-field.d.ts +33 -0
  47. package/lib/components/select/popup/selection-model-popup.directive.d.ts +3 -2
  48. package/lib/components/select/public_api.d.ts +3 -0
  49. package/lib/components/select/single/elder-clear-select.directive.d.ts +34 -0
  50. package/lib/components/select/{elder-select → single/elder-select}/elder-select.component.d.ts +24 -12
  51. package/lib/components/select/single/elder-select-form-field.d.ts +30 -0
  52. package/package.json +1 -1
  53. package/src/lib/components/select/multi/elder-multi-select-chips/elder-multi-select-chips.component.scss +54 -11
  54. package/src/lib/components/select/single/elder-select/elder-select.component.scss +114 -0
  55. package/theming/_elder-common.scss +48 -0
  56. package/esm2020/lib/components/select/elder-entity-value-accessor.mjs +0 -13
  57. package/esm2020/lib/components/select/elder-select/elder-select.component.mjs +0 -419
  58. package/src/lib/components/select/elder-select/elder-select.component.scss +0 -61
@@ -4,7 +4,7 @@ import { delay, map, takeUntil, tap } from 'rxjs/operators';
4
4
  import { merge } from 'rxjs';
5
5
  import { LoggerFactory } from '@elderbyte/ts-logger';
6
6
  import { ELDER_SELECT_BASE } from './elder-select-base';
7
- import { isElderEntityValueAccessor } from './elder-entity-value-accessor';
7
+ import { isElderEntityValueAccessor } from '../../common/forms/elder-entity-value-accessor';
8
8
  import * as i0 from "@angular/core";
9
9
  import * as i1 from "@angular/material/autocomplete";
10
10
  import * as i2 from "./elder-select-base";
@@ -130,4 +130,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.4", ngImpor
130
130
  type: HostListener,
131
131
  args: ['keydown.tab']
132
132
  }] } });
133
- //# 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, tap} 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, TValue = TEntity | TId> 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<void>();\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) private readonly elderSelectBase: ElderSelectBase<any, any, any>\n  ) {\n    this.controlValueAccessor = elderSelectBase;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Event Listener                                                          *\n   *                                                                         *\n   **************************************************************************/\n\n  @HostListener('keydown.arrowup')\n  @HostListener('keydown.arrowdown')\n  public onOptionSelect(){\n    this.userInput = true;\n  }\n\n  @HostListener('keydown.tab')\n  public onBlur(): void {\n\n    if (!this.panelOpen) { return; }\n\n    if (!this.userInput) {\n      // The user did not select anything in the auto-complete\n\n      if (this.controlValueAccessor.value) {\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      if(!this.elderSelectBase.required) {\n        // The user did not select any option and no current value is present.\n        // Since the input is NOT marked as required, we assume the user did not want to select a value\n        return;\n      }\n    }\n\n    const activeOption = this.autoTrigger.activeOption as MatOption<TEntity>;\n    if (activeOption) {\n      const entity = activeOption.value;\n      this.writeEntity(entity);\n    }\n\n    this.reset();\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      tap(opt => this.logger.debug('[optionSelections] CHANGED ', opt)),\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   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private reset(): void {\n    this.userInput = false;\n  }\n\n  private writeEntity(entity: TEntity): void {\n    const value = this.entityToValue(entity);\n    this.controlValueAccessor.updateValue(value);\n    this.autoTrigger.writeValue(value);\n    this.logger.warn('Wrote entity to control as value: ', value);\n  }\n\n  private entityToValue(entity: TEntity): TValue {\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      return this.controlValueAccessor.entityToValue(entity);\n    } else {\n      // By default, write the selected option to the control value\n      this.logger.warn('[controlValueAccessor] is not a ElderEntityValueAccessor!: ', this.controlValueAccessor);\n      return <any>entity;\n    }\n  }\n}\n"]}
133
+ //# 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, tap} 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 '../../common/forms/elder-entity-value-accessor';\n\n@Directive({\n  selector: '[elderSelectOnTab]'\n})\nexport class ElderSelectOnTabDirective<TEntity = any, TId = any, TValue = TEntity | TId> 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<void>();\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) private readonly elderSelectBase: ElderSelectBase<any, any, any>\n  ) {\n    this.controlValueAccessor = elderSelectBase;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Event Listener                                                          *\n   *                                                                         *\n   **************************************************************************/\n\n  @HostListener('keydown.arrowup')\n  @HostListener('keydown.arrowdown')\n  public onOptionSelect(){\n    this.userInput = true;\n  }\n\n  @HostListener('keydown.tab')\n  public onBlur(): void {\n\n    if (!this.panelOpen) { return; }\n\n    if (!this.userInput) {\n      // The user did not select anything in the auto-complete\n\n      if (this.controlValueAccessor.value) {\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      if(!this.elderSelectBase.required) {\n        // The user did not select any option and no current value is present.\n        // Since the input is NOT marked as required, we assume the user did not want to select a value\n        return;\n      }\n    }\n\n    const activeOption = this.autoTrigger.activeOption as MatOption<TEntity>;\n    if (activeOption) {\n      const entity = activeOption.value;\n      this.writeEntity(entity);\n    }\n\n    this.reset();\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      tap(opt => this.logger.debug('[optionSelections] CHANGED ', opt)),\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   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private reset(): void {\n    this.userInput = false;\n  }\n\n  private writeEntity(entity: TEntity): void {\n    const value = this.entityToValue(entity);\n    this.controlValueAccessor.updateValue(value);\n    this.autoTrigger.writeValue(value);\n    this.logger.warn('Wrote entity to control as value: ', value);\n  }\n\n  private entityToValue(entity: TEntity): TValue {\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      return this.controlValueAccessor.entityToValue(entity);\n    } else {\n      // By default, write the selected option to the control value\n      this.logger.warn('[controlValueAccessor] is not a ElderEntityValueAccessor!: ', this.controlValueAccessor);\n      return <any>entity;\n    }\n  }\n}\n"]}
@@ -1,6 +1,6 @@
1
1
  import { NgModule } from '@angular/core';
2
2
  import { CommonModule } from '@angular/common';
3
- import { ElderSelectComponent } from './elder-select/elder-select.component';
3
+ import { ElderSelectComponent } from './single/elder-select/elder-select.component';
4
4
  import { FormsModule, ReactiveFormsModule } from '@angular/forms';
5
5
  import { MatFormFieldModule } from '@angular/material/form-field';
6
6
  import { MatIconModule } from '@angular/material/icon';
@@ -24,19 +24,22 @@ import { ElderAutocompleteModule } from '../input/autocomplete/elder-autocomplet
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
- import { ElderSelectChipAvatarDirective, ElderSelectChipDirective } from './elder-select-chip.directive';
27
+ import { ElderSelectChipAvatarDirective, ElderSelectChipDirective, ElderSelectCustomInputDirective } from './elder-select-chip.directive';
28
28
  import { ElderAutoSelectFirstDirective } from './auto/elder-auto-select-first.directive';
29
29
  import { ElderPipesModule } from '../../pipes/elder-pipes.module';
30
+ import { ElderClearSelectDirective } from './single/elder-clear-select.directive';
31
+ import { MatTooltipModule } from '@angular/material/tooltip';
30
32
  import * as i0 from "@angular/core";
31
33
  export { ElderSelectValueDirective } from './elder-select-value.directive';
32
- export { ElderSelectChipDirective, ElderSelectChipAvatarDirective } from './elder-select-chip.directive';
34
+ export * from './elder-select-chip.directive';
33
35
  export { ElderSelectOnTabDirective } from './elder-select-on-tab.directive';
34
- export { ElderSelectComponent } from './elder-select/elder-select.component';
36
+ export { ElderSelectComponent } from './single/elder-select/elder-select.component';
35
37
  export { TemplatedSelectionDialogComponent } from './popup/templated-selection-dialog/templated-selection-dialog.component';
36
38
  export { SelectionModelPopupDirective, ElderDataViewSelectionMode } from './popup/selection-model-popup.directive';
37
39
  export { ElderMultiSelectBase } from './multi/elder-multi-select-base';
38
40
  export { ElderMultiSelectChipsComponent } from './multi/elder-multi-select-chips/elder-multi-select-chips.component';
39
41
  export { ElderAutoSelectFirstDirective } from './auto/elder-auto-select-first.directive';
42
+ export { ElderClearSelectDirective } from './single/elder-clear-select.directive';
40
43
  export class ElderSelectModule {
41
44
  }
42
45
  ElderSelectModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.4", ngImport: i0, type: ElderSelectModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
@@ -48,7 +51,9 @@ ElderSelectModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", versi
48
51
  ElderMultiSelectChipsComponent,
49
52
  ElderSelectChipDirective,
50
53
  ElderSelectChipAvatarDirective,
51
- ElderAutoSelectFirstDirective], imports: [CommonModule, FormsModule, ReactiveFormsModule,
54
+ ElderAutoSelectFirstDirective,
55
+ ElderClearSelectDirective,
56
+ ElderSelectCustomInputDirective], imports: [CommonModule, FormsModule, ReactiveFormsModule,
52
57
  MatFormFieldModule,
53
58
  MatIconModule,
54
59
  MatSelectModule,
@@ -57,14 +62,16 @@ ElderSelectModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", versi
57
62
  MatToolbarModule, MatProgressBarModule,
58
63
  MatProgressSpinnerModule,
59
64
  FlexModule, TranslateModule,
60
- ElderFormsDirectivesModule, ElderTableModule, ElderInfiniteScrollModule, ElderAutocompleteModule, MatChipsModule, ElderPipesModule], exports: [ElderSelectComponent,
65
+ ElderFormsDirectivesModule, ElderTableModule, ElderInfiniteScrollModule, ElderAutocompleteModule, MatChipsModule, ElderPipesModule, MatTooltipModule], exports: [ElderSelectComponent,
61
66
  ElderSelectValueDirective,
62
67
  TemplatedSelectionDialogComponent,
63
68
  SelectionModelPopupDirective,
64
69
  ElderMultiSelectChipsComponent,
65
70
  ElderSelectChipDirective,
66
71
  ElderSelectChipAvatarDirective,
67
- ElderAutoSelectFirstDirective] });
72
+ ElderAutoSelectFirstDirective,
73
+ ElderClearSelectDirective,
74
+ ElderSelectCustomInputDirective] });
68
75
  ElderSelectModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "14.0.4", ngImport: i0, type: ElderSelectModule, imports: [CommonModule, FormsModule, ReactiveFormsModule,
69
76
  MatFormFieldModule,
70
77
  MatIconModule,
@@ -74,7 +81,7 @@ ElderSelectModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", versi
74
81
  MatToolbarModule, MatProgressBarModule,
75
82
  MatProgressSpinnerModule,
76
83
  FlexModule, TranslateModule,
77
- ElderFormsDirectivesModule, ElderTableModule, ElderInfiniteScrollModule, ElderAutocompleteModule, MatChipsModule, ElderPipesModule] });
84
+ ElderFormsDirectivesModule, ElderTableModule, ElderInfiniteScrollModule, ElderAutocompleteModule, MatChipsModule, ElderPipesModule, MatTooltipModule] });
78
85
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.4", ngImport: i0, type: ElderSelectModule, decorators: [{
79
86
  type: NgModule,
80
87
  args: [{
@@ -88,7 +95,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.4", ngImpor
88
95
  MatToolbarModule, MatProgressBarModule,
89
96
  MatProgressSpinnerModule,
90
97
  FlexModule, TranslateModule,
91
- ElderFormsDirectivesModule, ElderTableModule, ElderInfiniteScrollModule, ElderAutocompleteModule, MatChipsModule, ElderPipesModule,
98
+ ElderFormsDirectivesModule, ElderTableModule, ElderInfiniteScrollModule, ElderAutocompleteModule, MatChipsModule, ElderPipesModule, MatTooltipModule,
92
99
  ],
93
100
  declarations: [
94
101
  ElderSelectComponent,
@@ -99,7 +106,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.4", ngImpor
99
106
  ElderMultiSelectChipsComponent,
100
107
  ElderSelectChipDirective,
101
108
  ElderSelectChipAvatarDirective,
102
- ElderAutoSelectFirstDirective
109
+ ElderAutoSelectFirstDirective,
110
+ ElderClearSelectDirective,
111
+ ElderSelectCustomInputDirective
103
112
  ],
104
113
  exports: [
105
114
  ElderSelectComponent,
@@ -109,8 +118,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.4", ngImpor
109
118
  ElderMultiSelectChipsComponent,
110
119
  ElderSelectChipDirective,
111
120
  ElderSelectChipAvatarDirective,
112
- ElderAutoSelectFirstDirective
121
+ ElderAutoSelectFirstDirective,
122
+ ElderClearSelectDirective,
123
+ ElderSelectCustomInputDirective
113
124
  ]
114
125
  }]
115
126
  }] });
116
- //# 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":"AAAA,OAAO,EAAC,QAAQ,EAAC,MAAM,eAAe,CAAC;AACvC,OAAO,EAAC,YAAY,EAAC,MAAM,iBAAiB,CAAC;AAC7C,OAAO,EAAC,oBAAoB,EAAC,MAAM,uCAAuC,CAAC;AAC3E,OAAO,EAAC,WAAW,EAAE,mBAAmB,EAAC,MAAM,gBAAgB,CAAC;AAChE,OAAO,EAAC,kBAAkB,EAAC,MAAM,8BAA8B,CAAC;AAChE,OAAO,EAAC,aAAa,EAAC,MAAM,wBAAwB,CAAC;AACrD,OAAO,EAAC,eAAe,EAAC,MAAM,0BAA0B,CAAC;AACzD,OAAO,EAAC,eAAe,EAAC,MAAM,qBAAqB,CAAC;AACpD,OAAO,EAAC,iCAAiC,EAAC,MAAM,yEAAyE,CAAC;AAC1H,OAAO,EAAC,UAAU,EAAC,MAAM,sBAAsB,CAAC;AAChD,OAAO,EAAC,eAAe,EAAC,MAAM,0BAA0B,CAAC;AACzD,OAAO,EAAC,eAAe,EAAC,MAAM,0BAA0B,CAAC;AACzD,OAAO,EAAC,4BAA4B,EAAC,MAAM,yCAAyC,CAAC;AACrF,OAAO,EAAC,0BAA0B,EAAC,MAAM,mDAAmD,CAAC;AAC7F,OAAO,EAAC,cAAc,EAAC,MAAM,yBAAyB,CAAC;AACvD,OAAO,EAAC,qBAAqB,EAAC,MAAM,gCAAgC,CAAC;AACrE,OAAO,EAAC,gBAAgB,EAAC,MAAM,2BAA2B,CAAC;AAC3D,OAAO,EAAC,oBAAoB,EAAC,MAAM,gCAAgC,CAAC;AACpE,OAAO,EAAC,gBAAgB,EAAC,MAAM,uCAAuC,CAAC;AACvE,OAAO,EAAC,wBAAwB,EAAC,MAAM,oCAAoC,CAAC;AAC5E,OAAO,EAAC,yBAAyB,EAAC,MAAM,gDAAgD,CAAC;AACzF,OAAO,EAAC,yBAAyB,EAAC,MAAM,iCAAiC,CAAC;AAC1E,OAAO,EAAC,uBAAuB,EAAC,MAAM,iDAAiD,CAAC;AACxF,OAAO,EAAC,yBAAyB,EAAC,MAAM,gCAAgC,CAAC;AACzE,OAAO,EAAC,8BAA8B,EAAC,MAAM,qEAAqE,CAAC;AACnH,OAAO,EAAC,cAAc,EAAC,MAAM,yBAAyB,CAAC;AACvD,OAAO,EAAC,8BAA8B,EAAE,wBAAwB,EAAC,MAAM,+BAA+B,CAAC;AACvG,OAAO,EAAC,6BAA6B,EAAC,MAAM,0CAA0C,CAAC;AACvF,OAAO,EAAC,gBAAgB,EAAC,MAAM,gCAAgC,CAAC;;AAEhE,OAAO,EAAC,yBAAyB,EAAC,MAAM,gCAAgC,CAAC;AACzE,OAAO,EAAC,wBAAwB,EAAE,8BAA8B,EAAC,MAAM,+BAA+B,CAAC;AACvG,OAAO,EAAC,yBAAyB,EAAC,MAAM,iCAAiC,CAAC;AAC1E,OAAO,EAAC,oBAAoB,EAAC,MAAM,uCAAuC,CAAC;AAC3E,OAAO,EACL,iCAAiC,EAClC,MAAM,yEAAyE,CAAC;AACjF,OAAO,EAAC,4BAA4B,EAAE,0BAA0B,EAAC,MAAM,yCAAyC,CAAC;AACjH,OAAO,EAAC,oBAAoB,EAAC,MAAM,iCAAiC,CAAC;AACrE,OAAO,EAAC,8BAA8B,EAAC,MAAM,qEAAqE,CAAC;AACnH,OAAO,EAAC,6BAA6B,EAAC,MAAM,0CAA0C,CAAC;AAwCvF,MAAM,OAAO,iBAAiB;;8GAAjB,iBAAiB;+GAAjB,iBAAiB,iBArB1B,oBAAoB;QACpB,yBAAyB;QACzB,iCAAiC;QACjC,4BAA4B;QAC5B,yBAAyB;QACzB,8BAA8B;QAC9B,wBAAwB;QACxB,8BAA8B;QAC9B,6BAA6B,aAvBzB,YAAY,EAAE,WAAW,EAAE,mBAAmB;QAE9C,kBAAkB;QAClB,aAAa;QACb,eAAe;QACf,eAAe,EAAE,eAAe;QAChC,cAAc,EAAE,qBAAqB;QACrC,gBAAgB,EAAE,oBAAoB;QACtC,wBAAwB;QAExB,UAAU,EAAE,eAAe;QAE3B,0BAA0B,EAAE,gBAAgB,EAAE,yBAAyB,EAAE,uBAAuB,EAAE,cAAc,EAAE,gBAAgB,aActI,oBAAoB;QACpB,yBAAyB;QACzB,iCAAiC;QACjC,4BAA4B;QAC5B,8BAA8B;QAC9B,wBAAwB;QACxB,8BAA8B;QAC9B,6BAA6B;+GAGpB,iBAAiB,YApCtB,YAAY,EAAE,WAAW,EAAE,mBAAmB;QAE9C,kBAAkB;QAClB,aAAa;QACb,eAAe;QACf,eAAe,EAAE,eAAe;QAChC,cAAc,EAAE,qBAAqB;QACrC,gBAAgB,EAAE,oBAAoB;QACtC,wBAAwB;QAExB,UAAU,EAAE,eAAe;QAE3B,0BAA0B,EAAE,gBAAgB,EAAE,yBAAyB,EAAE,uBAAuB,EAAE,cAAc,EAAE,gBAAgB;2FAwB7H,iBAAiB;kBAtC7B,QAAQ;mBAAC;oBACN,OAAO,EAAE;wBACL,YAAY,EAAE,WAAW,EAAE,mBAAmB;wBAE9C,kBAAkB;wBAClB,aAAa;wBACb,eAAe;wBACf,eAAe,EAAE,eAAe;wBAChC,cAAc,EAAE,qBAAqB;wBACrC,gBAAgB,EAAE,oBAAoB;wBACtC,wBAAwB;wBAExB,UAAU,EAAE,eAAe;wBAE3B,0BAA0B,EAAE,gBAAgB,EAAE,yBAAyB,EAAE,uBAAuB,EAAE,cAAc,EAAE,gBAAgB;qBACrI;oBACH,YAAY,EAAE;wBACZ,oBAAoB;wBACpB,yBAAyB;wBACzB,iCAAiC;wBACjC,4BAA4B;wBAC5B,yBAAyB;wBACzB,8BAA8B;wBAC9B,wBAAwB;wBACxB,8BAA8B;wBAC9B,6BAA6B;qBAC9B;oBACD,OAAO,EAAE;wBACP,oBAAoB;wBACpB,yBAAyB;wBACzB,iCAAiC;wBACjC,4BAA4B;wBAC5B,8BAA8B;wBAC9B,wBAAwB;wBACxB,8BAA8B;wBAC9B,6BAA6B;qBAC9B;iBACF","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 {ElderSelectChipAvatarDirective, ElderSelectChipDirective} from './elder-select-chip.directive';\nimport {ElderAutoSelectFirstDirective} from './auto/elder-auto-select-first.directive';\nimport {ElderPipesModule} from '../../pipes/elder-pipes.module';\n\nexport {ElderSelectValueDirective} from './elder-select-value.directive';\nexport {ElderSelectChipDirective, ElderSelectChipAvatarDirective} 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, ElderPipesModule,\n    ],\n  declarations: [\n    ElderSelectComponent,\n    ElderSelectValueDirective,\n    TemplatedSelectionDialogComponent,\n    SelectionModelPopupDirective,\n    ElderSelectOnTabDirective,\n    ElderMultiSelectChipsComponent,\n    ElderSelectChipDirective,\n    ElderSelectChipAvatarDirective,\n    ElderAutoSelectFirstDirective\n  ],\n  exports: [\n    ElderSelectComponent,\n    ElderSelectValueDirective,\n    TemplatedSelectionDialogComponent,\n    SelectionModelPopupDirective,\n    ElderMultiSelectChipsComponent,\n    ElderSelectChipDirective,\n    ElderSelectChipAvatarDirective,\n    ElderAutoSelectFirstDirective\n  ]\n})\nexport class ElderSelectModule {\n}\n"]}
127
+ //# 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 './single/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 {ElderSelectChipAvatarDirective, ElderSelectChipDirective, ElderSelectCustomInputDirective} from './elder-select-chip.directive';\nimport {ElderAutoSelectFirstDirective} from './auto/elder-auto-select-first.directive';\nimport {ElderPipesModule} from '../../pipes/elder-pipes.module';\nimport { ElderClearSelectDirective } from './single/elder-clear-select.directive';\nimport {MatTooltipModule} from '@angular/material/tooltip';\n\nexport {ElderSelectValueDirective} from './elder-select-value.directive';\nexport * from './elder-select-chip.directive';\nexport {ElderSelectOnTabDirective} from './elder-select-on-tab.directive';\nexport {ElderSelectComponent} from './single/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';\nexport {ElderClearSelectDirective} from './single/elder-clear-select.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, ElderPipesModule, MatTooltipModule,\n  ],\n  declarations: [\n    ElderSelectComponent,\n    ElderSelectValueDirective,\n    TemplatedSelectionDialogComponent,\n    SelectionModelPopupDirective,\n    ElderSelectOnTabDirective,\n    ElderMultiSelectChipsComponent,\n    ElderSelectChipDirective,\n    ElderSelectChipAvatarDirective,\n    ElderAutoSelectFirstDirective,\n    ElderClearSelectDirective,\n    ElderSelectCustomInputDirective\n  ],\n  exports: [\n    ElderSelectComponent,\n    ElderSelectValueDirective,\n    TemplatedSelectionDialogComponent,\n    SelectionModelPopupDirective,\n    ElderMultiSelectChipsComponent,\n    ElderSelectChipDirective,\n    ElderSelectChipAvatarDirective,\n    ElderAutoSelectFirstDirective,\n    ElderClearSelectDirective,\n    ElderSelectCustomInputDirective\n  ]\n})\nexport class ElderSelectModule {\n}\n"]}
@@ -1,11 +1,13 @@
1
1
  import { Directive, Input, Output } from '@angular/core';
2
2
  import { ElderSelectBase, ElderSelectComponentState } from '../elder-select-base';
3
- import { BehaviorSubject, EMPTY, of } from 'rxjs';
3
+ import { BehaviorSubject, EMPTY, of, throwError } from 'rxjs';
4
4
  import { LoggerFactory } from '@elderbyte/ts-logger';
5
- import { catchError, filter, map, switchMap, take, tap, timeout } from 'rxjs/operators';
5
+ import { catchError, filter, map, skip, switchMap, take, tap, timeout } from 'rxjs/operators';
6
6
  import { Sets } from '../../../common/sets';
7
7
  import { CollectionUtil } from '../../../common/utils/collection-util';
8
8
  import * as i0 from "@angular/core";
9
+ import * as i1 from "@angular/cdk/a11y";
10
+ import * as i2 from "@angular/forms";
9
11
  /**
10
12
  * Base component implementation of elder-multi-select.
11
13
  *
@@ -22,16 +24,19 @@ export class ElderMultiSelectBase extends ElderSelectBase {
22
24
  * Constructor *
23
25
  * *
24
26
  **************************************************************************/
25
- constructor(zone) {
26
- super(zone);
27
+ constructor(zone, hostRef, controlType, focusMonitor, ngControl) {
28
+ super(zone, hostRef, controlType, focusMonitor, ngControl);
27
29
  /***************************************************************************
28
30
  * *
29
31
  * Fields *
30
32
  * *
31
33
  **************************************************************************/
32
34
  this.logger = LoggerFactory.getLogger(this.constructor.name);
33
- this.entities$ = new BehaviorSubject([]);
35
+ this._entities$ = new BehaviorSubject([]);
34
36
  this.entityIdsChange = this.valueChange.pipe(map(values => this.entityIdsFromValues(values)));
37
+ this._entityIds$ = this.value$.pipe(map(values => this.entityIdsFromValues(values)));
38
+ this.entitiesChange = this._entities$.pipe(skip(1) // Skip the initial or current value
39
+ );
35
40
  this.entityIdsUpdated = this.valueUpdated.pipe(map(values => this.entityIdsFromValues(values)));
36
41
  this.entitiesUpdated = this.entityIdsUpdated.pipe(tap(entityIds => this.logger.debug('Awaiting entities with ids', entityIds)), switchMap(entityIds => this.awaitEntitiesWithId(entityIds, this.timeoutAfterMs)), catchError(err => {
37
42
  this.logger.warn(`awaitEntitiesWithId -> timed out after: ${this.timeoutAfterMs}`, err);
@@ -39,28 +44,25 @@ export class ElderMultiSelectBase extends ElderSelectBase {
39
44
  }));
40
45
  }
41
46
  awaitEntitiesWithId(entityIds, timeoutMs) {
42
- return this.entities$.pipe(tap(entities => this.logger.debug('Got entities for ids' + entityIds, entities)), filter(entities => this.equalIds(this.getEntityIds(entities), entityIds)), take(1), timeout(timeoutMs));
47
+ return this._entities$.pipe(tap(entities => this.logger.debug('Got entities for ids' + entityIds, entities)), filter(entities => this.equalIds(this.getEntityIds(entities), entityIds)), take(1), timeout(timeoutMs));
43
48
  }
44
49
  /***************************************************************************
45
50
  * *
46
51
  * Properties *
47
52
  * *
48
53
  **************************************************************************/
49
- updateValueByEntities(entities) {
50
- this.entities$.next(entities);
51
- this.updateValue(this.entitiesToValues(entities));
54
+ get entities$() {
55
+ return this._entities$.asObservable();
56
+ }
57
+ get entityIds$() {
58
+ return this._entityIds$;
52
59
  }
53
- /***************************************************************************
54
- * *
55
- * Properties *
56
- * *
57
- **************************************************************************/
58
60
  set entities(entities) {
59
- this.entities$.next(entities);
61
+ this._entities$.next(entities);
60
62
  this.writeValueInternal(this.entitiesToValues(entities));
61
63
  }
62
64
  get entities() {
63
- return this.entities$.getValue();
65
+ return this._entities$.getValue();
64
66
  }
65
67
  set entityIds(ids) {
66
68
  if (this.valueAsId) {
@@ -73,11 +75,38 @@ export class ElderMultiSelectBase extends ElderSelectBase {
73
75
  get entityIds() {
74
76
  return this.entityIdsFromValues(this.value);
75
77
  }
78
+ get isOptionDisabledInternalFn() {
79
+ return (option) => {
80
+ if (this.isOptionDisabledFn) {
81
+ return this.isOptionDisabledFn(option);
82
+ }
83
+ else {
84
+ return false;
85
+ }
86
+ };
87
+ }
88
+ get isOptionHiddenInternalFn() {
89
+ return (option) => {
90
+ if (this.isAlreadyPresent(option)) {
91
+ return true;
92
+ }
93
+ if (this.isOptionHiddenFn) {
94
+ return this.isOptionHiddenFn(option);
95
+ }
96
+ else {
97
+ return false;
98
+ }
99
+ };
100
+ }
76
101
  /***************************************************************************
77
102
  * *
78
103
  * Public API *
79
104
  * *
80
105
  **************************************************************************/
106
+ updateValueByEntities(entities) {
107
+ this._entities$.next(entities);
108
+ this.updateValue(this.entitiesToValues(entities));
109
+ }
81
110
  appendEntities(newEntities) {
82
111
  const current = this.entities ? this.entities : [];
83
112
  this.updateValueByEntities([...current, ...newEntities]);
@@ -98,12 +127,21 @@ export class ElderMultiSelectBase extends ElderSelectBase {
98
127
  this.logger.warn('Cant open selection browser popup, since [SelectionModelPopupDirective] was not found!');
99
128
  }
100
129
  }
130
+ entityToValue(entity) {
131
+ if (this.valueAsId) {
132
+ const id = this.getEntityId(entity);
133
+ return id;
134
+ }
135
+ else {
136
+ return entity;
137
+ }
138
+ }
101
139
  /***************************************************************************
102
140
  * *
103
141
  * Internal Methods *
104
142
  * *
105
143
  **************************************************************************/
106
- onSuggestionsDcChanged(data) {
144
+ onDataContextChanged(data) {
107
145
  if (this.valueAsId) {
108
146
  this.selectEntitiesByIds(this.entityIds);
109
147
  }
@@ -125,7 +163,7 @@ export class ElderMultiSelectBase extends ElderSelectBase {
125
163
  const newEntities = value;
126
164
  this.logger.debug('writeToControl: value was written as entities (size): ' + newEntities?.length, value);
127
165
  if (!this.equalEntities(currentEntities, newEntities)) {
128
- this.entities$.next(newEntities);
166
+ this._entities$.next(newEntities);
129
167
  }
130
168
  else {
131
169
  this.logger.warn('Ignored written entities as they are already set to entities$!');
@@ -138,6 +176,14 @@ export class ElderMultiSelectBase extends ElderSelectBase {
138
176
  * Private Methods *
139
177
  * *
140
178
  **************************************************************************/
179
+ isAlreadyPresent(queryValue) {
180
+ if (this.entities) {
181
+ return this.entities.some(v => this.isEqual(v, queryValue));
182
+ }
183
+ else {
184
+ return false;
185
+ }
186
+ }
141
187
  entityIdsFromValues(values) {
142
188
  if (values) {
143
189
  return values.map(v => this.entityIdFromValue(v));
@@ -159,15 +205,6 @@ export class ElderMultiSelectBase extends ElderSelectBase {
159
205
  }
160
206
  return null;
161
207
  }
162
- entityToValue(entity) {
163
- if (this.valueAsId) {
164
- const id = this.getEntityId(entity);
165
- return id;
166
- }
167
- else {
168
- return entity;
169
- }
170
- }
171
208
  getEntityIds(values) {
172
209
  if (values) {
173
210
  return values.map(v => this.getEntityId(v));
@@ -192,7 +229,7 @@ export class ElderMultiSelectBase extends ElderSelectBase {
192
229
  }
193
230
  }
194
231
  else {
195
- if (this.suggestionsDc) {
232
+ if (this.dataContext) {
196
233
  const existing = this.findInDataContext(ids);
197
234
  const missingIds = ids.filter(id => !existing.has(id));
198
235
  const valueLoadRequest$ = missingIds.length > 0 ? this.loadEntityByIds(missingIds) : of(new Map());
@@ -200,8 +237,9 @@ export class ElderMultiSelectBase extends ElderSelectBase {
200
237
  loaded.forEach((v, k) => existing.set(k, v));
201
238
  return existing;
202
239
  }), map(allValuesMap => Array.from(allValuesMap.values())) // TODO Sort?
203
- ).subscribe(entities => {
204
- this.entities = entities;
240
+ ).subscribe({
241
+ next: entities => this.entities = entities,
242
+ error: err => this.updateState(ElderSelectComponentState.error(err))
205
243
  });
206
244
  }
207
245
  else {
@@ -210,10 +248,9 @@ export class ElderMultiSelectBase extends ElderSelectBase {
210
248
  }
211
249
  }
212
250
  loadEntityByIds(ids) {
213
- return this.suggestionsDc.dataSource.findByIds(ids).pipe(catchError(err => {
214
- this.logger.error('Failed to load values by ids ' + ids, err);
215
- this.updateState(ElderSelectComponentState.error(err));
216
- return of([]);
251
+ return this.dataContext.dataSource.findByIds(ids).pipe(catchError(err => {
252
+ this.logger.error('Failed to load entities by ids: ' + ids, err);
253
+ return throwError(err);
217
254
  }), map(values => {
218
255
  this.logger.debug('Loaded entities by id:', values);
219
256
  const valueMap = new Map();
@@ -224,7 +261,7 @@ export class ElderMultiSelectBase extends ElderSelectBase {
224
261
  findInDataContext(ids) {
225
262
  const existingValues = new Map();
226
263
  if (ids?.length > 0) {
227
- const data = this.suggestionsDc.snapshot.data;
264
+ const data = this.dataContext.snapshot.data;
228
265
  const requiredIds = new Set(ids);
229
266
  data
230
267
  .filter(d => requiredIds.has(this.getEntityId(d)))
@@ -233,11 +270,13 @@ export class ElderMultiSelectBase extends ElderSelectBase {
233
270
  return existingValues;
234
271
  }
235
272
  }
236
- ElderMultiSelectBase.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.4", ngImport: i0, type: ElderMultiSelectBase, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Directive });
237
- ElderMultiSelectBase.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "14.0.4", type: ElderMultiSelectBase, inputs: { entities: "entities", entityIds: "entityIds" }, outputs: { entityIdsChange: "entityIdsChange", entityIdsUpdated: "entityIdsUpdated", entitiesUpdated: "entitiesUpdated" }, usesInheritance: true, ngImport: i0 });
273
+ ElderMultiSelectBase.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.4", ngImport: i0, type: ElderMultiSelectBase, deps: "invalid", target: i0.ɵɵFactoryTarget.Directive });
274
+ ElderMultiSelectBase.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "14.0.4", type: ElderMultiSelectBase, inputs: { entities: "entities", entityIds: "entityIds" }, outputs: { entityIdsChange: "entityIdsChange", entitiesChange: "entitiesChange", entityIdsUpdated: "entityIdsUpdated", entitiesUpdated: "entitiesUpdated" }, usesInheritance: true, ngImport: i0 });
238
275
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.4", ngImport: i0, type: ElderMultiSelectBase, decorators: [{
239
276
  type: Directive
240
- }], ctorParameters: function () { return [{ type: i0.NgZone }]; }, propDecorators: { entityIdsChange: [{
277
+ }], ctorParameters: function () { return [{ type: i0.NgZone }, { type: i0.ElementRef }, { type: undefined }, { type: i1.FocusMonitor }, { type: i2.NgControl }]; }, propDecorators: { entityIdsChange: [{
278
+ type: Output
279
+ }], entitiesChange: [{
241
280
  type: Output
242
281
  }], entityIdsUpdated: [{
243
282
  type: Output
@@ -248,4 +287,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.4", ngImpor
248
287
  }], entityIds: [{
249
288
  type: Input
250
289
  }] } });
251
- //# 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, EMPTY, Observable, of} from 'rxjs';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {catchError, filter, map, switchMap, take, tap, timeout} 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// eslint-disable-next-line @angular-eslint/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 = this.entityIdsUpdated.pipe(\n      tap(entityIds => this.logger.debug('Awaiting entities with ids', entityIds)),\n      switchMap(entityIds => this.awaitEntitiesWithId(entityIds, this.timeoutAfterMs)),\n      catchError(err => {\n        this.logger.warn(`awaitEntitiesWithId -> timed out after: ${this.timeoutAfterMs}`, err);\n        return EMPTY;\n      })\n    );\n  }\n\n  private awaitEntitiesWithId(entityIds: TId[], timeoutMs: number): Observable<TEntity[]> {\n    return this.entities$.pipe(\n      tap(entities => this.logger.debug('Got entities for ids' + entityIds, entities)),\n      filter(entities => this.equalIds(this.getEntityIds(entities), entityIds)),\n      take(1),\n      timeout(timeoutMs)\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public updateValueByEntities(entities: TEntity[]): void {\n    this.entities$.next(entities);\n    this.updateValue(\n      this.entitiesToValues(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 appendEntities(newEntities: TEntity[]) {\n    const current = this.entities ? this.entities : [];\n    this.updateValueByEntities([...current, ...newEntities]);\n  }\n\n  public appendEntity(newEntity: any) {\n    this.appendEntities([newEntity]);\n  }\n\n  public removeEntity(toRemove: TEntity): void {\n    const remaining = this.entities\n      .filter(l => !this.isEqual(l, toRemove));\n    this.updateValueByEntities(remaining);\n  }\n\n  public openSelectionPopup(event: Event): void {\n    if (this.selectionPopup) {\n      this.selectionPopup.choose(\n        (e: TEntity) => this.getEntityId(e),\n        this.entities,\n        this.filters,\n        this.sorts,\n        true\n      ).subscribe(\n        (selection: TEntity[]) => this.updateValueByEntities(selection)\n      );\n    } else {\n      this.logger.warn('Cant open selection browser popup, since [SelectionModelPopupDirective] was not found!');\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[] | null {\n    if (values) {\n      return values.map(v => this.entityIdFromValue(v));\n    }\n    return null;\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[] | null {\n    if (entities) {\n      return entities.map(e => this.entityToValue(e));\n    }\n    return null;\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[] | null): TId[] {\n    if (values) {\n      return values.map(v => this.getEntityId(v));\n    }\n    return null;\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    const currentEntities = this.entities;\n    this.logger.info('selectEntitiesByIds: ids' + ids + ', current entities:', currentEntities);\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"]}
290
+ //# 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, ElementRef, Input, NgZone, Output} from '@angular/core';\nimport {ElderSelectBase, ElderSelectComponentState} from '../elder-select-base';\nimport {IDataContext} from '../../../common/data/data-context/data-context';\nimport {BehaviorSubject, EMPTY, Observable, of, throwError} from 'rxjs';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {catchError, filter, map, skip, switchMap, take, tap, timeout} from 'rxjs/operators';\nimport {Sets} from '../../../common/sets';\nimport {CollectionUtil} from '../../../common/utils/collection-util';\nimport {NgControl} from '@angular/forms';\nimport {FocusMonitor} from '@angular/cdk/a11y';\nimport {IElderMultiEntityValueAccessor} from '../../../common/forms/elder-multi-entity-value-accessor';\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// eslint-disable-next-line @angular-eslint/directive-class-suffix\nexport abstract class ElderMultiSelectBase<TId, TEntity, TValue>\n  extends ElderSelectBase<TId, TEntity, TValue[]>\n  implements IElderMultiEntityValueAccessor<TEntity, TId, TValue> {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly _entityIds$: Observable<TId[]>;\n  private readonly _entities$ = new BehaviorSubject<TEntity[]>([]);\n\n  @Output()\n  public readonly entityIdsChange: Observable<TId[]>;\n\n  @Output()\n  public readonly entitiesChange: Observable<TEntity[]>;\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    hostRef: ElementRef,\n    controlType: string,\n    focusMonitor: FocusMonitor,\n    ngControl?: NgControl\n  ) {\n    super(zone, hostRef, controlType, focusMonitor, ngControl);\n\n    this.entityIdsChange = this.valueChange.pipe(\n      map(values => this.entityIdsFromValues(values))\n    );\n\n    this._entityIds$ = this.value$.pipe(\n      map(values => this.entityIdsFromValues(values))\n    );\n\n    this.entitiesChange = this._entities$.pipe(\n      skip(1) // Skip the initial or current value\n    );\n\n    this.entityIdsUpdated = this.valueUpdated.pipe(\n      map(values => this.entityIdsFromValues(values))\n    );\n\n    this.entitiesUpdated = this.entityIdsUpdated.pipe(\n      tap(entityIds => this.logger.debug('Awaiting entities with ids', entityIds)),\n      switchMap(entityIds => this.awaitEntitiesWithId(entityIds, this.timeoutAfterMs)),\n      catchError(err => {\n        this.logger.warn(`awaitEntitiesWithId -> timed out after: ${this.timeoutAfterMs}`, err);\n        return EMPTY;\n      })\n    );\n  }\n\n  private awaitEntitiesWithId(entityIds: TId[], timeoutMs: number): Observable<TEntity[]> {\n    return this._entities$.pipe(\n      tap(entities => this.logger.debug('Got entities for ids' + entityIds, entities)),\n      filter(entities => this.equalIds(this.getEntityIds(entities), entityIds)),\n      take(1),\n      timeout(timeoutMs)\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public get entities$(): Observable<TEntity[]> {\n    return this._entities$.asObservable();\n  }\n\n  public get entityIds$(): Observable<TId[]> {\n    return this._entityIds$;\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  public get isOptionDisabledInternalFn(): (option: TEntity) => boolean {\n    return (option: TEntity) => {\n      if (this.isOptionDisabledFn) {\n        return this.isOptionDisabledFn(option);\n      } else {\n        return false;\n      }\n    };\n  }\n\n  public get isOptionHiddenInternalFn(): (option: TEntity) => boolean {\n    return (option: TEntity) => {\n      if (this.isAlreadyPresent(option)) {\n        return true;\n      }\n      if (this.isOptionHiddenFn) {\n        return this.isOptionHiddenFn(option);\n      } else {\n        return false;\n      }\n    };\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public updateValueByEntities(entities: TEntity[]): void {\n    this._entities$.next(entities);\n    this.updateValue(\n      this.entitiesToValues(entities)\n    );\n  }\n\n  public appendEntities(newEntities: TEntity[]) {\n    const current = this.entities ? this.entities : [];\n    this.updateValueByEntities([...current, ...newEntities]);\n  }\n\n  public appendEntity(newEntity: any) {\n    this.appendEntities([newEntity]);\n  }\n\n  public removeEntity(toRemove: TEntity): void {\n    const remaining = this.entities\n      .filter(l => !this.isEqual(l, toRemove));\n    this.updateValueByEntities(remaining);\n  }\n\n  public openSelectionPopup(event: Event): void {\n    if (this.selectionPopup) {\n      this.selectionPopup.choose(\n        (e: TEntity) => this.getEntityId(e),\n        this.entities,\n        this.filters,\n        this.sorts,\n        true\n      ).subscribe(\n        (selection: TEntity[]) => this.updateValueByEntities(selection)\n      );\n    } else {\n      this.logger.warn('Cant open selection browser popup, since [SelectionModelPopupDirective] was not found!');\n    }\n  }\n\n  public 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  /***************************************************************************\n   *                                                                         *\n   * Internal Methods                                                        *\n   *                                                                         *\n   **************************************************************************/\n\n  protected onDataContextChanged(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\n  private isAlreadyPresent(queryValue: TEntity): boolean {\n    if (this.entities) {\n      return this.entities.some(v => this.isEqual(v, queryValue));\n    } else {\n      return false;\n    }\n  }\n\n  private entityIdsFromValues(values: TValue[]): TId[] | null {\n    if (values) {\n      return values.map(v => this.entityIdFromValue(v));\n    }\n    return null;\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[] | null {\n    if (entities) {\n      return entities.map(e => this.entityToValue(e));\n    }\n    return null;\n  }\n\n  private getEntityIds(values: TEntity[] | null): TId[] {\n    if (values) {\n      return values.map(v => this.getEntityId(v));\n    }\n    return null;\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    const currentEntities = this.entities;\n    this.logger.info('selectEntitiesByIds: ids' + ids + ', current entities:', currentEntities);\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.dataContext) {\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          next: entities => this.entities = entities,\n          error: err => this.updateState(ElderSelectComponentState.error(err))\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.dataContext.dataSource.findByIds(ids).pipe(\n      catchError(err => {\n        this.logger.error('Failed to load entities by ids: ' + ids, err);\n        return throwError(err);\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.dataContext.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"]}