@ethlete/cdk 4.35.0 → 4.36.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/CHANGELOG.md CHANGED
@@ -1,5 +1,11 @@
1
1
  # @ethlete/cdk
2
2
 
3
+ ## 4.36.0
4
+
5
+ ### Minor Changes
6
+
7
+ - [`51621d2`](https://github.com/ethlete-io/ethdk/commit/51621d2e35f20eb6cf08de42d2a28fb699a5e144) Thanks [@TomTomB](https://github.com/TomTomB)! - Add `updateQueryParam` method to the function returned by `createOverlayHandlerWithQueryParamLifecycle`
8
+
3
9
  ## 4.35.0
4
10
 
5
11
  ### Minor Changes
@@ -1,4 +1,4 @@
1
- import { DestroyRef, effect, inject, untracked, ViewContainerRef } from '@angular/core';
1
+ import { DestroyRef, effect, inject, Injector, untracked, ViewContainerRef, } from '@angular/core';
2
2
  import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
3
3
  import { Router } from '@angular/router';
4
4
  import { injectQueryParam } from '@ethlete/core';
@@ -49,7 +49,7 @@ export const createOverlayHandler = (rootConfig) => {
49
49
  export const OVERLAY_QUERY_PARAM_INPUT_NAME = 'overlayQueryParam';
50
50
  /**
51
51
  * This handler will automatically open the overlay when the query param is present.
52
- * The overlay can contain a required input with the name `overlayQueryParam` to receive the query param value.
52
+ * The overlay can contain a input or model with the name `overlayQueryParam` to receive the query param value.
53
53
  *
54
54
  * If you need to transfer more information (eg. a second query param), you need to combine them into a single query param string.
55
55
  * You can then split the string inside the overlay component using computed signals.
@@ -57,29 +57,42 @@ export const OVERLAY_QUERY_PARAM_INPUT_NAME = 'overlayQueryParam';
57
57
  * To open the overlay either use the `OverlayHandlerLinkDirective` or call the `.open` method of the returned handler.
58
58
  */
59
59
  export const createOverlayHandlerWithQueryParamLifecycle = (config) => {
60
- const handler = createOverlayHandler(config);
60
+ const handler = createOverlayHandler({ ...config, closeOnNavigation: false });
61
61
  let fnCalled = false;
62
+ let router = null;
63
+ const updateQueryParam = (value) => {
64
+ router?.navigate([], {
65
+ queryParams: {
66
+ [config.queryParamKey]: value,
67
+ },
68
+ queryParamsHandling: 'merge',
69
+ });
70
+ };
62
71
  const fn = (innerConfig) => {
63
72
  if (fnCalled) {
64
73
  throw new Error('The function returned by createOverlayHandlerWithQueryParamLifecycle can only be called once until the caller is destroyed');
65
74
  }
66
75
  fnCalled = true;
67
- const router = inject(Router);
76
+ router = inject(Router);
68
77
  const destroyRef = inject(DestroyRef);
69
- const overlayHandler = handler(innerConfig);
78
+ const injector = inject(Injector);
79
+ const overlayHandler = handler({ ...(innerConfig ?? {}) });
70
80
  const queryParamValue = injectQueryParam(config.queryParamKey);
71
81
  let currentOverlayRef = null;
82
+ let inputSignalEffect = null;
72
83
  destroyRef.onDestroy(() => {
73
84
  fnCalled = false;
85
+ router = null;
74
86
  cleanup();
75
87
  });
76
88
  const cleanup = () => {
77
- router.navigate([], {
78
- queryParams: {
79
- [config.queryParamKey]: null,
80
- },
81
- queryParamsHandling: 'merge',
82
- });
89
+ inputSignalEffect?.destroy();
90
+ inputSignalEffect = null;
91
+ updateQueryParam(null);
92
+ };
93
+ const getQueryParamInput = () => {
94
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
95
+ return currentOverlayRef?.componentInstance?.[OVERLAY_QUERY_PARAM_INPUT_NAME];
83
96
  };
84
97
  effect(() => {
85
98
  const value = queryParamValue();
@@ -91,28 +104,32 @@ export const createOverlayHandlerWithQueryParamLifecycle = (config) => {
91
104
  .afterClosed()
92
105
  .pipe(takeUntilDestroyed(destroyRef), tap(() => cleanup()))
93
106
  .subscribe();
107
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
108
+ const inputSignal = getQueryParamInput();
109
+ if (inputSignal) {
110
+ inputSignalEffect = effect(() => {
111
+ const inputVal = inputSignal();
112
+ untracked(() => {
113
+ updateQueryParam(inputVal);
114
+ });
115
+ }, { injector });
116
+ }
94
117
  }
95
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
96
- if (currentOverlayRef.componentInstance[OVERLAY_QUERY_PARAM_INPUT_NAME]) {
118
+ if (getQueryParamInput()) {
97
119
  currentOverlayRef.componentRef?.setInput(OVERLAY_QUERY_PARAM_INPUT_NAME, value);
98
120
  }
99
121
  }
100
122
  else {
101
123
  if (currentOverlayRef) {
124
+ inputSignalEffect?.destroy();
125
+ inputSignalEffect = null;
102
126
  currentOverlayRef.close();
103
127
  currentOverlayRef = null;
104
128
  }
105
129
  }
106
130
  });
107
131
  });
108
- const open = (queryParamValue) => {
109
- router.navigate([], {
110
- queryParams: {
111
- [config.queryParamKey]: queryParamValue,
112
- },
113
- queryParamsHandling: 'merge',
114
- });
115
- };
132
+ const open = (queryParamValue) => updateQueryParam(queryParamValue);
116
133
  const close = () => {
117
134
  cleanup();
118
135
  };
@@ -123,6 +140,7 @@ export const createOverlayHandlerWithQueryParamLifecycle = (config) => {
123
140
  return lifecycleHandler;
124
141
  };
125
142
  fn['injectOverlayRef'] = () => handler().injectOverlayRef();
143
+ fn['updateQueryParam'] = (value) => updateQueryParam(value);
126
144
  return fn;
127
145
  };
128
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"overlay-handler.js","sourceRoot":"","sources":["../../../../../../../../../../libs/cdk/src/lib/components/overlay/components/overlay/utils/overlay-handler.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,EAAe,SAAS,EAAE,gBAAgB,EAAE,MAAM,eAAe,CAAC;AACrG,OAAO,EAAE,kBAAkB,EAAE,MAAM,4BAA4B,CAAC;AAChE,OAAO,EAAE,MAAM,EAAE,MAAM,iBAAiB,CAAC;AACzC,OAAO,EAAE,gBAAgB,EAAE,MAAM,eAAe,CAAC;AACjD,OAAO,EAAE,GAAG,EAAE,MAAM,MAAM,CAAC;AAC3B,OAAO,EAAE,YAAY,EAAE,MAAM,cAAc,CAAC;AAC5C,OAAO,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;AAG7C,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAmC3C,MAAM,CAAC,MAAM,oBAAoB,GAAG,CAClC,UAAgE,EAChE,EAAE;IACF,MAAM,EAAE,GAAG,CAAC,WAA6D,EAAE,EAAE;QAC3E,MAAM,cAAc,GAAG,MAAM,CAAC,cAAc,CAAC,CAAC;QAC9C,MAAM,gBAAgB,GAAG,MAAM,CAAC,gBAAgB,CAAC,CAAC;QAClD,MAAM,UAAU,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;QAEtC,MAAM,GAAG,GAAG,UAAU,CAAC,SAAS,IAAI,UAAU,CAAC,QAAQ,CAAC;QAExD,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,MAAM,IAAI,KAAK,CAAC,+CAA+C,CAAC,CAAC;QACnE,CAAC;QAED,MAAM,IAAI,GAAG,CAAC,MAA4C,EAAE,EAAE;YAC5D,MAAM,GAAG,GAAG,cAAc,CAAC,IAAI,CAA2C,GAAG,EAAE;gBAC7E,gBAAgB;gBAChB,GAAG,UAAU;gBACb,SAAS,EAAE,UAAU,CAAC,SAAS,CAAC,cAAc,CAAC,SAAS,CAAC;gBACzD,GAAG,MAAM;aACV,CAAC,CAAC;YAEH,MAAM,aAAa,GAAG,WAAW,EAAE,WAAW,CAAC;YAE/C,IAAI,aAAa,EAAE,CAAC;gBAClB,GAAG;qBACA,WAAW,EAAE;qBACb,IAAI,CACH,kBAAkB,CAAC,UAAU,CAAC,EAC9B,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,aAAa,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,CACrC;qBACA,SAAS,EAAE,CAAC;YACjB,CAAC;YAED,OAAO,GAAG,CAAC;QACb,CAAC,CAAC;QAEF,MAAM,gBAAgB,GAAG,GAAG,EAAE;YAC5B,OAAO,MAAM,CAAyC,UAAU,CAAC,CAAC;QACpE,CAAC,CAAC;QAEF,MAAM,iBAAiB,GAAG,GAAG,EAAE;YAC7B,OAAO,MAAM,CAAe,YAAY,CAAC,CAAC;QAC5C,CAAC,CAAC;QAEF,MAAM,OAAO,GAA6D;YACxE,IAAI;YACJ,gBAAgB;YAChB,iBAAiB;SAClB,CAAC;QAEF,OAAO,OAAO,CAAC;IACjB,CAAC,CAAC;IAEF,OAAO,EAAE,CAAC;AACZ,CAAC,CAAC;AAwBF,MAAM,CAAC,MAAM,8BAA8B,GAAG,mBAAmB,CAAC;AAElE;;;;;;;;GAQG;AACH,MAAM,CAAC,MAAM,2CAA2C,GAAG,CAKzD,MAAqE,EACrE,EAAE;IACF,MAAM,OAAO,GAAG,oBAAoB,CAA4B,MAAM,CAAC,CAAC;IAExE,IAAI,QAAQ,GAAG,KAAK,CAAC;IAErB,MAAM,EAAE,GAAG,CAAC,WAAsD,EAAE,EAAE;QACpE,IAAI,QAAQ,EAAE,CAAC;YACb,MAAM,IAAI,KAAK,CACb,4HAA4H,CAC7H,CAAC;QACJ,CAAC;QAED,QAAQ,GAAG,IAAI,CAAC;QAEhB,MAAM,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC;QAC9B,MAAM,UAAU,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;QACtC,MAAM,cAAc,GAAG,OAAO,CAAC,WAAW,CAAC,CAAC;QAC5C,MAAM,eAAe,GAAG,gBAAgB,CAAc,MAAM,CAAC,aAAa,CAAC,CAAC;QAE5E,IAAI,iBAAiB,GAA2C,IAAI,CAAC;QAErE,UAAU,CAAC,SAAS,CAAC,GAAG,EAAE;YACxB,QAAQ,GAAG,KAAK,CAAC;YACjB,OAAO,EAAE,CAAC;QACZ,CAAC,CAAC,CAAC;QAEH,MAAM,OAAO,GAAG,GAAG,EAAE;YACnB,MAAM,CAAC,QAAQ,CAAC,EAAE,EAAE;gBAClB,WAAW,EAAE;oBACX,CAAC,MAAM,CAAC,aAAa,CAAC,EAAE,IAAI;iBAC7B;gBACD,mBAAmB,EAAE,OAAO;aAC7B,CAAC,CAAC;QACL,CAAC,CAAC;QAEF,MAAM,CAAC,GAAG,EAAE;YACV,MAAM,KAAK,GAAG,eAAe,EAAE,CAAC;YAEhC,SAAS,CAAC,GAAG,EAAE;gBACb,IAAI,KAAK,EAAE,CAAC;oBACV,IAAI,CAAC,iBAAiB,EAAE,CAAC;wBACvB,iBAAiB,GAAG,cAAc,CAAC,IAAI,EAAE,CAAC;wBAC1C,iBAAiB;6BACd,WAAW,EAAE;6BACb,IAAI,CACH,kBAAkB,CAAC,UAAU,CAAC,EAC9B,GAAG,CAAC,GAAG,EAAE,CAAC,OAAO,EAAE,CAAC,CACrB;6BACA,SAAS,EAAE,CAAC;oBACjB,CAAC;oBAED,8DAA8D;oBAC9D,IAAK,iBAAiB,CAAC,iBAAyB,CAAC,8BAA8B,CAAC,EAAE,CAAC;wBACjF,iBAAiB,CAAC,YAAY,EAAE,QAAQ,CAAC,8BAA8B,EAAE,KAAK,CAAC,CAAC;oBAClF,CAAC;gBACH,CAAC;qBAAM,CAAC;oBACN,IAAI,iBAAiB,EAAE,CAAC;wBACtB,iBAAiB,CAAC,KAAK,EAAE,CAAC;wBAC1B,iBAAiB,GAAG,IAAI,CAAC;oBAC3B,CAAC;gBACH,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,MAAM,IAAI,GAAG,CAAC,eAA4B,EAAE,EAAE;YAC5C,MAAM,CAAC,QAAQ,CAAC,EAAE,EAAE;gBAClB,WAAW,EAAE;oBACX,CAAC,MAAM,CAAC,aAAa,CAAC,EAAE,eAAe;iBACxC;gBACD,mBAAmB,EAAE,OAAO;aAC7B,CAAC,CAAC;QACL,CAAC,CAAC;QAEF,MAAM,KAAK,GAAG,GAAG,EAAE;YACjB,OAAO,EAAE,CAAC;QACZ,CAAC,CAAC;QAEF,MAAM,gBAAgB,GAAuD;YAC3E,IAAI;YACJ,KAAK;SACN,CAAC;QAEF,OAAO,gBAAgB,CAAC;IAC1B,CAAC,CAAC;IAEF,EAAE,CAAC,kBAAkB,CAAC,GAAG,GAAG,EAAE,CAAC,OAAO,EAAE,CAAC,gBAAgB,EAAE,CAAC;IAE5D,OAAO,EAAE,CAAC;AACZ,CAAC,CAAC","sourcesContent":["import { ComponentType } from '@angular/cdk/overlay';\nimport { DestroyRef, effect, inject, TemplateRef, untracked, ViewContainerRef } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { Router } from '@angular/router';\nimport { injectQueryParam } from '@ethlete/core';\nimport { tap } from 'rxjs';\nimport { OVERLAY_DATA } from '../constants';\nimport { OverlayService } from '../services';\nimport { OverlayBreakpointConfigEntry, OverlayConfig, OverlayConsumerConfig } from '../types';\nimport { OverlayPositionBuilder } from './overlay-position-builder';\nimport { OverlayRef } from './overlay-ref';\n\nexport type CreateOverlayHandlerConfig<T, D = unknown> = Omit<OverlayConfig<D>, 'positions'> & {\n  /** The overlay component. Use either this or the `template` property  */\n  component?: ComponentType<T>;\n\n  /** The overlay template. Use either this or the `component` property  */\n  template?: TemplateRef<T>;\n\n  /** The overlay positions using the position builder provided via argument  */\n  positions: (builder: OverlayPositionBuilder) => OverlayBreakpointConfigEntry[];\n};\n\nexport type OverlayHandler<T, D = unknown, R = unknown> = {\n  /** Open the overlay using a combination of the given configs  */\n  open: (config?: OverlayConsumerConfig<D>) => OverlayRef<T, R>;\n\n  /**\n   * Returns the typed overlay ref.\n   * @throws Error if the overlay ref gets accessed outside of the overlay component or templateRef\n   */\n  injectOverlayRef: () => OverlayRef<T, R>;\n\n  /**\n   * Returns the overlay data.\n   * @throws Error if the overlay data gets accessed outside of the overlay component or templateRef\n   */\n  injectOverlayData: () => D;\n};\n\nexport type CreateOverlayHandlerInnerConfig<R = unknown> = {\n  /** A callback function to be executed once the overlay has been closed */\n  afterClosed?: (result: R | null) => void;\n};\n\nexport const createOverlayHandler = <TComponent, TOverlayData = unknown, TOverlayResult = unknown>(\n  rootConfig: CreateOverlayHandlerConfig<TComponent, TOverlayData>,\n) => {\n  const fn = (innerConfig?: CreateOverlayHandlerInnerConfig<TOverlayResult>) => {\n    const overlayService = inject(OverlayService);\n    const viewContainerRef = inject(ViewContainerRef);\n    const destroyRef = inject(DestroyRef);\n\n    const tpl = rootConfig.component ?? rootConfig.template;\n\n    if (!tpl) {\n      throw new Error('Either component or template must be provided');\n    }\n\n    const open = (config?: OverlayConsumerConfig<TOverlayData>) => {\n      const ref = overlayService.open<TComponent, TOverlayData, TOverlayResult>(tpl, {\n        viewContainerRef,\n        ...rootConfig,\n        positions: rootConfig.positions(overlayService.positions),\n        ...config,\n      });\n\n      const afterClosedFn = innerConfig?.afterClosed;\n\n      if (afterClosedFn) {\n        ref\n          .afterClosed()\n          .pipe(\n            takeUntilDestroyed(destroyRef),\n            tap((r) => afterClosedFn(r ?? null)),\n          )\n          .subscribe();\n      }\n\n      return ref;\n    };\n\n    const injectOverlayRef = () => {\n      return inject<OverlayRef<TComponent, TOverlayResult>>(OverlayRef);\n    };\n\n    const injectOverlayData = () => {\n      return inject<TOverlayData>(OVERLAY_DATA);\n    };\n\n    const handler: OverlayHandler<TComponent, TOverlayData, TOverlayResult> = {\n      open,\n      injectOverlayRef,\n      injectOverlayData,\n    };\n\n    return handler;\n  };\n\n  return fn;\n};\n\nexport type OverlayHandlerWithQueryParamLifecycle<Q = string> = {\n  /** Open the overlay using the provided query param value  */\n  open: (queryParamValue: Q) => void;\n\n  /** Close the overlay and remove the query param  */\n  close: () => void;\n};\n\nexport type CreateOverlayHandlerWithQueryParamLifecycleConfig<T> = Omit<\n  OverlayConfig<unknown>,\n  'positions' | 'data'\n> & {\n  /** The overlay component  */\n  component: ComponentType<T>;\n\n  /** The overlay positions using the position builder provided via argument  */\n  positions: (builder: OverlayPositionBuilder) => OverlayBreakpointConfigEntry[];\n\n  /** The query param key to be used for the overlay  */\n  queryParamKey: string;\n};\n\nexport const OVERLAY_QUERY_PARAM_INPUT_NAME = 'overlayQueryParam';\n\n/**\n * This handler will automatically open the overlay when the query param is present.\n * The overlay can contain a required input with the name `overlayQueryParam` to receive the query param value.\n *\n * If you need to transfer more information (eg. a second query param), you need to combine them into a single query param string.\n * You can then split the string inside the overlay component using computed signals.\n *\n * To open the overlay either use the `OverlayHandlerLinkDirective` or call the `.open` method of the returned handler.\n */\nexport const createOverlayHandlerWithQueryParamLifecycle = <\n  TComponent,\n  TQueryParam extends string = string,\n  TResult = unknown,\n>(\n  config: CreateOverlayHandlerWithQueryParamLifecycleConfig<TComponent>,\n) => {\n  const handler = createOverlayHandler<TComponent, void, TResult>(config);\n\n  let fnCalled = false;\n\n  const fn = (innerConfig?: CreateOverlayHandlerInnerConfig<TResult>) => {\n    if (fnCalled) {\n      throw new Error(\n        'The function returned by createOverlayHandlerWithQueryParamLifecycle can only be called once until the caller is destroyed',\n      );\n    }\n\n    fnCalled = true;\n\n    const router = inject(Router);\n    const destroyRef = inject(DestroyRef);\n    const overlayHandler = handler(innerConfig);\n    const queryParamValue = injectQueryParam<TQueryParam>(config.queryParamKey);\n\n    let currentOverlayRef: OverlayRef<TComponent, TResult> | null = null;\n\n    destroyRef.onDestroy(() => {\n      fnCalled = false;\n      cleanup();\n    });\n\n    const cleanup = () => {\n      router.navigate([], {\n        queryParams: {\n          [config.queryParamKey]: null,\n        },\n        queryParamsHandling: 'merge',\n      });\n    };\n\n    effect(() => {\n      const value = queryParamValue();\n\n      untracked(() => {\n        if (value) {\n          if (!currentOverlayRef) {\n            currentOverlayRef = overlayHandler.open();\n            currentOverlayRef\n              .afterClosed()\n              .pipe(\n                takeUntilDestroyed(destroyRef),\n                tap(() => cleanup()),\n              )\n              .subscribe();\n          }\n\n          // eslint-disable-next-line @typescript-eslint/no-explicit-any\n          if ((currentOverlayRef.componentInstance as any)[OVERLAY_QUERY_PARAM_INPUT_NAME]) {\n            currentOverlayRef.componentRef?.setInput(OVERLAY_QUERY_PARAM_INPUT_NAME, value);\n          }\n        } else {\n          if (currentOverlayRef) {\n            currentOverlayRef.close();\n            currentOverlayRef = null;\n          }\n        }\n      });\n    });\n\n    const open = (queryParamValue: TQueryParam) => {\n      router.navigate([], {\n        queryParams: {\n          [config.queryParamKey]: queryParamValue,\n        },\n        queryParamsHandling: 'merge',\n      });\n    };\n\n    const close = () => {\n      cleanup();\n    };\n\n    const lifecycleHandler: OverlayHandlerWithQueryParamLifecycle<TQueryParam> = {\n      open,\n      close,\n    };\n\n    return lifecycleHandler;\n  };\n\n  fn['injectOverlayRef'] = () => handler().injectOverlayRef();\n\n  return fn;\n};\n"]}
146
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"overlay-handler.js","sourceRoot":"","sources":["../../../../../../../../../../libs/cdk/src/lib/components/overlay/components/overlay/utils/overlay-handler.ts"],"names":[],"mappings":"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","sourcesContent":["import { ComponentType } from '@angular/cdk/overlay';\nimport {\n  DestroyRef,\n  effect,\n  EffectRef,\n  inject,\n  Injector,\n  InputSignal,\n  ModelSignal,\n  TemplateRef,\n  untracked,\n  ViewContainerRef,\n} from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { Router } from '@angular/router';\nimport { injectQueryParam } from '@ethlete/core';\nimport { tap } from 'rxjs';\nimport { OVERLAY_DATA } from '../constants';\nimport { OverlayService } from '../services';\nimport { OverlayBreakpointConfigEntry, OverlayConfig, OverlayConsumerConfig } from '../types';\nimport { OverlayPositionBuilder } from './overlay-position-builder';\nimport { OverlayRef } from './overlay-ref';\n\nexport type CreateOverlayHandlerConfig<T, D = unknown> = Omit<OverlayConfig<D>, 'positions'> & {\n  /** The overlay component. Use either this or the `template` property  */\n  component?: ComponentType<T>;\n\n  /** The overlay template. Use either this or the `component` property  */\n  template?: TemplateRef<T>;\n\n  /** The overlay positions using the position builder provided via argument  */\n  positions: (builder: OverlayPositionBuilder) => OverlayBreakpointConfigEntry[];\n};\n\nexport type OverlayHandler<T, D = unknown, R = unknown> = {\n  /** Open the overlay using a combination of the given configs  */\n  open: (config?: OverlayConsumerConfig<D>) => OverlayRef<T, R>;\n\n  /**\n   * Returns the typed overlay ref.\n   * @throws Error if the overlay ref gets accessed outside of the overlay component or templateRef\n   */\n  injectOverlayRef: () => OverlayRef<T, R>;\n\n  /**\n   * Returns the overlay data.\n   * @throws Error if the overlay data gets accessed outside of the overlay component or templateRef\n   */\n  injectOverlayData: () => D;\n};\n\nexport type CreateOverlayHandlerInnerConfig<R = unknown> = {\n  /** A callback function to be executed once the overlay has been closed */\n  afterClosed?: (result: R | null) => void;\n};\n\nexport const createOverlayHandler = <TComponent, TOverlayData = unknown, TOverlayResult = unknown>(\n  rootConfig: CreateOverlayHandlerConfig<TComponent, TOverlayData>,\n) => {\n  const fn = (innerConfig?: CreateOverlayHandlerInnerConfig<TOverlayResult>) => {\n    const overlayService = inject(OverlayService);\n    const viewContainerRef = inject(ViewContainerRef);\n    const destroyRef = inject(DestroyRef);\n\n    const tpl = rootConfig.component ?? rootConfig.template;\n\n    if (!tpl) {\n      throw new Error('Either component or template must be provided');\n    }\n\n    const open = (config?: OverlayConsumerConfig<TOverlayData>) => {\n      const ref = overlayService.open<TComponent, TOverlayData, TOverlayResult>(tpl, {\n        viewContainerRef,\n        ...rootConfig,\n        positions: rootConfig.positions(overlayService.positions),\n        ...config,\n      });\n\n      const afterClosedFn = innerConfig?.afterClosed;\n\n      if (afterClosedFn) {\n        ref\n          .afterClosed()\n          .pipe(\n            takeUntilDestroyed(destroyRef),\n            tap((r) => afterClosedFn(r ?? null)),\n          )\n          .subscribe();\n      }\n\n      return ref;\n    };\n\n    const injectOverlayRef = () => {\n      return inject<OverlayRef<TComponent, TOverlayResult>>(OverlayRef);\n    };\n\n    const injectOverlayData = () => {\n      return inject<TOverlayData>(OVERLAY_DATA);\n    };\n\n    const handler: OverlayHandler<TComponent, TOverlayData, TOverlayResult> = {\n      open,\n      injectOverlayRef,\n      injectOverlayData,\n    };\n\n    return handler;\n  };\n\n  return fn;\n};\n\nexport type OverlayHandlerWithQueryParamLifecycle<Q = string> = {\n  /** Open the overlay using the provided query param value  */\n  open: (queryParamValue: Q) => void;\n\n  /** Close the overlay and remove the query param  */\n  close: () => void;\n};\n\nexport type CreateOverlayHandlerWithQueryParamLifecycleConfig<T> = Omit<\n  OverlayConfig<unknown>,\n  'positions' | 'data' | 'closeOnNavigation'\n> & {\n  /** The overlay component  */\n  component: ComponentType<T>;\n\n  /** The overlay positions using the position builder provided via argument  */\n  positions: (builder: OverlayPositionBuilder) => OverlayBreakpointConfigEntry[];\n\n  /** The query param key to be used for the overlay  */\n  queryParamKey: string;\n};\n\nexport const OVERLAY_QUERY_PARAM_INPUT_NAME = 'overlayQueryParam';\n\n/**\n * This handler will automatically open the overlay when the query param is present.\n * The overlay can contain a input or model with the name `overlayQueryParam` to receive the query param value.\n *\n * If you need to transfer more information (eg. a second query param), you need to combine them into a single query param string.\n * You can then split the string inside the overlay component using computed signals.\n *\n * To open the overlay either use the `OverlayHandlerLinkDirective` or call the `.open` method of the returned handler.\n */\nexport const createOverlayHandlerWithQueryParamLifecycle = <\n  TComponent,\n  TQueryParam extends string = string,\n  TResult = unknown,\n>(\n  config: CreateOverlayHandlerWithQueryParamLifecycleConfig<TComponent>,\n) => {\n  const handler = createOverlayHandler<TComponent, void, TResult>({ ...config, closeOnNavigation: false });\n\n  let fnCalled = false;\n  let router: Router | null = null;\n\n  const updateQueryParam = (value: TQueryParam | null) => {\n    router?.navigate([], {\n      queryParams: {\n        [config.queryParamKey]: value,\n      },\n      queryParamsHandling: 'merge',\n    });\n  };\n\n  const fn = (innerConfig?: CreateOverlayHandlerInnerConfig<TResult>) => {\n    if (fnCalled) {\n      throw new Error(\n        'The function returned by createOverlayHandlerWithQueryParamLifecycle can only be called once until the caller is destroyed',\n      );\n    }\n\n    fnCalled = true;\n\n    router = inject(Router);\n    const destroyRef = inject(DestroyRef);\n    const injector = inject(Injector);\n    const overlayHandler = handler({ ...(innerConfig ?? {}) });\n    const queryParamValue = injectQueryParam<TQueryParam>(config.queryParamKey);\n\n    let currentOverlayRef: OverlayRef<TComponent, TResult> | null = null;\n    let inputSignalEffect: EffectRef | null = null;\n\n    destroyRef.onDestroy(() => {\n      fnCalled = false;\n      router = null;\n      cleanup();\n    });\n\n    const cleanup = () => {\n      inputSignalEffect?.destroy();\n      inputSignalEffect = null;\n      updateQueryParam(null);\n    };\n\n    const getQueryParamInput = () => {\n      // eslint-disable-next-line @typescript-eslint/no-explicit-any\n      return (currentOverlayRef?.componentInstance as any)?.[OVERLAY_QUERY_PARAM_INPUT_NAME] as\n        | ModelSignal<TQueryParam>\n        | InputSignal<TQueryParam>\n        | undefined;\n    };\n\n    effect(() => {\n      const value = queryParamValue();\n\n      untracked(() => {\n        if (value) {\n          if (!currentOverlayRef) {\n            currentOverlayRef = overlayHandler.open();\n            currentOverlayRef\n              .afterClosed()\n              .pipe(\n                takeUntilDestroyed(destroyRef),\n                tap(() => cleanup()),\n              )\n              .subscribe();\n\n            // eslint-disable-next-line @typescript-eslint/no-explicit-any\n            const inputSignal = getQueryParamInput();\n\n            if (inputSignal) {\n              inputSignalEffect = effect(\n                () => {\n                  const inputVal = inputSignal();\n\n                  untracked(() => {\n                    updateQueryParam(inputVal);\n                  });\n                },\n                { injector },\n              );\n            }\n          }\n\n          if (getQueryParamInput()) {\n            currentOverlayRef.componentRef?.setInput(OVERLAY_QUERY_PARAM_INPUT_NAME, value);\n          }\n        } else {\n          if (currentOverlayRef) {\n            inputSignalEffect?.destroy();\n            inputSignalEffect = null;\n            currentOverlayRef.close();\n            currentOverlayRef = null;\n          }\n        }\n      });\n    });\n\n    const open = (queryParamValue: TQueryParam) => updateQueryParam(queryParamValue);\n\n    const close = () => {\n      cleanup();\n    };\n\n    const lifecycleHandler: OverlayHandlerWithQueryParamLifecycle<TQueryParam> = {\n      open,\n      close,\n    };\n\n    return lifecycleHandler;\n  };\n\n  fn['injectOverlayRef'] = () => handler().injectOverlayRef();\n  fn['updateQueryParam'] = (value: TQueryParam) => updateQueryParam(value);\n\n  return fn;\n};\n"]}
@@ -12460,7 +12460,7 @@ const createOverlayHandler = (rootConfig) => {
12460
12460
  const OVERLAY_QUERY_PARAM_INPUT_NAME = 'overlayQueryParam';
12461
12461
  /**
12462
12462
  * This handler will automatically open the overlay when the query param is present.
12463
- * The overlay can contain a required input with the name `overlayQueryParam` to receive the query param value.
12463
+ * The overlay can contain a input or model with the name `overlayQueryParam` to receive the query param value.
12464
12464
  *
12465
12465
  * If you need to transfer more information (eg. a second query param), you need to combine them into a single query param string.
12466
12466
  * You can then split the string inside the overlay component using computed signals.
@@ -12468,29 +12468,42 @@ const OVERLAY_QUERY_PARAM_INPUT_NAME = 'overlayQueryParam';
12468
12468
  * To open the overlay either use the `OverlayHandlerLinkDirective` or call the `.open` method of the returned handler.
12469
12469
  */
12470
12470
  const createOverlayHandlerWithQueryParamLifecycle = (config) => {
12471
- const handler = createOverlayHandler(config);
12471
+ const handler = createOverlayHandler({ ...config, closeOnNavigation: false });
12472
12472
  let fnCalled = false;
12473
+ let router = null;
12474
+ const updateQueryParam = (value) => {
12475
+ router?.navigate([], {
12476
+ queryParams: {
12477
+ [config.queryParamKey]: value,
12478
+ },
12479
+ queryParamsHandling: 'merge',
12480
+ });
12481
+ };
12473
12482
  const fn = (innerConfig) => {
12474
12483
  if (fnCalled) {
12475
12484
  throw new Error('The function returned by createOverlayHandlerWithQueryParamLifecycle can only be called once until the caller is destroyed');
12476
12485
  }
12477
12486
  fnCalled = true;
12478
- const router = inject(Router);
12487
+ router = inject(Router);
12479
12488
  const destroyRef = inject(DestroyRef);
12480
- const overlayHandler = handler(innerConfig);
12489
+ const injector = inject(Injector);
12490
+ const overlayHandler = handler({ ...(innerConfig ?? {}) });
12481
12491
  const queryParamValue = injectQueryParam(config.queryParamKey);
12482
12492
  let currentOverlayRef = null;
12493
+ let inputSignalEffect = null;
12483
12494
  destroyRef.onDestroy(() => {
12484
12495
  fnCalled = false;
12496
+ router = null;
12485
12497
  cleanup();
12486
12498
  });
12487
12499
  const cleanup = () => {
12488
- router.navigate([], {
12489
- queryParams: {
12490
- [config.queryParamKey]: null,
12491
- },
12492
- queryParamsHandling: 'merge',
12493
- });
12500
+ inputSignalEffect?.destroy();
12501
+ inputSignalEffect = null;
12502
+ updateQueryParam(null);
12503
+ };
12504
+ const getQueryParamInput = () => {
12505
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
12506
+ return currentOverlayRef?.componentInstance?.[OVERLAY_QUERY_PARAM_INPUT_NAME];
12494
12507
  };
12495
12508
  effect(() => {
12496
12509
  const value = queryParamValue();
@@ -12502,28 +12515,32 @@ const createOverlayHandlerWithQueryParamLifecycle = (config) => {
12502
12515
  .afterClosed()
12503
12516
  .pipe(takeUntilDestroyed(destroyRef), tap(() => cleanup()))
12504
12517
  .subscribe();
12518
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
12519
+ const inputSignal = getQueryParamInput();
12520
+ if (inputSignal) {
12521
+ inputSignalEffect = effect(() => {
12522
+ const inputVal = inputSignal();
12523
+ untracked(() => {
12524
+ updateQueryParam(inputVal);
12525
+ });
12526
+ }, { injector });
12527
+ }
12505
12528
  }
12506
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
12507
- if (currentOverlayRef.componentInstance[OVERLAY_QUERY_PARAM_INPUT_NAME]) {
12529
+ if (getQueryParamInput()) {
12508
12530
  currentOverlayRef.componentRef?.setInput(OVERLAY_QUERY_PARAM_INPUT_NAME, value);
12509
12531
  }
12510
12532
  }
12511
12533
  else {
12512
12534
  if (currentOverlayRef) {
12535
+ inputSignalEffect?.destroy();
12536
+ inputSignalEffect = null;
12513
12537
  currentOverlayRef.close();
12514
12538
  currentOverlayRef = null;
12515
12539
  }
12516
12540
  }
12517
12541
  });
12518
12542
  });
12519
- const open = (queryParamValue) => {
12520
- router.navigate([], {
12521
- queryParams: {
12522
- [config.queryParamKey]: queryParamValue,
12523
- },
12524
- queryParamsHandling: 'merge',
12525
- });
12526
- };
12543
+ const open = (queryParamValue) => updateQueryParam(queryParamValue);
12527
12544
  const close = () => {
12528
12545
  cleanup();
12529
12546
  };
@@ -12534,6 +12551,7 @@ const createOverlayHandlerWithQueryParamLifecycle = (config) => {
12534
12551
  return lifecycleHandler;
12535
12552
  };
12536
12553
  fn['injectOverlayRef'] = () => handler().injectOverlayRef();
12554
+ fn['updateQueryParam'] = (value) => updateQueryParam(value);
12537
12555
  return fn;
12538
12556
  };
12539
12557