@elderbyte/ngx-starter 17.11.0-beta4 → 17.11.0-beta6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (32) hide show
  1. package/esm2022/lib/components/forms/search/domain/input/search-input-state.mjs +6 -6
  2. package/esm2022/lib/components/forms/search/domain/input/search-input.mjs +1 -1
  3. package/esm2022/lib/components/forms/search/elder-search-context.directive.mjs +10 -10
  4. package/esm2022/lib/components/forms/search/elder-search-input.directive.mjs +14 -29
  5. package/esm2022/lib/components/forms/search/search-box/elder-search-box.component.mjs +4 -4
  6. package/esm2022/lib/components/select/elder-select.module.mjs +15 -15
  7. package/esm2022/lib/components/select/filter/elder-chip-filter-style-resolver.mjs +19 -15
  8. package/esm2022/lib/components/select/filter/elder-chips-include-exclude.directive.mjs +144 -0
  9. package/esm2022/lib/components/select/filter/elder-filter-chip-template/elder-filter-chip-template.component.mjs +24 -31
  10. package/esm2022/lib/components/select/filter/include-exclude-model-filter-adapter.mjs +63 -0
  11. package/esm2022/lib/components/select/filter/include-exclude-selection-model.mjs +76 -20
  12. package/esm2022/lib/components/select/multi/elder-multi-select-chips/elder-multi-select-chips.component.mjs +7 -2
  13. package/esm2022/lib/components/select/multi/elder-search-include-exclude.directive.mjs +117 -43
  14. package/fesm2022/elderbyte-ngx-starter.mjs +741 -649
  15. package/fesm2022/elderbyte-ngx-starter.mjs.map +1 -1
  16. package/lib/components/forms/search/domain/input/search-input-state.d.ts +3 -3
  17. package/lib/components/forms/search/domain/input/search-input.d.ts +2 -6
  18. package/lib/components/forms/search/elder-search-context.directive.d.ts +3 -3
  19. package/lib/components/forms/search/elder-search-input.directive.d.ts +7 -11
  20. package/lib/components/forms/search/search-box/elder-search-box.component.d.ts +2 -1
  21. package/lib/components/select/elder-select.module.d.ts +4 -4
  22. package/lib/components/select/filter/elder-chip-filter-style-resolver.d.ts +1 -1
  23. package/lib/components/select/filter/{elder-chips-filter.directive.d.ts → elder-chips-include-exclude.directive.d.ts} +20 -9
  24. package/lib/components/select/filter/elder-filter-chip-template/elder-filter-chip-template.component.d.ts +9 -14
  25. package/lib/components/select/filter/include-exclude-model-filter-adapter.d.ts +29 -0
  26. package/lib/components/select/filter/include-exclude-selection-model.d.ts +25 -7
  27. package/lib/components/select/multi/elder-multi-select-chips/elder-multi-select-chips.component.d.ts +2 -0
  28. package/lib/components/select/multi/elder-search-include-exclude.directive.d.ts +57 -17
  29. package/package.json +1 -1
  30. package/esm2022/lib/components/forms/search/filter/include-exclude-input-converter.mjs +0 -137
  31. package/esm2022/lib/components/select/filter/elder-chips-filter.directive.mjs +0 -93
  32. package/lib/components/forms/search/filter/include-exclude-input-converter.d.ts +0 -62
@@ -1,74 +1,148 @@
1
- import { Directive, Input } from '@angular/core';
1
+ import { Directive, Host, Input, Optional } from '@angular/core';
2
+ import { BehaviorSubject, concat, of, skipUntil, Subject } from 'rxjs';
3
+ import { SearchInputState } from '../../forms/search/domain/input/search-input-state';
4
+ import { LoggerFactory } from '@elderbyte/ts-logger';
2
5
  import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
3
- import { BehaviorSubject, combineLatest } from 'rxjs';
4
- import { filter } from 'rxjs/operators';
5
- import IncludeExcludeInputConverter from '../../forms/search/filter/include-exclude-input-converter';
6
- import { Objects } from '../../../common/objects';
6
+ import { debounceTime, delay, first, map, tap } from 'rxjs/operators';
7
+ import { IncludeExcludeModelFilterAdapter } from '../filter/include-exclude-model-filter-adapter';
7
8
  import * as i0 from "@angular/core";
