@elderbyte/ngx-starter 16.4.3 → 16.4.5

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.
@@ -2,7 +2,7 @@ import { Directive, Host, Input } from "@angular/core";
2
2
  import { LoggerFactory } from "@elderbyte/ts-logger";
3
3
  import { debounceTime, delay, first, map, takeUntil } from 'rxjs/operators';
4
4
  import { PropertyPathUtil } from "../../../common/utils/property-path-util";
5
- import { BehaviorSubject, concat, of, skipUntil, Subject } from 'rxjs';
5
+ import { concat, of, skipUntil, Subject } from 'rxjs';
6
6
  import { SearchInputState } from "./domain/input/search-input-state";
7
7
  import { Objects } from "../../../common/objects";
8
8
  import { CollectionUtil } from "../../../common/utils/collection-util";
@@ -22,10 +22,12 @@ export class ElderSearchInputDirective {
22
22
  constructor(searchContext, ngModel) {
23
23
  this.searchContext = searchContext;
24
24
  this.ngModel = ngModel;
25
- this._state = new BehaviorSubject(SearchInputState.unknown());
26
25
  this.destroy$ = new Subject();
27
26
  this.log = LoggerFactory.getLogger(this.constructor.name);
28
27
  this.initialValueDirective = new ElderInitialValueDirective(ngModel);
28
+ this._state$ = this.buildInputStateObservable().pipe(
29
+ // tap(state => this.log.error('EMIT SEARCH INPUT STATE [' + state.queryKey + ']: ' + state.queryValue + (state.pristine ? '(pristine)' : ''), state))
30
+ );
29
31
  }
30
32
  /***************************************************************************
31
33
  * *
@@ -39,11 +41,6 @@ export class ElderSearchInputDirective {
39
41
  ngAfterViewInit() {
40
42
  this._extractedName = this.extractName();
41
43
  this.log.debug(this._extractedName + "|" + this.queryKey + "|" + this.resolvePath);
42
- this.stateObservable()
43
- .subscribe(state => {
44
- // this.log.error('EMIT SEARCH INPUT STATE [' + state.queryKey + ']: ' + state.queryValue + (state.pristine ? '(pristine)' : ''), state)
45
- this.emitState(state);
46
- });
47
44
  }
48
45
  ngOnDestroy() {
49
46
  this.destroy$.next();
@@ -55,10 +52,7 @@ export class ElderSearchInputDirective {
55
52
  * *
56
53
  **************************************************************************/
57
54
  get state$() {
58
- return this._state.asObservable();
59
- }
60
- get stateSnapshot() {
61
- return this._state.getValue();
55
+ return this._state$;
62
56
  }
63
57
  get name() {
64
58
  if (this.queryKey) {
@@ -84,9 +78,9 @@ export class ElderSearchInputDirective {
84
78
  * Public API *
85
79
  * *
86
80
  **************************************************************************/
87
- applyInitialValue(queryString) {
81
+ setValue(queryString) {
88
82
  const value = this.convertQueryStringToRawModelValue(queryString);
89
- this.initialValueDirective.applyInitialValue(value);
83
+ this.initialValueDirective.setValue(value);
90
84
  }
91
85
  reset() {
92
86
  this.ngModel.reset();
@@ -100,10 +94,13 @@ export class ElderSearchInputDirective {
100
94
  return this.ngModel.control.valueChanges
101
95
  .pipe(takeUntil(this.destroy$));
102
96
  }
103
- stateObservable() {
97
+ buildInputStateObservable() {
104
98
  const viewStabilized$ = of('stabilized$').pipe(delay(20), first());
105
99
  const inputStateChanges$ = this.valueChanges$.pipe(takeUntil(this.destroy$), map(value => this.buildInputState(value, true)), skipUntil(viewStabilized$), debounceTime(20));
106
- const startWithAsync$ = of('start').pipe(map(start => this.buildInputState(this.ngModel.control.value, false)));
100
+ const startWithAsync$ = of('start').pipe(map(start => {
101
+ const state = this.buildInputState(this.ngModel.control.value, false);
102
+ return state;
103
+ }));
107
104
  return concat(startWithAsync$, inputStateChanges$);
108
105
  }
109
106
  buildInputState(controlValue, userEvent) {
@@ -135,9 +132,6 @@ export class ElderSearchInputDirective {
135
132
  }
136
133
  return queryValue;
137
134
  }
138
- emitState(state) {
139
- this._state.next(state);
140
- }
141
135
  checkAndConvertFaultyValuesToNull(value) {
142
136
  let fixedValue = value;
143
137
  fixedValue = this.convertEmptyArrayToNull(fixedValue);
@@ -237,4 +231,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
237
231
  type: Input,
238
232
  args: ["valueAsId"]
239
233
  }] } });
240
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-search-input.directive.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/forms/search/elder-search-input.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Directive, Host, Input, OnDestroy, OnInit } from \"@angular/core\";\nimport { LoggerFactory } from \"@elderbyte/ts-logger\";\nimport { ElderSearchContextDirective } from \"./elder-search-context.directive\";\nimport { Observable } from \"rxjs/internal/Observable\";\nimport { NgModel } from \"@angular/forms\";\nimport { SearchInput } from \"./domain/input/search-input\";\nimport {debounceTime, delay, first, map, startWith, takeUntil, takeWhile} from 'rxjs/operators';\nimport { PropertyPathUtil } from \"../../../common/utils/property-path-util\";\nimport {BehaviorSubject, concat, of, skipUntil, Subject} from 'rxjs';\nimport { SearchInputState } from \"./domain/input/search-input-state\";\nimport { Objects } from \"../../../common/objects\";\nimport { CollectionUtil } from \"../../../common/utils/collection-util\";\nimport { LocalPagedDataSource } from \"../../../common/data/datasource/local/local-paged-data-source\";\nimport { BooleanInput } from \"@angular/cdk/coercion\";\nimport { ElderInitialValueDirective } from \"../directives/elder-initial-value.directive\";\nimport {SearchContext} from './domain/context/search-context';\n\n/**\n * Search name adapter for input controls.\n */\n@Directive({\n  selector: \"[elderSearchInput]\",\n  exportAs: \"elderSearchInput\"\n})\nexport class ElderSearchInputDirective implements SearchInput, OnInit, AfterViewInit, OnDestroy {\n\n  private readonly initialValueDirective: ElderInitialValueDirective;\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * (Optional) Usually the control name is used, this allows a custom query key\n   */\n  @Input(\"elderSearchInputKey\")\n  public queryKey: string;\n\n  /**\n   * (Optional) Function which transforms the value object to a query param value\n   */\n  @Input(\"elderSearchInputTransform\")\n  public valueTransform: ((value: any) => string | string[] | null);\n\n  /**\n   * (Optional, Default) Path on the value object to use as query param value\n   *  value = \"type.id\"\n   */\n  @Input(\"elderSearchInput\")\n  public resolvePath: string;\n\n  @Input(\"dataSource\")\n  public dataSource: LocalPagedDataSource<any>;\n\n  /**\n   * (Optional) Use this value for the query if none is provided.\n   */\n  @Input(\"elderSearchInputFallback\")\n  public fallbackValue: string | string[];\n\n  @Input(\"valueAsId\")\n  public valueAsId: BooleanInput;\n\n  private readonly _state = new BehaviorSubject<SearchInputState>(SearchInputState.unknown());\n  private _extractedName: string | null;\n  private readonly destroy$ = new Subject<void>();\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private searchContext: ElderSearchContextDirective,\n    @Host() private ngModel: NgModel\n  ) {\n    this.initialValueDirective = new ElderInitialValueDirective(ngModel);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit() {\n    this._extractedName = this.extractName();\n    this.searchContext.register(this);\n  }\n\n  public ngAfterViewInit(): void {\n    this._extractedName = this.extractName();\n    this.log.debug(this._extractedName + \"|\" + this.queryKey + \"|\" + this.resolvePath);\n\n    this.stateObservable()\n      .subscribe(state => {\n      // this.log.error('EMIT SEARCH INPUT STATE [' + state.queryKey + ']: ' + state.queryValue + (state.pristine ? '(pristine)' : ''), state)\n      this.emitState(state);\n    });\n  }\n\n  public ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public get state$(): Observable<SearchInputState> {\n    return this._state.asObservable();\n  }\n\n  public get stateSnapshot(): SearchInputState {\n    return this._state.getValue();\n  }\n\n  public get name(): string {\n    if (this.queryKey) {\n      return this.queryKey;\n    }\n    if (this._extractedName) {\n      return this._extractedName;\n    }\n\n    throw new Error(\"Could not determine the search name key name.\" +\n      \" Either specify the name property or explicitly set [elderSearchInputKey].\");\n  }\n\n  public get value(): any {\n    return this.ngModel.value;\n  }\n\n  public get readonly(): boolean {\n    return this.ngModel.isDisabled;\n  }\n\n  private get hasFallback(): boolean {\n    return SearchInputState.isValueDefined(this.fallbackValue);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public applyInitialValue(queryString: string | string[] | null): void {\n    const value = this.convertQueryStringToRawModelValue(queryString);\n    this.initialValueDirective.applyInitialValue(value);\n  }\n\n  public reset(): void {\n    this.ngModel.reset();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private get valueChanges$() {\n    return this.ngModel.control.valueChanges\n      .pipe(\n        takeUntil(this.destroy$)\n      );\n  }\n\n  private stateObservable(): Observable<SearchInputState> {\n\n    const viewStabilized$ = of('stabilized$').pipe(\n      delay(20),\n      first()\n    );\n\n    const inputStateChanges$ = this.valueChanges$.pipe(\n      takeUntil(this.destroy$),\n      map(value => this.buildInputState(value, true)),\n      skipUntil(viewStabilized$),\n      debounceTime(20),\n    );\n\n    const startWithAsync$ = of('start').pipe(\n      map(start => this.buildInputState(this.ngModel.control.value, false))\n    );\n\n    return concat(startWithAsync$, inputStateChanges$);\n  }\n\n  private buildInputState(controlValue: any, userEvent: boolean): SearchInputState {\n    const queryValue = this.convertRawModelValueToQueryString(controlValue);\n    const pristine = !this.isAttributeValuePresent(controlValue);\n    return new SearchInputState(\n      this.name,\n      queryValue,\n      this.queryKey || this.name,\n      pristine,\n      userEvent\n    );\n  }\n\n  private convertQueryStringToRawModelValue(queryString: string | string[] | null) {\n    if (Objects.nonNull(queryString)) {\n      return queryString;\n    } else {\n      return null;\n    }\n  }\n\n  private convertRawModelValueToQueryString(model: any): string | string[] | null {\n\n    let queryValue: string | string[] | null = null;\n\n    if (this.valueAsId != undefined) {\n      queryValue = this.checkAndConvertFaultyValuesToNull(model);\n    } else if (this.isAttributeValuePresent(model)) {\n      // Attribute value is present\n      queryValue = this.resolveValue(model);\n    } else {\n      if (this.hasFallback) {\n        queryValue = this.fallbackValue;\n      }\n    }\n\n    return queryValue;\n  }\n\n  public emitState(state: SearchInputState): void {\n    this._state.next(state);\n  }\n\n  private checkAndConvertFaultyValuesToNull(value: any) {\n    let fixedValue = value;\n    fixedValue = this.convertEmptyArrayToNull(fixedValue);\n    fixedValue = this.convertEmptyStringToNull(fixedValue);\n\n    return fixedValue;\n  }\n\n  private convertEmptyArrayToNull(value: any) {\n    if (value instanceof Array && value.length === 0) {\n      return null;\n    }\n    return value;\n  }\n\n  private convertEmptyStringToNull(value: any) {\n    if (typeof value === \"string\" && value.length === 0) {\n      return null;\n    }\n    return value;\n  }\n\n  private isAttributeValuePresent(value: any): boolean {\n    return SearchInputState.isValueDefined(value)\n      && (value + \"\").length !== 0;\n  }\n\n\n  private resolveValue(value: any): string | string[] | null {\n    if (Objects.nonNull(value)) {\n      if (value instanceof Array) {\n\n        const resolvedValues = value\n          .map(v => this.resolveSingleValue(v))\n          .filter(v => Objects.nonNull(v));\n\n        return this.concatIfNecessary(resolvedValues);\n\n      } else {\n        return this.resolveSingleValue(value);\n      }\n    } else {\n      return null;\n    }\n  }\n\n  private resolveSingleValue(value: any): string | string[] | null {\n    value = PropertyPathUtil.resolveValue(value, this.resolvePath);\n    value = this.valueTransform ? this.valueTransform(value) : value;\n\n\n    if (Objects.nonNull(value)) {\n      if (value instanceof Array) {\n        return value\n          .filter(i => Objects.nonNull(i))\n          .map(i => i.toString());\n      } else {\n        return value.toString();\n      }\n    } else {\n      return null;\n    }\n  }\n\n  private extractName(): string | null {\n    return this.ngModel.name;\n  }\n\n  private concatIfNecessary(values: (string | string[])[]): any[] {\n    const arrays = values.map(v => {\n      if (v instanceof Array) {\n        return v;\n      } else {\n        return [v];\n      }\n    });\n    return CollectionUtil.flatten(arrays);\n  }\n\n}\n"]}
234
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-search-input.directive.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/forms/search/elder-search-input.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Directive, Host, Input, OnDestroy, OnInit } from \"@angular/core\";\nimport { LoggerFactory } from \"@elderbyte/ts-logger\";\nimport { ElderSearchContextDirective } from \"./elder-search-context.directive\";\nimport { Observable } from \"rxjs/internal/Observable\";\nimport { NgModel } from \"@angular/forms\";\nimport { SearchInput } from \"./domain/input/search-input\";\nimport {debounceTime, delay, first, map, startWith, takeUntil, takeWhile, tap} from 'rxjs/operators';\nimport { PropertyPathUtil } from \"../../../common/utils/property-path-util\";\nimport {BehaviorSubject, concat, of, skipUntil, Subject} from 'rxjs';\nimport { SearchInputState } from \"./domain/input/search-input-state\";\nimport { Objects } from \"../../../common/objects\";\nimport { CollectionUtil } from \"../../../common/utils/collection-util\";\nimport { LocalPagedDataSource } from \"../../../common/data/datasource/local/local-paged-data-source\";\nimport { BooleanInput } from \"@angular/cdk/coercion\";\nimport { ElderInitialValueDirective } from \"../directives/elder-initial-value.directive\";\nimport {SearchContext} from './domain/context/search-context';\n\n/**\n * Search name adapter for input controls.\n */\n@Directive({\n  selector: \"[elderSearchInput]\",\n  exportAs: \"elderSearchInput\"\n})\nexport class ElderSearchInputDirective implements SearchInput, OnInit, AfterViewInit, OnDestroy {\n\n  private readonly initialValueDirective: ElderInitialValueDirective;\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * (Optional) Usually the control name is used, this allows a custom query key\n   */\n  @Input(\"elderSearchInputKey\")\n  public queryKey: string;\n\n  /**\n   * (Optional) Function which transforms the value object to a query param value\n   */\n  @Input(\"elderSearchInputTransform\")\n  public valueTransform: ((value: any) => string | string[] | null);\n\n  /**\n   * (Optional, Default) Path on the value object to use as query param value\n   *  value = \"type.id\"\n   */\n  @Input(\"elderSearchInput\")\n  public resolvePath: string;\n\n  @Input(\"dataSource\")\n  public dataSource: LocalPagedDataSource<any>;\n\n  /**\n   * (Optional) Use this value for the query if none is provided.\n   */\n  @Input(\"elderSearchInputFallback\")\n  public fallbackValue: string | string[];\n\n  @Input(\"valueAsId\")\n  public valueAsId: BooleanInput;\n\n  private readonly _state$: Observable<SearchInputState>;\n  private _extractedName: string | null;\n  private readonly destroy$ = new Subject<void>();\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private searchContext: ElderSearchContextDirective,\n    @Host() private ngModel: NgModel\n  ) {\n    this.initialValueDirective = new ElderInitialValueDirective(ngModel);\n    this._state$ = this.buildInputStateObservable().pipe(\n      // tap(state => this.log.error('EMIT SEARCH INPUT STATE [' + state.queryKey + ']: ' + state.queryValue + (state.pristine ? '(pristine)' : ''), state))\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit() {\n    this._extractedName = this.extractName();\n    this.searchContext.register(this);\n  }\n\n  public ngAfterViewInit(): void {\n    this._extractedName = this.extractName();\n    this.log.debug(this._extractedName + \"|\" + this.queryKey + \"|\" + this.resolvePath);\n  }\n\n  public ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public get state$(): Observable<SearchInputState> {\n    return this._state$;\n  }\n\n  public get name(): string {\n    if (this.queryKey) {\n      return this.queryKey;\n    }\n    if (this._extractedName) {\n      return this._extractedName;\n    }\n\n    throw new Error(\"Could not determine the search name key name.\" +\n      \" Either specify the name property or explicitly set [elderSearchInputKey].\");\n  }\n\n  public get value(): any {\n    return this.ngModel.value;\n  }\n\n  public get readonly(): boolean {\n    return this.ngModel.isDisabled;\n  }\n\n  private get hasFallback(): boolean {\n    return SearchInputState.isValueDefined(this.fallbackValue);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public setValue(queryString: string | string[] | null): void {\n    const value = this.convertQueryStringToRawModelValue(queryString);\n    this.initialValueDirective.setValue(value);\n  }\n\n  public reset(): void {\n    this.ngModel.reset();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private get valueChanges$() {\n    return this.ngModel.control.valueChanges\n      .pipe(\n        takeUntil(this.destroy$)\n      );\n  }\n\n  private buildInputStateObservable(): Observable<SearchInputState> {\n\n    const viewStabilized$ = of('stabilized$').pipe(\n      delay(20),\n      first()\n    );\n\n    const inputStateChanges$ = this.valueChanges$.pipe(\n      takeUntil(this.destroy$),\n      map(value => this.buildInputState(value, true)),\n      skipUntil(viewStabilized$),\n      debounceTime(20),\n    );\n\n    const startWithAsync$ = of('start').pipe(\n      map(start => {\n        const state = this.buildInputState(this.ngModel.control.value, false);\n        return state;\n      })\n    );\n\n    return concat(startWithAsync$, inputStateChanges$);\n  }\n\n  private buildInputState(controlValue: any, userEvent: boolean): SearchInputState {\n    const queryValue = this.convertRawModelValueToQueryString(controlValue);\n    const pristine = !this.isAttributeValuePresent(controlValue);\n    return new SearchInputState(\n      this.name,\n      queryValue,\n      this.queryKey || this.name,\n      pristine,\n      userEvent\n    );\n  }\n\n  private convertQueryStringToRawModelValue(queryString: string | string[] | null) {\n    if (Objects.nonNull(queryString)) {\n      return queryString;\n    } else {\n      return null;\n    }\n  }\n\n  private convertRawModelValueToQueryString(model: any): string | string[] | null {\n\n    let queryValue: string | string[] | null = null;\n\n    if (this.valueAsId != undefined) {\n      queryValue = this.checkAndConvertFaultyValuesToNull(model);\n    } else if (this.isAttributeValuePresent(model)) {\n      // Attribute value is present\n      queryValue = this.resolveValue(model);\n    } else {\n      if (this.hasFallback) {\n        queryValue = this.fallbackValue;\n      }\n    }\n\n    return queryValue;\n  }\n\n  private checkAndConvertFaultyValuesToNull(value: any) {\n    let fixedValue = value;\n    fixedValue = this.convertEmptyArrayToNull(fixedValue);\n    fixedValue = this.convertEmptyStringToNull(fixedValue);\n\n    return fixedValue;\n  }\n\n  private convertEmptyArrayToNull(value: any) {\n    if (value instanceof Array && value.length === 0) {\n      return null;\n    }\n    return value;\n  }\n\n  private convertEmptyStringToNull(value: any) {\n    if (typeof value === \"string\" && value.length === 0) {\n      return null;\n    }\n    return value;\n  }\n\n  private isAttributeValuePresent(value: any): boolean {\n    return SearchInputState.isValueDefined(value)\n      && (value + \"\").length !== 0;\n  }\n\n\n  private resolveValue(value: any): string | string[] | null {\n    if (Objects.nonNull(value)) {\n      if (value instanceof Array) {\n\n        const resolvedValues = value\n          .map(v => this.resolveSingleValue(v))\n          .filter(v => Objects.nonNull(v));\n\n        return this.concatIfNecessary(resolvedValues);\n\n      } else {\n        return this.resolveSingleValue(value);\n      }\n    } else {\n      return null;\n    }\n  }\n\n  private resolveSingleValue(value: any): string | string[] | null {\n    value = PropertyPathUtil.resolveValue(value, this.resolvePath);\n    value = this.valueTransform ? this.valueTransform(value) : value;\n\n\n    if (Objects.nonNull(value)) {\n      if (value instanceof Array) {\n        return value\n          .filter(i => Objects.nonNull(i))\n          .map(i => i.toString());\n      } else {\n        return value.toString();\n      }\n    } else {\n      return null;\n    }\n  }\n\n  private extractName(): string | null {\n    return this.ngModel.name;\n  }\n\n  private concatIfNecessary(values: (string | string[])[]): any[] {\n    const arrays = values.map(v => {\n      if (v instanceof Array) {\n        return v;\n      } else {\n        return [v];\n      }\n    });\n    return CollectionUtil.flatten(arrays);\n  }\n\n}\n"]}
@@ -5,7 +5,7 @@ import { Pipe, Injectable, LOCALE_ID, Inject, NgModule, Optional, SkipSelf, Dire
5
5
  import * as i1 from '@angular/platform-browser';
6
6
  import { Duration, Period, TemporalQueries, LocalTime, Instant, LocalDate, nativeJs, ZoneId, DateTimeFormatter, convert, ZonedDateTime, Temporal } from '@js-joda/core';
7
7
  import { LoggerFactory } from '@elderbyte/ts-logger';
8
- import { timer, defer, ReplaySubject, concat, finalize, exhaustMap, BehaviorSubject, lastValueFrom, Subject, switchMap, of, combineLatest, EMPTY, merge, forkJoin, throwError, mergeWith, Observable, zip, mergeMap as mergeMap$1, fromEvent, combineLatestWith as combineLatestWith$1, skipUntil, NEVER } from 'rxjs';
8
+ import { timer, defer, ReplaySubject, concat, finalize, exhaustMap, BehaviorSubject, Subject, switchMap, of, combineLatest, EMPTY, merge, forkJoin, throwError, mergeWith, Observable, zip, mergeMap as mergeMap$1, fromEvent, combineLatestWith as combineLatestWith$1, skipUntil, NEVER } from 'rxjs';
9
9
  import { tap, takeUntil, takeWhile, map, filter, distinctUntilChanged, debounceTime, catchError, first, take, switchMap as switchMap$1, mergeMap, expand, reduce, startWith, skip, delay, share, combineLatestWith, timeout, skipWhile } from 'rxjs/operators';
10
10
  import * as i1$2 from '@angular/common/http';
11
11
  import { HttpParams, HttpEventType, HttpRequest, HttpClient, HttpErrorResponse, HTTP_INTERCEPTORS } from '@angular/common/http';
@@ -2704,10 +2704,9 @@ class FilterContext {
2704
2704
  }
2705
2705
  return Array.from(filterMap.values());
2706
2706
  }
