@elderbyte/ngx-starter 13.7.0 → 13.7.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.
- 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 +13 -8
- 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 +11 -5
- package/fesm2015/elderbyte-ngx-starter.mjs +119 -97
- package/fesm2015/elderbyte-ngx-starter.mjs.map +1 -1
- package/fesm2020/elderbyte-ngx-starter.mjs +119 -97
- package/fesm2020/elderbyte-ngx-starter.mjs.map +1 -1
- 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 +11 -1
- 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,77 @@
|
|
|
1
|
+
import { Directive, Input } from '@angular/core';
|
|
2
|
+
import { LoggerFactory } from '@elderbyte/ts-logger';
|
|
3
|
+
import { coerceBooleanProperty } from '@angular/cdk/coercion';
|
|
4
|
+
import { ElderSelectFirstUtil } from './elder-select-first-util';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
import * as i1 from "../elder-select/elder-select.component";
|
|
7
|
+
export class ElderAutoSelectFirstDirective {
|
|
8
|
+
/***************************************************************************
|
|
9
|
+
* *
|
|
10
|
+
* Constructor *
|
|
11
|
+
* *
|
|
12
|
+
**************************************************************************/
|
|
13
|
+
constructor(elderSelect) {
|
|
14
|
+
this.elderSelect = elderSelect;
|
|
15
|
+
/***************************************************************************
|
|
16
|
+
* *
|
|
17
|
+
* Fields *
|
|
18
|
+
* *
|
|
19
|
+
**************************************************************************/
|
|
20
|
+
this.log = LoggerFactory.getLogger(this.constructor.name);
|
|
21
|
+
}
|
|
22
|
+
/***************************************************************************
|
|
23
|
+
* *
|
|
24
|
+
* Life Cycle *
|
|
25
|
+
* *
|
|
26
|
+
**************************************************************************/
|
|
27
|
+
ngAfterViewInit() {
|
|
28
|
+
setTimeout(() => {
|
|
29
|
+
this.log.info('autoSelectFirst after-view init, auto selecting value');
|
|
30
|
+
this.handleAutoSelectFirst();
|
|
31
|
+
}, 10);
|
|
32
|
+
}
|
|
33
|
+
/***************************************************************************
|
|
34
|
+
* *
|
|
35
|
+
* Properties *
|
|
36
|
+
* *
|
|
37
|
+
**************************************************************************/
|
|
38
|
+
/**
|
|
39
|
+
* After data is loaded into this select, ensure that
|
|
40
|
+
* a value is selected or select the first automatically.
|
|
41
|
+
*/
|
|
42
|
+
set autoSelectFirstEnabled(val) {
|
|
43
|
+
this.enabled = coerceBooleanProperty(val);
|
|
44
|
+
}
|
|
45
|
+
set enabled(val) {
|
|
46
|
+
this._enabled = val;
|
|
47
|
+
}
|
|
48
|
+
get enabled() {
|
|
49
|
+
return this._enabled;
|
|
50
|
+
}
|
|
51
|
+
/***************************************************************************
|
|
52
|
+
* *
|
|
53
|
+
* Private methods *
|
|
54
|
+
* *
|
|
55
|
+
**************************************************************************/
|
|
56
|
+
handleAutoSelectFirst() {
|
|
57
|
+
if (!this.elderSelect.value) {
|
|
58
|
+
this.log.debug('Attempting to auto select first entity...');
|
|
59
|
+
ElderSelectFirstUtil.trySelectFirst(this.elderSelect);
|
|
60
|
+
}
|
|
61
|
+
else {
|
|
62
|
+
this.log.warn('Conflict avoided: Aborted auto selection because value id already present!');
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
ElderAutoSelectFirstDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.1", ngImport: i0, type: ElderAutoSelectFirstDirective, deps: [{ token: i1.ElderSelectComponent }], target: i0.ɵɵFactoryTarget.Directive });
|
|
67
|
+
ElderAutoSelectFirstDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.3.1", type: ElderAutoSelectFirstDirective, selector: "[autoSelectFirst]", inputs: { autoSelectFirstEnabled: ["autoSelectFirst", "autoSelectFirstEnabled"] }, ngImport: i0 });
|
|
68
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.1", ngImport: i0, type: ElderAutoSelectFirstDirective, decorators: [{
|
|
69
|
+
type: Directive,
|
|
70
|
+
args: [{
|
|
71
|
+
selector: '[autoSelectFirst]'
|
|
72
|
+
}]
|
|
73
|
+
}], ctorParameters: function () { return [{ type: i1.ElderSelectComponent }]; }, propDecorators: { autoSelectFirstEnabled: [{
|
|
74
|
+
type: Input,
|
|
75
|
+
args: ['autoSelectFirst']
|
|
76
|
+
}] } });
|
|
77
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZWxkZXItYXV0by1zZWxlY3QtZmlyc3QuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvZWxkZXJieXRlL25neC1zdGFydGVyL3NyYy9saWIvY29tcG9uZW50cy9zZWxlY3QvYXV0by9lbGRlci1hdXRvLXNlbGVjdC1maXJzdC5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFnQixTQUFTLEVBQUUsS0FBSyxFQUFDLE1BQU0sZUFBZSxDQUFDO0FBQzlELE9BQU8sRUFBQyxhQUFhLEVBQUMsTUFBTSxzQkFBc0IsQ0FBQztBQUNuRCxPQUFPLEVBQUMscUJBQXFCLEVBQUMsTUFBTSx1QkFBdUIsQ0FBQztBQU81RCxPQUFPLEVBQUMsb0JBQW9CLEVBQUMsTUFBTSwyQkFBMkIsQ0FBQzs7O0FBSy9ELE1BQU0sT0FBTyw2QkFBNkI7SUFZeEM7Ozs7Z0ZBSTRFO0lBRTVFLFlBQ2tCLFdBQW9EO1FBQXBELGdCQUFXLEdBQVgsV0FBVyxDQUF5QztRQWpCdEU7Ozs7b0ZBSTRFO1FBRTNELFFBQUcsR0FBRyxhQUFhLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLENBQUM7SUFZbEUsQ0FBQztJQUVMOzs7O2dGQUk0RTtJQUVyRSxlQUFlO1FBQ3BCLFVBQVUsQ0FBQyxHQUFHLEVBQUU7WUFDZCxJQUFJLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyx1REFBdUQsQ0FBQyxDQUFDO1lBQ3ZFLElBQUksQ0FBQyxxQkFBcUIsRUFBRSxDQUFDO1FBQy9CLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQztJQUNULENBQUM7SUFFRDs7OztnRkFJNEU7SUFFNUU7OztPQUdHO0lBQ0gsSUFDVyxzQkFBc0IsQ0FBQyxHQUFxQjtRQUNyRCxJQUFJLENBQUMsT0FBTyxHQUFHLHFCQUFxQixDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQzVDLENBQUM7SUFFRCxJQUFXLE9BQU8sQ0FBQyxHQUFZO1FBQzdCLElBQUksQ0FBQyxRQUFRLEdBQUcsR0FBRyxDQUFDO0lBQ3RCLENBQUM7SUFFRCxJQUFXLE9BQU87UUFDaEIsT0FBTyxJQUFJLENBQUMsUUFBUSxDQUFDO0lBQ3ZCLENBQUM7SUFFRDs7OztnRkFJNEU7SUFFcEUscUJBQXFCO1FBQzNCLElBQUksQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLEtBQUssRUFBRTtZQUMzQixJQUFJLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQywyQ0FBMkMsQ0FBQyxDQUFDO1lBQzVELG9CQUFvQixDQUFDLGNBQWMsQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLENBQUM7U0FDdkQ7YUFBTTtZQUNMLElBQUksQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLDRFQUE0RSxDQUFDLENBQUM7U0FDN0Y7SUFDSCxDQUFDOzswSEF2RVUsNkJBQTZCOzhHQUE3Qiw2QkFBNkI7MkZBQTdCLDZCQUE2QjtrQkFIekMsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsbUJBQW1CO2lCQUM5QjsyR0ErQ1ksc0JBQXNCO3NCQURoQyxLQUFLO3VCQUFDLGlCQUFpQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7QWZ0ZXJWaWV3SW5pdCwgRGlyZWN0aXZlLCBJbnB1dH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge0xvZ2dlckZhY3Rvcnl9IGZyb20gJ0BlbGRlcmJ5dGUvdHMtbG9nZ2VyJztcbmltcG9ydCB7Y29lcmNlQm9vbGVhblByb3BlcnR5fSBmcm9tICdAYW5ndWxhci9jZGsvY29lcmNpb24nO1xuaW1wb3J0IHtFTVBUWSwgT2JzZXJ2YWJsZSwgb2Z9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHtjYXRjaEVycm9yLCBmaWx0ZXIsIGZpcnN0LCBtYXAsIHN3aXRjaE1hcCwgdGFrZVVudGlsLCB0YXAsIHRpbWVvdXR9IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcbmltcG9ydCB7RWxkZXJTZWxlY3RDb21wb25lbnRTdGF0ZX0gZnJvbSAnLi4vZWxkZXItc2VsZWN0LWJhc2UnO1xuaW1wb3J0IHtJRGF0YUNvbnRleHR9IGZyb20gJy4uLy4uLy4uL2NvbW1vbi9kYXRhL2RhdGEtY29udGV4dC9kYXRhLWNvbnRleHQnO1xuaW1wb3J0IHtEYXRhU291cmNlQWRhcHRlcn0gZnJvbSAnLi4vLi4vLi4vY29tbW9uL2RhdGEvZGF0YXNvdXJjZS9kYXRhLXNvdXJjZS1hZGFwdGVyJztcbmltcG9ydCB7RWxkZXJTZWxlY3RDb21wb25lbnR9IGZyb20gJy4uL2VsZGVyLXNlbGVjdC9lbGRlci1zZWxlY3QuY29tcG9uZW50JztcbmltcG9ydCB7RWxkZXJTZWxlY3RGaXJzdFV0aWx9IGZyb20gJy4vZWxkZXItc2VsZWN0LWZpcnN0LXV0aWwnO1xuXG5ARGlyZWN0aXZlKHtcbiAgc2VsZWN0b3I6ICdbYXV0b1NlbGVjdEZpcnN0XSdcbn0pXG5leHBvcnQgY2xhc3MgRWxkZXJBdXRvU2VsZWN0Rmlyc3REaXJlY3RpdmU8VEVudGl0eSA9IGFueSwgVElkID0gYW55PiBpbXBsZW1lbnRzIEFmdGVyVmlld0luaXQge1xuXG4gIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKipcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiBGaWVsZHMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqL1xuXG4gIHByaXZhdGUgcmVhZG9ubHkgbG9nID0gTG9nZ2VyRmFjdG9yeS5nZXRMb2dnZXIodGhpcy5jb25zdHJ1Y3Rvci5uYW1lKTtcblxuICBwcml2YXRlIF9lbmFibGVkOiBib29sZWFuO1xuXG4gIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKipcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiBDb25zdHJ1Y3RvciAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqL1xuXG4gIGNvbnN0cnVjdG9yKFxuICAgIHB1YmxpYyByZWFkb25seSBlbGRlclNlbGVjdDogRWxkZXJTZWxlY3RDb21wb25lbnQ8VEVudGl0eSwgVElkLCBhbnk+XG4gICkgeyB9XG5cbiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqIExpZmUgQ3ljbGUgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgcHVibGljIG5nQWZ0ZXJWaWV3SW5pdCgpOiB2b2lkIHtcbiAgICBzZXRUaW1lb3V0KCgpID0+IHsgLy8gSEFDSzogV2FpdCB1bnRpbCBuZ01vZGVsIGhhcyBkb25lIGl0cyBzdHJhbmdlIG51bGwvZW1wdHkgdmFsdWUgaW5pdC4uLlxuICAgICAgdGhpcy5sb2cuaW5mbygnYXV0b1NlbGVjdEZpcnN0IGFmdGVyLXZpZXcgaW5pdCwgYXV0byBzZWxlY3RpbmcgdmFsdWUnKTtcbiAgICAgIHRoaXMuaGFuZGxlQXV0b1NlbGVjdEZpcnN0KCk7XG4gICAgfSwgMTApO1xuICB9XG5cbiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqIFByb3BlcnRpZXMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgLyoqXG4gICAqIEFmdGVyIGRhdGEgaXMgbG9hZGVkIGludG8gdGhpcyBzZWxlY3QsIGVuc3VyZSB0aGF0XG4gICAqIGEgdmFsdWUgaXMgc2VsZWN0ZWQgb3Igc2VsZWN0IHRoZSBmaXJzdCBhdXRvbWF0aWNhbGx5LlxuICAgKi9cbiAgQElucHV0KCdhdXRvU2VsZWN0Rmlyc3QnKVxuICBwdWJsaWMgc2V0IGF1dG9TZWxlY3RGaXJzdEVuYWJsZWQodmFsOiBib29sZWFuIHwgc3RyaW5nKSB7XG4gICAgdGhpcy5lbmFibGVkID0gY29lcmNlQm9vbGVhblByb3BlcnR5KHZhbCk7XG4gIH1cblxuICBwdWJsaWMgc2V0IGVuYWJsZWQodmFsOiBib29sZWFuKSB7XG4gICAgdGhpcy5fZW5hYmxlZCA9IHZhbDtcbiAgfVxuXG4gIHB1YmxpYyBnZXQgZW5hYmxlZCgpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5fZW5hYmxlZDtcbiAgfVxuXG4gIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKipcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiBQcml2YXRlIG1ldGhvZHMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqL1xuXG4gIHByaXZhdGUgaGFuZGxlQXV0b1NlbGVjdEZpcnN0KCk6IHZvaWQge1xuICAgIGlmICghdGhpcy5lbGRlclNlbGVjdC52YWx1ZSkge1xuICAgICAgdGhpcy5sb2cuZGVidWcoJ0F0dGVtcHRpbmcgdG8gYXV0byBzZWxlY3QgZmlyc3QgZW50aXR5Li4uJyk7XG4gICAgICBFbGRlclNlbGVjdEZpcnN0VXRpbC50cnlTZWxlY3RGaXJzdCh0aGlzLmVsZGVyU2VsZWN0KTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5sb2cud2FybignQ29uZmxpY3QgYXZvaWRlZDogQWJvcnRlZCBhdXRvIHNlbGVjdGlvbiBiZWNhdXNlIHZhbHVlIGlkIGFscmVhZHkgcHJlc2VudCEnKTtcbiAgICB9XG4gIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
import { EMPTY, of } from 'rxjs';
|
|
2
|
+
import { catchError, filter, first, map, switchMap, takeUntil, tap, timeout } from 'rxjs/operators';
|
|
3
|
+
import { ElderSelectComponentState } from '../elder-select-base';
|
|
4
|
+
import { DataSourceAdapter } from '../../../common/data/datasource/data-source-adapter';
|
|
5
|
+
import { LoggerFactory } from '@elderbyte/ts-logger';
|
|
6
|
+
export class ElderSelectFirstUtil {
|
|
7
|
+
/***************************************************************************
|
|
8
|
+
* *
|
|
9
|
+
* Public methods *
|
|
10
|
+
* *
|
|
11
|
+
**************************************************************************/
|
|
12
|
+
static trySelectFirst(elderSelect) {
|
|
13
|
+
const suggestionsDc = elderSelect.suggestionsDc;
|
|
14
|
+
const dataContextFirstItem = suggestionsDc?.isStarted
|
|
15
|
+
? this.firstItemInDataContextSoon(suggestionsDc)
|
|
16
|
+
: of(null);
|
|
17
|
+
dataContextFirstItem.pipe(takeUntil(elderSelect.valueChange.pipe(filter(value => !!value))), switchMap(firstItem => firstItem ? of(firstItem) : this.firstSuggestionItem(elderSelect))).subscribe(firstItem => {
|
|
18
|
+
if (firstItem && !elderSelect.entityId) {
|
|
19
|
+
elderSelect.updateValueByEntity(firstItem);
|
|
20
|
+
}
|
|
21
|
+
}, err => {
|
|
22
|
+
elderSelect.updateState(ElderSelectComponentState.error(err));
|
|
23
|
+
this.log.error('Failed to load first entity for auto-select-first flow!', err);
|
|
24
|
+
});
|
|
25
|
+
}
|
|
26
|
+
/***************************************************************************
|
|
27
|
+
* *
|
|
28
|
+
* Private methods *
|
|
29
|
+
* *
|
|
30
|
+
**************************************************************************/
|
|
31
|
+
static firstItemInDataContextSoon(dataContext) {
|
|
32
|
+
return dataContext.data.pipe(filter(data => data && data.length > 0), map(data => data[0]), timeout(250), catchError(timeoutError => {
|
|
33
|
+
this.log.warn('Gave up waiting for DataContext items, timeout reached. Falling back to sugestion lookup.', timeoutError);
|
|
34
|
+
return of(null);
|
|
35
|
+
}), first());
|
|
36
|
+
}
|
|
37
|
+
static firstSuggestionItem(elderSelect) {
|
|
38
|
+
const suggestionsDc = elderSelect.suggestionsDc;
|
|
39
|
+
const dataSource = suggestionsDc?.dataSource;
|
|
40
|
+
if (dataSource) {
|
|
41
|
+
elderSelect.updateState(ElderSelectComponentState.loading());
|
|
42
|
+
return DataSourceAdapter.from(dataSource)
|
|
43
|
+
.findFirst(elderSelect.filters, elderSelect.sorts).pipe(tap(() => elderSelect.updateState(ElderSelectComponentState.idle())), catchError(err => {
|
|
44
|
+
this.log.error('Failed to load suggestions!', err);
|
|
45
|
+
elderSelect.updateState(ElderSelectComponentState.error(err));
|
|
46
|
+
return of(null);
|
|
47
|
+
}));
|
|
48
|
+
}
|
|
49
|
+
else {
|
|
50
|
+
this.log.warn('Failed to load first suggestion!');
|
|
51
|
+
elderSelect.updateState(ElderSelectComponentState.idle());
|
|
52
|
+
return EMPTY;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
/***************************************************************************
|
|
57
|
+
* *
|
|
58
|
+
* Fields *
|
|
59
|
+
* *
|
|
60
|
+
**************************************************************************/
|
|
61
|
+
ElderSelectFirstUtil.log = LoggerFactory.getLogger('ElderSelectFirstUtil');
|
|
62
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-select-first-util.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/select/auto/elder-select-first-util.ts"],"names":[],"mappings":"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","sourcesContent":["import {EMPTY, Observable, of} from 'rxjs';\nimport {catchError, filter, first, map, switchMap, takeUntil, tap, timeout} from 'rxjs/operators';\nimport {ElderSelectComponentState} from '../elder-select-base';\nimport {IDataContext} from '../../../common/data/data-context/data-context';\nimport {DataSourceAdapter} from '../../../common/data/datasource/data-source-adapter';\nimport {ElderSelectComponent} from '../elder-select/elder-select.component';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\n\nexport class ElderSelectFirstUtil {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private static readonly log = LoggerFactory.getLogger('ElderSelectFirstUtil');\n\n  /***************************************************************************\n   *                                                                         *\n   * Public methods                                                          *\n   *                                                                         *\n   **************************************************************************/\n\n  public static trySelectFirst<TEntity, TId>(\n    elderSelect: ElderSelectComponent<TEntity, TId, any>\n  ): void {\n\n    const suggestionsDc = elderSelect.suggestionsDc;\n\n    const dataContextFirstItem: Observable<TEntity> = suggestionsDc?.isStarted\n      ? this.firstItemInDataContextSoon(suggestionsDc)\n      : of(null);\n\n    dataContextFirstItem.pipe(\n      takeUntil(elderSelect.valueChange.pipe(filter(value => !!value))),\n      switchMap(firstItem => firstItem ? of(firstItem) : this.firstSuggestionItem(elderSelect))\n    ).subscribe(\n      firstItem => {\n        if (firstItem && !elderSelect.entityId) {\n          elderSelect.updateValueByEntity(firstItem);\n        }\n      },\n      err => {\n        elderSelect.updateState(ElderSelectComponentState.error(err));\n        this.log.error('Failed to load first entity for auto-select-first flow!', err);\n      }\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private static firstItemInDataContextSoon<TEntity>(dataContext: IDataContext<TEntity>): Observable<TEntity | null> {\n    return dataContext.data.pipe(\n      filter(data => data && data.length > 0),\n      map(data => data[0]),\n      timeout(250),\n      catchError(timeoutError => {\n        this.log.warn('Gave up waiting for DataContext items, timeout reached. Falling back to sugestion lookup.', timeoutError);\n        return of(null);\n      }),\n      first()\n    );\n  }\n\n  private static firstSuggestionItem<TEntity, TId>(\n    elderSelect: ElderSelectComponent<TEntity, TId, any>\n  ): Observable<TEntity | null> {\n\n    const suggestionsDc = elderSelect.suggestionsDc;\n    const dataSource = suggestionsDc?.dataSource;\n\n    if (dataSource) {\n\n      elderSelect.updateState(ElderSelectComponentState.loading());\n\n      return DataSourceAdapter.from(dataSource)\n        .findFirst(elderSelect.filters, elderSelect.sorts).pipe(\n          tap(() => elderSelect.updateState(ElderSelectComponentState.idle())),\n          catchError(err => {\n            this.log.error('Failed to load suggestions!', err);\n            elderSelect.updateState(ElderSelectComponentState.error(err));\n            return of(null);\n          }),\n        );\n    } else {\n      this.log.warn('Failed to load first suggestion!');\n      elderSelect.updateState(ElderSelectComponentState.idle());\n      return EMPTY;\n    }\n  }\n\n\n}\n"]}
|
|
@@ -2,10 +2,10 @@ import { ChangeDetectionStrategy, Component, forwardRef, Input, Output, } from '
|
|
|
2
2
|
import { buildFormIntegrationProviders } from '../../../common/forms/template-composite-control';
|
|
3
3
|
import { LoggerFactory } from '@elderbyte/ts-logger';
|
|
4
4
|
import { BehaviorSubject, combineLatest, Subject } from 'rxjs';
|
|
5
|
-
import { map, takeUntil, tap, skip, } from 'rxjs/operators';
|
|
5
|
+
import { map, takeUntil, tap, skip, filter, } from 'rxjs/operators';
|
|
6
6
|
import { coerceBooleanProperty } from '@angular/cdk/coercion';
|
|
7
7
|
import { ELDER_SELECT_BASE, ElderSelectBase, ElderSelectComponentState } from '../elder-select-base';
|
|
8
|
-
import {
|
|
8
|
+
import { ElderSelectFirstUtil } from '../auto/elder-select-first-util';
|
|
9
9
|
import * as i0 from "@angular/core";
|
|
10
10
|
import * as i1 from "@angular/material/form-field";
|
|
11
11
|
import * as i2 from "@angular/material/icon";
|
|
@@ -16,7 +16,7 @@ import * as i6 from "@angular/flex-layout/flex";
|
|
|
16
16
|
import * as i7 from "@angular/material/input";
|
|
17
17
|
import * as i8 from "@angular/material/autocomplete";
|
|
18
18
|
import * as i9 from "@angular/forms";
|
|
19
|
-
import * as i10 from "../select-on-tab.directive";
|
|
19
|
+
import * as i10 from "../elder-select-on-tab.directive";
|
|
20
20
|
import * as i11 from "../../input/autocomplete/elder-autocomplete.directive";
|
|
21
21
|
import * as i12 from "../../forms/directives/elder-stop-event-propagation.directive";
|
|
22
22
|
import * as i13 from "@ngx-translate/core";
|
|
@@ -63,6 +63,8 @@ export class ElderSelectComponent extends ElderSelectBase {
|
|
|
63
63
|
this.entityIdChange = this.valueChange.pipe(map(v => this.entityIdFromValue(v)));
|
|
64
64
|
this.entityChange = this.entity$.pipe(skip(1) // Skip the initial NULL value
|
|
65
65
|
);
|
|
66
|
+
this.entityIdUpdated = this.valueUpdated.pipe(map(value => this.entityIdFromValue(value)));
|
|
67
|
+
this.entityUpdated = combineLatest([this.entityIdUpdated, this.entity$]).pipe(filter(([updatedId, entity]) => this.getEntityId(entity) === updatedId), map(([updatedId, entity]) => entity));
|
|
66
68
|
this.entityWrapped$ = combineLatest([
|
|
67
69
|
this.entity$,
|
|
68
70
|
this.displayPropertyResolver$,
|
|
@@ -221,8 +223,7 @@ export class ElderSelectComponent extends ElderSelectBase {
|
|
|
221
223
|
}
|
|
222
224
|
forceReloadFirst() {
|
|
223
225
|
this.value = null; // valueId is set to null automatically
|
|
224
|
-
|
|
225
|
-
autoReloadFirst.trySelectFirst();
|
|
226
|
+
ElderSelectFirstUtil.trySelectFirst(this);
|
|
226
227
|
}
|
|
227
228
|
updateValueByEntity(entity) {
|
|
228
229
|
this.updateValue(this.entityToValue(entity));
|
|
@@ -351,13 +352,13 @@ export class ElderSelectComponent extends ElderSelectBase {
|
|
|
351
352
|
}
|
|
352
353
|
}
|
|
353
354
|
ElderSelectComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.1", ngImport: i0, type: ElderSelectComponent, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component });
|
|
354
|
-
ElderSelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.1", type: ElderSelectComponent, selector: "elder-select", inputs: { nullDisplay: "nullDisplay", autocomplete: "autocomplete", allowNull: "allowNull", entity: "entity", entityId: "entityId", hintProperty: "hintProperty", hintPropertyResolver: "hintPropertyResolver" }, outputs: { entityIdChange: "entityIdChange", entityChange: "entityChange" }, providers: [
|
|
355
|
+
ElderSelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.1", type: ElderSelectComponent, selector: "elder-select", inputs: { nullDisplay: "nullDisplay", autocomplete: "autocomplete", allowNull: "allowNull", entity: "entity", entityId: "entityId", hintProperty: "hintProperty", hintPropertyResolver: "hintPropertyResolver" }, outputs: { entityIdChange: "entityIdChange", entityChange: "entityChange", entityIdUpdated: "entityIdUpdated", entityUpdated: "entityUpdated" }, providers: [
|
|
355
356
|
{
|
|
356
357
|
provide: ELDER_SELECT_BASE,
|
|
357
358
|
useExisting: forwardRef(() => ElderSelectComponent)
|
|
358
359
|
},
|
|
359
360
|
...buildFormIntegrationProviders(ElderSelectComponent)
|
|
360
|
-
], usesInheritance: true, ngImport: i0, template: "<mat-form-field *ngIf=\"(entityWrapped$ | async) as entityWrapper\" fxFlex\n class=\"elder-std-form-field\"\n [appearance]=\"appearance\"\n [floatLabel]=\"floatLabel\"\n [color]=\"color\"\n>\n\n <mat-label *ngIf=\"label\">{{label | translate}}</mat-label>\n\n\n <ng-container matPrefix *ngIf=\"(mergedState$ | async) as state\">\n\n <mat-icon *ngIf=\"icon\" disabled\n class=\"leading-icon prefix-padding noselect\"\n [class.loading]=\"state.loading\"\n [color]=\"state?.error ? 'warn' : color\">\n {{icon}}\n </mat-icon>\n\n\n <mat-icon *ngIf=\"!icon && state?.error\"\n class=\"leading-icon prefix-padding noselect\"\n color=\"warn\">\n warning\n </mat-icon>\n </ng-container>\n\n <!-- A dynamic input -->\n <input\n matInput type=\"text\" fxFlex=\"grow\"\n [disabled]=\"disabled\"\n [required]=\"required\"\n [readonly]=\"readonly || !autocomplete\"\n [name]=\"name + '-inner-input'\"\n [placeholder]=\"placeholder | translate\"\n [matAutocomplete] #autoTrigger=\"matAutocompleteTrigger\"\n [elderElderAutocomplete]=\"elderAuto\" [queryFilter]=\"queryFilter\" [filters]=\"filters\" [sorts]=\"sorts\"\n elderSelectOnTab\n [class.select]=\"!autocomplete\" (click)=\"onInputClicked(autoTrigger)\"\n [ngModel]=\"inputText$ | async\" [ngModelOptions]=\"{standalone: true, updateOn: 'submit'}\"\n (blur)=\"onInputBlur($event)\"\n >\n\n <!-- This breaks stuff: [displayWith]=\"displayPropertyResolver$ | async\" -->\n\n <elder-autocomplete\n #elderAuto=\"elderAutocomplete\"\n [suggestionsDc]=\"suggestionsDc$ | async\"\n [valueTemplate]=\"valueTemplate\"\n [enabled]=\"autocomplete\"\n [displayPropertyResolver]=\"displayPropertyResolver$ | async\"\n [isOptionDisabledFn]=\"isOptionDisabledInternalFn\"\n [isOptionHiddenFn]=\"isOptionHiddenInternalFn\"\n (optionSelected)=\"onOptionSelected($any($event))\"\n ></elder-autocomplete>\n\n\n <ng-container matSuffix>\n\n <mat-icon\n *ngIf=\"!autocomplete\" class=\"select-arrow noselect\"\n (click)=\"onInputClicked(autoTrigger)\">\n arrow_drop_down\n </mat-icon>\n\n <button mat-icon-button type=\"button\"\n *ngIf=\"selectionPopup && (!entityWrapper.value || !allowNull)\"\n [disabled]=\"isLocked\"\n (click)=\"openSelectionPopup($event)\" aria-label=\"Search\"\n elderStopEventPropagation\n tabIndex=\"-1\"\n >\n <mat-icon>search</mat-icon>\n </button>\n\n <button mat-icon-button type=\"button\"\n *ngIf=\"entityWrapper.value && allowNull\"\n [disabled]=\"isLocked\"\n (click)=\"clear($event)\" aria-label=\"Clear\"\n elderStopEventPropagation\n tabIndex=\"-1\"\n >\n <mat-icon>close</mat-icon>\n </button>\n\n </ng-container>\n\n <mat-hint *ngIf=\"entityWrapper.hintText\">{{entityWrapper.hintText}}</mat-hint>\n\n</mat-form-field>\n\n\n\n", styles: [".prefix-container{width:24px;height:16px}@-webkit-keyframes shrink{0%{transform:scale(1)}to{transform:scale(.75)}}@keyframes shrink{0%{transform:scale(1)}to{transform:scale(.75)}}.loading{animation:shrink .3s ease-in-out infinite alternate;-webkit-animation:shrink .3s ease-in-out infinite alternate}.prefix-padding{padding-right:4px}.leading-icon{font-size:16px;width:16px;height:16px}.full-width{width:100%}.select{cursor:pointer;width:162px}.select-arrow{font-size:18px;width:18px;height:18px;cursor:pointer}\n"], components: [{ type: i1.MatFormField, selector: "mat-form-field", inputs: ["color", "appearance", "hideRequiredMarker", "hintLabel", "floatLabel"], exportAs: ["matFormField"] }, { type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i3.ElderAutocompleteComponent, selector: "elder-autocomplete", inputs: ["isOptionDisabledFn", "isOptionHiddenFn", "enabled", "valueTemplate", "suggestionsDc", "displayPropertyResolver"], outputs: ["optionSelected"], exportAs: ["elderAutocomplete"] }, { type: i4.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }], directives: [{ type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.DefaultFlexDirective, selector: " [fxFlex], [fxFlex.xs], [fxFlex.sm], [fxFlex.md], [fxFlex.lg], [fxFlex.xl], [fxFlex.lt-sm], [fxFlex.lt-md], [fxFlex.lt-lg], [fxFlex.lt-xl], [fxFlex.gt-xs], [fxFlex.gt-sm], [fxFlex.gt-md], [fxFlex.gt-lg]", inputs: ["fxFlex", "fxFlex.xs", "fxFlex.sm", "fxFlex.md", "fxFlex.lg", "fxFlex.xl", "fxFlex.lt-sm", "fxFlex.lt-md", "fxFlex.lt-lg", "fxFlex.lt-xl", "fxFlex.gt-xs", "fxFlex.gt-sm", "fxFlex.gt-md", "fxFlex.gt-lg"] }, { type: i1.MatLabel, selector: "mat-label" }, { type: i1.MatPrefix, selector: "[matPrefix]" }, { type: i7.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { type: i8.MatAutocompleteTrigger, selector: "input[matAutocomplete], textarea[matAutocomplete]", exportAs: ["matAutocompleteTrigger"] }, { type: i9.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i10.
|
|
361
|
+
], usesInheritance: true, ngImport: i0, template: "<mat-form-field *ngIf=\"(entityWrapped$ | async) as entityWrapper\" fxFlex\n class=\"elder-std-form-field\"\n [appearance]=\"appearance\"\n [floatLabel]=\"floatLabel\"\n [color]=\"color\"\n>\n\n <mat-label *ngIf=\"label\">{{label | translate}}</mat-label>\n\n\n <ng-container matPrefix *ngIf=\"(mergedState$ | async) as state\">\n\n <mat-icon *ngIf=\"icon\" disabled\n class=\"leading-icon prefix-padding noselect\"\n [class.loading]=\"state.loading\"\n [color]=\"state?.error ? 'warn' : color\">\n {{icon}}\n </mat-icon>\n\n\n <mat-icon *ngIf=\"!icon && state?.error\"\n class=\"leading-icon prefix-padding noselect\"\n color=\"warn\">\n warning\n </mat-icon>\n </ng-container>\n\n <!-- A dynamic input -->\n <input\n matInput type=\"text\" fxFlex=\"grow\"\n [disabled]=\"disabled\"\n [required]=\"required\"\n [readonly]=\"readonly || !autocomplete\"\n [name]=\"name + '-inner-input'\"\n [placeholder]=\"placeholder | translate\"\n [matAutocomplete] #autoTrigger=\"matAutocompleteTrigger\"\n [elderElderAutocomplete]=\"elderAuto\" [queryFilter]=\"queryFilter\" [filters]=\"filters\" [sorts]=\"sorts\"\n elderSelectOnTab\n [class.select]=\"!autocomplete\" (click)=\"onInputClicked(autoTrigger)\"\n [ngModel]=\"inputText$ | async\" [ngModelOptions]=\"{standalone: true, updateOn: 'submit'}\"\n (blur)=\"onInputBlur($event)\"\n >\n\n <!-- This breaks stuff: [displayWith]=\"displayPropertyResolver$ | async\" -->\n\n <elder-autocomplete\n #elderAuto=\"elderAutocomplete\"\n [suggestionsDc]=\"suggestionsDc$ | async\"\n [valueTemplate]=\"valueTemplate\"\n [enabled]=\"autocomplete\"\n [displayPropertyResolver]=\"displayPropertyResolver$ | async\"\n [isOptionDisabledFn]=\"isOptionDisabledInternalFn\"\n [isOptionHiddenFn]=\"isOptionHiddenInternalFn\"\n (optionSelected)=\"onOptionSelected($any($event))\"\n ></elder-autocomplete>\n\n\n <ng-container matSuffix>\n\n <mat-icon\n *ngIf=\"!autocomplete\" class=\"select-arrow noselect\"\n (click)=\"onInputClicked(autoTrigger)\">\n arrow_drop_down\n </mat-icon>\n\n <button mat-icon-button type=\"button\"\n *ngIf=\"selectionPopup && (!entityWrapper.value || !allowNull)\"\n [disabled]=\"isLocked\"\n (click)=\"openSelectionPopup($event)\" aria-label=\"Search\"\n elderStopEventPropagation\n tabIndex=\"-1\"\n >\n <mat-icon>search</mat-icon>\n </button>\n\n <button mat-icon-button type=\"button\"\n *ngIf=\"entityWrapper.value && allowNull\"\n [disabled]=\"isLocked\"\n (click)=\"clear($event)\" aria-label=\"Clear\"\n elderStopEventPropagation\n tabIndex=\"-1\"\n >\n <mat-icon>close</mat-icon>\n </button>\n\n </ng-container>\n\n <mat-hint *ngIf=\"entityWrapper.hintText\">{{entityWrapper.hintText}}</mat-hint>\n\n</mat-form-field>\n\n\n\n", styles: [".prefix-container{width:24px;height:16px}@-webkit-keyframes shrink{0%{transform:scale(1)}to{transform:scale(.75)}}@keyframes shrink{0%{transform:scale(1)}to{transform:scale(.75)}}.loading{animation:shrink .3s ease-in-out infinite alternate;-webkit-animation:shrink .3s ease-in-out infinite alternate}.prefix-padding{padding-right:4px}.leading-icon{font-size:16px;width:16px;height:16px}.full-width{width:100%}.select{cursor:pointer;width:162px}.select-arrow{font-size:18px;width:18px;height:18px;cursor:pointer}\n"], components: [{ type: i1.MatFormField, selector: "mat-form-field", inputs: ["color", "appearance", "hideRequiredMarker", "hintLabel", "floatLabel"], exportAs: ["matFormField"] }, { type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { type: i3.ElderAutocompleteComponent, selector: "elder-autocomplete", inputs: ["isOptionDisabledFn", "isOptionHiddenFn", "enabled", "valueTemplate", "suggestionsDc", "displayPropertyResolver"], outputs: ["optionSelected"], exportAs: ["elderAutocomplete"] }, { type: i4.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }], directives: [{ type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.DefaultFlexDirective, selector: " [fxFlex], [fxFlex.xs], [fxFlex.sm], [fxFlex.md], [fxFlex.lg], [fxFlex.xl], [fxFlex.lt-sm], [fxFlex.lt-md], [fxFlex.lt-lg], [fxFlex.lt-xl], [fxFlex.gt-xs], [fxFlex.gt-sm], [fxFlex.gt-md], [fxFlex.gt-lg]", inputs: ["fxFlex", "fxFlex.xs", "fxFlex.sm", "fxFlex.md", "fxFlex.lg", "fxFlex.xl", "fxFlex.lt-sm", "fxFlex.lt-md", "fxFlex.lt-lg", "fxFlex.lt-xl", "fxFlex.gt-xs", "fxFlex.gt-sm", "fxFlex.gt-md", "fxFlex.gt-lg"] }, { type: i1.MatLabel, selector: "mat-label" }, { type: i1.MatPrefix, selector: "[matPrefix]" }, { type: i7.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { type: i8.MatAutocompleteTrigger, selector: "input[matAutocomplete], textarea[matAutocomplete]", exportAs: ["matAutocompleteTrigger"] }, { type: i9.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i10.ElderSelectOnTabDirective, selector: "[elderSelectOnTab]" }, { type: i9.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { type: i11.ElderAutocompleteDirective, selector: "[elderElderAutocomplete]", inputs: ["queryFilter", "filters", "sorts", "elderElderAutocomplete"] }, { type: i9.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i9.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { type: i1.MatSuffix, selector: "[matSuffix]" }, { type: i12.ElderStopEventPropagationDirective, selector: "[elderStopEventPropagation]" }, { type: i1.MatHint, selector: "mat-hint", inputs: ["align", "id"] }], pipes: { "async": i5.AsyncPipe, "translate": i13.TranslatePipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
361
362
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.1", ngImport: i0, type: ElderSelectComponent, decorators: [{
|
|
362
363
|
type: Component,
|
|
363
364
|
args: [{ selector: 'elder-select', changeDetection: ChangeDetectionStrategy.OnPush, providers: [
|
|
@@ -373,6 +374,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.1", ngImpor
|
|
|
373
374
|
type: Output
|
|
374
375
|
}], entityChange: [{
|
|
375
376
|
type: Output
|
|
377
|
+
}], entityIdUpdated: [{
|
|
378
|
+
type: Output
|
|
379
|
+
}], entityUpdated: [{
|
|
380
|
+
type: Output
|
|
376
381
|
}], autocomplete: [{
|
|
377
382
|
type: Input
|
|
378
383
|
}], allowNull: [{
|
|
@@ -386,4 +391,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.1", ngImpor
|
|
|
386
391
|
}], hintPropertyResolver: [{
|
|
387
392
|
type: Input
|
|
388
393
|
}] } });
|
|
389
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-select.component.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/select/elder-select/elder-select.component.ts","../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/select/elder-select/elder-select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component, forwardRef,\n  Input, NgZone,\n  OnDestroy,\n  OnInit,\n  Output,\n} from '@angular/core';\nimport {buildFormIntegrationProviders} from '../../../common/forms/template-composite-control';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {IDataContext} from '../../../common/data/data-context/data-context';\nimport {BehaviorSubject, combineLatest, Observable, Subject} from 'rxjs';\nimport {\n  map,\n  takeUntil,\n  tap, skip,\n} from 'rxjs/operators';\nimport {coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {MatAutocompleteTrigger} from '@angular/material/autocomplete';\nimport {ELDER_SELECT_BASE, ElderSelectBase, ElderSelectComponentState, TextResolverFn} from '../elder-select-base';\nimport {AutoSelectFirstDirective} from '../auto-select-first.directive';\nimport {IElderEntityValueAccessor} from '../elder-entity-value-accessor';\n\nclass EntityContext<T> {\n  constructor(\n    public readonly value: T,\n    public readonly hintText: string,\n    public readonly displayText: string\n  ) {\n  }\n}\n\n\n/**\n * The elder-select control allows to select a single\n * entity.\n *\n * Primary focus is on easy usage for common scenarios:\n *\n * - Integrated into angular forms and validation\n * - Support entity id handling, input / output is the entity id vs full value.\n *\n */\n@Component({\n  selector: 'elder-select',\n  templateUrl: './elder-select.component.html',\n  styleUrls: ['./elder-select.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    {\n      provide: ELDER_SELECT_BASE,\n      useExisting: forwardRef(() => ElderSelectComponent)\n    },\n    ...buildFormIntegrationProviders(ElderSelectComponent)\n  ]\n})\nexport class ElderSelectComponent<TEntity = any, TId = any, TValue = TEntity | TId>\n  extends ElderSelectBase<TId, TEntity, TValue>\n  implements IElderEntityValueAccessor<TEntity, TId>, OnInit, OnDestroy {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly hintPropertyResolver$ = new BehaviorSubject<TextResolverFn>((value: TEntity) => null);\n\n  private _allowNull = false;\n\n  /**\n   * String to display for the 'nothing' / null value.\n   */\n  @Input()\n  public nullDisplay = '-';\n\n  @Output()\n  public readonly entityIdChange: Observable<TId>;\n\n  @Output()\n  public readonly entityChange: Observable<TEntity>;\n\n  private readonly entity$ = new BehaviorSubject<TEntity>(undefined);\n\n  private _autocomplete = false;\n\n  public readonly entityWrapped$: Observable<EntityContext<TEntity>>;\n\n  private readonly unsubscribe$ = new Subject();\n\n  public readonly inputText$: Observable<string>;\n\n  private entityContext: EntityContext<TEntity>;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    zone: NgZone,\n  ) {\n    super(zone);\n\n    this.entityIdChange = this.valueChange.pipe(\n      map(v => this.entityIdFromValue(v))\n    );\n\n    this.entityChange = this.entity$.pipe(\n      skip(1) // Skip the initial NULL value\n    );\n\n    this.entityWrapped$ = combineLatest(\n      [\n        this.entity$,\n        this.displayPropertyResolver$,\n        this.hintPropertyResolver$\n      ]\n    ).pipe(\n      map(([value, dPR, hPR]) => new EntityContext<TEntity>(value, hPR(value), dPR(value)))\n    );\n\n    this.inputText$ = this.entityWrapped$.pipe(\n      takeUntil(this.unsubscribe$),\n      tap(valueContext => this.entityContext = valueContext),\n      map(valueContext => {\n        if (valueContext.value) {\n          return valueContext.displayText;\n        } else {\n          return null;\n        }\n      })\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n\n  }\n\n  public ngOnDestroy(): void {\n    this.unsubscribe$.next();\n    this.unsubscribe$.complete();\n    this.autoCleanUp();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Behaviour Properties                                                    *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\n  public set autocomplete(value: boolean) {\n    this._autocomplete = coerceBooleanProperty(value);\n  }\n\n  public get autocomplete(): boolean {\n    return this._autocomplete;\n  }\n\n  /**\n   * Allows the user to select 'nothing' / null.\n   */\n  @Input()\n  public set allowNull(val: boolean) {\n    this._allowNull = coerceBooleanProperty(val);\n  }\n\n  public get allowNull(): boolean {\n    return this._allowNull;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\n  public set entity(entity: TEntity) {\n    this.entity$.next(entity);\n    this.writeValueInternal(\n      this.entityToValue(entity)\n    );\n  }\n\n  public get entity(): TEntity {\n    return this.entity$.getValue();\n  }\n\n  @Input()\n  public set entityId(id: TId) {\n    if (this.valueAsId) {\n      this.writeValueInternal(<any>id as TValue);\n    } else {\n      this.selectEntityById(id);\n    }\n  }\n\n  public get entityId(): TId {\n    return this.entityIdFromValue(this.value);\n  }\n\n  @Input()\n  public set hintProperty(property: string) {\n    this.hintPropertyResolver = (value => this.propertyStringValue(value, property));\n  }\n\n  @Input()\n  public set hintPropertyResolver(resolver: TextResolverFn) {\n    this.hintPropertyResolver$.next(resolver);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public onInputBlur(event: FocusEvent): void {\n    const input = event.target as HTMLInputElement;\n    if (this.entityContext.value) {\n      input.value = this.entityContext.displayText;\n    } else {\n      input.value = null;\n    }\n  }\n\n  public get isOptionDisabledInternalFn(): (option: TEntity) => boolean {\n    return (option: TEntity) => {\n      if (!this.isEntitySelected(option)) {\n        if (this.isOptionDisabledFn) {\n          return this.isOptionDisabledFn(option);\n        } else {\n          return false;\n        }\n      }\n      return true;\n    };\n  }\n\n  public get isOptionHiddenInternalFn(): (option: TEntity) => boolean {\n    return (option: TEntity) => {\n      if (this.isOptionHiddenFn) {\n        return this.isOptionHiddenFn(option);\n      } else {\n        return false;\n      }\n    };\n  }\n\n\n  public togglePanel(autoTrigger: MatAutocompleteTrigger): void {\n    if (autoTrigger.autocomplete.isOpen) {\n      autoTrigger.closePanel();\n    } else {\n      this.logger.debug('Requesting to open auto-complete panel...', autoTrigger);\n      if (this.suggestionsDc) {\n        try {\n          this.suggestionsDc.start(this.sorts, this.filters);\n        } catch (err) {\n          this.logger.error('Failed to start DC!', err);\n          this.updateState(ElderSelectComponentState.error(err));\n        }\n      }\n      autoTrigger.openPanel();\n    }\n  }\n\n  public onInputClicked(autoTrigger: MatAutocompleteTrigger): void {\n\n    // this.logger.debug('onInputClicked, locked: ' + this.isLocked + ', autocomplete: ' + this.autocomplete, autoTrigger);\n    if (this.isLocked || this.autocomplete) {\n      return;\n    }\n    this.togglePanel(autoTrigger);\n  }\n\n  public onOptionSelected(selectedValue: TEntity): void {\n    if (this.isEntitySelected(selectedValue)) {\n      this.writeValueInternal(null); // HACK: Ensure we trigger view redraw\n    }\n    this.updateValueByEntity(selectedValue);\n  }\n\n  public openSelectionPopup(event: Event): void {\n    if (this.selectionPopup) {\n      this.selectionPopup.choose(\n        this.filters,\n        this.sorts\n      ).subscribe(\n        (selection: TEntity[]) => {\n          this.updateValueByEntity(selection[0]);\n        }\n      );\n    } else {\n      this.logger.warn('Cant open selection browser popup, since [SelectionModelPopupDirective] was not found!');\n    }\n  }\n\n  public clear(event: Event): void {\n    this.updateValue(null);\n  }\n\n  public forceReloadFirst() {\n    this.value = null; // valueId is set to null automatically\n    const autoReloadFirst = new AutoSelectFirstDirective(this);\n    autoReloadFirst.trySelectFirst();\n  }\n\n  public updateValueByEntity(entity: TEntity): void {\n    this.updateValue(\n      this.entityToValue(entity)\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Abstract Implementation                                                 *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Occurs when the suggestions data-context has changed\n   */\n  protected onSuggestionsDcChanged(data: IDataContext<TEntity>): void {\n    if (this.valueAsId) {\n      this.selectEntityById(this.entityId);\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private Methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  protected valuesEquals(a: TValue, b: TValue): boolean {\n    return this.entityIdFromValue(a) === this.entityIdFromValue(b);\n  }\n\n  private isEntitySelected(entity: TEntity): boolean {\n    return this.getEntityId(entity) === this.entityId;\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 selectEntityById(id: TId): void {\n\n    const currentEntity = this.entity;\n    if (currentEntity && this.getEntityId(currentEntity) === id) {\n      return; // Entity already loaded\n    }\n\n    if (id === null || id === undefined) {\n      if (currentEntity !== null && currentEntity !== undefined) {\n        this.entity = null;\n      }\n    } else {\n      if (this.suggestionsDc) {\n        const existing = this.findInDataContext(id);\n        if (existing) {\n          this.entity = existing;\n        } else {\n          this.loadEntityById(id);\n        }\n      } else {\n        this.logger.warn('Failed to select value by Id: ' + id + ' - DataContext not available.');\n      }\n    }\n  }\n\n  private loadEntityById(id: TId): void {\n    this.suggestionsDc.dataSource.findById(<any>id)\n      .subscribe(\n        entity => {\n          this.logger.debug('Loaded missing entity by id ' + id + ':', entity);\n          this.entity = entity;\n        },\n        err => {\n          this.logger.error('Failed to load entity by id ' + id, err);\n          this.updateState(ElderSelectComponentState.error(err));\n        }\n      );\n  }\n\n  private findInDataContext(id: TId): TEntity | null {\n    return this.suggestionsDc.snapshot.data\n      .find(v => this.getEntityId(v) === id);\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  /**\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 entityId = <any>value as TId;\n      // Value was written as entity id, ensure we select entity by id\n      this.logger.debug('writeToControl: value was written as id: ' + entityId, value);\n      this.selectEntityById(entityId);\n    } else {\n      // Value was written as entity, ensure entity is updated\n      const currentEntity = this.entity;\n      const newEntity = <any>value as TEntity;\n      this.logger.debug('writeToControl: value was written as entity: ' + JSON.stringify(newEntity), value);\n      if (!this.isEqual(currentEntity, newEntity)) {\n        this.entity$.next(newEntity);\n      } else {\n        this.logger.warn('Ignored written entity as it is already set to entity$!');\n      }\n    }\n    super.writeToControl(value);\n  }\n\n  /**\n   * @description\n   * Writes a new value to the element.\n   *\n   * This method is called by the forms API to write to the view when programmatic\n   * changes from model to view are requested.\n   *\n   * @param value The new value for the element\n   */\n  public writeValue(value: TValue): void {\n    // this.logger.debug('ElderSelectBase.writeValue(): Current value-id ' + this._valueId + ', new value: ', value);\n    super.writeValue(value);\n  }\n\n  private isNullOrEmpty(value: TValue): boolean {\n    return (value == null || (<any>value) === '');\n  }\n\n}\n","<mat-form-field *ngIf=\"(entityWrapped$ | async) as entityWrapper\" fxFlex\n                class=\"elder-std-form-field\"\n                [appearance]=\"appearance\"\n                [floatLabel]=\"floatLabel\"\n                [color]=\"color\"\n>\n\n  <mat-label *ngIf=\"label\">{{label | translate}}</mat-label>\n\n\n  <ng-container matPrefix *ngIf=\"(mergedState$ | async) as state\">\n\n    <mat-icon *ngIf=\"icon\" disabled\n              class=\"leading-icon prefix-padding noselect\"\n              [class.loading]=\"state.loading\"\n              [color]=\"state?.error ? 'warn' : color\">\n      {{icon}}\n    </mat-icon>\n\n\n    <mat-icon *ngIf=\"!icon && state?.error\"\n              class=\"leading-icon prefix-padding noselect\"\n              color=\"warn\">\n      warning\n    </mat-icon>\n  </ng-container>\n\n  <!-- A dynamic input -->\n  <input\n    matInput type=\"text\" fxFlex=\"grow\"\n    [disabled]=\"disabled\"\n    [required]=\"required\"\n    [readonly]=\"readonly || !autocomplete\"\n    [name]=\"name + '-inner-input'\"\n    [placeholder]=\"placeholder | translate\"\n    [matAutocomplete] #autoTrigger=\"matAutocompleteTrigger\"\n    [elderElderAutocomplete]=\"elderAuto\" [queryFilter]=\"queryFilter\" [filters]=\"filters\" [sorts]=\"sorts\"\n    elderSelectOnTab\n    [class.select]=\"!autocomplete\" (click)=\"onInputClicked(autoTrigger)\"\n    [ngModel]=\"inputText$ | async\" [ngModelOptions]=\"{standalone: true, updateOn: 'submit'}\"\n    (blur)=\"onInputBlur($event)\"\n  >\n\n  <!-- This breaks stuff: [displayWith]=\"displayPropertyResolver$ | async\" -->\n\n  <elder-autocomplete\n    #elderAuto=\"elderAutocomplete\"\n    [suggestionsDc]=\"suggestionsDc$ | async\"\n    [valueTemplate]=\"valueTemplate\"\n    [enabled]=\"autocomplete\"\n    [displayPropertyResolver]=\"displayPropertyResolver$ | async\"\n    [isOptionDisabledFn]=\"isOptionDisabledInternalFn\"\n    [isOptionHiddenFn]=\"isOptionHiddenInternalFn\"\n    (optionSelected)=\"onOptionSelected($any($event))\"\n  ></elder-autocomplete>\n\n\n  <ng-container matSuffix>\n\n    <mat-icon\n      *ngIf=\"!autocomplete\" class=\"select-arrow noselect\"\n      (click)=\"onInputClicked(autoTrigger)\">\n      arrow_drop_down\n    </mat-icon>\n\n    <button mat-icon-button type=\"button\"\n            *ngIf=\"selectionPopup && (!entityWrapper.value || !allowNull)\"\n            [disabled]=\"isLocked\"\n            (click)=\"openSelectionPopup($event)\" aria-label=\"Search\"\n            elderStopEventPropagation\n            tabIndex=\"-1\"\n    >\n      <mat-icon>search</mat-icon>\n    </button>\n\n    <button mat-icon-button type=\"button\"\n            *ngIf=\"entityWrapper.value && allowNull\"\n            [disabled]=\"isLocked\"\n            (click)=\"clear($event)\" aria-label=\"Clear\"\n            elderStopEventPropagation\n            tabIndex=\"-1\"\n    >\n      <mat-icon>close</mat-icon>\n    </button>\n\n  </ng-container>\n\n  <mat-hint *ngIf=\"entityWrapper.hintText\">{{entityWrapper.hintText}}</mat-hint>\n\n</mat-form-field>\n\n\n\n"]}
|
|
394
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-select.component.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/select/elder-select/elder-select.component.ts","../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/select/elder-select/elder-select.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component, forwardRef,\n  Input, NgZone,\n  OnDestroy,\n  OnInit,\n  Output,\n} from '@angular/core';\nimport {buildFormIntegrationProviders} from '../../../common/forms/template-composite-control';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {IDataContext} from '../../../common/data/data-context/data-context';\nimport {BehaviorSubject, combineLatest, Observable, Subject} from 'rxjs';\nimport {\n  map,\n  takeUntil,\n  tap, skip, filter,\n} from 'rxjs/operators';\nimport {coerceBooleanProperty} from '@angular/cdk/coercion';\nimport {MatAutocompleteTrigger} from '@angular/material/autocomplete';\nimport {ELDER_SELECT_BASE, ElderSelectBase, ElderSelectComponentState, TextResolverFn} from '../elder-select-base';\nimport {IElderEntityValueAccessor} from '../elder-entity-value-accessor';\nimport {ElderSelectFirstUtil} from '../auto/elder-select-first-util';\n\nclass EntityContext<T> {\n  constructor(\n    public readonly value: T,\n    public readonly hintText: string,\n    public readonly displayText: string\n  ) {\n  }\n}\n\n\n/**\n * The elder-select control allows to select a single\n * entity.\n *\n * Primary focus is on easy usage for common scenarios:\n *\n * - Integrated into angular forms and validation\n * - Support entity id handling, input / output is the entity id vs full value.\n *\n */\n@Component({\n  selector: 'elder-select',\n  templateUrl: './elder-select.component.html',\n  styleUrls: ['./elder-select.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    {\n      provide: ELDER_SELECT_BASE,\n      useExisting: forwardRef(() => ElderSelectComponent)\n    },\n    ...buildFormIntegrationProviders(ElderSelectComponent)\n  ]\n})\nexport class ElderSelectComponent<TEntity = any, TId = any, TValue = TEntity | TId>\n  extends ElderSelectBase<TId, TEntity, TValue>\n  implements IElderEntityValueAccessor<TEntity, TId>, OnInit, OnDestroy {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly hintPropertyResolver$ = new BehaviorSubject<TextResolverFn>((value: TEntity) => null);\n\n  private _allowNull = false;\n\n  /**\n   * String to display for the 'nothing' / null value.\n   */\n  @Input()\n  public nullDisplay = '-';\n\n  @Output()\n  public readonly entityIdChange: Observable<TId>;\n\n  @Output()\n  public readonly entityChange: 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 entityIdUpdated: 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 entityUpdated: Observable<TEntity>;\n\n  private readonly entity$ = new BehaviorSubject<TEntity>(undefined);\n\n  private _autocomplete = false;\n\n  public readonly entityWrapped$: Observable<EntityContext<TEntity>>;\n\n  private readonly unsubscribe$ = new Subject();\n\n  public readonly inputText$: Observable<string>;\n\n  private entityContext: EntityContext<TEntity>;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    zone: NgZone,\n  ) {\n    super(zone);\n\n    this.entityIdChange = this.valueChange.pipe(\n      map(v => this.entityIdFromValue(v))\n    );\n\n    this.entityChange = this.entity$.pipe(\n      skip(1) // Skip the initial NULL value\n    );\n\n    this.entityIdUpdated = this.valueUpdated.pipe(\n      map(value => this.entityIdFromValue(value))\n    );\n\n    this.entityUpdated = combineLatest(\n      [this.entityIdUpdated, this.entity$]\n    ).pipe(\n       filter(([updatedId, entity]) => this.getEntityId(entity) === updatedId),\n        map(([updatedId, entity]) => entity)\n    );\n\n    this.entityWrapped$ = combineLatest(\n      [\n        this.entity$,\n        this.displayPropertyResolver$,\n        this.hintPropertyResolver$\n      ]\n    ).pipe(\n      map(([value, dPR, hPR]) => new EntityContext<TEntity>(value, hPR(value), dPR(value)))\n    );\n\n    this.inputText$ = this.entityWrapped$.pipe(\n      takeUntil(this.unsubscribe$),\n      tap(valueContext => this.entityContext = valueContext),\n      map(valueContext => {\n        if (valueContext.value) {\n          return valueContext.displayText;\n        } else {\n          return null;\n        }\n      })\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n\n  }\n\n  public ngOnDestroy(): void {\n    this.unsubscribe$.next();\n    this.unsubscribe$.complete();\n    this.autoCleanUp();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Behaviour Properties                                                    *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\n  public set autocomplete(value: boolean) {\n    this._autocomplete = coerceBooleanProperty(value);\n  }\n\n  public get autocomplete(): boolean {\n    return this._autocomplete;\n  }\n\n  /**\n   * Allows the user to select 'nothing' / null.\n   */\n  @Input()\n  public set allowNull(val: boolean) {\n    this._allowNull = coerceBooleanProperty(val);\n  }\n\n  public get allowNull(): boolean {\n    return this._allowNull;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\n  public set entity(entity: TEntity) {\n    this.entity$.next(entity);\n    this.writeValueInternal(\n      this.entityToValue(entity)\n    );\n  }\n\n  public get entity(): TEntity {\n    return this.entity$.getValue();\n  }\n\n  @Input()\n  public set entityId(id: TId) {\n    if (this.valueAsId) {\n      this.writeValueInternal(<any>id as TValue);\n    } else {\n      this.selectEntityById(id);\n    }\n  }\n\n  public get entityId(): TId {\n    return this.entityIdFromValue(this.value);\n  }\n\n  @Input()\n  public set hintProperty(property: string) {\n    this.hintPropertyResolver = (value => this.propertyStringValue(value, property));\n  }\n\n  @Input()\n  public set hintPropertyResolver(resolver: TextResolverFn) {\n    this.hintPropertyResolver$.next(resolver);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public onInputBlur(event: FocusEvent): void {\n    const input = event.target as HTMLInputElement;\n    if (this.entityContext.value) {\n      input.value = this.entityContext.displayText;\n    } else {\n      input.value = null;\n    }\n  }\n\n  public get isOptionDisabledInternalFn(): (option: TEntity) => boolean {\n    return (option: TEntity) => {\n      if (!this.isEntitySelected(option)) {\n        if (this.isOptionDisabledFn) {\n          return this.isOptionDisabledFn(option);\n        } else {\n          return false;\n        }\n      }\n      return true;\n    };\n  }\n\n  public get isOptionHiddenInternalFn(): (option: TEntity) => boolean {\n    return (option: TEntity) => {\n      if (this.isOptionHiddenFn) {\n        return this.isOptionHiddenFn(option);\n      } else {\n        return false;\n      }\n    };\n  }\n\n\n  public togglePanel(autoTrigger: MatAutocompleteTrigger): void {\n    if (autoTrigger.autocomplete.isOpen) {\n      autoTrigger.closePanel();\n    } else {\n      this.logger.debug('Requesting to open auto-complete panel...', autoTrigger);\n      if (this.suggestionsDc) {\n        try {\n          this.suggestionsDc.start(this.sorts, this.filters);\n        } catch (err) {\n          this.logger.error('Failed to start DC!', err);\n          this.updateState(ElderSelectComponentState.error(err));\n        }\n      }\n      autoTrigger.openPanel();\n    }\n  }\n\n  public onInputClicked(autoTrigger: MatAutocompleteTrigger): void {\n\n    // this.logger.debug('onInputClicked, locked: ' + this.isLocked + ', autocomplete: ' + this.autocomplete, autoTrigger);\n    if (this.isLocked || this.autocomplete) {\n      return;\n    }\n    this.togglePanel(autoTrigger);\n  }\n\n  public onOptionSelected(selectedValue: TEntity): void {\n    if (this.isEntitySelected(selectedValue)) {\n      this.writeValueInternal(null); // HACK: Ensure we trigger view redraw\n    }\n    this.updateValueByEntity(selectedValue);\n  }\n\n  public openSelectionPopup(event: Event): void {\n    if (this.selectionPopup) {\n      this.selectionPopup.choose(\n        this.filters,\n        this.sorts\n      ).subscribe(\n        (selection: TEntity[]) => {\n          this.updateValueByEntity(selection[0]);\n        }\n      );\n    } else {\n      this.logger.warn('Cant open selection browser popup, since [SelectionModelPopupDirective] was not found!');\n    }\n  }\n\n  public clear(event: Event): void {\n    this.updateValue(null);\n  }\n\n  public forceReloadFirst() {\n    this.value = null; // valueId is set to null automatically\n    ElderSelectFirstUtil.trySelectFirst(this);\n  }\n\n  public updateValueByEntity(entity: TEntity): void {\n    this.updateValue(\n      this.entityToValue(entity)\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Abstract Implementation                                                 *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Occurs when the suggestions data-context has changed\n   */\n  protected onSuggestionsDcChanged(data: IDataContext<TEntity>): void {\n    if (this.valueAsId) {\n      this.selectEntityById(this.entityId);\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private Methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  protected valuesEquals(a: TValue, b: TValue): boolean {\n    return this.entityIdFromValue(a) === this.entityIdFromValue(b);\n  }\n\n  private isEntitySelected(entity: TEntity): boolean {\n    return this.getEntityId(entity) === this.entityId;\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 selectEntityById(id: TId): void {\n\n    const currentEntity = this.entity;\n    if (currentEntity && this.getEntityId(currentEntity) === id) {\n      return; // Entity already loaded\n    }\n\n    if (id === null || id === undefined) {\n      if (currentEntity !== null && currentEntity !== undefined) {\n        this.entity = null;\n      }\n    } else {\n      if (this.suggestionsDc) {\n        const existing = this.findInDataContext(id);\n        if (existing) {\n          this.entity = existing;\n        } else {\n          this.loadEntityById(id);\n        }\n      } else {\n        this.logger.warn('Failed to select value by Id: ' + id + ' - DataContext not available.');\n      }\n    }\n  }\n\n  private loadEntityById(id: TId): void {\n    this.suggestionsDc.dataSource.findById(<any>id)\n      .subscribe(\n        entity => {\n          this.logger.debug('Loaded missing entity by id ' + id + ':', entity);\n          this.entity = entity;\n        },\n        err => {\n          this.logger.error('Failed to load entity by id ' + id, err);\n          this.updateState(ElderSelectComponentState.error(err));\n        }\n      );\n  }\n\n  private findInDataContext(id: TId): TEntity | null {\n    return this.suggestionsDc.snapshot.data\n      .find(v => this.getEntityId(v) === id);\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  /**\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 entityId = <any>value as TId;\n      // Value was written as entity id, ensure we select entity by id\n      this.logger.debug('writeToControl: value was written as id: ' + entityId, value);\n      this.selectEntityById(entityId);\n    } else {\n      // Value was written as entity, ensure entity is updated\n      const currentEntity = this.entity;\n      const newEntity = <any>value as TEntity;\n      this.logger.debug('writeToControl: value was written as entity: ' + JSON.stringify(newEntity), value);\n      if (!this.isEqual(currentEntity, newEntity)) {\n        this.entity$.next(newEntity);\n      } else {\n        this.logger.warn('Ignored written entity as it is already set to entity$!');\n      }\n    }\n    super.writeToControl(value);\n  }\n\n  /**\n   * @description\n   * Writes a new value to the element.\n   *\n   * This method is called by the forms API to write to the view when programmatic\n   * changes from model to view are requested.\n   *\n   * @param value The new value for the element\n   */\n  public writeValue(value: TValue): void {\n    // this.logger.debug('ElderSelectBase.writeValue(): Current value-id ' + this._valueId + ', new value: ', value);\n    super.writeValue(value);\n  }\n\n  private isNullOrEmpty(value: TValue): boolean {\n    return (value == null || (<any>value) === '');\n  }\n\n}\n","<mat-form-field *ngIf=\"(entityWrapped$ | async) as entityWrapper\" fxFlex\n                class=\"elder-std-form-field\"\n                [appearance]=\"appearance\"\n                [floatLabel]=\"floatLabel\"\n                [color]=\"color\"\n>\n\n  <mat-label *ngIf=\"label\">{{label | translate}}</mat-label>\n\n\n  <ng-container matPrefix *ngIf=\"(mergedState$ | async) as state\">\n\n    <mat-icon *ngIf=\"icon\" disabled\n              class=\"leading-icon prefix-padding noselect\"\n              [class.loading]=\"state.loading\"\n              [color]=\"state?.error ? 'warn' : color\">\n      {{icon}}\n    </mat-icon>\n\n\n    <mat-icon *ngIf=\"!icon && state?.error\"\n              class=\"leading-icon prefix-padding noselect\"\n              color=\"warn\">\n      warning\n    </mat-icon>\n  </ng-container>\n\n  <!-- A dynamic input -->\n  <input\n    matInput type=\"text\" fxFlex=\"grow\"\n    [disabled]=\"disabled\"\n    [required]=\"required\"\n    [readonly]=\"readonly || !autocomplete\"\n    [name]=\"name + '-inner-input'\"\n    [placeholder]=\"placeholder | translate\"\n    [matAutocomplete] #autoTrigger=\"matAutocompleteTrigger\"\n    [elderElderAutocomplete]=\"elderAuto\" [queryFilter]=\"queryFilter\" [filters]=\"filters\" [sorts]=\"sorts\"\n    elderSelectOnTab\n    [class.select]=\"!autocomplete\" (click)=\"onInputClicked(autoTrigger)\"\n    [ngModel]=\"inputText$ | async\" [ngModelOptions]=\"{standalone: true, updateOn: 'submit'}\"\n    (blur)=\"onInputBlur($event)\"\n  >\n\n  <!-- This breaks stuff: [displayWith]=\"displayPropertyResolver$ | async\" -->\n\n  <elder-autocomplete\n    #elderAuto=\"elderAutocomplete\"\n    [suggestionsDc]=\"suggestionsDc$ | async\"\n    [valueTemplate]=\"valueTemplate\"\n    [enabled]=\"autocomplete\"\n    [displayPropertyResolver]=\"displayPropertyResolver$ | async\"\n    [isOptionDisabledFn]=\"isOptionDisabledInternalFn\"\n    [isOptionHiddenFn]=\"isOptionHiddenInternalFn\"\n    (optionSelected)=\"onOptionSelected($any($event))\"\n  ></elder-autocomplete>\n\n\n  <ng-container matSuffix>\n\n    <mat-icon\n      *ngIf=\"!autocomplete\" class=\"select-arrow noselect\"\n      (click)=\"onInputClicked(autoTrigger)\">\n      arrow_drop_down\n    </mat-icon>\n\n    <button mat-icon-button type=\"button\"\n            *ngIf=\"selectionPopup && (!entityWrapper.value || !allowNull)\"\n            [disabled]=\"isLocked\"\n            (click)=\"openSelectionPopup($event)\" aria-label=\"Search\"\n            elderStopEventPropagation\n            tabIndex=\"-1\"\n    >\n      <mat-icon>search</mat-icon>\n    </button>\n\n    <button mat-icon-button type=\"button\"\n            *ngIf=\"entityWrapper.value && allowNull\"\n            [disabled]=\"isLocked\"\n            (click)=\"clear($event)\" aria-label=\"Clear\"\n            elderStopEventPropagation\n            tabIndex=\"-1\"\n    >\n      <mat-icon>close</mat-icon>\n    </button>\n\n  </ng-container>\n\n  <mat-hint *ngIf=\"entityWrapper.hintText\">{{entityWrapper.hintText}}</mat-hint>\n\n</mat-form-field>\n\n\n\n"]}
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
import { Directive, HostListener, Inject, SkipSelf } from '@angular/core';
|
|
2
|
+
import { Subject } from 'rxjs/internal/Subject';
|
|
3
|
+
import { delay, map, takeUntil } from 'rxjs/operators';
|
|
4
|
+
import { merge } from 'rxjs';
|
|
5
|
+
import { LoggerFactory } from '@elderbyte/ts-logger';
|
|
6
|
+
import { ELDER_SELECT_BASE } from './elder-select-base';
|
|
7
|
+
import { isElderEntityValueAccessor } from './elder-entity-value-accessor';
|
|
8
|
+
import * as i0 from "@angular/core";
|
|
9
|
+
import * as i1 from "@angular/material/autocomplete";
|
|
10
|
+
import * as i2 from "./elder-select-base";
|
|
11
|
+
export class ElderSelectOnTabDirective {
|
|
12
|
+
/***************************************************************************
|
|
13
|
+
* *
|
|
14
|
+
* Constructor *
|
|
15
|
+
* *
|
|
16
|
+
**************************************************************************/
|
|
17
|
+
constructor(autoTrigger, elderSelectBase) {
|
|
18
|
+
this.autoTrigger = autoTrigger;
|
|
19
|
+
/***************************************************************************
|
|
20
|
+
* *
|
|
21
|
+
* Fields *
|
|
22
|
+
* *
|
|
23
|
+
**************************************************************************/
|
|
24
|
+
this.logger = LoggerFactory.getLogger(this.constructor.name);
|
|
25
|
+
this.destroy$ = new Subject();
|
|
26
|
+
/**
|
|
27
|
+
* Whether the autocomplete panel was open before the event
|
|
28
|
+
*/
|
|
29
|
+
this.panelOpen = false;
|
|
30
|
+
/**
|
|
31
|
+
* Whether the user selected an option.
|
|
32
|
+
* (We want to ignore selections if there is already a selection present and the user tabs away)
|
|
33
|
+
*/
|
|
34
|
+
this.userInput = false;
|
|
35
|
+
this.controlValueAccessor = elderSelectBase;
|
|
36
|
+
}
|
|
37
|
+
/***************************************************************************
|
|
38
|
+
* *
|
|
39
|
+
* Event Listener *
|
|
40
|
+
* *
|
|
41
|
+
**************************************************************************/
|
|
42
|
+
onBlur() {
|
|
43
|
+
if (!this.panelOpen) {
|
|
44
|
+
return;
|
|
45
|
+
}
|
|
46
|
+
if (this.controlValueAccessor.value) {
|
|
47
|
+
// If there is already a value present, only allow tab based value setting if the user actually modified the value
|
|
48
|
+
if (!this.userInput) {
|
|
49
|
+
this.logger.warn('Discarding TAB since the user did probably not intend to change the value! userInput:', this.userInput);
|
|
50
|
+
return;
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
const activeOption = this.autoTrigger.activeOption;
|
|
54
|
+
if (activeOption) {
|
|
55
|
+
const entity = activeOption.value;
|
|
56
|
+
this.updateValueByEntity(entity);
|
|
57
|
+
this.autoTrigger.writeValue(entity);
|
|
58
|
+
this.logger.warn('Wrote option.value to control: ', entity);
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
updateValueByEntity(entity) {
|
|
62
|
+
if (isElderEntityValueAccessor(this.controlValueAccessor)) {
|
|
63
|
+
// Since elder-select may use the entity id as value (valueAsId),
|
|
64
|
+
// we can not directly write to the value safely.
|
|
65
|
+
this.controlValueAccessor.updateValueByEntity(entity);
|
|
66
|
+
}
|
|
67
|
+
else {
|
|
68
|
+
// By default, write the selected option to the control value
|
|
69
|
+
this.controlValueAccessor.updateValue(entity);
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
/***************************************************************************
|
|
73
|
+
* *
|
|
74
|
+
* Life Cycle *
|
|
75
|
+
* *
|
|
76
|
+
**************************************************************************/
|
|
77
|
+
ngAfterViewInit() {
|
|
78
|
+
const autocomplete = this.autoTrigger.autocomplete;
|
|
79
|
+
merge(autocomplete.opened.pipe(map(() => true)), autocomplete.closed.pipe(map(() => false))).pipe(takeUntil(this.destroy$), delay(0)).subscribe(value => this.panelOpen = value);
|
|
80
|
+
this.autoTrigger.optionSelections.pipe(// TODO https://github.com/angular/components/pull/14813
|
|
81
|
+
takeUntil(this.destroy$), map(opt => opt.isUserInput)).subscribe(isUserInput => this.userInput = isUserInput);
|
|
82
|
+
}
|
|
83
|
+
ngOnDestroy() {
|
|
84
|
+
this.destroy$.next();
|
|
85
|
+
this.destroy$.complete();
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
ElderSelectOnTabDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.1", ngImport: i0, type: ElderSelectOnTabDirective, deps: [{ token: i1.MatAutocompleteTrigger }, { token: ELDER_SELECT_BASE, skipSelf: true }], target: i0.ɵɵFactoryTarget.Directive });
|
|
89
|
+
ElderSelectOnTabDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.3.1", type: ElderSelectOnTabDirective, selector: "[elderSelectOnTab]", host: { listeners: { "keydown.tab": "onBlur()" } }, ngImport: i0 });
|
|
90
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.1", ngImport: i0, type: ElderSelectOnTabDirective, decorators: [{
|
|
91
|
+
type: Directive,
|
|
92
|
+
args: [{
|
|
93
|
+
selector: '[elderSelectOnTab]'
|
|
94
|
+
}]
|
|
95
|
+
}], ctorParameters: function () { return [{ type: i1.MatAutocompleteTrigger }, { type: i2.ElderSelectBase, decorators: [{
|
|
96
|
+
type: SkipSelf
|
|
97
|
+
}, {
|
|
98
|
+
type: Inject,
|
|
99
|
+
args: [ELDER_SELECT_BASE]
|
|
100
|
+
}] }]; }, propDecorators: { onBlur: [{
|
|
101
|
+
type: HostListener,
|
|
102
|
+
args: ['keydown.tab']
|
|
103
|
+
}] } });
|
|
104
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-select-on-tab.directive.js","sourceRoot":"","sources":["../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/select/elder-select-on-tab.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {AfterViewInit, Directive, Host, HostListener, Inject, OnDestroy, SkipSelf} from '@angular/core';\nimport {MatAutocompleteTrigger} from '@angular/material/autocomplete';\nimport {Subject} from 'rxjs/internal/Subject';\nimport {delay, map, takeUntil} from 'rxjs/operators';\nimport {merge} from 'rxjs';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {ValueAccessorBase} from '../../common/forms/value-accessor-base';\nimport {ELDER_SELECT_BASE, ElderSelectBase} from './elder-select-base';\nimport {MatOption} from '@angular/material/core';\nimport {isElderEntityValueAccessor} from './elder-entity-value-accessor';\n\n@Directive({\n  selector: '[elderSelectOnTab]'\n})\nexport class ElderSelectOnTabDirective<TEntity = any, TId = any> implements AfterViewInit, OnDestroy {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly destroy$ = new Subject<never>();\n\n  private readonly controlValueAccessor: ValueAccessorBase<any>;\n\n  /**\n   * Whether the autocomplete panel was open before the event\n   */\n  private panelOpen = false;\n\n  /**\n   * Whether the user selected an option.\n   * (We want to ignore selections if there is already a selection present and the user tabs away)\n   */\n  private userInput = false;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private readonly autoTrigger: MatAutocompleteTrigger,\n    @SkipSelf() @Inject(ELDER_SELECT_BASE) elderSelectBase: ElderSelectBase<any, any, any>\n  ) {\n    this.controlValueAccessor = elderSelectBase;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Event Listener                                                          *\n   *                                                                         *\n   **************************************************************************/\n\n  @HostListener('keydown.tab')\n  public onBlur(): void {\n\n    if (!this.panelOpen) { return; }\n\n    if (this.controlValueAccessor.value) {\n      // If there is already a value present, only allow tab based value setting if the user actually modified the value\n      if (!this.userInput) {\n        this.logger.warn('Discarding TAB since the user did probably not intend to change the value! userInput:', this.userInput);\n        return;\n      }\n    }\n\n    const activeOption = this.autoTrigger.activeOption as MatOption<TEntity>;\n\n    if (activeOption) {\n      const entity = activeOption.value;\n      this.updateValueByEntity(entity);\n      this.autoTrigger.writeValue(entity);\n      this.logger.warn('Wrote option.value to control: ', entity);\n    }\n  }\n\n  private updateValueByEntity(entity: TEntity): void {\n    if (isElderEntityValueAccessor(this.controlValueAccessor)) {\n      // Since elder-select may use the entity id as value (valueAsId),\n      // we can not directly write to the value safely.\n      this.controlValueAccessor.updateValueByEntity(entity);\n    } else {\n      // By default, write the selected option to the control value\n      this.controlValueAccessor.updateValue(entity);\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngAfterViewInit() {\n    const autocomplete = this.autoTrigger.autocomplete;\n    merge(\n      autocomplete.opened.pipe(map(() => true)),\n      autocomplete.closed.pipe(map(() => false))\n    ).pipe(\n      takeUntil(this.destroy$),\n      delay(0)\n    ).subscribe(value => this.panelOpen = value);\n\n    this.autoTrigger.optionSelections.pipe( // TODO https://github.com/angular/components/pull/14813\n      takeUntil(this.destroy$),\n      map(opt => opt.isUserInput)\n    ).subscribe(isUserInput => this.userInput = isUserInput);\n  }\n\n  public ngOnDestroy() {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n}\n"]}
|