@elderbyte/ngx-starter 16.3.4 → 16.4.1

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 (48) hide show
  1. package/esm2022/lib/common/data/data-context/data-context-active-page.mjs +5 -5
  2. package/esm2022/lib/common/data/data-context/data-context-auto-starter.mjs +2 -2
  3. package/esm2022/lib/common/data/data-context/data-context-base.mjs +58 -28
  4. package/esm2022/lib/common/data/data-context/data-context-continuable-base.mjs +7 -7
  5. package/esm2022/lib/common/data/data-context/data-context-continuable-paged.mjs +10 -10
  6. package/esm2022/lib/common/data/data-context/data-context-continuable-token.mjs +8 -8
  7. package/esm2022/lib/common/data/data-context/data-context-simple.mjs +6 -5
  8. package/esm2022/lib/common/data/data-context/data-context-source-event-binding.mjs +2 -2
  9. package/esm2022/lib/common/data/data-context/data-context.mjs +1 -1
  10. package/esm2022/lib/common/data/filters/filter-context.mjs +12 -26
  11. package/esm2022/lib/common/data/required-filter-evaluator.mjs +2 -3
  12. package/esm2022/lib/common/data/sort-context.mjs +2 -2
  13. package/esm2022/lib/common/utils/filter-util.mjs +19 -1
  14. package/esm2022/lib/components/data-view/table/elder-table/elder-table.component.mjs +2 -2
  15. package/esm2022/lib/components/forms/search/domain/context/search-context.mjs +57 -0
  16. package/esm2022/lib/components/forms/search/domain/context/search-context.service.mjs +42 -0
  17. package/esm2022/lib/components/forms/search/domain/input/search-input-state.mjs +81 -0
  18. package/esm2022/lib/components/forms/search/domain/input/search-input.mjs +2 -0
  19. package/esm2022/lib/components/forms/search/domain/url/elder-search-url.directive.mjs +74 -0
  20. package/esm2022/lib/components/forms/search/domain/url/elder-search-url.service.mjs +162 -0
  21. package/esm2022/lib/components/forms/search/domain/url/search-query-params-parser.mjs +81 -0
  22. package/esm2022/lib/components/forms/search/elder-search-context.directive.mjs +88 -50
  23. package/esm2022/lib/components/forms/search/elder-search-input.directive.mjs +6 -5
  24. package/esm2022/lib/components/forms/search/elder-search.module.mjs +30 -9
  25. package/fesm2022/elderbyte-ngx-starter.mjs +663 -432
  26. package/fesm2022/elderbyte-ngx-starter.mjs.map +1 -1
  27. package/lib/common/data/data-context/data-context-base.d.ts +10 -3
  28. package/lib/common/data/data-context/data-context.d.ts +6 -1
  29. package/lib/common/data/filters/filter-context.d.ts +1 -6
  30. package/lib/common/utils/filter-util.d.ts +1 -0
  31. package/lib/components/forms/search/domain/context/search-context.d.ts +38 -0
  32. package/lib/components/forms/search/domain/context/search-context.service.d.ts +25 -0
  33. package/lib/components/forms/search/{elder-search-context-url-binding.directive.d.ts → domain/url/elder-search-url.directive.d.ts} +12 -13
  34. package/lib/components/forms/search/domain/url/elder-search-url.service.d.ts +60 -0
  35. package/lib/components/forms/search/domain/url/search-query-params-parser.d.ts +36 -0
  36. package/lib/components/forms/search/elder-search-context.directive.d.ts +29 -16
  37. package/lib/components/forms/search/elder-search-input.directive.d.ts +2 -2
  38. package/lib/components/forms/search/elder-search.module.d.ts +5 -3
  39. package/package.json +1 -1
  40. package/esm2022/lib/components/forms/search/elder-search-context-url-binding.directive.mjs +0 -71
  41. package/esm2022/lib/components/forms/search/model/search-input-state.mjs +0 -81
  42. package/esm2022/lib/components/forms/search/model/search-input.mjs +0 -2
  43. package/esm2022/lib/components/forms/search/search-box/elder-search-context-filters.mjs +0 -10
  44. package/esm2022/lib/components/forms/search/search-box/elder-search-url-binding.service.mjs +0 -184
  45. package/lib/components/forms/search/search-box/elder-search-context-filters.d.ts +0 -7
  46. package/lib/components/forms/search/search-box/elder-search-url-binding.service.d.ts +0 -74
  47. /package/lib/components/forms/search/{model → domain/input}/search-input-state.d.ts +0 -0
  48. /package/lib/components/forms/search/{model → domain/input}/search-input.d.ts +0 -0
