@elderbyte/ngx-starter 16.4.5 → 16.4.6

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.
@@ -1,6 +1,6 @@
1
1
  import { FilterContext } from '../../../../../common/data/filters/filter-context';
2
2
  import { BehaviorSubject } from 'rxjs';
3
- import { combineLatestWith, filter, map, skip, tap } from 'rxjs/operators';
3
+ import { combineLatestWith, distinctUntilChanged, filter, map, skip, tap } from 'rxjs/operators';
4
4
  import { FilterUtil } from '../../../../../common/utils/filter-util';
5
5
  import { LoggerFactory } from '@elderbyte/ts-logger';
6
6
  export class SearchContext {
@@ -22,12 +22,6 @@ export class SearchContext {
22
22
  this.userFilterContext = new FilterContext();
23
23
  this.forcedFilterContext = new FilterContext();
24
24
  this.allFilters$ = new BehaviorSubject(undefined);
25
- /***************************************************************************
26
- * *
27
- * Public API *
28
- * *
29
- **************************************************************************/
30
- this.acceptDefaultFilters = true;
31
25
  this.id = id;
32
26
  this.userFilterContext.filters.pipe(combineLatestWith(this.forcedFilterContext.filters), skip(1), map(([userFilters, forcedFilters]) => FilterUtil.mergeCombine(userFilters, forcedFilters)), tap(x => this.log.debug('allFilters changed: ', x))).subscribe(merged => this.allFilters$.next(merged));
33
27
  }
@@ -37,7 +31,7 @@ export class SearchContext {
37
31
  * *
38
32
  **************************************************************************/
39
33
  get allFiltersChanged() {
40
- return this.allFilters$.pipe(filter(filters => !!filters));
34
+ return this.allFilters$.pipe(filter(filters => !!filters), distinctUntilChanged((a, b) => FilterUtil.equals(a, b)));
41
35
  }
42
36
  get userFilters() {
43
37
  return this.userFilterContext.filtersSnapshot;
@@ -48,23 +42,23 @@ export class SearchContext {
48
42
  get forcedFilters() {
49
43
  return this.forcedFilterContext.filtersSnapshot;
50
44
  }
51
- updateDefaultFilters(filters) {
45
+ /***************************************************************************
46
+ * *
47
+ * Public API *
48
+ * *
49
+ **************************************************************************/
50
+ updateUserFiltersSave(filters) {
52
51
  // this.log.error('updateDefaultFilters, acceptDefaultFilters: ' + this.acceptDefaultFilters, filters);
53
- if (this.acceptDefaultFilters) {
54
- this.userFilterContext.updateFilters(FilterUtil.strip(filters, this.forcedFilters));
55
- }
56
- return this.acceptDefaultFilters;
52
+ this.userFilterContext.updateFilters(FilterUtil.strip(filters, this.forcedFilters));
57
53
  }
58
54
  updateUserFilters(filters) {
59
55
  if (filters.length > 0) {
60
56
  // this.log.error('updateUserFilters', filters);
61
- this.acceptDefaultFilters = false;
62
57
  this.userFilterContext.updateFilters(filters);
63
58
  }
64
59
  }
65
60
  replaceUserFilters(filters) {
66
61
  // this.log.error('replaceUserFilters', filters);
67
- this.acceptDefaultFilters = false;
68
62
  this.userFilterContext.replaceFilters(filters);
69
63
  }
70
64
  replaceForcedFilters(filters) {
@@ -72,4 +66,4 @@ export class SearchContext {
72
66
  // Maybe clean up existing user filters here
73
67
  }
74
68
  }
75
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"search-context.js","sourceRoot":"","sources":["../../../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/forms/search/domain/context/search-context.ts"],"names":[],"mappings":"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","sourcesContent":["import {FilterContext} from '../../../../../common/data/filters/filter-context';\nimport {BehaviorSubject, Observable} from 'rxjs';\nimport {Filter} from '../../../../../common/data/filters/filter';\nimport {combineLatestWith, filter, map, skip, tap} from 'rxjs/operators';\nimport {FilterUtil} from '../../../../../common/utils/filter-util';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\n\nexport class SearchContext {\n\n  public static standalone(): SearchContext {\n    return new SearchContext(null);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n\n  public readonly id: string;\n  private readonly userFilterContext = new FilterContext();\n  private readonly forcedFilterContext = new FilterContext();\n\n  private readonly allFilters$ = new BehaviorSubject<Filter[]>(undefined);\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    id: string\n  ) {\n    this.id = id;\n    this.userFilterContext.filters.pipe(\n      combineLatestWith(this.forcedFilterContext.filters),\n      skip(1),\n      map(\n        ([userFilters, forcedFilters]) =>\n          FilterUtil.mergeCombine(userFilters, forcedFilters)\n      ),\n      tap(x => this.log.debug('allFilters changed: ', x)),\n    ).subscribe(\n      merged => this.allFilters$.next(merged)\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public get allFiltersChanged(): Observable<Filter[]> {\n    return this.allFilters$.pipe(\n      filter(filters => !!filters)\n    );\n  }\n\n  public get userFilters(): Filter[] {\n    return this.userFilterContext.filtersSnapshot;\n  }\n\n  public get userFilters$(): Observable<Filter[]> {\n    return this.userFilterContext.filters;\n  }\n\n  public get forcedFilters(): Filter[] {\n    return this.forcedFilterContext.filtersSnapshot;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  private acceptDefaultFilters = true;\n\n  public updateDefaultFilters(filters: Filter[]): boolean {\n    // this.log.error('updateDefaultFilters, acceptDefaultFilters: ' + this.acceptDefaultFilters, filters);\n    if (this.acceptDefaultFilters) {\n      this.userFilterContext.updateFilters(\n        FilterUtil.strip(\n          filters,\n          this.forcedFilters\n        )\n      );\n    }\n    return this.acceptDefaultFilters;\n  }\n\n  public updateUserFilters(filters: Filter[]): void { // called by search inputs\n    if (filters.length > 0) {\n      // this.log.error('updateUserFilters', filters);\n      this.acceptDefaultFilters = false;\n      this.userFilterContext.updateFilters(filters);\n    }\n  }\n\n  public replaceUserFilters(filters: Filter[]): void { // called by url search\n    // this.log.error('replaceUserFilters', filters);\n    this.acceptDefaultFilters = false;\n    this.userFilterContext.replaceFilters(filters);\n  }\n\n  public replaceForcedFilters(filters: Filter[]): void {\n    this.forcedFilterContext.replaceFilters(filters);\n    // Maybe clean up existing user filters here\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n\n}\n"]}
69
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"search-context.js","sourceRoot":"","sources":["../../../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/forms/search/domain/context/search-context.ts"],"names":[],"mappings":"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","sourcesContent":["import {FilterContext} from '../../../../../common/data/filters/filter-context';\nimport {BehaviorSubject, Observable} from 'rxjs';\nimport {Filter} from '../../../../../common/data/filters/filter';\nimport {combineLatestWith, distinctUntilChanged, filter, map, skip, tap} from 'rxjs/operators';\nimport {FilterUtil} from '../../../../../common/utils/filter-util';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\n\nexport class SearchContext {\n\n  public static standalone(): SearchContext {\n    return new SearchContext(null);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n\n  public readonly id: string;\n\n  private readonly userFilterContext = new FilterContext();\n  private readonly forcedFilterContext = new FilterContext();\n\n  private readonly allFilters$ = new BehaviorSubject<Filter[]>(undefined);\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    id: string\n  ) {\n    this.id = id;\n    this.userFilterContext.filters.pipe(\n      combineLatestWith(this.forcedFilterContext.filters),\n      skip(1),\n      map(\n        ([userFilters, forcedFilters]) =>\n          FilterUtil.mergeCombine(userFilters, forcedFilters)\n      ),\n      tap(x => this.log.debug('allFilters changed: ', x)),\n    ).subscribe(\n      merged => this.allFilters$.next(merged)\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public get allFiltersChanged(): Observable<Filter[]> {\n    return this.allFilters$.pipe(\n      filter(filters => !!filters),\n      distinctUntilChanged((a, b) => FilterUtil.equals(a, b))\n    );\n  }\n\n  public get userFilters(): Filter[] {\n    return this.userFilterContext.filtersSnapshot;\n  }\n\n  public get userFilters$(): Observable<Filter[]> {\n    return this.userFilterContext.filters;\n  }\n\n  public get forcedFilters(): Filter[] {\n    return this.forcedFilterContext.filtersSnapshot;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public updateUserFiltersSave(filters: Filter[]): void {\n    // this.log.error('updateDefaultFilters, acceptDefaultFilters: ' + this.acceptDefaultFilters, filters);\n    this.userFilterContext.updateFilters(\n      FilterUtil.strip(\n        filters,\n        this.forcedFilters\n      )\n    );\n  }\n\n  public updateUserFilters(filters: Filter[]): void { // called by search inputs\n    if (filters.length > 0) {\n      // this.log.error('updateUserFilters', filters);\n      this.userFilterContext.updateFilters(filters);\n    }\n  }\n\n  public replaceUserFilters(filters: Filter[]): void { // called by url search\n    // this.log.error('replaceUserFilters', filters);\n    this.userFilterContext.replaceFilters(filters);\n  }\n\n  public replaceForcedFilters(filters: Filter[]): void {\n    this.forcedFilterContext.replaceFilters(filters);\n    // Maybe clean up existing user filters here\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n\n}\n"]}
@@ -1,10 +1,11 @@
1
1
  import { Directive, Input } from '@angular/core';
2
2
  import { LoggerFactory } from '@elderbyte/ts-logger';
3
- import { combineLatestWith, debounceTime, filter, map, switchMap, takeUntil, takeWhile } from 'rxjs/operators';
4
- import { BehaviorSubject, Subject, combineLatest } from 'rxjs';
3
+ import { combineLatestWith, debounceTime, filter, map, switchMap } from 'rxjs/operators';
4
+ import { BehaviorSubject, combineLatest } from 'rxjs';
5
5
  import { Filter } from '../../../common/data/filters/filter';
6
6
  import { FilterContext } from '../../../common/data/filters/filter-context';
7
7
  import { SearchContext } from './domain/context/search-context';
8
+ import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
8
9
  import * as i0 from "@angular/core";
9
10
  import * as i1 from "./domain/context/search-context.service";
10
11
  /**
@@ -20,21 +21,21 @@ export class ElderSearchContextDirective {
20
21
  * Constructor *
21
22
  * *
22
23
  **************************************************************************/
23
- constructor(searchContextService) {
24
+ constructor(searchContextService, destroyRef) {
25
+ this.destroyRef = destroyRef;
24
26
  /***************************************************************************
25
27
  * *
26
28
  * Fields *
27
29
  * *
28
30
  **************************************************************************/
29
31
  this.log = LoggerFactory.getLogger(this.constructor.name);
30
- this.destroy$ = new Subject();
31
32
  this.filtersToApplyAtRegistration = new Map;
32
33
  this._searchInputs$ = new BehaviorSubject([]);
33
34
  this._searchStates$ = new BehaviorSubject([]);
34
35
  this._filterContext$ = new BehaviorSubject(null);
35
36
  this._searchContext$ = new BehaviorSubject(SearchContext.standalone());
36
37
  this.searchContextId$ = new BehaviorSubject(null);
37
- this.searchContextId$.pipe(takeUntil(this.destroy$), filter(contextId => !!contextId), map(contextId => searchContextService.context(contextId))).subscribe(ctx => this._searchContext$.next(ctx));
38
+ this.searchContextId$.pipe(takeUntilDestroyed(this.destroyRef), filter(contextId => !!contextId), map(contextId => searchContextService.context(contextId))).subscribe(ctx => this._searchContext$.next(ctx));
38
39
  }
39
40
  /***************************************************************************
40
41
  * *
@@ -50,10 +51,6 @@ export class ElderSearchContextDirective {
50
51
  this.applySearchContextToInputs(searchContext);
51
52
  this.syncSearchInputsToSearchContext();
52
53
  }
53
- ngOnDestroy() {
54
- this.destroy$.next();
55
- this.destroy$.complete();
56
- }
57
54
  /***************************************************************************
58
55
  * *
59
56
  * Properties *
@@ -160,21 +157,21 @@ export class ElderSearchContextDirective {
160
157
  }
161
158
  }
162
159
  applyAsDefaultsUntilUserAction(searchContext, dcFilters) {
163
- dcFilters.filters.pipe(map(filters => searchContext.updateDefaultFilters(filters)), takeWhile(acceptDefaultFilters => acceptDefaultFilters)).subscribe();
160
+ dcFilters.filters.pipe(takeUntilDestroyed(this.destroyRef), map(filters => searchContext.updateUserFiltersSave(filters))).subscribe();
164
161
  }
165
162
  syncSearchInputsToSearchContext() {
166
- this._searchInputs$.pipe(takeUntil(this.destroy$), switchMap(inputs => combineLatest(inputs.map(i => i.state$))), debounceTime(5)).subscribe(states => {
163
+ this._searchInputs$.pipe(takeUntilDestroyed(this.destroyRef), switchMap(inputs => combineLatest(inputs.map(i => i.state$))), debounceTime(5)).subscribe(states => {
167
164
  this._searchStates$.next(states);
168
165
  });
169
166
  const inputFilters$ = this._searchStates$.pipe(map(states => this.convertToFilters(states)), filter(filters => filters.length > 0));
170
167
  inputFilters$
171
- .pipe(takeUntil(this.destroy$), combineLatestWith(this._searchContext$))
168
+ .pipe(takeUntilDestroyed(this.destroyRef), combineLatestWith(this._searchContext$))
172
169
  .subscribe(([inputFilters, searchContext]) => {
173
170
  searchContext.updateUserFilters(inputFilters);
174
171
  });
175
172
  }
176
173
  syncSearchContextToBoundFilterContext() {
177
- this._searchContext$.pipe(takeUntil(this.destroy$), switchMap(context => context.allFiltersChanged)).subscribe(allFilters => this.applyToBoundFilterContext(allFilters));
174
+ this._searchContext$.pipe(takeUntilDestroyed(this.destroyRef), switchMap(context => context.allFiltersChanged)).subscribe(allFilters => this.applyToBoundFilterContext(allFilters));
178
175
  }
179
176
  applyToBoundFilterContext(filters) {
180
177
  const filterCtx = this.filterContext;
@@ -219,7 +216,7 @@ export class ElderSearchContextDirective {
219
216
  .filter(s => !!s.queryKey)
220
217
  .map(s => new Filter(s.queryKey, s.queryValue));
221
218
  }
222
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ElderSearchContextDirective, deps: [{ token: i1.SearchContextService }], target: i0.ɵɵFactoryTarget.Directive }); }
219
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ElderSearchContextDirective, deps: [{ token: i1.SearchContextService }, { token: i0.DestroyRef }], target: i0.ɵɵFactoryTarget.Directive }); }
223
220
  static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: ElderSearchContextDirective, selector: "[elderSearchContext]", inputs: { searchContextId: "searchContextId", filterContext: ["elderSearchContext", "filterContext"], forcedFilters: "forcedFilters" }, exportAs: ["elderSearchContext"], ngImport: i0 }); }
224
221
  }
225
222
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ElderSearchContextDirective, decorators: [{
@@ -228,7 +225,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
228
225
  selector: '[elderSearchContext]',
229
226
  exportAs: 'elderSearchContext'
230
227
  }]
231
- }], ctorParameters: function () { return [{ type: i1.SearchContextService }]; }, propDecorators: { searchContextId: [{
228
+ }], ctorParameters: function () { return [{ type: i1.SearchContextService }, { type: i0.DestroyRef }]; }, propDecorators: { searchContextId: [{
232
229
  type: Input
233
230
  }], filterContext: [{
234
231
  type: Input,
@@ -236,4 +233,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
236
233
  }], forcedFilters: [{
237
234
  type: Input
238
235
  }] } });
239
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-search-context.directive.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/forms/search/elder-search-context.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {AfterContentInit, AfterViewInit, Directive, Input, OnDestroy, OnInit} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {combineLatestWith, debounceTime, filter, map, switchMap, takeUntil, takeWhile, tap} from 'rxjs/operators';\nimport {BehaviorSubject, Subject, Observable, combineLatest} from 'rxjs';\nimport {SearchInput} from './domain/input/search-input';\nimport {Filter} from '../../../common/data/filters/filter';\nimport {FilterContext} from '../../../common/data/filters/filter-context';\nimport {SearchInputState} from './domain/input/search-input-state';\nimport {SearchContextService} from './domain/context/search-context.service';\nimport {SearchContext} from './domain/context/search-context';\n\n/**\n * The SearchContextDirective binds a group of search-inputs\n * to a SearchContext with a two-way binding.\n *\n * It also binds the SearchContext to a FilterContext (DataContext).\n * TODO Maybe separate this??\n */\n@Directive({\n  selector: '[elderSearchContext]',\n  exportAs: 'elderSearchContext'\n})\nexport class ElderSearchContextDirective implements OnInit, OnDestroy, AfterViewInit {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly destroy$ = new Subject<void>();\n\n  private readonly filtersToApplyAtRegistration = new Map<string, Filter>;\n\n  private readonly _searchInputs$ = new BehaviorSubject<SearchInput[]>([]);\n  private readonly _searchStates$ = new BehaviorSubject<SearchInputState[]>([]);\n  private readonly _filterContext$ = new BehaviorSubject<FilterContext>(null);\n\n  private readonly _searchContext$ = new BehaviorSubject<SearchContext>(SearchContext.standalone());\n  private readonly searchContextId$ = new BehaviorSubject<string>(null);\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    searchContextService: SearchContextService\n  ) {\n    this.searchContextId$.pipe(\n      takeUntil(this.destroy$),\n      filter(contextId => !!contextId),\n      map(contextId => searchContextService.context(contextId)),\n    ).subscribe(ctx => this._searchContext$.next(ctx));\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n    this.applyDefaultFiltersToSearchContext();\n    this.syncSearchContextToBoundFilterContext();\n  }\n\n  public ngAfterViewInit(): void {\n    const searchContext = this.searchContext;\n    this.applySearchContextToInputs(searchContext);\n    this.syncSearchInputsToSearchContext();\n  }\n\n  public ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\n  public set searchContextId(contextId: string) {\n    this.searchContextId$.next(contextId);\n  }\n\n  public get searchContextId(): string {\n    return this.searchContextId$.getValue();\n  }\n\n  public get searchContext$(): Observable<SearchContext> {\n    return this._searchContext$.asObservable();\n  }\n\n  public get searchContext(): SearchContext {\n    return this._searchContext$.getValue();\n  }\n\n  @Input('elderSearchContext')\n  public set filterContext(value: FilterContext | '') {\n    let context: FilterContext;\n    if (value) {\n      if (typeof value !== 'string') {\n        context = value;\n      } else {\n        this.log.warn('Illegal value provided for property filterContext: ', JSON.stringify(value));\n        throw new Error('Illegal value provided for property filterContext! ' + value);\n      }\n    } else {\n      context = FilterContext.empty();\n    }\n    this._filterContext$.next(context);\n  }\n\n  public get filterContext(): FilterContext {\n    return this._filterContext$.getValue();\n  }\n\n  /**\n   * Forced filters are always merged into the final FilterContext.\n   *\n   * This means they override user defined filters, but attempt\n   * to also keep the users intent (by merging) them.\n   */\n  @Input()\n  public set forcedFilters(filters: Filter[]) {\n    this.searchContext.replaceForcedFilters(filters);\n  }\n\n  public get forcedFilters(): Filter[] {\n    return this.searchContext.forcedFilters;\n  }\n\n  public get attributes(): Observable<SearchInput[]> {\n    return this._searchInputs$.asObservable();\n  }\n\n  public get attributesSnapshot(): SearchInput[] {\n    return this._searchInputs$.getValue();\n  }\n\n  public get states$(): Observable<SearchInputState[]> {\n    return this._searchStates$.asObservable();\n  }\n\n  public get statesSnapshot(): SearchInputState[] {\n    return this._searchStates$.getValue();\n  }\n\n  /**\n   * Returns the current user touched attributes. (ignoring fallbacks)\n   */\n  public get userDefinedAttributes$(): Observable<SearchInputState[]> {\n    return this.states$.pipe(\n      map(states => states.filter(s => !s.pristine))\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Register a new search name in this container\n   */\n  public register(searchInput: SearchInput): void {\n    const current = this._searchInputs$.getValue();\n    const toApply = this.filtersToApplyAtRegistration.get(searchInput.name);\n    if (toApply) {\n      searchInput.setValue(toApply.value);\n    }\n    this.log.debug('Registering search input [' + searchInput.name + ']. Late registration value:' + toApply?.value);\n    this._searchInputs$.next([...current, searchInput]);\n  }\n\n  public reset(): void {\n    this.attributesSnapshot\n      .filter(attr => !attr.readonly)\n      .forEach(a => {\n        a.reset();\n      });\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private                                                                 *\n   *                                                                         *\n   **************************************************************************/\n\n  private applyDefaultFiltersToSearchContext(): void {\n\n    const dcFilters = this.filterContext;\n    const searchContext = this.searchContext;\n    if (searchContext.userFilters.length === 0) {\n      this.applyAsDefaultsUntilUserAction(searchContext, dcFilters);\n    } else {\n      dcFilters.replaceFilters(\n        searchContext.userFilters\n      );\n    }\n  }\n\n  private applyAsDefaultsUntilUserAction(\n    searchContext: SearchContext,\n    dcFilters: FilterContext\n  ): void {\n    dcFilters.filters.pipe(\n      map(filters => searchContext.updateDefaultFilters(filters)),\n      takeWhile(acceptDefaultFilters => acceptDefaultFilters)\n    ).subscribe();\n  }\n\n  private syncSearchInputsToSearchContext(): void {\n\n    this._searchInputs$.pipe(\n      takeUntil(this.destroy$),\n      switchMap(inputs => combineLatest(inputs.map(i => i.state$))),\n      debounceTime(5)\n    ).subscribe(states => {\n      this._searchStates$.next(states);\n    });\n\n    const inputFilters$ = this._searchStates$.pipe(\n      map(states => this.convertToFilters(states)),\n      filter(filters => filters.length > 0)\n    );\n\n    inputFilters$\n      .pipe(\n        takeUntil(this.destroy$),\n        combineLatestWith(this._searchContext$)\n      )\n      .subscribe(([inputFilters, searchContext]) => {\n        searchContext.updateUserFilters(inputFilters);\n      });\n  }\n\n  private syncSearchContextToBoundFilterContext(): void {\n    this._searchContext$.pipe(\n      takeUntil(this.destroy$),\n      switchMap(context => context.allFiltersChanged)\n    ).subscribe(allFilters => this.applyToBoundFilterContext(allFilters));\n  }\n\n  private applyToBoundFilterContext(filters: Filter[]): void {\n    const filterCtx = this.filterContext;\n    if (filterCtx) {\n      filterCtx.replaceFilters(filters);\n      this.log.debug('Applied Filters of SearchContext ' + this.searchContextId + ' to DataContext FilterContext.', filters);\n    } else {\n      this.log.warn('Failed to apply filters since no FilterContext is available!', filters);\n    }\n  }\n\n  private applySearchContextToInputs(searchContext: SearchContext): void {\n    this.applyFiltersToInputs(\n      searchContext.userFilters\n    );\n  }\n\n  private applyFiltersToInputs(filters: Filter[]): void {\n    if (filters.length > 0) {\n      const inputsByName = this.searchInputsByName();\n      this.log.warn('applyFiltersToInputs [' + Array.from(inputsByName.keys()) + ']', {\n        filters: filters,\n        inputsByName: inputsByName\n      });\n      filters.forEach(f => {\n        const input = inputsByName.get(f.key);\n        if (input) {\n          input.setValue(f.value);\n        } else {\n          this.filtersToApplyAtRegistration.set(f.key, f);\n        }\n      });\n    }\n  }\n\n  private searchInputsByName(): Map<string, SearchInput> {\n    const inputs = this._searchInputs$.getValue();\n    const map = new Map<string, SearchInput>();\n    inputs.forEach(input => map.set(input.name, input));\n    return map;\n  }\n\n  private convertToFilters(states: SearchInputState[]): Filter[] {\n    return states\n      .filter(s => s.userEvent)\n      .filter(s => !!s.queryKey)\n      .map(s => new Filter(s.queryKey, s.queryValue));\n  }\n}\n\n"]}
236
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-search-context.directive.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/forms/search/elder-search-context.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {AfterViewInit, DestroyRef, Directive, Input, OnInit} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {combineLatestWith, debounceTime, filter, map, switchMap} from 'rxjs/operators';\nimport {BehaviorSubject, Observable, combineLatest} from 'rxjs';\nimport {SearchInput} from './domain/input/search-input';\nimport {Filter} from '../../../common/data/filters/filter';\nimport {FilterContext} from '../../../common/data/filters/filter-context';\nimport {SearchInputState} from './domain/input/search-input-state';\nimport {SearchContextService} from './domain/context/search-context.service';\nimport {SearchContext} from './domain/context/search-context';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\n\n/**\n * The SearchContextDirective binds a group of search-inputs\n * to a SearchContext with a two-way binding.\n *\n * It also binds the SearchContext to a FilterContext (DataContext).\n * TODO Maybe separate this??\n */\n@Directive({\n  selector: '[elderSearchContext]',\n  exportAs: 'elderSearchContext'\n})\nexport class ElderSearchContextDirective implements OnInit, AfterViewInit {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly filtersToApplyAtRegistration = new Map<string, Filter>;\n\n  private readonly _searchInputs$ = new BehaviorSubject<SearchInput[]>([]);\n  private readonly _searchStates$ = new BehaviorSubject<SearchInputState[]>([]);\n  private readonly _filterContext$ = new BehaviorSubject<FilterContext>(null);\n\n  private readonly _searchContext$ = new BehaviorSubject<SearchContext>(SearchContext.standalone());\n  private readonly searchContextId$ = new BehaviorSubject<string>(null);\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    searchContextService: SearchContextService,\n    private readonly destroyRef: DestroyRef\n  ) {\n    this.searchContextId$.pipe(\n      takeUntilDestroyed(this.destroyRef),\n      filter(contextId => !!contextId),\n      map(contextId => searchContextService.context(contextId)),\n    ).subscribe(ctx => this._searchContext$.next(ctx));\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n    this.applyDefaultFiltersToSearchContext();\n    this.syncSearchContextToBoundFilterContext();\n  }\n\n  public ngAfterViewInit(): void {\n    const searchContext = this.searchContext;\n    this.applySearchContextToInputs(searchContext);\n    this.syncSearchInputsToSearchContext();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\n  public set searchContextId(contextId: string) {\n    this.searchContextId$.next(contextId);\n  }\n\n  public get searchContextId(): string {\n    return this.searchContextId$.getValue();\n  }\n\n  public get searchContext$(): Observable<SearchContext> {\n    return this._searchContext$.asObservable();\n  }\n\n  public get searchContext(): SearchContext {\n    return this._searchContext$.getValue();\n  }\n\n  @Input('elderSearchContext')\n  public set filterContext(value: FilterContext | '') {\n    let context: FilterContext;\n    if (value) {\n      if (typeof value !== 'string') {\n        context = value;\n      } else {\n        this.log.warn('Illegal value provided for property filterContext: ', JSON.stringify(value));\n        throw new Error('Illegal value provided for property filterContext! ' + value);\n      }\n    } else {\n      context = FilterContext.empty();\n    }\n    this._filterContext$.next(context);\n  }\n\n  public get filterContext(): FilterContext {\n    return this._filterContext$.getValue();\n  }\n\n  /**\n   * Forced filters are always merged into the final FilterContext.\n   *\n   * This means they override user defined filters, but attempt\n   * to also keep the users intent (by merging) them.\n   */\n  @Input()\n  public set forcedFilters(filters: Filter[]) {\n    this.searchContext.replaceForcedFilters(filters);\n  }\n\n  public get forcedFilters(): Filter[] {\n    return this.searchContext.forcedFilters;\n  }\n\n  public get attributes(): Observable<SearchInput[]> {\n    return this._searchInputs$.asObservable();\n  }\n\n  public get attributesSnapshot(): SearchInput[] {\n    return this._searchInputs$.getValue();\n  }\n\n  public get states$(): Observable<SearchInputState[]> {\n    return this._searchStates$.asObservable();\n  }\n\n  public get statesSnapshot(): SearchInputState[] {\n    return this._searchStates$.getValue();\n  }\n\n  /**\n   * Returns the current user touched attributes. (ignoring fallbacks)\n   */\n  public get userDefinedAttributes$(): Observable<SearchInputState[]> {\n    return this.states$.pipe(\n      map(states => states.filter(s => !s.pristine))\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Register a new search name in this container\n   */\n  public register(searchInput: SearchInput): void {\n    const current = this._searchInputs$.getValue();\n    const toApply = this.filtersToApplyAtRegistration.get(searchInput.name);\n    if (toApply) {\n      searchInput.setValue(toApply.value);\n    }\n    this.log.debug('Registering search input [' + searchInput.name + ']. Late registration value:' + toApply?.value);\n    this._searchInputs$.next([...current, searchInput]);\n  }\n\n  public reset(): void {\n    this.attributesSnapshot\n      .filter(attr => !attr.readonly)\n      .forEach(a => {\n        a.reset();\n      });\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private                                                                 *\n   *                                                                         *\n   **************************************************************************/\n\n  private applyDefaultFiltersToSearchContext(): void {\n\n    const dcFilters = this.filterContext;\n    const searchContext = this.searchContext;\n    if (searchContext.userFilters.length === 0) {\n      this.applyAsDefaultsUntilUserAction(searchContext, dcFilters);\n    } else {\n      dcFilters.replaceFilters(\n        searchContext.userFilters\n      );\n    }\n  }\n\n  private applyAsDefaultsUntilUserAction(\n    searchContext: SearchContext,\n    dcFilters: FilterContext\n  ): void {\n    dcFilters.filters.pipe(\n      takeUntilDestroyed(this.destroyRef),\n      map(filters => searchContext.updateUserFiltersSave(filters)),\n      // takeWhile(acceptDefaultFilters => acceptDefaultFilters)\n    ).subscribe();\n  }\n\n  private syncSearchInputsToSearchContext(): void {\n\n    this._searchInputs$.pipe(\n      takeUntilDestroyed(this.destroyRef),\n      switchMap(inputs => combineLatest(inputs.map(i => i.state$))),\n      debounceTime(5)\n    ).subscribe(states => {\n      this._searchStates$.next(states);\n    });\n\n    const inputFilters$ = this._searchStates$.pipe(\n      map(states => this.convertToFilters(states)),\n      filter(filters => filters.length > 0)\n    );\n\n    inputFilters$\n      .pipe(\n        takeUntilDestroyed(this.destroyRef),\n        combineLatestWith(this._searchContext$)\n      )\n      .subscribe(([inputFilters, searchContext]) => {\n        searchContext.updateUserFilters(inputFilters);\n      });\n  }\n\n  private syncSearchContextToBoundFilterContext(): void {\n    this._searchContext$.pipe(\n      takeUntilDestroyed(this.destroyRef),\n      switchMap(context => context.allFiltersChanged)\n    ).subscribe(allFilters => this.applyToBoundFilterContext(allFilters));\n  }\n\n  private applyToBoundFilterContext(filters: Filter[]): void {\n    const filterCtx = this.filterContext;\n    if (filterCtx) {\n      filterCtx.replaceFilters(filters);\n      this.log.debug('Applied Filters of SearchContext ' + this.searchContextId + ' to DataContext FilterContext.', filters);\n    } else {\n      this.log.warn('Failed to apply filters since no FilterContext is available!', filters);\n    }\n  }\n\n  private applySearchContextToInputs(searchContext: SearchContext): void {\n    this.applyFiltersToInputs(\n      searchContext.userFilters\n    );\n  }\n\n  private applyFiltersToInputs(filters: Filter[]): void {\n    if (filters.length > 0) {\n      const inputsByName = this.searchInputsByName();\n      this.log.warn('applyFiltersToInputs [' + Array.from(inputsByName.keys()) + ']', {\n        filters: filters,\n        inputsByName: inputsByName\n      });\n      filters.forEach(f => {\n        const input = inputsByName.get(f.key);\n        if (input) {\n          input.setValue(f.value);\n        } else {\n          this.filtersToApplyAtRegistration.set(f.key, f);\n        }\n      });\n    }\n  }\n\n  private searchInputsByName(): Map<string, SearchInput> {\n    const inputs = this._searchInputs$.getValue();\n    const map = new Map<string, SearchInput>();\n    inputs.forEach(input => map.set(input.name, input));\n    return map;\n  }\n\n  private convertToFilters(states: SearchInputState[]): Filter[] {\n    return states\n      .filter(s => s.userEvent)\n      .filter(s => !!s.queryKey)\n      .map(s => new Filter(s.queryKey, s.queryValue));\n  }\n}\n\n"]}
@@ -84,6 +84,7 @@ import { MatSidenavModule } from '@angular/material/sidenav';
84
84
  import * as i6 from '@angular/material/datepicker';
85
85
  import { MatDatepicker, MatDatepickerModule } from '@angular/material/datepicker';
86
86
  import localeDECH from '@angular/common/locales/de-CH';
87
+ import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
87
88
  import { TemplatePortal } from '@angular/cdk/portal';
88
89
  import * as i1$a from '@angular/cdk/overlay';
89
90
  import { OverlayModule } from '@angular/cdk/overlay';
@@ -25934,12 +25935,6 @@ class SearchContext {
25934
25935
  this.userFilterContext = new FilterContext();
25935
25936
  this.forcedFilterContext = new FilterContext();
25936
25937
  this.allFilters$ = new BehaviorSubject(undefined);
25937
- /***************************************************************************
25938
- * *
25939
- * Public API *
25940
- * *
25941
- **************************************************************************/
25942
- this.acceptDefaultFilters = true;
25943
25938
  this.id = id;
25944
25939
  this.userFilterContext.filters.pipe(combineLatestWith(this.forcedFilterContext.filters), skip(1), map(([userFilters, forcedFilters]) => FilterUtil.mergeCombine(userFilters, forcedFilters)), tap(x => this.log.debug('allFilters changed: ', x))).subscribe(merged => this.allFilters$.next(merged));
25945
25940
  }
@@ -25949,7 +25944,7 @@ class SearchContext {
25949
25944
  * *
25950
25945
  **************************************************************************/
25951
25946
  get allFiltersChanged() {
25952
- return this.allFilters$.pipe(filter(filters => !!filters));
25947
+ return this.allFilters$.pipe(filter(filters => !!filters), distinctUntilChanged((a, b) => FilterUtil.equals(a, b)));
25953
25948
  }
25954
25949
  get userFilters() {
25955
25950
  return this.userFilterContext.filtersSnapshot;
@@ -25960,23 +25955,23 @@ class SearchContext {
25960
25955
  get forcedFilters() {
25961
25956
  return this.forcedFilterContext.filtersSnapshot;
25962
25957
  }
25963
- updateDefaultFilters(filters) {
25958
+ /***************************************************************************
25959
+ * *
25960
+ * Public API *
25961
+ * *
25962
+ **************************************************************************/
25963
+ updateUserFiltersSave(filters) {
25964
25964
  // this.log.error('updateDefaultFilters, acceptDefaultFilters: ' + this.acceptDefaultFilters, filters);
25965
- if (this.acceptDefaultFilters) {
25966
- this.userFilterContext.updateFilters(FilterUtil.strip(filters, this.forcedFilters));
25967
- }
25968
- return this.acceptDefaultFilters;
25965
+ this.userFilterContext.updateFilters(FilterUtil.strip(filters, this.forcedFilters));
25969
25966
  }
25970
25967
  updateUserFilters(filters) {
25971
25968
  if (filters.length > 0) {
25972
25969
  // this.log.error('updateUserFilters', filters);
25973
- this.acceptDefaultFilters = false;
25974
25970
  this.userFilterContext.updateFilters(filters);
25975
25971
  }
25976
25972
  }
25977
25973
  replaceUserFilters(filters) {
25978
25974
  // this.log.error('replaceUserFilters', filters);
25979
- this.acceptDefaultFilters = false;
25980
25975
  this.userFilterContext.replaceFilters(filters);
25981
25976
  }
25982
25977
  replaceForcedFilters(filters) {
@@ -26036,21 +26031,21 @@ class ElderSearchContextDirective {
26036
26031
  * Constructor *
26037
26032
  * *
26038
26033
  **************************************************************************/
26039
- constructor(searchContextService) {
26034
+ constructor(searchContextService, destroyRef) {
26035
+ this.destroyRef = destroyRef;
26040
26036
  /***************************************************************************
26041
26037
  * *
26042
26038
  * Fields *
26043
26039
  * *
26044
26040
  **************************************************************************/
26045
26041
  this.log = LoggerFactory.getLogger(this.constructor.name);
26046
- this.destroy$ = new Subject();
26047
26042
  this.filtersToApplyAtRegistration = new Map;
26048
26043
  this._searchInputs$ = new BehaviorSubject([]);
26049
26044
  this._searchStates$ = new BehaviorSubject([]);
26050
26045
  this._filterContext$ = new BehaviorSubject(null);
26051
26046
  this._searchContext$ = new BehaviorSubject(SearchContext.standalone());
26052
26047
  this.searchContextId$ = new BehaviorSubject(null);
26053
- this.searchContextId$.pipe(takeUntil(this.destroy$), filter(contextId => !!contextId), map(contextId => searchContextService.context(contextId))).subscribe(ctx => this._searchContext$.next(ctx));
26048
+ this.searchContextId$.pipe(takeUntilDestroyed(this.destroyRef), filter(contextId => !!contextId), map(contextId => searchContextService.context(contextId))).subscribe(ctx => this._searchContext$.next(ctx));
26054
26049
  }
26055
26050
  /***************************************************************************
26056
26051
  * *
@@ -26066,10 +26061,6 @@ class ElderSearchContextDirective {
26066
26061
  this.applySearchContextToInputs(searchContext);
26067
26062
  this.syncSearchInputsToSearchContext();
26068
26063
  }
26069
- ngOnDestroy() {
26070
- this.destroy$.next();
26071
- this.destroy$.complete();
26072
- }
26073
26064
  /***************************************************************************
26074
26065
  * *
26075
26066
  * Properties *
@@ -26176,21 +26167,21 @@ class ElderSearchContextDirective {
26176
26167
  }
26177
26168
  }
26178
26169
  applyAsDefaultsUntilUserAction(searchContext, dcFilters) {
26179
- dcFilters.filters.pipe(map(filters => searchContext.updateDefaultFilters(filters)), takeWhile(acceptDefaultFilters => acceptDefaultFilters)).subscribe();
26170
+ dcFilters.filters.pipe(takeUntilDestroyed(this.destroyRef), map(filters => searchContext.updateUserFiltersSave(filters))).subscribe();
26180
26171
  }
26181
26172
  syncSearchInputsToSearchContext() {
26182
- this._searchInputs$.pipe(takeUntil(this.destroy$), switchMap$1(inputs => combineLatest(inputs.map(i => i.state$))), debounceTime(5)).subscribe(states => {
26173
+ this._searchInputs$.pipe(takeUntilDestroyed(this.destroyRef), switchMap$1(inputs => combineLatest(inputs.map(i => i.state$))), debounceTime(5)).subscribe(states => {
26183
26174
  this._searchStates$.next(states);
26184
26175
  });
26185
26176
  const inputFilters$ = this._searchStates$.pipe(map(states => this.convertToFilters(states)), filter(filters => filters.length > 0));
26186
26177
  inputFilters$
26187
- .pipe(takeUntil(this.destroy$), combineLatestWith(this._searchContext$))
26178
+ .pipe(takeUntilDestroyed(this.destroyRef), combineLatestWith(this._searchContext$))
26188
26179
  .subscribe(([inputFilters, searchContext]) => {
26189
26180
  searchContext.updateUserFilters(inputFilters);
26190
26181
  });
26191
26182
  }
26192
26183
  syncSearchContextToBoundFilterContext() {
26193
- this._searchContext$.pipe(takeUntil(this.destroy$), switchMap$1(context => context.allFiltersChanged)).subscribe(allFilters => this.applyToBoundFilterContext(allFilters));
26184
+ this._searchContext$.pipe(takeUntilDestroyed(this.destroyRef), switchMap$1(context => context.allFiltersChanged)).subscribe(allFilters => this.applyToBoundFilterContext(allFilters));
26194
26185
  }
26195
26186
  applyToBoundFilterContext(filters) {
26196
26187
  const filterCtx = this.filterContext;
@@ -26235,7 +26226,7 @@ class ElderSearchContextDirective {
26235
26226
  .filter(s => !!s.queryKey)
26236
26227
  .map(s => new Filter(s.queryKey, s.queryValue));
26237
26228
  }
26238
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ElderSearchContextDirective, deps: [{ token: SearchContextService }], target: i0.ɵɵFactoryTarget.Directive }); }
26229
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ElderSearchContextDirective, deps: [{ token: SearchContextService }, { token: i0.DestroyRef }], target: i0.ɵɵFactoryTarget.Directive }); }
26239
26230
  static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: ElderSearchContextDirective, selector: "[elderSearchContext]", inputs: { searchContextId: "searchContextId", filterContext: ["elderSearchContext", "filterContext"], forcedFilters: "forcedFilters" }, exportAs: ["elderSearchContext"], ngImport: i0 }); }
26240
26231
  }
26241
26232
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ElderSearchContextDirective, decorators: [{
@@ -26244,7 +26235,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
26244
26235
  selector: '[elderSearchContext]',
26245
26236
  exportAs: 'elderSearchContext'
26246
26237
  }]
26247
- }], ctorParameters: function () { return [{ type: SearchContextService }]; }, propDecorators: { searchContextId: [{
26238
+ }], ctorParameters: function () { return [{ type: SearchContextService }, { type: i0.DestroyRef }]; }, propDecorators: { searchContextId: [{
26248
26239
  type: Input
26249
26240
  }], filterContext: [{
26250
26241
  type: Input,