@ethlete/cdk 4.25.0 → 4.25.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CHANGELOG.md CHANGED
@@ -1,5 +1,11 @@
1
1
  # @ethlete/cdk
2
2
 
3
+ ## 4.25.1
4
+
5
+ ### Patch Changes
6
+
7
+ - [`8bc0c1d`](https://github.com/ethlete-io/ethdk/commit/8bc0c1dfa133c2de51980f3ff75e1465201289f2) Thanks [@TomTomB](https://github.com/TomTomB)! - Do not throw errors in filter overlays during init phase
8
+
3
9
  ## 4.25.0
4
10
 
5
11
  ### Minor Changes
@@ -1,14 +1,14 @@
1
1
  import { Injectable, InjectionToken, computed, inject, isDevMode } from '@angular/core';
2
2
  import { toObservable, toSignal } from '@angular/core/rxjs-interop';
3
- import { cloneFormGroup, getFormGroupValue } from '@ethlete/core';
3
+ import { cloneFormGroup, controlValueSignal, getFormGroupValue } from '@ethlete/core';
4
4
  import { isQueryStateFailure, isQueryStateLoading, isQueryStateSuccess, switchQueryState, } from '@ethlete/query';
5
- import { debounceTime, map, merge, startWith, take } from 'rxjs';
6
5
  import { OverlayRef } from './overlay-ref';
7
6
  import * as i0 from "@angular/core";
8
7
  export const FILTER_OVERLAY_CONFIG = new InjectionToken('FILTER_OVERLAY_CONFIG');
9
8
  export const defaultSubmitButtonConfigFn = (config) => {
10
9
  const { queryState, totalHits, locale = 'en' } = config;
11
- if (isQueryStateLoading(queryState)) {
10
+ const isInitializing = queryState === null && totalHits === null;
11
+ if (isQueryStateLoading(queryState) || isInitializing) {
12
12
  return {
13
13
  disabled: true,
14
14
  label: locale === 'en' ? 'Loading results...' : 'Lade Ergebnisse...',
@@ -60,7 +60,8 @@ export class FilterOverlayService {
60
60
  this.config = inject(FILTER_OVERLAY_CONFIG);
61
61
  this.overlayRef = inject(OverlayRef);
62
62
  this.form = cloneFormGroup(this.config.form);
63
- this.searchPreviewQuery = toSignal(merge(this.form.valueChanges.pipe(startWith(this.form.getRawValue()), take(1)), this.form.valueChanges.pipe(debounceTime(300))).pipe(map((value) => this.config.searchPreviewQueryFn?.(value) ?? null)), { initialValue: null });
63
+ this.formValue = controlValueSignal(this.form);
64
+ this.searchPreviewQuery = computed(() => this.config.searchPreviewQueryFn?.(this.formValue()) ?? null);
64
65
  this.searchPreviewQueryState = toSignal(toObservable(this.searchPreviewQuery).pipe(switchQueryState()), {
65
66
  initialValue: null,
66
67
  });
@@ -114,4 +115,4 @@ export const provideFilterOverlayConfig = (config) => {
114
115
  },
115
116
  ];
116
117
  };
117
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filter-overlay.js","sourceRoot":"","sources":["../../../../../../../../../../libs/cdk/src/lib/components/overlay/components/overlay/utils/filter-overlay.ts"],"names":[],"mappings":"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","sourcesContent":["import { ComponentType } from '@angular/cdk/portal';\nimport { Injectable, InjectionToken, Provider, computed, inject, isDevMode } from '@angular/core';\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { FormGroup } from '@angular/forms';\nimport { cloneFormGroup, getFormGroupValue } from '@ethlete/core';\nimport {\n  AnyQuery,\n  QueryResponseOf,\n  QueryState,\n  isQueryStateFailure,\n  isQueryStateLoading,\n  isQueryStateSuccess,\n  switchQueryState,\n} from '@ethlete/query';\nimport { debounceTime, map, merge, startWith, take } from 'rxjs';\nimport { OverlayRef } from './overlay-ref';\n\nexport const FILTER_OVERLAY_CONFIG = new InjectionToken<FilterOverlayConfig>('FILTER_OVERLAY_CONFIG');\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport type FilterOverlayConfig<F extends FormGroup<any> = FormGroup<any>, Q extends AnyQuery = AnyQuery> = {\n  /**\n   * The form to use.\n   */\n  form: F;\n\n  /**\n   * The default form value. Used when the form is reset.\n   */\n  defaults?: ReturnType<F['getRawValue']>;\n\n  /**\n   * A function to create a query based on the form value.\n   */\n  searchPreviewQueryFn?: (formValue: ReturnType<F['getRawValue']>) => Q;\n\n  /**\n   * A function to extract the total hits from the response.\n   * @default `response.totalHits`\n   */\n  totalHitsExtractorFn?: (response: QueryResponseOf<Q>) => number;\n\n  /**\n   * A function to create the submit button config based on the query state and the total hits.\n   * If a german translation is needed, manually provide the `defaultSubmitButtonConfigFn` and set the locale to `de`.\n   * @default defaultSubmitButtonConfigFn()\n   */\n  submitButtonConfigFn?: (\n    config: DefaultSubmitButtonConfigFnConfig<QueryResponseOf<Q>>,\n  ) => FilterOverlaySubmitButtonConfig;\n};\n\nexport type FilterOverlaySubmitButtonConfig = {\n  label: string;\n  disabled: boolean;\n};\n\nexport type DefaultSubmitButtonConfigFnConfig<ResponseType = unknown> = {\n  queryState: QueryState<ResponseType> | null;\n  totalHits: number | null;\n  locale?: 'en' | 'de';\n};\n\nexport const defaultSubmitButtonConfigFn = (\n  config: DefaultSubmitButtonConfigFnConfig,\n): FilterOverlaySubmitButtonConfig => {\n  const { queryState, totalHits, locale = 'en' } = config;\n\n  if (isQueryStateLoading(queryState)) {\n    return {\n      disabled: true,\n      label: locale === 'en' ? 'Loading results...' : 'Lade Ergebnisse...',\n    };\n  } else if (isQueryStateFailure(queryState)) {\n    return {\n      disabled: true,\n      label: locale === 'en' ? 'An error occurred' : 'Ein Fehler ist aufgetreten',\n    };\n  } else if (totalHits !== null) {\n    if (totalHits === 0) {\n      return {\n        disabled: true,\n        label: locale === 'en' ? 'No results found' : 'Keine Ergebnisse gefunden',\n      };\n    } else if (totalHits === 1) {\n      return {\n        disabled: false,\n        label: locale === 'en' ? 'Show one result' : 'Zeige ein Ergebnis',\n      };\n    } else if (totalHits > 250) {\n      return {\n        disabled: false,\n        label: locale === 'en' ? 'Show more than 250 results' : 'Zeige mehr als 250 Ergebnisse',\n      };\n    } else {\n      return {\n        disabled: false,\n        label: locale === 'en' ? `Show ${totalHits} results` : `Zeige ${totalHits} Ergebnisse`,\n      };\n    }\n  }\n\n  if (isDevMode()) {\n    console.error(\n      'The default submit button config function resulted in an unknown state. The fallback state might be invalid and should be checked.',\n      config,\n    );\n  }\n\n  // This should never happen\n  return {\n    disabled: true,\n    label: locale === 'en' ? 'No results found' : 'Keine Ergebnisse gefunden',\n  };\n};\n\nexport type FilterOverlayResult<FormValue = unknown> =\n  | {\n      didUpdate: false;\n    }\n  | {\n      didUpdate: true;\n      formValue: FormValue;\n    };\n\n@Injectable()\nexport class FilterOverlayService<F extends FormGroup, C extends ComponentType<unknown> | unknown = unknown> {\n  config = inject<FilterOverlayConfig<F>>(FILTER_OVERLAY_CONFIG);\n  overlayRef = inject<OverlayRef<C, FilterOverlayResult>>(OverlayRef);\n\n  form = cloneFormGroup(this.config.form as F);\n\n  searchPreviewQuery = toSignal(\n    merge(\n      this.form.valueChanges.pipe(startWith(this.form.getRawValue()), take(1)),\n      this.form.valueChanges.pipe(debounceTime(300)),\n    ).pipe(map((value) => this.config.searchPreviewQueryFn?.(value) ?? null)),\n    { initialValue: null },\n  );\n\n  searchPreviewQueryState = toSignal(toObservable(this.searchPreviewQuery).pipe(switchQueryState()), {\n    initialValue: null,\n  });\n\n  searchPreviewTotalHits = computed(() => {\n    const state = this.searchPreviewQueryState();\n\n    if (isQueryStateSuccess(state)) {\n      if (this.config.totalHitsExtractorFn) {\n        return this.config.totalHitsExtractorFn(state.response);\n      }\n\n      if (!state.response || typeof state.response !== 'object' || !('totalHits' in state.response)) {\n        console.error(`The response does not contain a totalHits property.`, state.response);\n        return null;\n      }\n\n      return state.response.totalHits as number;\n    }\n\n    return null;\n  });\n\n  submitButtonConfig = computed(() => {\n    const state = this.searchPreviewQueryState();\n    const totalHits = this.searchPreviewTotalHits();\n\n    return (\n      this.config.submitButtonConfigFn?.({ queryState: state, totalHits }) ??\n      defaultSubmitButtonConfigFn({ queryState: state, totalHits })\n    );\n  });\n\n  submit() {\n    const value = getFormGroupValue(this.form);\n    this.config.form.setValue(value);\n\n    this.close({ didUpdate: true, formValue: value });\n  }\n\n  reset() {\n    if (!this.config.defaults) {\n      throw new Error(`The default form value is not defined.`);\n    }\n\n    this.form.patchValue(this.config.defaults);\n  }\n\n  close(data?: FilterOverlayResult) {\n    this.overlayRef?.close(data);\n  }\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport const provideFilterOverlayConfig = <F extends FormGroup<any> = FormGroup<any>, Q extends AnyQuery = AnyQuery>(\n  config: FilterOverlayConfig<F, Q>,\n): Provider[] => {\n  return [\n    {\n      provide: FILTER_OVERLAY_CONFIG,\n      useValue: config,\n    },\n  ];\n};\n"]}
118
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filter-overlay.js","sourceRoot":"","sources":["../../../../../../../../../../libs/cdk/src/lib/components/overlay/components/overlay/utils/filter-overlay.ts"],"names":[],"mappings":"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","sourcesContent":["import { ComponentType } from '@angular/cdk/portal';\nimport { Injectable, InjectionToken, Provider, computed, inject, isDevMode } from '@angular/core';\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { FormGroup } from '@angular/forms';\nimport { cloneFormGroup, controlValueSignal, getFormGroupValue } from '@ethlete/core';\nimport {\n  AnyQuery,\n  QueryResponseOf,\n  QueryState,\n  isQueryStateFailure,\n  isQueryStateLoading,\n  isQueryStateSuccess,\n  switchQueryState,\n} from '@ethlete/query';\nimport { OverlayRef } from './overlay-ref';\n\nexport const FILTER_OVERLAY_CONFIG = new InjectionToken<FilterOverlayConfig>('FILTER_OVERLAY_CONFIG');\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport type FilterOverlayConfig<F extends FormGroup<any> = FormGroup<any>, Q extends AnyQuery = AnyQuery> = {\n  /**\n   * The form to use.\n   */\n  form: F;\n\n  /**\n   * The default form value. Used when the form is reset.\n   */\n  defaults?: ReturnType<F['getRawValue']>;\n\n  /**\n   * A function to create a query based on the form value.\n   */\n  searchPreviewQueryFn?: (formValue: ReturnType<F['getRawValue']>) => Q;\n\n  /**\n   * A function to extract the total hits from the response.\n   * @default `response.totalHits`\n   */\n  totalHitsExtractorFn?: (response: QueryResponseOf<Q>) => number;\n\n  /**\n   * A function to create the submit button config based on the query state and the total hits.\n   * If a german translation is needed, manually provide the `defaultSubmitButtonConfigFn` and set the locale to `de`.\n   * @default defaultSubmitButtonConfigFn()\n   */\n  submitButtonConfigFn?: (\n    config: DefaultSubmitButtonConfigFnConfig<QueryResponseOf<Q>>,\n  ) => FilterOverlaySubmitButtonConfig;\n};\n\nexport type FilterOverlaySubmitButtonConfig = {\n  label: string;\n  disabled: boolean;\n};\n\nexport type DefaultSubmitButtonConfigFnConfig<ResponseType = unknown> = {\n  queryState: QueryState<ResponseType> | null;\n  totalHits: number | null;\n  locale?: 'en' | 'de';\n};\n\nexport const defaultSubmitButtonConfigFn = (\n  config: DefaultSubmitButtonConfigFnConfig,\n): FilterOverlaySubmitButtonConfig => {\n  const { queryState, totalHits, locale = 'en' } = config;\n\n  const isInitializing = queryState === null && totalHits === null;\n\n  if (isQueryStateLoading(queryState) || isInitializing) {\n    return {\n      disabled: true,\n      label: locale === 'en' ? 'Loading results...' : 'Lade Ergebnisse...',\n    };\n  } else if (isQueryStateFailure(queryState)) {\n    return {\n      disabled: true,\n      label: locale === 'en' ? 'An error occurred' : 'Ein Fehler ist aufgetreten',\n    };\n  } else if (totalHits !== null) {\n    if (totalHits === 0) {\n      return {\n        disabled: true,\n        label: locale === 'en' ? 'No results found' : 'Keine Ergebnisse gefunden',\n      };\n    } else if (totalHits === 1) {\n      return {\n        disabled: false,\n        label: locale === 'en' ? 'Show one result' : 'Zeige ein Ergebnis',\n      };\n    } else if (totalHits > 250) {\n      return {\n        disabled: false,\n        label: locale === 'en' ? 'Show more than 250 results' : 'Zeige mehr als 250 Ergebnisse',\n      };\n    } else {\n      return {\n        disabled: false,\n        label: locale === 'en' ? `Show ${totalHits} results` : `Zeige ${totalHits} Ergebnisse`,\n      };\n    }\n  }\n\n  if (isDevMode()) {\n    console.error(\n      'The default submit button config function resulted in an unknown state. The fallback state might be invalid and should be checked.',\n      config,\n    );\n  }\n\n  // This should never happen\n  return {\n    disabled: true,\n    label: locale === 'en' ? 'No results found' : 'Keine Ergebnisse gefunden',\n  };\n};\n\nexport type FilterOverlayResult<FormValue = unknown> =\n  | {\n      didUpdate: false;\n    }\n  | {\n      didUpdate: true;\n      formValue: FormValue;\n    };\n\n@Injectable()\nexport class FilterOverlayService<F extends FormGroup, C extends ComponentType<unknown> | unknown = unknown> {\n  config = inject<FilterOverlayConfig<F>>(FILTER_OVERLAY_CONFIG);\n  overlayRef = inject<OverlayRef<C, FilterOverlayResult>>(OverlayRef);\n\n  form = cloneFormGroup(this.config.form as F);\n\n  formValue = controlValueSignal(this.form);\n  searchPreviewQuery = computed(() => this.config.searchPreviewQueryFn?.(this.formValue()) ?? null);\n\n  searchPreviewQueryState = toSignal(toObservable(this.searchPreviewQuery).pipe(switchQueryState()), {\n    initialValue: null,\n  });\n\n  searchPreviewTotalHits = computed(() => {\n    const state = this.searchPreviewQueryState();\n\n    if (isQueryStateSuccess(state)) {\n      if (this.config.totalHitsExtractorFn) {\n        return this.config.totalHitsExtractorFn(state.response);\n      }\n\n      if (!state.response || typeof state.response !== 'object' || !('totalHits' in state.response)) {\n        console.error(`The response does not contain a totalHits property.`, state.response);\n        return null;\n      }\n\n      return state.response.totalHits as number;\n    }\n\n    return null;\n  });\n\n  submitButtonConfig = computed(() => {\n    const state = this.searchPreviewQueryState();\n    const totalHits = this.searchPreviewTotalHits();\n\n    return (\n      this.config.submitButtonConfigFn?.({ queryState: state, totalHits }) ??\n      defaultSubmitButtonConfigFn({ queryState: state, totalHits })\n    );\n  });\n\n  submit() {\n    const value = getFormGroupValue(this.form);\n    this.config.form.setValue(value);\n\n    this.close({ didUpdate: true, formValue: value });\n  }\n\n  reset() {\n    if (!this.config.defaults) {\n      throw new Error(`The default form value is not defined.`);\n    }\n\n    this.form.patchValue(this.config.defaults);\n  }\n\n  close(data?: FilterOverlayResult) {\n    this.overlayRef?.close(data);\n  }\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport const provideFilterOverlayConfig = <F extends FormGroup<any> = FormGroup<any>, Q extends AnyQuery = AnyQuery>(\n  config: FilterOverlayConfig<F, Q>,\n): Provider[] => {\n  return [\n    {\n      provide: FILTER_OVERLAY_CONFIG,\n      useValue: config,\n    },\n  ];\n};\n"]}
@@ -4,7 +4,7 @@ import { AsyncPipe, NgClass, NgTemplateOutlet, JsonPipe, NgComponentOutlet, DOCU
4
4
  import * as i0 from '@angular/core';
5
5
  import { Component, ViewEncapsulation, ChangeDetectionStrategy, InjectionToken, Directive, booleanAttribute, Input, ContentChild, ContentChildren, inject, ElementRef, Injector, HostBinding, input, numberAttribute, computed, signal, contentChildren, viewChild, viewChildren, isDevMode, Injectable, ChangeDetectorRef, TemplateRef, ViewContainerRef, ViewChild, forwardRef, EventEmitter, Output, ViewChildren, Optional, Inject, SkipSelf, HostListener, contentChild, effect, NgZone, NgModule, isSignal, DestroyRef, assertInInjectionContext, Attribute } from '@angular/core';
6
6
  import * as i1$1 from '@ethlete/core';
7
- import { LetDirective, createDestroy, Memo, signalHostAttributes, signalHostClasses, previousSignalValue, signalHostStyles, nextFrame, syncSignal, signalStyles, injectHostElement, ObserveVisibilityDirective, signalVisibilityChangeClasses, IS_EMAIL, MUST_MATCH, IS_ARRAY_NOT_EMPTY, AT_LEAST_ONE_REQUIRED, equal, switchQueryListChanges, signalAttributes, ResizeObserverService, createFlipAnimation, AnimatedOverlayDirective, RuntimeError, SelectionModel, ActiveSelectionModel, KeyPressManager, signalClasses, scrollToElement, isEmptyArray, isObjectArray, isPrimitiveArray, createComponentId, ClickOutsideDirective, ANIMATED_LIFECYCLE_TOKEN, AnimatedLifecycleDirective, ObserveContentDirective, clamp, DELAYABLE_TOKEN, ObserveResizeDirective, SmartBlockScrollStrategy, RouterStateService, signalElementDimensions, signalElementScrollState, createIsRenderedSignal, createCanAnimateSignal, useCursorDragScroll, signalElementIntersection, signalElementChildren, getIntersectionInfo, getElementScrollCoordinates, ObserveScrollStateDirective, ClickObserverService, RootBoundaryDirective, elementCanScroll, cloneFormGroup, getFormGroupValue, fromNextFrame, ViewportService, ROOT_BOUNDARY_TOKEN, isElementVisible, AnimatedIfDirective, FocusVisibleService, inferMimeType, ScrollObserverIgnoreTargetDirective, TypedQueryList } from '@ethlete/core';
7
+ import { LetDirective, createDestroy, Memo, signalHostAttributes, signalHostClasses, previousSignalValue, signalHostStyles, nextFrame, syncSignal, signalStyles, injectHostElement, ObserveVisibilityDirective, signalVisibilityChangeClasses, IS_EMAIL, MUST_MATCH, IS_ARRAY_NOT_EMPTY, AT_LEAST_ONE_REQUIRED, equal, switchQueryListChanges, signalAttributes, ResizeObserverService, createFlipAnimation, AnimatedOverlayDirective, RuntimeError, SelectionModel, ActiveSelectionModel, KeyPressManager, signalClasses, scrollToElement, isEmptyArray, isObjectArray, isPrimitiveArray, createComponentId, ClickOutsideDirective, ANIMATED_LIFECYCLE_TOKEN, AnimatedLifecycleDirective, ObserveContentDirective, clamp, DELAYABLE_TOKEN, ObserveResizeDirective, SmartBlockScrollStrategy, RouterStateService, signalElementDimensions, signalElementScrollState, createIsRenderedSignal, createCanAnimateSignal, useCursorDragScroll, signalElementIntersection, signalElementChildren, getIntersectionInfo, getElementScrollCoordinates, ObserveScrollStateDirective, ClickObserverService, RootBoundaryDirective, elementCanScroll, cloneFormGroup, controlValueSignal, getFormGroupValue, fromNextFrame, ViewportService, ROOT_BOUNDARY_TOKEN, isElementVisible, AnimatedIfDirective, FocusVisibleService, inferMimeType, ScrollObserverIgnoreTargetDirective, TypedQueryList } from '@ethlete/core';
8
8
  import { BehaviorSubject, startWith, map, switchMap, combineLatest, pairwise, tap, takeUntil, skip, of, merge, timer, takeWhile, filter, fromEvent, Subject, Observable, debounceTime, withLatestFrom, distinctUntilChanged, take, skipUntil, skipWhile, catchError, throwError, defer, partition, from, finalize, Subscription } from 'rxjs';
9
9
  import { trigger, state, style, transition, animate } from '@angular/animations';
10
10
  import { __decorate, __metadata } from 'tslib';
@@ -11746,7 +11746,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.2", ngImpor
11746
11746
  const FILTER_OVERLAY_CONFIG = new InjectionToken('FILTER_OVERLAY_CONFIG');
11747
11747
  const defaultSubmitButtonConfigFn = (config) => {
11748
11748
  const { queryState, totalHits, locale = 'en' } = config;
11749
- if (isQueryStateLoading(queryState)) {
11749
+ const isInitializing = queryState === null && totalHits === null;
11750
+ if (isQueryStateLoading(queryState) || isInitializing) {
11750
11751
  return {
11751
11752
  disabled: true,
11752
11753
  label: locale === 'en' ? 'Loading results...' : 'Lade Ergebnisse...',
@@ -11798,7 +11799,8 @@ class FilterOverlayService {
11798
11799
  this.config = inject(FILTER_OVERLAY_CONFIG);
11799
11800
  this.overlayRef = inject(OverlayRef);
11800
11801
  this.form = cloneFormGroup(this.config.form);
11801
- this.searchPreviewQuery = toSignal(merge(this.form.valueChanges.pipe(startWith(this.form.getRawValue()), take(1)), this.form.valueChanges.pipe(debounceTime(300))).pipe(map((value) => this.config.searchPreviewQueryFn?.(value) ?? null)), { initialValue: null });
11802
+ this.formValue = controlValueSignal(this.form);
11803
+ this.searchPreviewQuery = computed(() => this.config.searchPreviewQueryFn?.(this.formValue()) ?? null);
11802
11804
  this.searchPreviewQueryState = toSignal(toObservable(this.searchPreviewQuery).pipe(switchQueryState()), {
11803
11805
  initialValue: null,
11804
11806
  });