@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.
- package/esm2020/lib/common/utils/collection-util.mjs +4 -1
- package/esm2020/lib/components/select/auto/elder-auto-select-first.directive.mjs +77 -0
- package/esm2020/lib/components/select/auto/elder-select-first-util.mjs +62 -0
- package/esm2020/lib/components/select/elder-select/elder-select.component.mjs +14 -9
- package/esm2020/lib/components/select/elder-select-on-tab.directive.mjs +104 -0
- package/esm2020/lib/components/select/elder-select.module.mjs +11 -11
- package/esm2020/lib/components/select/multi/elder-multi-select-base.mjs +103 -39
- package/esm2020/lib/components/select/multi/elder-multi-select-chips/elder-multi-select-chips.component.mjs +13 -13
- package/fesm2015/elderbyte-ngx-starter.mjs +226 -144
- package/fesm2015/elderbyte-ngx-starter.mjs.map +1 -1
- package/fesm2020/elderbyte-ngx-starter.mjs +226 -144
- package/fesm2020/elderbyte-ngx-starter.mjs.map +1 -1
- package/lib/common/utils/collection-util.d.ts +1 -0
- package/lib/components/select/{auto-select-first.directive.d.ts → auto/elder-auto-select-first.directive.d.ts} +4 -13
- package/lib/components/select/auto/elder-select-first-util.d.ts +22 -0
- package/lib/components/select/elder-select/elder-select.component.d.ts +11 -1
- package/lib/components/select/{select-on-tab.directive.d.ts → elder-select-on-tab.directive.d.ts} +3 -3
- package/lib/components/select/elder-select.module.d.ts +5 -5
- package/lib/components/select/multi/elder-multi-select-base.d.ts +39 -15
- package/lib/components/select/multi/elder-multi-select-chips/elder-multi-select-chips.component.d.ts +11 -11
- package/package.json +1 -1
- package/esm2020/lib/components/select/auto-select-first.directive.mjs +0 -126
- 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":"AAAA,OAAO,EAAgB,SAAS,EAAQ,YAAY,EAAE,MAAM,EAAa,QAAQ,EAAC,MAAM,eAAe,CAAC;AAExG,OAAO,EAAC,OAAO,EAAC,MAAM,uBAAuB,CAAC;AAC9C,OAAO,EAAC,KAAK,EAAE,GAAG,EAAE,SAAS,EAAC,MAAM,gBAAgB,CAAC;AACrD,OAAO,EAAC,KAAK,EAAC,MAAM,MAAM,CAAC;AAC3B,OAAO,EAAC,aAAa,EAAC,MAAM,sBAAsB,CAAC;AAEnD,OAAO,EAAC,iBAAiB,EAAkB,MAAM,qBAAqB,CAAC;AAEvE,OAAO,EAAC,0BAA0B,EAAC,MAAM,+BAA+B,CAAC;;;;AAKzE,MAAM,OAAO,yBAAyB;IAyBpC;;;;gFAI4E;IAE5E,YACmB,WAAmC,EACb,eAA+C;QADrE,gBAAW,GAAX,WAAW,CAAwB;QA9BtD;;;;oFAI4E;QAE3D,WAAM,GAAG,aAAa,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;QAExD,aAAQ,GAAG,IAAI,OAAO,EAAS,CAAC;QAIjD;;WAEG;QACK,cAAS,GAAG,KAAK,CAAC;QAE1B;;;WAGG;QACK,cAAS,GAAG,KAAK,CAAC;QAYxB,IAAI,CAAC,oBAAoB,GAAG,eAAe,CAAC;IAC9C,CAAC;IAED;;;;gFAI4E;IAGrE,MAAM;QAEX,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE;YAAE,OAAO;SAAE;QAEhC,IAAI,IAAI,CAAC,oBAAoB,CAAC,KAAK,EAAE;YACnC,kHAAkH;YAClH,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE;gBACnB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,uFAAuF,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;gBAC1H,OAAO;aACR;SACF;QAED,MAAM,YAAY,GAAG,IAAI,CAAC,WAAW,CAAC,YAAkC,CAAC;QAEzE,IAAI,YAAY,EAAE;YAChB,MAAM,MAAM,GAAG,YAAY,CAAC,KAAK,CAAC;YAClC,IAAI,CAAC,mBAAmB,CAAC,MAAM,CAAC,CAAC;YACjC,IAAI,CAAC,WAAW,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;YACpC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,iCAAiC,EAAE,MAAM,CAAC,CAAC;SAC7D;IACH,CAAC;IAEO,mBAAmB,CAAC,MAAe;QACzC,IAAI,0BAA0B,CAAC,IAAI,CAAC,oBAAoB,CAAC,EAAE;YACzD,iEAAiE;YACjE,iDAAiD;YACjD,IAAI,CAAC,oBAAoB,CAAC,mBAAmB,CAAC,MAAM,CAAC,CAAC;SACvD;aAAM;YACL,6DAA6D;YAC7D,IAAI,CAAC,oBAAoB,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;SAC/C;IACH,CAAC;IAED;;;;gFAI4E;IAErE,eAAe;QACpB,MAAM,YAAY,GAAG,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC;QACnD,KAAK,CACH,YAAY,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,CAAC,EACzC,YAAY,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,CAAC,CAC3C,CAAC,IAAI,CACJ,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,EACxB,KAAK,CAAC,CAAC,CAAC,CACT,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC,CAAC;QAE7C,IAAI,CAAC,WAAW,CAAC,gBAAgB,CAAC,IAAI,CAAE,wDAAwD;QAC9F,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,EACxB,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,WAAW,CAAC,CAC5B,CAAC,SAAS,CAAC,WAAW,CAAC,EAAE,CAAC,IAAI,CAAC,SAAS,GAAG,WAAW,CAAC,CAAC;IAC3D,CAAC;IAEM,WAAW;QAChB,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;IAC3B,CAAC;;sHAvGU,yBAAyB,wDAiCd,iBAAiB;0GAjC5B,yBAAyB;2FAAzB,yBAAyB;kBAHrC,SAAS;mBAAC;oBACT,QAAQ,EAAE,oBAAoB;iBAC/B;;0BAkCI,QAAQ;;0BAAI,MAAM;2BAAC,iBAAiB;4CAYhC,MAAM;sBADZ,YAAY;uBAAC,aAAa","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 {
|
|
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 {
|
|
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 {
|
|
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 {
|
|
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
|
-
|
|
46
|
+
ElderSelectOnTabDirective,
|
|
47
47
|
ElderMultiSelectChipsComponent,
|
|
48
48
|
ElderSelectChipDirective,
|
|
49
|
-
|
|
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
|
-
|
|
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
|
-
|
|
97
|
+
ElderSelectOnTabDirective,
|
|
98
98
|
ElderMultiSelectChipsComponent,
|
|
99
99
|
ElderSelectChipDirective,
|
|
100
|
-
|
|
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
|
-
|
|
109
|
+
ElderAutoSelectFirstDirective
|
|
110
110
|
]
|
|
111
111
|
}]
|
|
112
112
|
}] });
|
|
113
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
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.
|
|
33
|
-
this.
|
|
34
|
-
|
|
35
|
-
|
|
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
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
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
|
-
|
|
58
|
-
if (this.
|
|
59
|
-
|
|
50
|
+
set entityIds(ids) {
|
|
51
|
+
if (this.valueAsId) {
|
|
52
|
+
this.writeValueInternal(ids);
|
|
60
53
|
}
|
|
61
54
|
else {
|
|
62
|
-
|
|
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.
|
|
72
|
-
this.
|
|
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
|
-
|
|
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
|
-
|
|
87
|
-
|
|
88
|
-
|
|
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
|
|
91
|
-
if (
|
|
92
|
-
this.writeValueInternal([]);
|
|
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(
|
|
105
|
-
this.
|
|
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: {
|
|
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: {
|
|
198
|
+
}], ctorParameters: function () { return [{ type: i0.NgZone }]; }, propDecorators: { entityIdsChange: [{
|
|
199
|
+
type: Output
|
|
200
|
+
}], entityIdsUpdated: [{
|
|
141
201
|
type: Output
|
|
142
|
-
}],
|
|
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"]}
|