2707
- static async of(filters) {
2707
+ static of(filters) {
2708
2708
  const filterContext = new FilterContext();
2709
- filterContext.updateFilters(filters);
2710
- await lastValueFrom(filterContext.filters);
2709
+ filterContext.replaceFilters(filters);
2711
2710
  return filterContext;
2712
2711
  }
2713
2712
  /***************************************************************************
@@ -25935,6 +25934,12 @@ class SearchContext {
25935
25934
  this.userFilterContext = new FilterContext();
25936
25935
  this.forcedFilterContext = new FilterContext();
25937
25936
  this.allFilters$ = new BehaviorSubject(undefined);
25937
+ /***************************************************************************
25938
+ * *
25939
+ * Public API *
25940
+ * *
25941
+ **************************************************************************/
25942
+ this.acceptDefaultFilters = true;
25938
25943
  this.id = id;
25939
25944
  this.userFilterContext.filters.pipe(combineLatestWith(this.forcedFilterContext.filters), skip(1), map(([userFilters, forcedFilters]) => FilterUtil.mergeCombine(userFilters, forcedFilters)), tap(x => this.log.debug('allFilters changed: ', x))).subscribe(merged => this.allFilters$.next(merged));
25940
25945
  }
@@ -25947,24 +25952,36 @@ class SearchContext {
25947
25952
  return this.allFilters$.pipe(filter(filters => !!filters));
25948
25953
  }
