@elderbyte/ngx-starter 16.3.4 → 16.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/common/data/data-context/data-context-active-page.mjs +5 -5
- package/esm2022/lib/common/data/data-context/data-context-auto-starter.mjs +2 -2
- package/esm2022/lib/common/data/data-context/data-context-base.mjs +58 -28
- package/esm2022/lib/common/data/data-context/data-context-continuable-base.mjs +7 -7
- package/esm2022/lib/common/data/data-context/data-context-continuable-paged.mjs +10 -10
- package/esm2022/lib/common/data/data-context/data-context-continuable-token.mjs +8 -8
- package/esm2022/lib/common/data/data-context/data-context-simple.mjs +6 -5
- package/esm2022/lib/common/data/data-context/data-context-source-event-binding.mjs +2 -2
- package/esm2022/lib/common/data/data-context/data-context.mjs +1 -1
- package/esm2022/lib/common/data/filters/filter-context.mjs +12 -26
- package/esm2022/lib/common/data/sort-context.mjs +2 -2
- package/esm2022/lib/common/utils/filter-util.mjs +19 -1
- package/esm2022/lib/components/data-view/table/elder-table/elder-table.component.mjs +2 -2
- package/esm2022/lib/components/forms/search/domain/context/search-context.mjs +52 -0
- package/esm2022/lib/components/forms/search/domain/context/search-context.service.mjs +42 -0
- package/esm2022/lib/components/forms/search/domain/input/search-input-state.mjs +81 -0
- package/esm2022/lib/components/forms/search/domain/input/search-input.mjs +2 -0
- package/esm2022/lib/components/forms/search/domain/url/elder-search-url.directive.mjs +74 -0
- package/esm2022/lib/components/forms/search/domain/url/elder-search-url.service.mjs +162 -0
- package/esm2022/lib/components/forms/search/domain/url/search-query-params-parser.mjs +81 -0
- package/esm2022/lib/components/forms/search/elder-search-context.directive.mjs +87 -51
- package/esm2022/lib/components/forms/search/elder-search-input.directive.mjs +6 -5
- package/esm2022/lib/components/forms/search/elder-search.module.mjs +30 -9
- package/fesm2022/elderbyte-ngx-starter.mjs +657 -430
- package/fesm2022/elderbyte-ngx-starter.mjs.map +1 -1
- package/lib/common/data/data-context/data-context-base.d.ts +10 -3
- package/lib/common/data/data-context/data-context.d.ts +6 -1
- package/lib/common/data/filters/filter-context.d.ts +1 -6
- package/lib/common/utils/filter-util.d.ts +1 -0
- package/lib/components/forms/search/domain/context/search-context.d.ts +37 -0
- package/lib/components/forms/search/domain/context/search-context.service.d.ts +25 -0
- package/lib/components/forms/search/{elder-search-context-url-binding.directive.d.ts → domain/url/elder-search-url.directive.d.ts} +12 -13
- package/lib/components/forms/search/domain/url/elder-search-url.service.d.ts +60 -0
- package/lib/components/forms/search/domain/url/search-query-params-parser.d.ts +36 -0
- package/lib/components/forms/search/elder-search-context.directive.d.ts +29 -16
- package/lib/components/forms/search/elder-search-input.directive.d.ts +2 -2
- package/lib/components/forms/search/elder-search.module.d.ts +5 -3
- package/package.json +1 -1
- package/esm2022/lib/components/forms/search/elder-search-context-url-binding.directive.mjs +0 -71
- package/esm2022/lib/components/forms/search/model/search-input-state.mjs +0 -81
- package/esm2022/lib/components/forms/search/model/search-input.mjs +0 -2
- package/esm2022/lib/components/forms/search/search-box/elder-search-context-filters.mjs +0 -10
- package/esm2022/lib/components/forms/search/search-box/elder-search-url-binding.service.mjs +0 -184
- package/lib/components/forms/search/search-box/elder-search-context-filters.d.ts +0 -7
- package/lib/components/forms/search/search-box/elder-search-url-binding.service.d.ts +0 -74
- /package/lib/components/forms/search/{model → domain/input}/search-input-state.d.ts +0 -0
- /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":"AAAA,OAAO,EAAC,aAAa,EAAC,MAAM,sBAAsB,CAAC;AACnD,OAAO,EAAC,MAAM,EAAC,MAAM,2CAA2C,CAAC;AACjE,OAAO,EAAC,iBAAiB,EAAmB,MAAM,iBAAiB,CAAC;AAQpE,MAAM,OAAO,uBAAuB;IAWlC;;;;gFAI4E;IAE5E,YACE,WAAmB;QAhBrB;;;;oFAI4E;QAE3D,QAAG,GAAG,aAAa,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;QAYpE,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC;IACjC,CAAC;IAED;;;;gFAI4E;IAErE,KAAK,CAAC,MAAc;QACzB,MAAM,QAAQ,GAAG,iBAAiB,CAAC,MAAM,CAAC,CAAC;QAC3C,MAAM,eAAe,GAAG,IAAI,CAAC,uBAAuB,CAAC,QAAQ,CAAC,CAAC;QAC/D,OAAO,IAAI,CAAC,qBAAqB,CAAC,QAAQ,EAAE,eAAe,CAAC,CAAC;IAC/D,CAAC;IAED;;;;gFAI4E;IAEpE,uBAAuB,CAAC,QAAkB;QAChD,OAAO,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,UAAU,CAAC,IAAI,CAAC,WAAW,GAAG,GAAG,CAAC,CAAC,CAAC;IACjF,CAAC;IAEO,qBAAqB,CAAC,QAAkB,EAAE,eAAyB;QACzE,MAAM,kBAAkB,GAAG,IAAI,GAAG,EAAoB,CAAC;QACvD,KAAK,MAAM,GAAG,IAAI,eAAe,EAAE;YACjC,MAAM,YAAY,GAAG,QAAQ,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;YAC1C,MAAM,WAAW,GAAG,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,+BAA+B;YACpE,MAAM,eAAe,GAAG,IAAI,CAAC,kBAAkB,CAAC,GAAG,EAAE,WAAW,CAAC,CAAC;YAClE,MAAM,MAAM,GAAG,IAAI,CAAC,2BAA2B,CAAC,WAAW,EAAE,eAAe,CAAC,CAAC;YAC9E,IAAI,CAAC,iBAAiB,CAAC,kBAAkB,EAAE,eAAe,CAAC,eAAe,EAAE,MAAM,CAAC,CAAC;SACrF;QACD,OAAO,kBAAkB,CAAC;IAC5B,CAAC;IAEO,iBAAiB,CACvB,kBAAyC,EACzC,eAAuB,EACvB,MAAc;QAEd,IAAI,gBAAgB,GAAG,kBAAkB,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;QAC/D,IAAI,CAAC,gBAAgB,EAAE;YACrB,gBAAgB,GAAG,EAAE,CAAC;YACtB,kBAAkB,CAAC,GAAG,CAAC,eAAe,EAAE,gBAAgB,CAAC,CAAC;SAC3D;QACD,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAChC,CAAC;IAGO,kBAAkB,CAAC,GAAW,EAAE,WAAmB;QACzD,MAAM,WAAW,GAAG,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAEnC,OAAO;YACL,eAAe,EAAE,WAAW,CAAC,CAAC,CAAC;YAC/B,QAAQ,EAAE,WAAW,CAAC,CAAC,CAAC;YACxB,OAAO,EAAE,IAAI,CAAC,mBAAmB,CAAC,WAAW,CAAC;SAC/C,CAAC;IACJ,CAAC;IAEO,2BAA2B,CAAC,UAAkB,EAAE,eAAgC;QACtF,IAAI,KAAK,GAAsB,UAAU,CAAC;QAE1C,IAAI,IAAI,CAAC,oBAAoB,CAAC,KAAK,EAAE,eAAe,CAAC,EAAE;YACrD,KAAK,GAAG,IAAI,CAAC,mBAAmB,CAAC,KAAK,CAAC,CAAC;SACzC;QAED,OAAO,IAAI,MAAM,CAAC,eAAe,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;IACrD,CAAC;IAEO,oBAAoB,CAAC,KAAwB,EAAE,eAAgC;QACrF,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,IAAI,eAAe,CAAC,OAAO,CAAC;IAC1D,CAAC;IAEO,mBAAmB,CAAC,KAAa;QACvC,OAAO,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IAChD,CAAC;IAEO,mBAAmB,CAAC,WAAmB;QAC7C,OAAO,WAAW,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,WAAW,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;IAClE,CAAC;CAEF","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,18 @@
|
|
|
1
|
-
import { Directive, Input } from
|
|
2
|
-
import { LoggerFactory } from
|
|
3
|
-
import { combineLatestWith, debounceTime, map, switchMap, takeUntil } from
|
|
4
|
-
import { BehaviorSubject } from
|
|
5
|
-
import {
|
|
6
|
-
import {
|
|
7
|
-
import {
|
|
8
|
-
import { combineLatest } from 'rxjs';
|
|
9
|
-
import { FilterUtil } from '../../../common/utils/filter-util';
|
|
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';
|
|
10
8
|
import * as i0 from "@angular/core";
|
|
9
|
+
import * as i1 from "./domain/context/search-context.service";
|
|
11
10
|
/**
|
|
12
|
-
* The
|
|
13
|
-
*
|
|
11
|
+
* The SearchContextDirective binds a group of search-inputs
|
|
12
|
+
* to a SearchContext with a two-way binding.
|
|
13
|
+
*
|
|
14
|
+
* It also binds the SearchContext to a FilterContext (DataContext).
|
|
15
|
+
* TODO Maybe separate this??
|
|
14
16
|
*/
|
|
15
17
|
export class ElderSearchContextDirective {
|
|
16
18
|
/***************************************************************************
|
|
@@ -18,50 +20,35 @@ export class ElderSearchContextDirective {
|
|
|
18
20
|
* Constructor *
|
|
19
21
|
* *
|
|
20
22
|
**************************************************************************/
|
|
21
|
-
constructor() {
|
|
23
|
+
constructor(searchContextService) {
|
|
22
24
|
/***************************************************************************
|
|
23
25
|
* *
|
|
24
26
|
* Fields *
|
|
25
27
|
* *
|
|
26
28
|
**************************************************************************/
|
|
27
|
-
this._forcedFilters$ = new BehaviorSubject([]);
|
|
28
29
|
this.log = LoggerFactory.getLogger(this.constructor.name);
|
|
29
30
|
this.destroy$ = new Subject();
|
|
30
31
|
this._searchInputs$ = new BehaviorSubject([]);
|
|
31
32
|
this._searchStates$ = new BehaviorSubject([]);
|
|
32
33
|
this._filterContext$ = new BehaviorSubject(null);
|
|
33
|
-
this.
|
|
34
|
-
|
|
35
|
-
|
|
34
|
+
this._searchContext$ = new BehaviorSubject(SearchContext.standalone());
|
|
35
|
+
this.searchContextId$ = new BehaviorSubject(null);
|
|
36
|
+
this.searchContextId$.pipe(takeUntil(this.destroy$), filter(contextId => !!contextId), map(contextId => searchContextService.context(contextId))).subscribe(ctx => this._searchContext$.next(ctx));
|
|
36
37
|
}
|
|
37
38
|
/***************************************************************************
|
|
38
39
|
* *
|
|
39
40
|
* Life Cycle *
|
|
40
41
|
* *
|
|
41
42
|
**************************************************************************/
|
|
42
|
-
|
|
43
|
-
this.
|
|
44
|
-
|
|
45
|
-
const userFilters = this.convertToFilters(states);
|
|
46
|
-
this.applyFilters(userFilters, forcedFilters);
|
|
47
|
-
});
|
|
43
|
+
ngOnInit() {
|
|
44
|
+
this.applyDefaultFiltersToSearchContext();
|
|
45
|
+
this.syncSearchContextToBoundFilterContext();
|
|
48
46
|
}
|
|
49
|
-
|
|
50
|
-
|
|
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
|
-
}
|
|
47
|
+
ngAfterContentInit() {
|
|
48
|
+
this.syncSearchInputsToSearchContext();
|
|
62
49
|
}
|
|
63
50
|
ngAfterViewInit() {
|
|
64
|
-
this.
|
|
51
|
+
this.applySearchContextToInputs();
|
|
65
52
|
}
|
|
66
53
|
ngOnDestroy() {
|
|
67
54
|
this.destroy$.next();
|
|
@@ -72,15 +59,27 @@ export class ElderSearchContextDirective {
|
|
|
72
59
|
* Properties *
|
|
73
60
|
* *
|
|
74
61
|
**************************************************************************/
|
|
62
|
+
set searchContextId(contextId) {
|
|
63
|
+
this.searchContextId$.next(contextId);
|
|
64
|
+
}
|
|
65
|
+
get searchContextId() {
|
|
66
|
+
return this.searchContextId$.getValue();
|
|
67
|
+
}
|
|
68
|
+
get searchContext$() {
|
|
69
|
+
return this._searchContext$.asObservable();
|
|
70
|
+
}
|
|
71
|
+
get searchContext() {
|
|
72
|
+
return this._searchContext$.getValue();
|
|
73
|
+
}
|
|
75
74
|
set filterContext(value) {
|
|
76
75
|
let context;
|
|
77
76
|
if (value) {
|
|
78
|
-
if (typeof value !==
|
|
77
|
+
if (typeof value !== 'string') {
|
|
79
78
|
context = value;
|
|
80
79
|
}
|
|
81
80
|
else {
|
|
82
|
-
this.log.warn(
|
|
83
|
-
throw new Error(
|
|
81
|
+
this.log.warn('Illegal value provided for property filterContext: ', JSON.stringify(value));
|
|
82
|
+
throw new Error('Illegal value provided for property filterContext! ' + value);
|
|
84
83
|
}
|
|
85
84
|
}
|
|
86
85
|
else {
|
|
@@ -98,10 +97,10 @@ export class ElderSearchContextDirective {
|
|
|
98
97
|
* to also keep the users intent (by merging) them.
|
|
99
98
|
*/
|
|
100
99
|
set forcedFilters(filters) {
|
|
101
|
-
this.
|
|
100
|
+
this.searchContext.forcedFilters.replaceFilters(filters);
|
|
102
101
|
}
|
|
103
102
|
get forcedFilters() {
|
|
104
|
-
return this.
|
|
103
|
+
return this.searchContext.forcedFilters.filtersSnapshot;
|
|
105
104
|
}
|
|
106
105
|
get attributes() {
|
|
107
106
|
return this._searchInputs$.asObservable();
|
|
@@ -130,7 +129,7 @@ export class ElderSearchContextDirective {
|
|
|
130
129
|
* Register a new search name in this container
|
|
131
130
|
*/
|
|
132
131
|
register(searchInput) {
|
|
133
|
-
this.log.debug(
|
|
132
|
+
this.log.debug('Registering search input [' + searchInput.name + ']');
|
|
134
133
|
const current = this._searchInputs$.getValue();
|
|
135
134
|
this._searchInputs$.next([...current, searchInput]);
|
|
136
135
|
}
|
|
@@ -146,8 +145,43 @@ export class ElderSearchContextDirective {
|
|
|
146
145
|
* Private *
|
|
147
146
|
* *
|
|
148
147
|
**************************************************************************/
|
|
149
|
-
|
|
150
|
-
this.
|
|
148
|
+
applyDefaultFiltersToSearchContext() {
|
|
149
|
+
const dcFilters = this.filterContext;
|
|
150
|
+
const searchContext = this.searchContext;
|
|
151
|
+
if (searchContext.userFilters.isEmpty) {
|
|
152
|
+
if (!dcFilters.isEmpty) {
|
|
153
|
+
searchContext.updateUserFilters(dcFilters.filtersSnapshot);
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
else {
|
|
157
|
+
dcFilters.replaceFilters(searchContext.userFilters.filtersSnapshot);
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
syncSearchInputsToSearchContext() {
|
|
161
|
+
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)));
|
|
162
|
+
inputFilters$
|
|
163
|
+
.pipe(takeUntil(this.destroy$), combineLatestWith(this._searchContext$))
|
|
164
|
+
.subscribe(([inputFilters, searchContext]) => {
|
|
165
|
+
searchContext.updateUserFilters(inputFilters);
|
|
166
|
+
});
|
|
167
|
+
}
|
|
168
|
+
syncSearchContextToBoundFilterContext() {
|
|
169
|
+
this._searchContext$.pipe(takeUntil(this.destroy$), switchMap(context => context.allFiltersChanged)).subscribe(allFilters => this.applyToBoundFilterContext(allFilters));
|
|
170
|
+
}
|
|
171
|
+
applyToBoundFilterContext(filters) {
|
|
172
|
+
const filterCtx = this.filterContext;
|
|
173
|
+
if (filterCtx) {
|
|
174
|
+
filterCtx.replaceFilters(filters);
|
|
175
|
+
this.log.debug('Applied Filters of SearchContext ' + this.searchContextId + ' to DataContext FilterContext.', filters);
|
|
176
|
+
}
|
|
177
|
+
else {
|
|
178
|
+
this.log.warn('Failed to apply filters since no FilterContext is available!', filters);
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
applySearchContextToInputs() {
|
|
182
|
+
const ctx = this.searchContext;
|
|
183
|
+
this.applyFiltersToInputs(ctx.userFilters.filtersSnapshot // this.filterContext.filtersSnapshot (old)
|
|
184
|
+
);
|
|
151
185
|
}
|
|
152
186
|
applyFiltersToInputs(filters) {
|
|
153
187
|
this.log.warn('applyFiltersToInputs', filters);
|
|
@@ -171,19 +205,21 @@ export class ElderSearchContextDirective {
|
|
|
171
205
|
.filter(s => !!s.queryKey)
|
|
172
206
|
.map(s => new Filter(s.queryKey, s.queryValue));
|
|
173
207
|
}
|
|
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 }); }
|
|
208
|
+
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 }); }
|
|
209
|
+
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
210
|
}
|
|
177
211
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ElderSearchContextDirective, decorators: [{
|
|
178
212
|
type: Directive,
|
|
179
213
|
args: [{
|
|
180
|
-
selector:
|
|
181
|
-
exportAs:
|
|
214
|
+
selector: '[elderSearchContext]',
|
|
215
|
+
exportAs: 'elderSearchContext'
|
|
182
216
|
}]
|
|
183
|
-
}], ctorParameters: function () { return []; }, propDecorators: {
|
|
217
|
+
}], ctorParameters: function () { return [{ type: i1.SearchContextService }]; }, propDecorators: { searchContextId: [{
|
|
218
|
+
type: Input
|
|
219
|
+
}], filterContext: [{
|
|
184
220
|
type: Input,
|
|
185
|
-
args: [
|
|
221
|
+
args: ['elderSearchContext']
|
|
186
222
|
}], forcedFilters: [{
|
|
187
223
|
type: Input
|
|
188
224
|
}] } });
|
|
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"]}
|
|
225
|
+
//# 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":"AAAA,OAAO,EAAkC,SAAS,EAAE,KAAK,EAAoB,MAAM,eAAe,CAAC;AACnG,OAAO,EAAC,aAAa,EAAC,MAAM,sBAAsB,CAAC;AACnD,OAAO,EAAC,iBAAiB,EAAE,YAAY,EAAE,MAAM,EAAE,GAAG,EAAE,SAAS,EAAE,SAAS,EAAE,GAAG,EAAC,MAAM,gBAAgB,CAAC;AACvG,OAAO,EAAC,eAAe,EAAE,OAAO,EAAc,aAAa,EAAC,MAAM,MAAM,CAAC;AAEzE,OAAO,EAAC,MAAM,EAAC,MAAM,qCAAqC,CAAC;AAC3D,OAAO,EAAC,aAAa,EAAC,MAAM,6CAA6C,CAAC;AAG1E,OAAO,EAAC,aAAa,EAAC,MAAM,iCAAiC,CAAC;;;AAE9D;;;;;;GAMG;AAKH,MAAM,OAAO,2BAA2B;IAmBtC;;;;gFAI4E;IAE5E,YACE,oBAA0C;QAxB5C;;;;oFAI4E;QAE3D,QAAG,GAAG,aAAa,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;QAErD,aAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;QAE/B,mBAAc,GAAG,IAAI,eAAe,CAAgB,EAAE,CAAC,CAAC;QACxD,mBAAc,GAAG,IAAI,eAAe,CAAqB,EAAE,CAAC,CAAC;QAC7D,oBAAe,GAAG,IAAI,eAAe,CAAgB,IAAI,CAAC,CAAC;QAE3D,oBAAe,GAAG,IAAI,eAAe,CAAgB,aAAa,CAAC,UAAU,EAAE,CAAC,CAAC;QACjF,qBAAgB,GAAG,IAAI,eAAe,CAAS,IAAI,CAAC,CAAC;QAWpE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CACxB,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,EACxB,MAAM,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,EAChC,GAAG,CAAC,SAAS,CAAC,EAAE,CAAC,oBAAoB,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,CAC1D,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;IACrD,CAAC;IAED;;;;gFAI4E;IAErE,QAAQ;QACb,IAAI,CAAC,kCAAkC,EAAE,CAAC;QAC1C,IAAI,CAAC,qCAAqC,EAAE,CAAC;IAC/C,CAAC;IAEM,kBAAkB;QACvB,IAAI,CAAC,+BAA+B,EAAE,CAAC;IACzC,CAAC;IAEM,eAAe;QACpB,IAAI,CAAC,0BAA0B,EAAE,CAAC;IACpC,CAAC;IAEM,WAAW;QAChB,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;IAC3B,CAAC;IAGD;;;;gFAI4E;IAE5E,IACW,eAAe,CAAC,SAAiB;QAC1C,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IACxC,CAAC;IAED,IAAW,eAAe;QACxB,OAAO,IAAI,CAAC,gBAAgB,CAAC,QAAQ,EAAE,CAAC;IAC1C,CAAC;IAED,IAAW,cAAc;QACvB,OAAO,IAAI,CAAC,eAAe,CAAC,YAAY,EAAE,CAAC;IAC7C,CAAC;IAED,IAAW,aAAa;QACtB,OAAO,IAAI,CAAC,eAAe,CAAC,QAAQ,EAAE,CAAC;IACzC,CAAC;IAED,IACW,aAAa,CAAC,KAAyB;QAChD,IAAI,OAAsB,CAAC;QAC3B,IAAI,KAAK,EAAE;YACT,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE;gBAC7B,OAAO,GAAG,KAAK,CAAC;aACjB;iBAAM;gBACL,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,qDAAqD,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC;gBAC5F,MAAM,IAAI,KAAK,CAAC,qDAAqD,GAAG,KAAK,CAAC,CAAC;aAChF;SACF;aAAM;YACL,OAAO,GAAG,aAAa,CAAC,KAAK,EAAE,CAAC;SACjC;QACD,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACrC,CAAC;IAED,IAAW,aAAa;QACtB,OAAO,IAAI,CAAC,eAAe,CAAC,QAAQ,EAAE,CAAC;IACzC,CAAC;IAED;;;;;OAKG;IACH,IACW,aAAa,CAAC,OAAiB;QACxC,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC;IAC3D,CAAC;IAED,IAAW,aAAa;QACtB,OAAO,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,eAAe,CAAC;IAC1D,CAAC;IAED,IAAW,UAAU;QACnB,OAAO,IAAI,CAAC,cAAc,CAAC,YAAY,EAAE,CAAC;IAC5C,CAAC;IAED,IAAW,kBAAkB;QAC3B,OAAO,IAAI,CAAC,cAAc,CAAC,QAAQ,EAAE,CAAC;IACxC,CAAC;IAED,IAAW,OAAO;QAChB,OAAO,IAAI,CAAC,cAAc,CAAC,YAAY,EAAE,CAAC;IAC5C,CAAC;IAED,IAAW,cAAc;QACvB,OAAO,IAAI,CAAC,cAAc,CAAC,QAAQ,EAAE,CAAC;IACxC,CAAC;IAED;;OAEG;IACH,IAAW,sBAAsB;QAC/B,OAAO,IAAI,CAAC,OAAO,CAAC,IAAI,CACtB,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAC/C,CAAC;IACJ,CAAC;IAED;;;;gFAI4E;IAE5E;;OAEG;IACI,QAAQ,CAAC,WAAwB;QACtC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,4BAA4B,GAAG,WAAW,CAAC,IAAI,GAAG,GAAG,CAAC,CAAC;QACtE,MAAM,OAAO,GAAG,IAAI,CAAC,cAAc,CAAC,QAAQ,EAAE,CAAC;QAC/C,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,GAAG,OAAO,EAAE,WAAW,CAAC,CAAC,CAAC;IACtD,CAAC;IAEM,KAAK;QACV,IAAI,CAAC,kBAAkB;aACpB,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC;aAC9B,OAAO,CAAC,CAAC,CAAC,EAAE;YACX,CAAC,CAAC,KAAK,EAAE,CAAC;QACZ,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;;;gFAI4E;IAEpE,kCAAkC;QACxC,MAAM,SAAS,GAAG,IAAI,CAAC,aAAa,CAAC;QACrC,MAAM,aAAa,GAAG,IAAI,CAAC,aAAa,CAAC;QACzC,IAAI,aAAa,CAAC,WAAW,CAAC,OAAO,EAAE;YACrC,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE;gBACtB,aAAa,CAAC,iBAAiB,CAAC,SAAS,CAAC,eAAe,CAAC,CAAC;aAC5D;SACF;aAAM;YACL,SAAS,CAAC,cAAc,CACtB,aAAa,CAAC,WAAW,CAAC,eAAe,CAC1C,CAAC;SACH;IACH,CAAC;IAEO,+BAA+B;QACrC,MAAM,aAAa,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,CAC5C,SAAS,CAAC,MAAM,CAAC,EAAE,CAAC,aAAa,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAC7D,YAAY,CAAC,CAAC,CAAC,EACf,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,EAC/C,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,CAAC,CAC7C,CAAC;QAEF,aAAa;aACV,IAAI,CACH,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,EACxB,iBAAiB,CAAC,IAAI,CAAC,eAAe,CAAC,CACxC;aACA,SAAS,CAAC,CAAC,CAAC,YAAY,EAAE,aAAa,CAAC,EAAE,EAAE;YAC3C,aAAa,CAAC,iBAAiB,CAAC,YAAY,CAAC,CAAC;QAChD,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,qCAAqC;QAC3C,IAAI,CAAC,eAAe,CAAC,IAAI,CACvB,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,EACxB,SAAS,CAAC,OAAO,CAAC,EAAE,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAChD,CAAC,SAAS,CAAC,UAAU,CAAC,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,UAAU,CAAC,CAAC,CAAC;IACxE,CAAC;IAEO,yBAAyB,CAAC,OAAiB;QACjD,MAAM,SAAS,GAAG,IAAI,CAAC,aAAa,CAAC;QACrC,IAAI,SAAS,EAAE;YACb,SAAS,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC;YAClC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,mCAAmC,GAAC,IAAI,CAAC,eAAe,GAAC,gCAAgC,EAAE,OAAO,CAAC,CAAC;SACpH;aAAM;YACL,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,8DAA8D,EAAE,OAAO,CAAC,CAAC;SACxF;IACH,CAAC;IAEM,0BAA0B;QAC/B,MAAM,GAAG,GAAG,IAAI,CAAC,aAAa,CAAC;QAC/B,IAAI,CAAC,oBAAoB,CACvB,GAAG,CAAC,WAAW,CAAC,eAAe,CAAC,2CAA2C;SAC5E,CAAC;IACJ,CAAC;IAEO,oBAAoB,CAAC,OAAiB;QAC5C,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,sBAAsB,EAAE,OAAO,CAAC,CAAC;QAC/C,IAAI,CAAC,cAAc,CAAC,QAAQ,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE;YAC7C,MAAM,MAAM,GAAG,IAAI,CAAC,kBAAkB,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;YACvD,IAAI,MAAM,EAAE;gBACV,KAAK,CAAC,iBAAiB,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;aACvC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,kBAAkB,CAAC,KAAkB,EAAE,OAAiB;QAC9D,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE;YAC5B,IAAI,MAAM,CAAC,GAAG,KAAK,KAAK,CAAC,IAAI,EAAE;gBAC7B,OAAO,MAAM,CAAC;aACf;SACF;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAEO,gBAAgB,CAAC,MAA0B;QACjD,OAAO,MAAM;aACV,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC;aACzB,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,MAAM,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC;IACpD,CAAC;+GA3PU,2BAA2B;mGAA3B,2BAA2B;;4FAA3B,2BAA2B;kBAJvC,SAAS;mBAAC;oBACT,QAAQ,EAAE,sBAAsB;oBAChC,QAAQ,EAAE,oBAAoB;iBAC/B;2GAoEY,eAAe;sBADzB,KAAK;gBAkBK,aAAa;sBADvB,KAAK;uBAAC,oBAAoB;gBA2BhB,aAAa;sBADvB,KAAK","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';\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.forcedFilters.replaceFilters(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        searchContext.updateUserFilters(dcFilters.filtersSnapshot);\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"]}
|