8
- import * as i1 from "../../forms/search/elder-search-input.directive";
9
- import * as i2 from "../filter/elder-chips-filter.directive";
9
+ import * as i1 from "../../forms/search/elder-search-context.directive";
10
+ import * as i2 from "../filter/include-exclude-selection-model";
11
+ import * as i3 from "@angular/forms";
12
+ /**
13
+ * Binds a IncludeExcludeModel to a SearchContexts Filter.
14
+ */
10
15
  export class ElderSearchIncludeExcludeDirective {
11
- /***************************************************************************
12
- * *
13
- * Properties *
14
- * *
15
- **************************************************************************/
16
- set includeSuffix(suffix) {
17
- this._includeSuffix$.next(suffix);
18
- }
19
- set excludeSuffix(suffix) {
20
- this._excludeSuffix$.next(suffix);
21
- }
22
16
  /***************************************************************************
23
17
  * *
24
18
  * Constructor *
25
19
  * *
26
20
  **************************************************************************/
27
- constructor(elderSearchInput, elderChipsFilter, destroyRef) {
28
- this.elderSearchInput = elderSearchInput;
29
- this.elderChipsFilter = elderChipsFilter;
21
+ constructor(searchContext, destroyRef, selectionModel, ngModel) {
22
+ this.searchContext = searchContext;
30
23
  this.destroyRef = destroyRef;
24
+ this.selectionModel = selectionModel;
25
+ this.ngModel = ngModel;
31
26
  /***************************************************************************
32
27
  * *
33
28
  * Fields *
34
29
  * *
35
30
  **************************************************************************/
36
- this._includeSuffix$ = new BehaviorSubject('');
37
- this._excludeSuffix$ = new BehaviorSubject(null);
38
- this.setupSearchQueryTransforms();
31
+ this.log = LoggerFactory.getLogger(this.constructor.name);
32
+ this._queryKey$ = new BehaviorSubject(null);
33
+ this.includeSuffix = '';
34
+ this.initialDone$ = new Subject();
35
+ this._state$ = this.buildInputStateObservable().pipe(
36
+ // tap(state => this.log.error('EMIT SEARCH INPUT STATE [' + state.queryKey + ']: ' + state.queryValue + (state.pristine ? '(pristine)' : ''), state))
37
+ );
38
+ }
39
+ /***************************************************************************
40
+ * *
41
+ * Life Cycle *
42
+ * *
43
+ **************************************************************************/
44
+ ngOnInit() {
45
+ this._extractedName = this.extractName();
46
+ this.updateQueryKey();
47
+ this.includeExcludeAdapter = new IncludeExcludeModelFilterAdapter(this.selectionModel, this.queryKey + this.includeSuffix, this.queryKey + this.excludeSuffix);
48
+ this.searchContext.register(this);
49
+ }
50
+ /***************************************************************************
51
+ * *
52
+ * Properties *
53
+ * *
54
+ **************************************************************************/
55
+ /**
56
+ * (Optional) Usually the control name is used, this allows a custom query key
57
+ */
58
+ set queryKey(value) {
59
+ this._userQueryKey = value;
60
+ this.updateQueryKey();
61
+ }
62
+ get queryKey() {
63
+ return this._queryKey$.getValue();
64
+ }
65
+ get queryKey$() {
66
+ return this._queryKey$;
67
+ }
68
+ get state$() {
69
+ return this._state$;
70
+ }
71
+ get readonly() {
72
+ return this.ngModel?.isDisabled;
73
+ }
74
+ /***************************************************************************
75
+ * *
76
+ * Public API *
77
+ * *
78
+ **************************************************************************/
79
+ applyFilters(filters) {
80
+ this.includeExcludeAdapter.applyFilters(filters);
81
+ }
82
+ reset() {
83
+ this.selectionModel.replaceSelection([]);
39
84
  }
40
85
  /***************************************************************************
41
86
  * *
42
- * Private Methods *
87
+ * Private methods *
43
88
  * *
44
89
  **************************************************************************/
45
- setupSearchQueryTransforms() {
46
- combineLatest([
47
- this.elderSearchInput.queryKey$,
48
- this._includeSuffix$,
49
- this._excludeSuffix$
50
- ])
51
- .pipe(takeUntilDestroyed(this.destroyRef), filter(([queryKey, includeSuffix, excludeSuffix]) => !!queryKey && Objects.nonNull(includeSuffix) && Objects.nonNull(excludeSuffix)))
52
- .subscribe(([queryKey, includeSuffix, excludeSuffix]) => {
53
- this.elderSearchInput.inputConverter = this.buildConverter(queryKey, includeSuffix, excludeSuffix);
54
- });
55
- }
56
- buildConverter(queryKey, includeSuffix, excludeSuffix) {
57
- return new IncludeExcludeInputConverter(this.elderChipsFilter.selectionModel, queryKey, includeSuffix, excludeSuffix);
58
- }
59
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: ElderSearchIncludeExcludeDirective, deps: [{ token: i1.ElderSearchInputDirective }, { token: i2.ElderChipsFilterDirective }, { token: i0.DestroyRef }], target: i0.ɵɵFactoryTarget.Directive }); }
60
- static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.3.11", type: ElderSearchIncludeExcludeDirective, isStandalone: true, selector: "[elderSearchIncludeExclude]", inputs: { includeSuffix: "includeSuffix", excludeSuffix: "excludeSuffix" }, ngImport: i0 }); }
90
+ updateQueryKey() {
91
+ const newKey = this._userQueryKey ?? this._extractedName;
92
+ if (this._queryKey$.getValue() !== newKey) {
93
+ this._queryKey$.next(newKey);
94
+ }
95
+ }
96
+ buildInputStateObservable() {
97
+ const viewStabilized$ = of('stabilized$').pipe(delay(20), first());
98
+ const inputStateChanges$ = this.selectionModel.selection$.pipe(takeUntilDestroyed(this.destroyRef), map(value => this.buildInputState(true)), skipUntil(viewStabilized$), tap(() => this.initialDone$.next()), debounceTime(20));
99
+ const startDelayed$ = of('start').pipe(delay(15), map(start => {
100
+ return this.buildInputState(false);
101
+ }));
102
+ return concat(startDelayed$, inputStateChanges$);
103
+ }
104
+ buildInputState(userEvent) {
105
+ const filters = this.includeExcludeAdapter.toFilters();
106
+ return new SearchInputState(this.queryKey, filters, this.areAllPristine(filters), userEvent);
107
+ }
108
+ areAllPristine(filters) {
109
+ if (filters.length > 0) {
110
+ return filters.every(f => this.isPristine(f));
111
+ }
112
+ else {
113
+ return true;
114
+ }
115
+ }
116
+ isPristine(filter) {
117
+ return !this.isAttributeValuePresent(filter.value);
118
+ }
119
+ isAttributeValuePresent(value) {
120
+ return SearchInputState.isValueDefined(value)
121
+ && (value + '').length !== 0;
122
+ }
123
+ extractName() {
124
+ return this.ngModel?.name;
125
+ }
126
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: ElderSearchIncludeExcludeDirective, deps: [{ token: i1.ElderSearchContextDirective }, { token: i0.DestroyRef }, { token: i2.IncludeExcludeSelectionModel }, { token: i3.NgModel, host: true, optional: true }], target: i0.ɵɵFactoryTarget.Directive }); }
127
+ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.3.11", type: ElderSearchIncludeExcludeDirective, isStandalone: true, selector: "[elderSearchIncludeExclude]", inputs: { includeSuffix: "includeSuffix", excludeSuffix: "excludeSuffix", queryKey: "queryKey" }, exportAs: ["elderSearchInput"], ngImport: i0 }); }
61
128
  }