25949
25954
  get userFilters() {
25950
- return this.userFilterContext;
25955
+ return this.userFilterContext.filtersSnapshot;
25956
+ }
25957
+ get userFilters$() {
25958
+ return this.userFilterContext.filters;
25951
25959
  }
25952
25960
  get forcedFilters() {
25953
- return this.forcedFilterContext;
25961
+ return this.forcedFilterContext.filtersSnapshot;
25962
+ }
25963
+ updateDefaultFilters(filters) {
25964
+ // this.log.error('updateDefaultFilters, acceptDefaultFilters: ' + this.acceptDefaultFilters, filters);
25965
+ if (this.acceptDefaultFilters) {
25966
+ this.userFilterContext.updateFilters(FilterUtil.strip(filters, this.forcedFilters));
25967
+ }
25968
+ return this.acceptDefaultFilters;
25954
25969
  }
25955
- /***************************************************************************
25956
- * *
25957
- * Public API *
25958
- * *
25959
- **************************************************************************/
25960
25970
  updateUserFilters(filters) {
25961
- this.userFilterContext.updateFilters(filters);
25971
+ if (filters.length > 0) {
25972
+ // this.log.error('updateUserFilters', filters);
25973
+ this.acceptDefaultFilters = false;
25974
+ this.userFilterContext.updateFilters(filters);
25975
+ }
25962
25976
  }