@@ -0,0 +1,162 @@
1
+ import { Injectable } from '@angular/core';
2
+ import { LoggerFactory } from '@elderbyte/ts-logger';
3
+ import { filter, map } from 'rxjs/operators';
4
+ import { SearchQueryParamsParser } from './search-query-params-parser';
5
+ import { BehaviorSubject } from 'rxjs';
6
+ import { FilterUtil } from '../../../../../common/utils/filter-util';
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "@angular/router";
9
+ export class UrlFilterByContext {
10
+ constructor(urlFiltersByContext) {
11
+ this.byContext = urlFiltersByContext ?? new Map();
12
+ }
13
+ filtersOf(contextId) {
14
+ return this.byContext.get(contextId);
15
+ }
16
+ static empty() {
17
+ return new UrlFilterByContext();
18
+ }
19
+ withUrlFilters(current) {
20
+ const copy = new Map(this.byContext);
21
+ current
22
+ .forEach((v, k) => copy.set(k, new UrlFilters(v)));
23
+ return new UrlFilterByContext(copy);
24
+ }
25
+ }
26
+ export class UrlFilters {
27
+ constructor(filters) {
28
+ this.consumed = false;
29
+ this.filters = [...filters];
30
+ }
31
+ consumeFilters() {
32
+ this.consumed = true;
33
+ return this.filters;
34
+ }
35
+ }
36
+ export class ElderSearchUrlService {
37
+ /***************************************************************************
38
+ * *
39
+ * Constructor *
40
+ * *
41
+ **************************************************************************/
42
+ constructor(router) {
43
+ this.router = router;
44
+ /***************************************************************************
45
+ * *
46
+ * Fields *
47
+ * *
48
+ **************************************************************************/
49
+ this.log = LoggerFactory.getLogger(this.constructor.name);
50
+ this.SEARCH_QUERY_KEY = 'q';
51
+ this.paramsParser = new SearchQueryParamsParser(this.SEARCH_QUERY_KEY);
52
+ this.urlFiltersByContext$ = new BehaviorSubject(UrlFilterByContext.empty());
53
+ }
54
+ /***************************************************************************
55
+ * *
56
+ * Public API *
57
+ * *
58
+ **************************************************************************/
59
+ init() {
60
+ // this.logRouterEvents();
61
+ this.router.events.pipe(filter(event => event.type === 1 /* EventType.NavigationEnd */), map(event => this.extractUrlFilters(this.router.routerState.snapshot))).subscribe(currentContextFilters => this.updateSeenUrlFilters(currentContextFilters));
62
+ }
63
+ urlFiltersOfContext$(contextId) {
64
+ if (!contextId) {
65
+ throw new Error('Illegal contextId Argument: ' + contextId);
66
+ }
67
+ return this.urlFiltersByContext$.pipe(map(byContext => byContext.filtersOf(contextId)), filter(filters => !!filters));
68
+ }
69
+ updateQueryParams(contextId, filters) {
70
+ if (!contextId)
71
+ throw new Error('Illegal searchContextId Argument: ' + contextId);
72
+ if (!filters)
73
+ throw new Error('Illegal filters Argument: ' + contextId);
74
+ filters = filters.filter(f => f.value !== null && f.value !== undefined);
75
+ const currentUrlAll = this.extractUrlFilters(this.router.routerState.snapshot);
76
+ const currentUrl = currentUrlAll.get(contextId) ?? [];
77
+ if (!FilterUtil.equals(filters, currentUrl)) {
78
+ const resetParams = this.resetSearchParams(contextId, currentUrl);
79
+ const queryParams = { ...resetParams, ...this.buildQueryParams(contextId, filters) };
80
+ this.log.info('writeUrlQueryParams: ' + contextId, {
81
+ searchMap: filters,
82
+ queryParams: queryParams
83
+ });
84
+ this.router.navigate([], {
85
+ queryParams: queryParams,
86
+ replaceUrl: true,
87
+ queryParamsHandling: 'merge'
88
+ });
89
+ }
90
+ else {
91
+ // this.log.info('writeUrlQueryParams PREVENTED')
92
+ }
93
+ }
94
+ /***************************************************************************
95
+ * *
96
+ * Private methods *
97
+ * *
98
+ **************************************************************************/
99
+ updateSeenUrlFilters(current) {
100
+ if (current.size > 0) {
101
+ const filtersByContext = this.urlFiltersByContext$.getValue();
102
+ const newUrlFilters = filtersByContext.withUrlFilters(current);
103
+ this.log.info('updateSeenUrlFilters:', newUrlFilters);
104
+ this.urlFiltersByContext$.next(newUrlFilters);
105
+ }
106
+ }
107
+ logRouterEvents() {
108
+ this.router.events.pipe(map(e => e)).subscribe(e => this.log.debug('ROUTER-EVENT [' + (e.constructor.name) + ']#' + e.id + ': ' + e.url));
109
+ }
110
+ parseUrlFilters(params) {
111
+ return this.paramsParser.parse(params);
112
+ }
113
+ collectAllRouteParams(routerState) {
114
+ return this.collectRouteQueryParams(routerState.root);
115
+ }
116
+ collectRouteQueryParams(root) {
117
+ let params = {};
118
+ let stack = [root];
119
+ while (stack.length > 0) {
120
+ const route = stack.pop();
121
+ params = { ...params, ...route.queryParams };
122
+ stack.push(...route.children);
123
+ }
124
+ return params;
125
+ }
126
+ extractUrlFilters(routerState) {
127
+ const params = this.collectAllRouteParams(routerState);
128
+ return this.parseUrlFilters(params);
129
+ }
130
+ resetSearchParams(searchContextId, urlFiltersOf) {
131
+ if (urlFiltersOf && urlFiltersOf.length > 0) {
132
+ const params = {};
133
+ urlFiltersOf.forEach(toRemove => {
134
+ params[this.queryParamKey(searchContextId, toRemove)] = null;
135
+ });
136
+ return params;
137
+ }
138
+ return {};
139
+ }
140
+ buildQueryParams(searchContextId, filters) {
141
+ const params = {};
142
+ for (const filter of filters) {
143
+ params[this.queryParamKey(searchContextId, filter)] = this.convertValueToArrayOrSingleValue(filter.value);
144
+ }
145
+ return params;
146
+ }
147
+ queryParamKey(searchContextId, filter) {
148
+ return `${this.SEARCH_QUERY_KEY}-${searchContextId}-${filter.key}`;
149
+ }
150
+ convertValueToArrayOrSingleValue(value) {
151
+ return Array.isArray(value) ? `(${value.toString()})` : value;
152
+ }
153
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ElderSearchUrlService, deps: [{ token: i1.Router }], target: i0.ɵɵFactoryTarget.Injectable }); }
154
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ElderSearchUrlService, providedIn: 'root' }); }
155
+ }
156
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ElderSearchUrlService, decorators: [{
157
+ type: Injectable,
158
+ args: [{
159
+ providedIn: 'root'
160
+ }]
161
+ }], ctorParameters: function () { return [{ type: i1.Router }]; } });
162
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-search-url.service.js","sourceRoot":"","sources":["../../../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/forms/search/domain/url/elder-search-url.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {Injectable} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {ActivatedRoute, ActivatedRouteSnapshot, EventType, Params, Router, RouterEvent, RouterStateSnapshot} from '@angular/router';\nimport {filter, map, tap} from 'rxjs/operators';\nimport {Filter} from '../../../../../common/data/filters/filter';\nimport {SearchQueryParamsParser} from './search-query-params-parser';\nimport {BehaviorSubject, Observable} from 'rxjs';\nimport {by} from 'ng-packagr/lib/graph/select';\nimport {FilterUtil} from '../../../../../common/utils/filter-util';\n\nexport class UrlFilterByContext {\n  private readonly byContext: Map<string, UrlFilters>;\n\n  constructor(\n    urlFiltersByContext?: Map<string, UrlFilters>\n  ) {\n    this.byContext = urlFiltersByContext ?? new Map();\n  }\n\n  public filtersOf(contextId: string): UrlFilters {\n    return this.byContext.get(contextId);\n  }\n\n\n  public static empty(): UrlFilterByContext {\n    return new UrlFilterByContext();\n  }\n\n  public withUrlFilters(current: Map<string, Filter[]>): UrlFilterByContext {\n    const copy = new Map(this.byContext);\n    current\n      .forEach((v, k) => copy.set(k, new UrlFilters(v)));\n    return new UrlFilterByContext(copy);\n  }\n}\n\nexport class UrlFilters {\n\n  private readonly filters: Filter[];\n  public consumed = false;\n\n  constructor(\n    filters: Filter[]\n  ) {\n    this.filters = [...filters];\n  }\n\n  public consumeFilters(): Filter[] {\n    this.consumed = true;\n    return this.filters;\n  }\n\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class ElderSearchUrlService {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n  private readonly SEARCH_QUERY_KEY = 'q';\n  private readonly paramsParser = new SearchQueryParamsParser(this.SEARCH_QUERY_KEY);\n\n  private readonly urlFiltersByContext$ = new BehaviorSubject<UrlFilterByContext>(UrlFilterByContext.empty());\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private router: Router\n  ) {\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public init(): void {\n    // this.logRouterEvents();\n    this.router.events.pipe(\n      filter(event => event.type === EventType.NavigationEnd),\n      map(event => this.extractUrlFilters(this.router.routerState.snapshot)),\n    ).subscribe(\n      currentContextFilters => this.updateSeenUrlFilters(currentContextFilters)\n    );\n  }\n\n  public urlFiltersOfContext$(contextId: string): Observable<UrlFilters> {\n    if (!contextId) {\n      throw new Error('Illegal contextId Argument: ' + contextId);\n    }\n    return this.urlFiltersByContext$.pipe(\n      map(byContext => byContext.filtersOf(contextId)),\n      filter(filters => !!filters)\n    );\n  }\n\n  public updateQueryParams(contextId: string, filters: Filter[]): void {\n\n    if(!contextId)\n      throw new Error('Illegal searchContextId Argument: ' + contextId);\n    if(!filters)\n      throw new Error('Illegal filters Argument: ' + contextId);\n\n    filters = filters.filter(f => f.value !== null && f.value !== undefined);\n    const currentUrlAll = this.extractUrlFilters(this.router.routerState.snapshot);\n    const currentUrl = currentUrlAll.get(contextId) ?? [];\n\n    if(!FilterUtil.equals(filters, currentUrl)){\n      const resetParams = this.resetSearchParams(contextId, currentUrl);\n      const queryParams = {...resetParams, ...this.buildQueryParams(contextId, filters)};\n\n      this.log.info('writeUrlQueryParams: ' + contextId, {\n        searchMap: filters,\n        queryParams: queryParams\n      });\n\n      this.router.navigate(\n        [],\n        {\n          queryParams: queryParams,\n          replaceUrl: true,\n          queryParamsHandling: 'merge'\n        });\n    } else {\n      // this.log.info('writeUrlQueryParams PREVENTED')\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private updateSeenUrlFilters(current: Map<string, Filter[]>): void {\n    if (current.size > 0) {\n      const filtersByContext = this.urlFiltersByContext$.getValue();\n      const newUrlFilters = filtersByContext.withUrlFilters(current);\n      this.log.info('updateSeenUrlFilters:', newUrlFilters);\n      this.urlFiltersByContext$.next(newUrlFilters);\n    }\n  }\n\n  private logRouterEvents(): void {\n    this.router.events.pipe(\n      map(e => e as RouterEvent)\n    ).subscribe(\n      e => this.log.debug(\n        'ROUTER-EVENT [' + (e.constructor.name) + ']#' + e.id + ': ' + e.url\n      )\n    );\n  }\n\n  private parseUrlFilters(params: Params): Map<string, Filter[]> {\n    return this.paramsParser.parse(params);\n  }\n\n  private collectAllRouteParams(routerState: RouterStateSnapshot): Params {\n    return this.collectRouteQueryParams(routerState.root);\n  }\n\n  private collectRouteQueryParams(root: ActivatedRouteSnapshot): Params {\n    let params: Params = {};\n    let stack: ActivatedRouteSnapshot[] = [root];\n    while (stack.length > 0) {\n      const route = stack.pop()!;\n      params = {...params, ...route.queryParams};\n      stack.push(...route.children);\n    }\n    return params;\n  }\n\n  private extractUrlFilters(routerState: RouterStateSnapshot): Map<string, Filter[]> {\n    const params = this.collectAllRouteParams(routerState);\n    return this.parseUrlFilters(params);\n  }\n\n\n  private resetSearchParams(searchContextId: string, urlFiltersOf: Filter[]): Params {\n    if (urlFiltersOf && urlFiltersOf.length > 0) {\n      const params: Params = {};\n      urlFiltersOf.forEach(\n        toRemove => {\n          params[this.queryParamKey(searchContextId, toRemove)] = null;\n        }\n      );\n      return params;\n    }\n    return {};\n  }\n\n  private buildQueryParams(searchContextId: string, filters: Filter[]): Params {\n    const params: Params = {};\n    for (const filter of filters) {\n      params[this.queryParamKey(searchContextId, filter)] = this.convertValueToArrayOrSingleValue(filter.value);\n    }\n    return params;\n  }\n\n  private queryParamKey(searchContextId: string, filter: Filter): string {\n    return `${this.SEARCH_QUERY_KEY}-${searchContextId}-${filter.key}`;\n  }\n\n  private convertValueToArrayOrSingleValue(value: string | string[]): string {\n    return Array.isArray(value) ? `(${value.toString()})` : value;\n  }\n\n\n}\n"]}
@@ -0,0 +1,81 @@
1
+ import { LoggerFactory } from '@elderbyte/ts-logger';
2
+ import { Filter } from '../../../../../common/data/filters/filter';
3
+ import { convertToParamMap } from '@angular/router';
4
+ export class SearchQueryParamsParser {
5
+ /***************************************************************************
6
+ * *
7
+ * Constructor *
8
+ * *
9
+ **************************************************************************/
10
+ constructor(queryPrefix) {
11
+ /***************************************************************************
12
+ * *
13
+ * Fields *
14
+ * *
15
+ **************************************************************************/
16
+ this.log = LoggerFactory.getLogger(this.constructor.name);
17
+ this.queryPrefix = queryPrefix;
18
+ }
19
+ /***************************************************************************
20
+ * *
21
+ * Public API *
22
+ * *
23
+ **************************************************************************/
24
+ parse(params) {
25
+ const paramMap = convertToParamMap(params);
26
+ const searchParamKeys = this.extractSearchParameters(paramMap);
27
+ return this.buildSearchContextMap(paramMap, searchParamKeys);
28
+ }
29
+ /***************************************************************************
30
+ * *
31
+ * Private methods *
32
+ * *
33
+ **************************************************************************/
34
+ extractSearchParameters(paramMap) {
35
+ return paramMap.keys.filter(param => param.startsWith(this.queryPrefix + '-'));
36
+ }
37
+ buildSearchContextMap(paramMap, searchParamKeys) {
38
+ const filtersByContextId = new Map();
39
+ for (const key of searchParamKeys) {
40
+ const filterValues = paramMap.getAll(key);
41
+ const filterValue = filterValues[0]; // TODO Support multiple values
42
+ const filterParamData = this.extractFilterParam(key, filterValue);
43
+ const filter = this.convertFilterStringToFilter(filterValue, filterParamData);
44
+ this.checkAndSetFilter(filtersByContextId, filterParamData.searchContextId, filter);
45
+ }
46
+ return filtersByContextId;
47
+ }
48
+ checkAndSetFilter(filtersByContextId, searchContextId, filter) {
49
+ let filtersInContext = filtersByContextId.get(searchContextId);
50
+ if (!filtersInContext) {
51
+ filtersInContext = [];
52
+ filtersByContextId.set(searchContextId, filtersInContext);
53
+ }
54
+ filtersInContext.push(filter);
55
+ }
56
+ extractFilterParam(key, filterValue) {
57
+ const splitParams = key.split('-');
58
+ return {
59
+ searchContextId: splitParams[1],
60
+ filterId: splitParams[2],
61
+ isArray: this.checkIfParamIsArray(filterValue)
62
+ };
63
+ }
64
+ convertFilterStringToFilter(paramValue, filterParamData) {
65
+ let value = paramValue;
66
+ if (this.isSingularArrayValue(value, filterParamData)) {
67
+ value = this.convertValueToArray(value);
68
+ }
69
+ return new Filter(filterParamData.filterId, value);
70
+ }
71
+ isSingularArrayValue(value, filterParamData) {
72
+ return !Array.isArray(value) && filterParamData.isArray;
73
+ }
74
+ convertValueToArray(value) {
75
+ return value.slice(1).slice(0, -1).split(',');
76
+ }
77
+ checkIfParamIsArray(filterValue) {
78
+ return filterValue.startsWith('(') && filterValue.endsWith(')');
79
+ }
80
+ }
81
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"search-query-params-parser.js","sourceRoot":"","sources":["../../../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/forms/search/domain/url/search-query-params-parser.ts"],"names":[],"mappings":"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","sourcesContent":["import {LoggerFactory} from '@elderbyte/ts-logger';\nimport {Filter} from '../../../../../common/data/filters/filter';\nimport {convertToParamMap, ParamMap, Params} from '@angular/router';\n\ninterface FilterParamData {\n  searchContextId: string,\n  filterId: string,\n  isArray: boolean\n}\n\nexport class SearchQueryParamsParser {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n  private readonly queryPrefix: string; // \"q\"\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    queryPrefix: string\n  ) {\n    this.queryPrefix = queryPrefix;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public parse(params: Params): Map<string, Filter[]> {\n    const paramMap = convertToParamMap(params);\n    const searchParamKeys = this.extractSearchParameters(paramMap);\n    return this.buildSearchContextMap(paramMap, searchParamKeys);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private extractSearchParameters(paramMap: ParamMap): string[] {\n    return paramMap.keys.filter(param => param.startsWith(this.queryPrefix + '-'));\n  }\n\n  private buildSearchContextMap(paramMap: ParamMap, searchParamKeys: string[]): Map<string, Filter[]> {\n    const filtersByContextId = new Map<string, Filter[]>();\n    for (const key of searchParamKeys) {\n      const filterValues = paramMap.getAll(key);\n      const filterValue = filterValues[0]; // TODO Support multiple values\n      const filterParamData = this.extractFilterParam(key, filterValue);\n      const filter = this.convertFilterStringToFilter(filterValue, filterParamData);\n      this.checkAndSetFilter(filtersByContextId, filterParamData.searchContextId, filter);\n    }\n    return filtersByContextId;\n  }\n\n  private checkAndSetFilter(\n    filtersByContextId: Map<string, Filter[]>,\n    searchContextId: string,\n    filter: Filter\n  ): void {\n    let filtersInContext = filtersByContextId.get(searchContextId);\n    if (!filtersInContext) {\n      filtersInContext = [];\n      filtersByContextId.set(searchContextId, filtersInContext);\n    }\n    filtersInContext.push(filter);\n  }\n\n\n  private extractFilterParam(key: string, filterValue: string): FilterParamData {\n    const splitParams = key.split('-');\n\n    return {\n      searchContextId: splitParams[1],\n      filterId: splitParams[2],\n      isArray: this.checkIfParamIsArray(filterValue)\n    };\n  }\n\n  private convertFilterStringToFilter(paramValue: string, filterParamData: FilterParamData): Filter {\n    let value: string | string[] = paramValue;\n\n    if (this.isSingularArrayValue(value, filterParamData)) {\n      value = this.convertValueToArray(value);\n    }\n\n    return new Filter(filterParamData.filterId, value);\n  }\n\n  private isSingularArrayValue(value: string | string[], filterParamData: FilterParamData): boolean {\n    return !Array.isArray(value) && filterParamData.isArray;\n  }\n\n  private convertValueToArray(value: string): string[] {\n    return value.slice(1).slice(0, -1).split(',');\n  }\n\n  private checkIfParamIsArray(filterValue: string): boolean {\n    return filterValue.startsWith('(') && filterValue.endsWith(')');\n  }\n\n}\n"]}
@@ -1,16 +1,19 @@
1
- import { Directive, Input } from "@angular/core";
2
- import { LoggerFactory } from "@elderbyte/ts-logger";
3
- import { combineLatestWith, debounceTime, map, switchMap, takeUntil } from "rxjs/operators";
4
- import { BehaviorSubject } from "rxjs/internal/BehaviorSubject";
5
- import { Subject } from "rxjs/internal/Subject";
6
- import { Filter } from "../../../common/data/filters/filter";
7
- import { FilterContext } from "../../../common/data/filters/filter-context";
8
- import { combineLatest } from 'rxjs';
1
+ import { Directive, Input } from '@angular/core';
2
+ import { LoggerFactory } from '@elderbyte/ts-logger';
3
+ import { combineLatestWith, debounceTime, filter, map, switchMap, takeUntil, tap } from 'rxjs/operators';
4
+ import { BehaviorSubject, Subject, combineLatest } from 'rxjs';
5
+ import { Filter } from '../../../common/data/filters/filter';
6
+ import { FilterContext } from '../../../common/data/filters/filter-context';
7
+ import { SearchContext } from './domain/context/search-context';
9
8
  import { FilterUtil } from '../../../common/utils/filter-util';
10
9
  import * as i0 from "@angular/core";
10
+ import * as i1 from "./domain/context/search-context.service";
11
11
  /**
12
- * The search container manages a group of search-inputs
13
- * and holds their values in a central search model.
12
+ * The SearchContextDirective binds a group of search-inputs
13
+ * to a SearchContext with a two-way binding.
14
+ *
15
+ * It also binds the SearchContext to a FilterContext (DataContext).
16
+ * TODO Maybe separate this??
14
17
  */
15
18
  export class ElderSearchContextDirective {
16
19
  /***************************************************************************
@@ -18,50 +21,35 @@ export class ElderSearchContextDirective {
18
21
  * Constructor *
19
22
  * *
20
23
  **************************************************************************/
21
- constructor() {
24
+ constructor(searchContextService) {
22
25
  /***************************************************************************
23
26
  * *
24
27
  * Fields *
25
28
  * *
26
29
  **************************************************************************/
27
- this._forcedFilters$ = new BehaviorSubject([]);
28
30
  this.log = LoggerFactory.getLogger(this.constructor.name);
29
31
  this.destroy$ = new Subject();
30
32
  this._searchInputs$ = new BehaviorSubject([]);
31
33
  this._searchStates$ = new BehaviorSubject([]);
32
34
  this._filterContext$ = new BehaviorSubject(null);
33
- this.userFilters$ = this._filterContext$.pipe(switchMap(ctx => ctx.filters), combineLatestWith(this._forcedFilters$), map(([allFilters, forcedFilters]) => {
34
- return FilterUtil.strip(allFilters, forcedFilters);
35
- }));
35
+ this._searchContext$ = new BehaviorSubject(SearchContext.standalone());
36
+ 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));
36
38
  }
37
39
  /***************************************************************************
38
40
  * *
39
41
  * Life Cycle *
40
42
  * *
41
43
  **************************************************************************/
42
- ngAfterContentInit() {
43
- this._searchInputs$.pipe(takeUntil(this.destroy$), switchMap(inputs => combineLatest(inputs.map(i => i.state$))), combineLatestWith(this._forcedFilters$), debounceTime(5)).subscribe(([states, forcedFilters]) => {
44
- this._searchStates$.next(states);
45
- const userFilters = this.convertToFilters(states);
46
- this.applyFilters(userFilters, forcedFilters);
47
- });
44
+ ngOnInit() {
45
+ this.applyDefaultFiltersToSearchContext();
46
+ this.syncSearchContextToBoundFilterContext();
48
47
  }
49
- applyFilters(userFilters, forcedFilters) {
50
- const context = this.filterContext;
51
- if (context) {
52
- context.updateFilters(userFilters);
53
- context.mergeFilters(forcedFilters);
54
- this.log.trace("Search-Model filters updated:", Array.from(context.filtersSnapshot));
55
- }
56
- else {
57
- this.log.warn("Failed to apply filters since no FilterContext is available!", {
58
- userFilters: userFilters,
59
- forcedFilters: forcedFilters
60
- });
61
- }
48
+ ngAfterContentInit() {
49
+ this.syncSearchInputsToSearchContext();
62
50
  }
63
51
  ngAfterViewInit() {
64
- this.applyModelToInputs();
52
+ this.applySearchContextToInputs();
65
53
  }
66
54
  ngOnDestroy() {
67
55
  this.destroy$.next();
@@ -72,15 +60,27 @@ export class ElderSearchContextDirective {
72
60
  * Properties *
73
61
  * *
74
62
  **************************************************************************/
63
+ set searchContextId(contextId) {
64
+ this.searchContextId$.next(contextId);
65
+ }
66
+ get searchContextId() {
67
+ return this.searchContextId$.getValue();
68
+ }
69
+ get searchContext$() {
70
+ return this._searchContext$.asObservable();
71
+ }
72
+ get searchContext() {
73
+ return this._searchContext$.getValue();
74
+ }
75
75
  set filterContext(value) {
76
76
  let context;
77
77
  if (value) {
78
- if (typeof value !== "string") {
78
+ if (typeof value !== 'string') {
79
79
  context = value;
80
80
  }
81
81
  else {
82
- this.log.warn("Illegal value provided for property filterContext: ", JSON.stringify(value));
83
- throw new Error("Illegal value provided for property filterContext! " + value);
82
+ this.log.warn('Illegal value provided for property filterContext: ', JSON.stringify(value));
83
+ throw new Error('Illegal value provided for property filterContext! ' + value);
84
84
  }
85
85
  }
86
86
  else {
@@ -98,10 +98,10 @@ export class ElderSearchContextDirective {
98
98
  * to also keep the users intent (by merging) them.
99
99
  */
100
100
  set forcedFilters(filters) {
101
- this._forcedFilters$.next(filters);
101
+ this.searchContext.replaceForcedFilters(filters);
102
102
  }
103
103
  get forcedFilters() {
104
- return this._forcedFilters$.getValue();
104
+ return this.searchContext.forcedFilters.filtersSnapshot;
105
105
  }
106
106
  get attributes() {
107
107
  return this._searchInputs$.asObservable();
@@ -130,7 +130,7 @@ export class ElderSearchContextDirective {
130
130
  * Register a new search name in this container
131
131
  */
132
132
  register(searchInput) {
133
- this.log.debug("Registering search input [" + searchInput.name + "]");
133
+ this.log.debug('Registering search input [' + searchInput.name + ']');
134
134
  const current = this._searchInputs$.getValue();
135
135
  this._searchInputs$.next([...current, searchInput]);
136
136
  }
@@ -146,8 +146,44 @@ export class ElderSearchContextDirective {
146
146
  * Private *
147
147
  * *
148
148
  **************************************************************************/
149
- applyModelToInputs() {
150
- this.applyFiltersToInputs(this.filterContext.filtersSnapshot);
149
+ applyDefaultFiltersToSearchContext() {
150
+ const dcFilters = this.filterContext;
151
+ const searchContext = this.searchContext;
152
+ if (searchContext.userFilters.isEmpty) {
153
+ if (!dcFilters.isEmpty) {
154
+ // Consider all filters in DC as defaults, expect for foced filters.
155
+ searchContext.updateUserFilters(FilterUtil.strip(dcFilters.filtersSnapshot, this.forcedFilters));
156
+ }
157
+ }
158
+ else {
159
+ dcFilters.replaceFilters(searchContext.userFilters.filtersSnapshot);
160
+ }
161
+ }
162
+ syncSearchInputsToSearchContext() {
163
+ const inputFilters$ = this._searchInputs$.pipe(switchMap(inputs => combineLatest(inputs.map(i => i.state$))), debounceTime(5), tap(states => this._searchStates$.next(states)), map(states => this.convertToFilters(states)));
164
+ inputFilters$
165
+ .pipe(takeUntil(this.destroy$), combineLatestWith(this._searchContext$))
166
+ .subscribe(([inputFilters, searchContext]) => {
167
+ searchContext.updateUserFilters(inputFilters);
168
+ });
169
+ }
170
+ syncSearchContextToBoundFilterContext() {
171
+ this._searchContext$.pipe(takeUntil(this.destroy$), switchMap(context => context.allFiltersChanged)).subscribe(allFilters => this.applyToBoundFilterContext(allFilters));
172
+ }
173
+ applyToBoundFilterContext(filters) {
174
+ const filterCtx = this.filterContext;
175
+ if (filterCtx) {
176
+ filterCtx.replaceFilters(filters);
177
+ this.log.debug('Applied Filters of SearchContext ' + this.searchContextId + ' to DataContext FilterContext.', filters);
178
+ }
179
+ else {
180
+ this.log.warn('Failed to apply filters since no FilterContext is available!', filters);
181
+ }
182
+ }
183
+ applySearchContextToInputs() {
184
+ const ctx = this.searchContext;
185
+ this.applyFiltersToInputs(ctx.userFilters.filtersSnapshot // this.filterContext.filtersSnapshot (old)
186
+ );
151
187
  }
152
188
  applyFiltersToInputs(filters) {
153
189
  this.log.warn('applyFiltersToInputs', filters);
@@ -171,19 +207,21 @@ export class ElderSearchContextDirective {
171
207
  .filter(s => !!s.queryKey)
172
208
  .map(s => new Filter(s.queryKey, s.queryValue));
173
209
  }
174
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ElderSearchContextDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
175
- static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: ElderSearchContextDirective, selector: "[elderSearchContext]", inputs: { filterContext: ["elderSearchContext", "filterContext"], forcedFilters: "forcedFilters" }, exportAs: ["elderSearchContext"], ngImport: i0 }); }
210
+ 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 }); }
211
+ 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 }); }
176
212
  }
177
213
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ElderSearchContextDirective, decorators: [{
178
214
  type: Directive,
179
215
  args: [{
180
- selector: "[elderSearchContext]",
181
- exportAs: "elderSearchContext"
216
+ selector: '[elderSearchContext]',
217
+ exportAs: 'elderSearchContext'
182
218
  }]
183
- }], ctorParameters: function () { return []; }, propDecorators: { filterContext: [{
219
+ }], ctorParameters: function () { return [{ type: i1.SearchContextService }]; }, propDecorators: { searchContextId: [{
220
+ type: Input
221
+ }], filterContext: [{
184
222
  type: Input,
185
- args: ["elderSearchContext"]
223
+ args: ['elderSearchContext']
186
224
  }], forcedFilters: [{
187
225
  type: Input
188
226
  }] } });
189
- //# 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 } from \"@angular/core\";\nimport { LoggerFactory } from \"@elderbyte/ts-logger\";\nimport { Observable } from \"rxjs/internal/Observable\";\nimport { combineLatestWith, debounceTime, map, switchMap, takeUntil } from \"rxjs/operators\";\nimport { SearchInput } from \"./model/search-input\";\nimport { BehaviorSubject } from \"rxjs/internal/BehaviorSubject\";\nimport { Subject } from \"rxjs/internal/Subject\";\nimport { Filter } from \"../../../common/data/filters/filter\";\nimport { FilterContext } from \"../../../common/data/filters/filter-context\";\nimport { SearchInputState } from \"./model/search-input-state\";\nimport {combineLatest} from 'rxjs';\nimport {FilterUtil} from '../../../common/utils/filter-util';\n\n/**\n * The search container manages a group of search-inputs\n * and holds their values in a central search model.\n */\n@Directive({\n  selector: \"[elderSearchContext]\",\n  exportAs: \"elderSearchContext\"\n})\nexport class ElderSearchContextDirective implements AfterViewInit, OnDestroy, AfterContentInit {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private _forcedFilters$ = new BehaviorSubject<Filter[]>([]);\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly destroy$ = new Subject<void>();\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  public readonly userFilters$: Observable<Filter[]>;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor() {\n    this.userFilters$ = this._filterContext$.pipe(\n      switchMap(ctx => ctx.filters),\n      combineLatestWith(this._forcedFilters$),\n      map(([allFilters, forcedFilters]) => {\n        return FilterUtil.strip(allFilters, forcedFilters);\n      })\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngAfterContentInit() {\n    this._searchInputs$.pipe(\n      takeUntil(this.destroy$),\n      switchMap(inputs => combineLatest(inputs.map(i => i.state$))),\n      combineLatestWith(this._forcedFilters$),\n      debounceTime(5)\n    ).subscribe(([states, forcedFilters]) => {\n      this._searchStates$.next(states);\n      const userFilters = this.convertToFilters(states);\n      this.applyFilters(userFilters, forcedFilters);\n    });\n  }\n\n  private applyFilters(\n    userFilters: Filter[],\n    forcedFilters: Filter[]\n  ): void {\n    const context = this.filterContext;\n    if (context) {\n      context.updateFilters(userFilters);\n      context.mergeFilters(forcedFilters);\n      this.log.trace(\"Search-Model filters updated:\", Array.from(context.filtersSnapshot));\n    } else {\n      this.log.warn(\"Failed to apply filters since no FilterContext is available!\", {\n        userFilters: userFilters,\n        forcedFilters: forcedFilters\n      });\n    }\n  }\n\n  public ngAfterViewInit(): void {\n    this.applyModelToInputs();\n  }\n\n  public ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\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._forcedFilters$.next(filters);\n  }\n\n  public get forcedFilters(): Filter[] {\n    return this._forcedFilters$.getValue();\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    this.log.debug(\"Registering search input [\" + searchInput.name + \"]\");\n    const current = this._searchInputs$.getValue();\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  public applyModelToInputs(): void {\n    this.applyFiltersToInputs(\n        this.filterContext.filtersSnapshot\n    );\n  }\n\n  private applyFiltersToInputs(filters: Filter[]): void {\n    this.log.warn('applyFiltersToInputs', filters);\n    this._searchInputs$.getValue().forEach(input => {\n      const filter = this.findFilterForInput(input, filters);\n      if (filter) {\n        input.applyInitialValue(filter.value);\n      }\n    });\n  }\n\n  private findFilterForInput(input: SearchInput, filters: Filter[]): Filter {\n    for (const filter of filters) {\n      if (filter.key === input.name) {\n        return filter;\n      }\n    }\n    return null;\n  }\n\n  private convertToFilters(states: SearchInputState[]): Filter[] {\n    return states\n      .filter(s => !!s.queryKey)\n      .map(s => new Filter(s.queryKey, s.queryValue));\n  }\n}\n\n"]}
227
+ //# 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, 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';\nimport {FilterUtil} from '../../../common/utils/filter-util';\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, OnDestroy, AfterContentInit {\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 _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 ngAfterContentInit(): void {\n    this.syncSearchInputsToSearchContext();\n  }\n\n  public ngAfterViewInit(): void {\n    this.applySearchContextToInputs();\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.filtersSnapshot;\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    this.log.debug('Registering search input [' + searchInput.name + ']');\n    const current = this._searchInputs$.getValue();\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    const dcFilters = this.filterContext;\n    const searchContext = this.searchContext;\n    if (searchContext.userFilters.isEmpty) {\n      if (!dcFilters.isEmpty) {\n        // Consider all filters in DC as defaults, expect for foced filters.\n        searchContext.updateUserFilters(\n          FilterUtil.strip(\n            dcFilters.filtersSnapshot,\n            this.forcedFilters\n          )\n        );\n      }\n    } else {\n      dcFilters.replaceFilters(\n        searchContext.userFilters.filtersSnapshot\n      );\n    }\n  }\n\n  private syncSearchInputsToSearchContext(): void {\n    const inputFilters$ = this._searchInputs$.pipe(\n      switchMap(inputs => combineLatest(inputs.map(i => i.state$))),\n      debounceTime(5),\n      tap(states => this._searchStates$.next(states)),\n      map(states => this.convertToFilters(states))\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  public applySearchContextToInputs(): void {\n    const ctx = this.searchContext;\n    this.applyFiltersToInputs(\n      ctx.userFilters.filtersSnapshot // this.filterContext.filtersSnapshot (old)\n    );\n  }\n\n  private applyFiltersToInputs(filters: Filter[]): void {\n    this.log.warn('applyFiltersToInputs', filters);\n    this._searchInputs$.getValue().forEach(input => {\n      const filter = this.findFilterForInput(input, filters);\n      if (filter) {\n        input.applyInitialValue(filter.value);\n      }\n    });\n  }\n\n  private findFilterForInput(input: SearchInput, filters: Filter[]): Filter {\n    for (const filter of filters) {\n      if (filter.key === input.name) {\n        return filter;\n      }\n    }\n    return null;\n  }\n\n  private convertToFilters(states: SearchInputState[]): Filter[] {\n    return states\n      .filter(s => !!s.queryKey)\n      .map(s => new Filter(s.queryKey, s.queryValue));\n  }\n\n\n}\n\n"]}