62
129
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: ElderSearchIncludeExcludeDirective, decorators: [{
63
130
  type: Directive,
64
131
  args: [{
65
132
  selector: '[elderSearchIncludeExclude]',
133
+ exportAs: 'elderSearchInput',
66
134
  standalone: true
67
135
  }]
68
- }], ctorParameters: () => [{ type: i1.ElderSearchInputDirective }, { type: i2.ElderChipsFilterDirective }, { type: i0.DestroyRef }], propDecorators: { includeSuffix: [{
136
+ }], ctorParameters: () => [{ type: i1.ElderSearchContextDirective }, { type: i0.DestroyRef }, { type: i2.IncludeExcludeSelectionModel }, { type: i3.NgModel, decorators: [{
137
+ type: Optional
138
+ }, {
139
+ type: Host
140
+ }] }], propDecorators: { includeSuffix: [{
69
141
  type: Input
70
142
  }], excludeSuffix: [{
71
143
  type: Input,
72
144
  args: [{ required: true }]
145
+ }], queryKey: [{
146
+ type: Input
73
147
  }] } });
74
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZWxkZXItc2VhcmNoLWluY2x1ZGUtZXhjbHVkZS5kaXJlY3RpdmUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9lbGRlcmJ5dGUvbmd4LXN0YXJ0ZXIvc3JjL2xpYi9jb21wb25lbnRzL3NlbGVjdC9tdWx0aS9lbGRlci1zZWFyY2gtaW5jbHVkZS1leGNsdWRlLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQWEsU0FBUyxFQUFFLEtBQUssRUFBQyxNQUFNLGVBQWUsQ0FBQztBQUMzRCxPQUFPLEVBQUMsa0JBQWtCLEVBQUMsTUFBTSw0QkFBNEIsQ0FBQztBQUM5RCxPQUFPLEVBQUMsZUFBZSxFQUFFLGFBQWEsRUFBQyxNQUFNLE1BQU0sQ0FBQztBQUNwRCxPQUFPLEVBQUMsTUFBTSxFQUFDLE1BQU0sZ0JBQWdCLENBQUM7QUFFdEMsT0FBTyw0QkFBNEIsTUFBTSwyREFBMkQsQ0FBQztBQUVyRyxPQUFPLEVBQUMsT0FBTyxFQUFDLE1BQU0seUJBQXlCLENBQUM7Ozs7QUFNaEQsTUFBTSxPQUFPLGtDQUFrQztJQVc3Qzs7OztnRkFJNEU7SUFFNUUsSUFDVyxhQUFhLENBQUMsTUFBYztRQUNyQyxJQUFJLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUNwQyxDQUFDO0lBRUQsSUFDVyxhQUFhLENBQUMsTUFBYztRQUNyQyxJQUFJLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUNwQyxDQUFDO0lBRUQ7Ozs7Z0ZBSTRFO0lBRTVFLFlBQ21CLGdCQUEyQyxFQUMzQyxnQkFBMkMsRUFDcEQsVUFBc0I7UUFGYixxQkFBZ0IsR0FBaEIsZ0JBQWdCLENBQTJCO1FBQzNDLHFCQUFnQixHQUFoQixnQkFBZ0IsQ0FBMkI7UUFDcEQsZUFBVSxHQUFWLFVBQVUsQ0FBWTtRQWxDaEM7Ozs7b0ZBSTRFO1FBRTNELG9CQUFlLEdBQUcsSUFBSSxlQUFlLENBQVMsRUFBRSxDQUFDLENBQUM7UUFDbEQsb0JBQWUsR0FBRyxJQUFJLGVBQWUsQ0FBUyxJQUFJLENBQUMsQ0FBQztRQTZCbkUsSUFBSSxDQUFDLDBCQUEwQixFQUFFLENBQUM7SUFDcEMsQ0FBQztJQUVEOzs7O2dGQUk0RTtJQUVwRSwwQkFBMEI7UUFDaEMsYUFBYSxDQUFDO1lBQ1osSUFBSSxDQUFDLGdCQUFnQixDQUFDLFNBQVM7WUFDL0IsSUFBSSxDQUFDLGVBQWU7WUFDcEIsSUFBSSxDQUFDLGVBQWU7U0FDckIsQ0FBQzthQUNDLElBQUksQ0FDSCxrQkFBa0IsQ0FBQyxJQUFJLENBQUMsVUFBVSxDQUFDLEVBQ25DLE1BQU0sQ0FBQyxDQUFDLENBQUMsUUFBUSxFQUFFLGFBQWEsRUFBRSxhQUFhLENBQUMsRUFBRSxFQUFFLENBQ2xELENBQUMsQ0FBQyxRQUFRLElBQUksT0FBTyxDQUFDLE9BQU8sQ0FBQyxhQUFhLENBQUMsSUFBSSxPQUFPLENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FBQyxDQUMvRSxDQUNGO2FBQ0EsU0FBUyxDQUFDLENBQUMsQ0FBQyxRQUFRLEVBQUUsYUFBYSxFQUFFLGFBQWEsQ0FBQyxFQUFFLEVBQUU7WUFDdEQsSUFBSSxDQUFDLGdCQUFnQixDQUFDLGNBQWMsR0FBRyxJQUFJLENBQUMsY0FBYyxDQUFDLFFBQVEsRUFBRSxhQUFhLEVBQUUsYUFBYSxDQUFDLENBQUM7UUFDckcsQ0FBQyxDQUFDLENBQUM7SUFDUCxDQUFDO0lBSU8sY0FBYyxDQUNwQixRQUFnQixFQUNoQixhQUFxQixFQUNyQixhQUFxQjtRQUVyQixPQUFPLElBQUksNEJBQTRCLENBQ3JDLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxjQUFjLEVBQ3BDLFFBQVEsRUFDUixhQUFhLEVBQ2IsYUFBYSxDQUNkLENBQUM7SUFDSixDQUFDOytHQTdFVSxrQ0FBa0M7bUdBQWxDLGtDQUFrQzs7NEZBQWxDLGtDQUFrQztrQkFKOUMsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsNkJBQTZCO29CQUN2QyxVQUFVLEVBQUUsSUFBSTtpQkFDakI7K0pBbUJZLGFBQWE7c0JBRHZCLEtBQUs7Z0JBTUssYUFBYTtzQkFEdkIsS0FBSzt1QkFBQyxFQUFDLFFBQVEsRUFBRSxJQUFJLEVBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0Rlc3Ryb3lSZWYsIERpcmVjdGl2ZSwgSW5wdXR9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHt0YWtlVW50aWxEZXN0cm95ZWR9IGZyb20gJ0Bhbmd1bGFyL2NvcmUvcnhqcy1pbnRlcm9wJztcbmltcG9ydCB7QmVoYXZpb3JTdWJqZWN0LCBjb21iaW5lTGF0ZXN0fSBmcm9tICdyeGpzJztcbmltcG9ydCB7ZmlsdGVyfSBmcm9tICdyeGpzL29wZXJhdG9ycyc7XG5pbXBvcnQge0VsZGVyU2VhcmNoSW5wdXREaXJlY3RpdmV9IGZyb20gJy4uLy4uL2Zvcm1zL3NlYXJjaC9lbGRlci1zZWFyY2gtaW5wdXQuZGlyZWN0aXZlJztcbmltcG9ydCBJbmNsdWRlRXhjbHVkZUlucHV0Q29udmVydGVyIGZyb20gJy4uLy4uL2Zvcm1zL3NlYXJjaC9maWx0ZXIvaW5jbHVkZS1leGNsdWRlLWlucHV0LWNvbnZlcnRlcic7XG5pbXBvcnQge0VsZGVyQ2hpcHNGaWx0ZXJEaXJlY3RpdmV9IGZyb20gJy4uL2ZpbHRlci9lbGRlci1jaGlwcy1maWx0ZXIuZGlyZWN0aXZlJztcbmltcG9ydCB7T2JqZWN0c30gZnJvbSAnLi4vLi4vLi4vY29tbW9uL29iamVjdHMnO1xuXG5ARGlyZWN0aXZlKHtcbiAgc2VsZWN0b3I6ICdbZWxkZXJTZWFyY2hJbmNsdWRlRXhjbHVkZV0nLFxuICBzdGFuZGFsb25lOiB0cnVlXG59KVxuZXhwb3J0IGNsYXNzIEVsZGVyU2VhcmNoSW5jbHVkZUV4Y2x1ZGVEaXJlY3RpdmUge1xuXG4gIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKipcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiBGaWVsZHMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqL1xuXG4gIHByaXZhdGUgcmVhZG9ubHkgX2luY2x1ZGVTdWZmaXgkID0gbmV3IEJlaGF2aW9yU3ViamVjdDxzdHJpbmc+KCcnKTtcbiAgcHJpdmF0ZSByZWFkb25seSBfZXhjbHVkZVN1ZmZpeCQgPSBuZXcgQmVoYXZpb3JTdWJqZWN0PHN0cmluZz4obnVsbCk7XG5cbiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqIFByb3BlcnRpZXMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgQElucHV0KClcbiAgcHVibGljIHNldCBpbmNsdWRlU3VmZml4KHN1ZmZpeDogc3RyaW5nKSB7XG4gICAgdGhpcy5faW5jbHVkZVN1ZmZpeCQubmV4dChzdWZmaXgpO1xuICB9XG5cbiAgQElucHV0KHtyZXF1aXJlZDogdHJ1ZX0pXG4gIHB1YmxpYyBzZXQgZXhjbHVkZVN1ZmZpeChzdWZmaXg6IHN0cmluZykge1xuICAgIHRoaXMuX2V4Y2x1ZGVTdWZmaXgkLm5leHQoc3VmZml4KTtcbiAgfVxuXG4gIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKipcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiBDb25zdHJ1Y3RvciAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqL1xuXG4gIGNvbnN0cnVjdG9yKFxuICAgIHByaXZhdGUgcmVhZG9ubHkgZWxkZXJTZWFyY2hJbnB1dDogRWxkZXJTZWFyY2hJbnB1dERpcmVjdGl2ZSxcbiAgICBwcml2YXRlIHJlYWRvbmx5IGVsZGVyQ2hpcHNGaWx0ZXI6IEVsZGVyQ2hpcHNGaWx0ZXJEaXJlY3RpdmUsXG4gICAgcHJpdmF0ZSBkZXN0cm95UmVmOiBEZXN0cm95UmVmXG4gICkge1xuICAgIHRoaXMuc2V0dXBTZWFyY2hRdWVyeVRyYW5zZm9ybXMoKTtcbiAgfVxuXG4gIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKipcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiBQcml2YXRlIE1ldGhvZHMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqL1xuXG4gIHByaXZhdGUgc2V0dXBTZWFyY2hRdWVyeVRyYW5zZm9ybXMoKSB7XG4gICAgY29tYmluZUxhdGVzdChbXG4gICAgICB0aGlzLmVsZGVyU2VhcmNoSW5wdXQucXVlcnlLZXkkLFxuICAgICAgdGhpcy5faW5jbHVkZVN1ZmZpeCQsXG4gICAgICB0aGlzLl9leGNsdWRlU3VmZml4JFxuICAgIF0pXG4gICAgICAucGlwZShcbiAgICAgICAgdGFrZVVudGlsRGVzdHJveWVkKHRoaXMuZGVzdHJveVJlZiksXG4gICAgICAgIGZpbHRlcigoW3F1ZXJ5S2V5LCBpbmNsdWRlU3VmZml4LCBleGNsdWRlU3VmZml4XSkgPT5cbiAgICAgICAgICAhIXF1ZXJ5S2V5ICYmIE9iamVjdHMubm9uTnVsbChpbmNsdWRlU3VmZml4KSAmJiBPYmplY3RzLm5vbk51bGwoZXhjbHVkZVN1ZmZpeClcbiAgICAgICAgKVxuICAgICAgKVxuICAgICAgLnN1YnNjcmliZSgoW3F1ZXJ5S2V5LCBpbmNsdWRlU3VmZml4LCBleGNsdWRlU3VmZml4XSkgPT4ge1xuICAgICAgICB0aGlzLmVsZGVyU2VhcmNoSW5wdXQuaW5wdXRDb252ZXJ0ZXIgPSB0aGlzLmJ1aWxkQ29udmVydGVyKHF1ZXJ5S2V5LCBpbmNsdWRlU3VmZml4LCBleGNsdWRlU3VmZml4KTtcbiAgICAgIH0pO1xuICB9XG5cblxuXG4gIHByaXZhdGUgYnVpbGRDb252ZXJ0ZXIoXG4gICAgcXVlcnlLZXk6IHN0cmluZyxcbiAgICBpbmNsdWRlU3VmZml4OiBzdHJpbmcsXG4gICAgZXhjbHVkZVN1ZmZpeDogc3RyaW5nXG4gICk6IEluY2x1ZGVFeGNsdWRlSW5wdXRDb252ZXJ0ZXIge1xuICAgIHJldHVybiBuZXcgSW5jbHVkZUV4Y2x1ZGVJbnB1dENvbnZlcnRlcihcbiAgICAgIHRoaXMuZWxkZXJDaGlwc0ZpbHRlci5zZWxlY3Rpb25Nb2RlbCxcbiAgICAgIHF1ZXJ5S2V5LFxuICAgICAgaW5jbHVkZVN1ZmZpeCxcbiAgICAgIGV4Y2x1ZGVTdWZmaXhcbiAgICApO1xuICB9XG59XG5cbiJdfQ==
148
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-search-include-exclude.directive.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/select/multi/elder-search-include-exclude.directive.ts"],"names":[],"mappings":"AAAA,OAAO,EAAa,SAAS,EAAE,IAAI,EAAE,KAAK,EAAU,QAAQ,EAAC,MAAM,eAAe,CAAC;AAEnF,OAAO,EAAC,eAAe,EAAE,MAAM,EAAc,EAAE,EAAE,SAAS,EAAE,OAAO,EAAC,MAAM,MAAM,CAAC;AACjF,OAAO,EAAC,gBAAgB,EAAC,MAAM,oDAAoD,CAAC;AACpF,OAAO,EAAC,aAAa,EAAC,MAAM,sBAAsB,CAAC;AAInD,OAAO,EAAC,kBAAkB,EAAC,MAAM,4BAA4B,CAAC;AAE9D,OAAO,EAAC,YAAY,EAAE,KAAK,EAAE,KAAK,EAAE,GAAG,EAAE,GAAG,EAAC,MAAM,gBAAgB,CAAC;AACpE,OAAO,EAAC,gCAAgC,EAAC,MAAM,gDAAgD,CAAC;;;;;AAGhG;;GAEG;AAMH,MAAM,OAAO,kCAAkC;IA0B7C;;;;gFAI4E;IAE5E,YACU,aAA0C,EAC1C,UAAsB,EACb,cAA4C,EACjC,OAAgB;QAHpC,kBAAa,GAAb,aAAa,CAA6B;QAC1C,eAAU,GAAV,UAAU,CAAY;QACb,mBAAc,GAAd,cAAc,CAA8B;QACjC,YAAO,GAAP,OAAO,CAAS;QAlC9C;;;;oFAI4E;QAE3D,QAAG,GAAG,aAAa,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;QAErD,eAAU,GAAG,IAAI,eAAe,CAAS,IAAI,CAAC,CAAC;QAKzD,kBAAa,GAAG,EAAE,CAAC;QASV,iBAAY,GAAG,IAAI,OAAO,EAAQ,CAAC;QAcjD,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,yBAAyB,EAAE,CAAC,IAAI;QAClD,sJAAsJ;SACvJ,CAAC;IACJ,CAAC;IAED;;;;gFAI4E;IAErE,QAAQ;QACb,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QACzC,IAAI,CAAC,cAAc,EAAE,CAAC;QAEtB,IAAI,CAAC,qBAAqB,GAAG,IAAI,gCAAgC,CAC/D,IAAI,CAAC,cAAc,EACnB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,aAAa,EAClC,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,aAAa,CACnC,CAAC;QAEF,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;IACpC,CAAC;IAED;;;;gFAI4E;IAE5E;;OAEG;IACH,IACW,QAAQ,CAAC,KAAa;QAC/B,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;QAC3B,IAAI,CAAC,cAAc,EAAE,CAAC;IACxB,CAAC;IAED,IAAW,QAAQ;QACjB,OAAO,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC;IACpC,CAAC;IAED,IAAW,SAAS;QAClB,OAAO,IAAI,CAAC,UAAU,CAAC;IACzB,CAAC;IAED,IAAW,MAAM;QACf,OAAO,IAAI,CAAC,OAAO,CAAC;IACtB,CAAC;IAED,IAAW,QAAQ;QACjB,OAAO,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC;IAClC,CAAC;IAED;;;;gFAI4E;IAErE,YAAY,CAAC,OAAiB;QACnC,IAAI,CAAC,qBAAqB,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC;IACnD,CAAC;IAEM,KAAK;QACV,IAAI,CAAC,cAAc,CAAC,gBAAgB,CAAC,EAAE,CAAC,CAAC;IAC3C,CAAC;IAED;;;;gFAI4E;IAEpE,cAAc;QACpB,MAAM,MAAM,GAAG,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,cAAc,CAAC;QACzD,IAAI,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,KAAK,MAAM,EAAE;YACzC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;SAC9B;IACH,CAAC;IAEO,yBAAyB;QAE/B,MAAM,eAAe,GAAG,EAAE,CAAC,aAAa,CAAC,CAAC,IAAI,CAC5C,KAAK,CAAC,EAAE,CAAC,EACT,KAAK,EAAE,CACR,CAAC;QAEF,MAAM,kBAAkB,GAAG,IAAI,CAAC,cAAc,CAAC,UAAU,CAAC,IAAI,CAC5D,kBAAkB,CAAC,IAAI,CAAC,UAAU,CAAC,EACnC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC,EACxC,SAAS,CAAC,eAAe,CAAC,EAC1B,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC,EACnC,YAAY,CAAC,EAAE,CAAC,CACjB,CAAC;QAEF,MAAM,aAAa,GAAG,EAAE,CAAC,OAAO,CAAC,CAAC,IAAI,CACpC,KAAK,CAAC,EAAE,CAAC,EACT,GAAG,CAAC,KAAK,CAAC,EAAE;YACV,OAAO,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;QACrC,CAAC,CAAC,CACH,CAAC;QAEF,OAAO,MAAM,CAAC,aAAa,EAAE,kBAAkB,CAAC,CAAC;IACnD,CAAC;IAEO,eAAe,CAAC,SAAkB;QACxC,MAAM,OAAO,GAAG,IAAI,CAAC,qBAAqB,CAAC,SAAS,EAAE,CAAC;QACvD,OAAO,IAAI,gBAAgB,CACzB,IAAI,CAAC,QAAQ,EACb,OAAO,EACP,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,EAC5B,SAAS,CACV,CAAC;IACJ,CAAC;IAEO,cAAc,CAAC,OAAiB;QACtC,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE;YACtB,OAAO,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC;SAC/C;aAAM;YACL,OAAO,IAAI,CAAC;SACb;IACH,CAAC;IAEO,UAAU,CAAC,MAAc;QAC/B,OAAO,CAAC,IAAI,CAAC,uBAAuB,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;IACrD,CAAC;IAEO,uBAAuB,CAAC,KAAU;QACxC,OAAO,gBAAgB,CAAC,cAAc,CAAC,KAAK,CAAC;eACxC,CAAC,KAAK,GAAG,EAAE,CAAC,CAAC,MAAM,KAAK,CAAC,CAAC;IACjC,CAAC;IAEO,WAAW;QACjB,OAAO,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC;IAC5B,CAAC;+GA9KU,kCAAkC;mGAAlC,kCAAkC;;4FAAlC,kCAAkC;kBAL9C,SAAS;mBAAC;oBACT,QAAQ,EAAE,6BAA6B;oBACvC,QAAQ,EAAE,kBAAkB;oBAC5B,UAAU,EAAE,IAAI;iBACjB;;0BAqCI,QAAQ;;0BAAI,IAAI;yCArBZ,aAAa;sBADnB,KAAK;gBAIC,aAAa;sBADnB,KAAK;uBAAC,EAAC,QAAQ,EAAE,IAAI,EAAC;gBAuDZ,QAAQ;sBADlB,KAAK","sourcesContent":["import {DestroyRef, Directive, Host, Input, OnInit, Optional} from '@angular/core';\nimport {SearchInput} from '../../forms/search/domain/input/search-input';\nimport {BehaviorSubject, concat, Observable, of, skipUntil, Subject} from 'rxjs';\nimport {SearchInputState} from '../../forms/search/domain/input/search-input-state';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {ElderSearchContextDirective} from '../../forms/search/elder-search-context.directive';\nimport {NgModel} from '@angular/forms';\nimport {Filter} from '../../../common/data/filters/filter';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {IncludeExcludeSelectionModel} from '../filter/include-exclude-selection-model';\nimport {debounceTime, delay, first, map, tap} from 'rxjs/operators';\nimport {IncludeExcludeModelFilterAdapter} from '../filter/include-exclude-model-filter-adapter';\n\n\n/**\n * Binds a IncludeExcludeModel to a SearchContexts Filter.\n */\n@Directive({\n  selector: '[elderSearchIncludeExclude]',\n  exportAs: 'elderSearchInput',\n  standalone: true\n})\nexport class ElderSearchIncludeExcludeDirective implements SearchInput, OnInit {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly _queryKey$ = new BehaviorSubject<string>(null);\n  private _extractedName: string | null;\n  private _userQueryKey: string | null;\n\n  @Input()\n  public includeSuffix = '';\n\n  @Input({required: true})\n  public excludeSuffix: string;\n\n  private readonly _state$: Observable<SearchInputState>;\n\n  private includeExcludeAdapter: IncludeExcludeModelFilterAdapter;\n\n  public readonly initialDone$ = new Subject<void>();\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private searchContext: ElderSearchContextDirective,\n    private destroyRef: DestroyRef,\n    private readonly selectionModel: IncludeExcludeSelectionModel,\n    @Optional() @Host() private ngModel: NgModel\n  ) {\n    this._state$ = this.buildInputStateObservable().pipe(\n      // tap(state => this.log.error('EMIT SEARCH INPUT STATE [' + state.queryKey + ']: ' + state.queryValue + (state.pristine ? '(pristine)' : ''), state))\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit() {\n    this._extractedName = this.extractName();\n    this.updateQueryKey();\n\n    this.includeExcludeAdapter = new IncludeExcludeModelFilterAdapter(\n      this.selectionModel,\n      this.queryKey + this.includeSuffix,\n      this.queryKey + this.excludeSuffix\n    );\n\n    this.searchContext.register(this);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * (Optional) Usually the control name is used, this allows a custom query key\n   */\n  @Input()\n  public set queryKey(value: string) {\n    this._userQueryKey = value;\n    this.updateQueryKey();\n  }\n\n  public get queryKey() {\n    return this._queryKey$.getValue();\n  }\n\n  public get queryKey$(): Observable<string> {\n    return this._queryKey$;\n  }\n\n  public get state$(): Observable<SearchInputState> {\n    return this._state$;\n  }\n\n  public get readonly(): boolean {\n    return this.ngModel?.isDisabled;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public applyFilters(filters: Filter[]) {\n    this.includeExcludeAdapter.applyFilters(filters);\n  }\n\n  public reset(): void {\n    this.selectionModel.replaceSelection([]);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private updateQueryKey(): void {\n    const newKey = this._userQueryKey ?? this._extractedName;\n    if (this._queryKey$.getValue() !== newKey) {\n      this._queryKey$.next(newKey);\n    }\n  }\n\n  private buildInputStateObservable(): Observable<SearchInputState> {\n\n    const viewStabilized$ = of('stabilized$').pipe(\n      delay(20),\n      first()\n    );\n\n    const inputStateChanges$ = this.selectionModel.selection$.pipe(\n      takeUntilDestroyed(this.destroyRef),\n      map(value => this.buildInputState(true)),\n      skipUntil(viewStabilized$),\n      tap(() => this.initialDone$.next()),\n      debounceTime(20)\n    );\n\n    const startDelayed$ = of('start').pipe(\n      delay(15),\n      map(start => {\n        return this.buildInputState(false);\n      })\n    );\n\n    return concat(startDelayed$, inputStateChanges$);\n  }\n\n  private buildInputState(userEvent: boolean): SearchInputState {\n    const filters = this.includeExcludeAdapter.toFilters();\n    return new SearchInputState(\n      this.queryKey,\n      filters,\n      this.areAllPristine(filters),\n      userEvent\n    );\n  }\n\n  private areAllPristine(filters: Filter[]): boolean {\n    if (filters.length > 0) {\n      return filters.every(f => this.isPristine(f));\n    } else {\n      return true;\n    }\n  }\n\n  private isPristine(filter: Filter): boolean {\n    return !this.isAttributeValuePresent(filter.value);\n  }\n\n  private isAttributeValuePresent(value: any): boolean {\n    return SearchInputState.isValueDefined(value)\n      && (value + '').length !== 0;\n  }\n\n  private extractName(): string | null {\n    return this.ngModel?.name;\n  }\n}\n"]}