25963
25977
  replaceUserFilters(filters) {
25978
+ // this.log.error('replaceUserFilters', filters);
25979
+ this.acceptDefaultFilters = false;
25964
25980
  this.userFilterContext.replaceFilters(filters);
25965
25981
  }
25966
25982
  replaceForcedFilters(filters) {
25967
25983
  this.forcedFilterContext.replaceFilters(filters);
25984
+ // Maybe clean up existing user filters here
25968
25985
  }
25969
25986
  }
25970
25987
 
@@ -26027,6 +26044,7 @@ class ElderSearchContextDirective {
26027
26044
  **************************************************************************/
26028
26045
  this.log = LoggerFactory.getLogger(this.constructor.name);
26029
26046
  this.destroy$ = new Subject();
26047
+ this.filtersToApplyAtRegistration = new Map;
26030
26048
  this._searchInputs$ = new BehaviorSubject([]);
26031
26049
  this._searchStates$ = new BehaviorSubject([]);
26032
26050
  this._filterContext$ = new BehaviorSubject(null);
@@ -26043,7 +26061,7 @@ class ElderSearchContextDirective {
26043
26061
  this.applyDefaultFiltersToSearchContext();
26044
26062
  this.syncSearchContextToBoundFilterContext();
26045
26063
  }
26046
- ngAfterContentInit() {
26064
+ ngAfterViewInit() {
26047
26065
  const searchContext = this.searchContext;
26048
26066
  this.applySearchContextToInputs(searchContext);
26049
26067
  this.syncSearchInputsToSearchContext();
@@ -26098,7 +26116,7 @@ class ElderSearchContextDirective {
26098
26116
  this.searchContext.replaceForcedFilters(filters);
26099
26117
  }
26100
26118
  get forcedFilters() {
26101
- return this.searchContext.forcedFilters.filtersSnapshot;
26119
+ return this.searchContext.forcedFilters;
26102
26120
  }
26103
26121
  get attributes() {
26104
26122
  return this._searchInputs$.asObservable();
@@ -26127,8 +26145,12 @@ class ElderSearchContextDirective {
26127
26145
  * Register a new search name in this container
26128
26146
  */
26129
26147
  register(searchInput) {
26130
- this.log.debug('Registering search input [' + searchInput.name + ']');
26131
26148
  const current = this._searchInputs$.getValue();
26149
+ const toApply = this.filtersToApplyAtRegistration.get(searchInput.name);
26150
+ if (toApply) {
26151
+ searchInput.setValue(toApply.value);
26152
+ }
26153
+ this.log.debug('Registering search input [' + searchInput.name + ']. Late registration value:' + toApply?.value);
26132
26154
  this._searchInputs$.next([...current, searchInput]);
26133
26155
  }
26134
26156
  reset() {
@@ -26146,18 +26168,21 @@ class ElderSearchContextDirective {
26146
26168
  applyDefaultFiltersToSearchContext() {
26147
26169
  const dcFilters = this.filterContext;
26148
26170
  const searchContext = this.searchContext;
26149
- if (searchContext.userFilters.isEmpty) {
26150
- if (!dcFilters.isEmpty) {
26151
- // Consider all filters in DC as defaults, expect for foced filters.
26152
- searchContext.updateUserFilters(FilterUtil.strip(dcFilters.filtersSnapshot, this.forcedFilters));
26153
- }
26171
+ if (searchContext.userFilters.length === 0) {
26172
+ this.applyAsDefaultsUntilUserAction(searchContext, dcFilters);
26154
26173
  }
26155
26174
  else {
26156
- dcFilters.replaceFilters(searchContext.userFilters.filtersSnapshot);
26175
+ dcFilters.replaceFilters(searchContext.userFilters);
26157
26176
  }
26158
26177
  }
26178
+ applyAsDefaultsUntilUserAction(searchContext, dcFilters) {
26179
+ dcFilters.filters.pipe(map(filters => searchContext.updateDefaultFilters(filters)), takeWhile(acceptDefaultFilters => acceptDefaultFilters)).subscribe();
26180
+ }
26159
26181
  syncSearchInputsToSearchContext() {
26160
- const inputFilters$ = this._searchInputs$.pipe(switchMap$1(inputs => combineLatest(inputs.map(i => i.state$))), debounceTime(5), tap(states => this._searchStates$.next(states)), map(states => this.convertToFilters(states)));
26182
+ this._searchInputs$.pipe(takeUntil(this.destroy$), switchMap$1(inputs => combineLatest(inputs.map(i => i.state$))), debounceTime(5)).subscribe(states => {
26183
+ this._searchStates$.next(states);
26184
+ });
26185
+ const inputFilters$ = this._searchStates$.pipe(map(states => this.convertToFilters(states)), filter(filters => filters.length > 0));
26161
26186
  inputFilters$
26162
26187
  .pipe(takeUntil(this.destroy$), combineLatestWith(this._searchContext$))
26163
26188
  .subscribe(([inputFilters, searchContext]) => {
@@ -26178,7 +26203,7 @@ class ElderSearchContextDirective {
26178
26203
  }
26179
26204
  }
26180
26205
  applySearchContextToInputs(searchContext) {
26181
- this.applyFiltersToInputs(searchContext.userFilters.filtersSnapshot);
26206
+ this.applyFiltersToInputs(searchContext.userFilters);
26182
26207
  }
26183
26208
  applyFiltersToInputs(filters) {
26184
26209
  if (filters.length > 0) {
@@ -26190,7 +26215,10 @@ class ElderSearchContextDirective {
26190
26215
  filters.forEach(f => {
26191
26216
  const input = inputsByName.get(f.key);
26192
26217
  if (input) {
26193
- input.applyInitialValue(f.value);
26218
+ input.setValue(f.value);
26219
+ }
26220
+ else {
26221
+ this.filtersToApplyAtRegistration.set(f.key, f);
26194
26222
  }
26195
26223
  });
26196
26224
  }
@@ -26342,28 +26370,6 @@ class ElderInitialValueDirective {
26342
26370
  this.initialDone$.next();
26343
26371
  this.initialDone$.complete();
26344
26372
  });
26345
- // this.ngModelValue = this.ngModel.model;
26346
- // ngModel.control.valueChanges
26347
- // .pipe(
26348
- // takeUntil(this.initialDone$),
26349
- // takeUntil(this.unsubscribe$),
26350
- // combineLatestWith(this.newValue$),
26351
- // debounceTime(500)
26352
- // ).subscribe({
26353
- // next: ([ngModelValue, newValue]) => {
26354
- // if (this.initialValue === ngModelValue) {
26355
- // this.initialDone$.next();
26356
- // this.initialDone$.complete();
26357
- // ngModel.control.setValue(newValue ? newValue : ngModelValue); // Why set ngModelValue again??
26358
- // } else {
26359
- // this.log.warn('Initialisation not yet done!', {
26360
- // initialValue: this.initialValue,
26361
- // ngModelValue: ngModelValue,
26362
- // newValue: newValue
26363
- // });
26364
- // }
26365
- // }
26366
- // });
26367
26373
  }
26368
26374
  /***************************************************************************
26369
26375
  * *
@@ -26379,7 +26385,7 @@ class ElderInitialValueDirective {
26379
26385
  return this.ngModel.control.valueChanges
26380
26386
  .pipe(takeUntil(this.destroy$));
26381
26387
  }
26382
- applyInitialValue(value) {
26388
+ setValue(value) {
26383
26389
  if (value) {
26384
26390
  this.ngModel.control.setValue(value);
26385
26391
  }
@@ -26414,10 +26420,12 @@ class ElderSearchInputDirective {
26414
26420
  constructor(searchContext, ngModel) {
26415
26421
  this.searchContext = searchContext;
26416
26422
  this.ngModel = ngModel;
26417
- this._state = new BehaviorSubject(SearchInputState.unknown());
26418
26423
  this.destroy$ = new Subject();
26419
26424
  this.log = LoggerFactory.getLogger(this.constructor.name);
26420
26425
  this.initialValueDirective = new ElderInitialValueDirective(ngModel);
26426
+ this._state$ = this.buildInputStateObservable().pipe(
26427
+ // tap(state => this.log.error('EMIT SEARCH INPUT STATE [' + state.queryKey + ']: ' + state.queryValue + (state.pristine ? '(pristine)' : ''), state))
26428
+ );
26421
26429
  }
26422
26430
  /***************************************************************************
26423
26431
  * *
@@ -26431,11 +26439,6 @@ class ElderSearchInputDirective {
26431
26439
  ngAfterViewInit() {
26432
26440
  this._extractedName = this.extractName();
26433
26441
  this.log.debug(this._extractedName + "|" + this.queryKey + "|" + this.resolvePath);
26434
- this.stateObservable()
26435
- .subscribe(state => {
26436
- // this.log.error('EMIT SEARCH INPUT STATE [' + state.queryKey + ']: ' + state.queryValue + (state.pristine ? '(pristine)' : ''), state)
26437
- this.emitState(state);
26438
- });
26439
26442
  }
26440
26443
  ngOnDestroy() {
26441
26444
  this.destroy$.next();
@@ -26447,10 +26450,7 @@ class ElderSearchInputDirective {
26447
26450
  * *
26448
26451
  **************************************************************************/
26449
26452
  get state$() {
26450
- return this._state.asObservable();
26451
- }
26452
- get stateSnapshot() {
26453
- return this._state.getValue();
26453
+ return this._state$;
26454
26454
  }
26455
26455
  get name() {
26456
26456
  if (this.queryKey) {
@@ -26476,9 +26476,9 @@ class ElderSearchInputDirective {
26476
26476
  * Public API *
26477
26477
  * *
26478
26478
  **************************************************************************/
26479
- applyInitialValue(queryString) {
26479
+ setValue(queryString) {
26480
26480
  const value = this.convertQueryStringToRawModelValue(queryString);
26481
- this.initialValueDirective.applyInitialValue(value);
26481
+ this.initialValueDirective.setValue(value);
26482
26482
  }
26483
26483
  reset() {
26484
26484
  this.ngModel.reset();
@@ -26492,10 +26492,13 @@ class ElderSearchInputDirective {
26492
26492
  return this.ngModel.control.valueChanges
26493
26493
  .pipe(takeUntil(this.destroy$));
26494
26494
  }
26495
- stateObservable() {
26495
+ buildInputStateObservable() {
26496
26496
  const viewStabilized$ = of('stabilized$').pipe(delay(20), first());
26497
26497
  const inputStateChanges$ = this.valueChanges$.pipe(takeUntil(this.destroy$), map(value => this.buildInputState(value, true)), skipUntil(viewStabilized$), debounceTime(20));
26498
- const startWithAsync$ = of('start').pipe(map(start => this.buildInputState(this.ngModel.control.value, false)));
26498
+ const startWithAsync$ = of('start').pipe(map(start => {
26499
+ const state = this.buildInputState(this.ngModel.control.value, false);
26500
+ return state;
26501
+ }));
26499
26502
  return concat(startWithAsync$, inputStateChanges$);
26500
26503
  }
26501
26504
  buildInputState(controlValue, userEvent) {
@@ -26527,9 +26530,6 @@ class ElderSearchInputDirective {
26527
26530
  }
26528
26531
  return queryValue;
26529
26532
  }
26530
- emitState(state) {
26531
- this._state.next(state);
26532
- }
26533
26533
  checkAndConvertFaultyValuesToNull(value) {
26534
26534
  let fixedValue = value;
26535
26535
  fixedValue = this.convertEmptyArrayToNull(fixedValue);
@@ -27360,7 +27360,7 @@ class ElderSearchUrlDirective {
27360
27360
  * *
27361
27361
  **************************************************************************/
27362
27362
  ngAfterViewInit() {
27363
- this.searchContext$.pipe(takeUntil(this.destroy$), switchMap(searchContext => searchContext.userFilters.filters.pipe(map(filters => new ContextWithUserFilters(searchContext, filters))))).subscribe(ctxAndFilters => this.searchUrlService.updateQueryParams(ctxAndFilters.context.id, ctxAndFilters.userFilters));
27363
+ this.searchContext$.pipe(takeUntil(this.destroy$), switchMap(searchContext => searchContext.userFilters$.pipe(map(filters => new ContextWithUserFilters(searchContext, filters))))).subscribe(ctxAndFilters => this.searchUrlService.updateQueryParams(ctxAndFilters.context.id, ctxAndFilters.userFilters));
27364
27364
  }
27365
27365
  ngOnDestroy() {
27366
27366
  this.destroy$.next();