@ethlete/cdk 2.3.2 → 2.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (57) hide show
  1. package/CHANGELOG.md +12 -0
  2. package/esm2022/lib/components/accordion/components/accordion-group/accordion-group.component.mjs +6 -6
  3. package/esm2022/lib/components/button/directives/button/button.directive.mjs +4 -5
  4. package/esm2022/lib/components/button/directives/query-button/query-button.directive.mjs +4 -5
  5. package/esm2022/lib/components/forms/components/checkbox/components/checkbox-group/checkbox-group.component.mjs +3 -4
  6. package/esm2022/lib/components/forms/components/checkbox/directives/checkbox/checkbox.directive.mjs +4 -4
  7. package/esm2022/lib/components/forms/components/checkbox/directives/checkbox-field/checkbox-field.directive.mjs +4 -4
  8. package/esm2022/lib/components/forms/components/checkbox/directives/checkbox-group/checkbox-group.directive.mjs +6 -6
  9. package/esm2022/lib/components/forms/components/checkbox/directives/checkbox-group-control/checkbox-group-control.directive.mjs +4 -4
  10. package/esm2022/lib/components/forms/components/error/components/error/error.component.mjs +5 -5
  11. package/esm2022/lib/components/forms/components/input/components/email-input/email-input.component.mjs +3 -4
  12. package/esm2022/lib/components/forms/components/input/components/number-input/number-input.component.mjs +4 -5
  13. package/esm2022/lib/components/forms/components/input/components/password-input/password-input.component.mjs +4 -5
  14. package/esm2022/lib/components/forms/components/input/components/search-input/search-input.component.mjs +4 -5
  15. package/esm2022/lib/components/forms/components/input/components/text-input/text-input.component.mjs +4 -5
  16. package/esm2022/lib/components/forms/components/input/components/textarea-input/textarea-input.component.mjs +4 -5
  17. package/esm2022/lib/components/forms/components/input/directives/autosize-textarea/autosize-textarea.directive.mjs +4 -5
  18. package/esm2022/lib/components/forms/components/input/directives/email-input/email-input.directive.mjs +5 -5
  19. package/esm2022/lib/components/forms/components/input/directives/number-input/number-input.directive.mjs +5 -5
  20. package/esm2022/lib/components/forms/components/input/directives/password-input/password-input.directive.mjs +5 -5
  21. package/esm2022/lib/components/forms/components/input/directives/search-input/search-input.directive.mjs +5 -5
  22. package/esm2022/lib/components/forms/components/input/directives/text-input/text-input.directive.mjs +5 -5
  23. package/esm2022/lib/components/forms/components/input/directives/textarea-input/textarea-input.directive.mjs +5 -5
  24. package/esm2022/lib/components/forms/components/radio/directives/radio/radio.directive.mjs +5 -5
  25. package/esm2022/lib/components/forms/components/radio/directives/radio-field/radio-field.directive.mjs +4 -4
  26. package/esm2022/lib/components/forms/components/radio/directives/radio-group/radio-group.directive.mjs +4 -4
  27. package/esm2022/lib/components/forms/components/segmented-button/directives/segmented-button/segmented-button.directive.mjs +4 -4
  28. package/esm2022/lib/components/forms/components/segmented-button/directives/segmented-button-field/segmented-button-field.directive.mjs +4 -4
  29. package/esm2022/lib/components/forms/components/segmented-button/directives/segmented-button-group/segmented-button-group.directive.mjs +5 -5
  30. package/esm2022/lib/components/forms/components/select/components/native-select/components/native-select/native-select.component.mjs +3 -4
  31. package/esm2022/lib/components/forms/components/select/components/native-select/directives/native-select-input/native-select-input.directive.mjs +5 -5
  32. package/esm2022/lib/components/forms/components/select/components/select/components/select/select.component.mjs +24 -12
  33. package/esm2022/lib/components/forms/components/select/components/select/partials/select-body/select-body.component.mjs +11 -16
  34. package/esm2022/lib/components/forms/components/slider/components/slider/slider.component.mjs +6 -6
  35. package/esm2022/lib/components/forms/directives/if-input-empty/if-input-empty.directive.mjs +4 -5
  36. package/esm2022/lib/components/forms/directives/if-input-filled/if-input-filled.directive.mjs +4 -5
  37. package/esm2022/lib/components/forms/directives/input/input.directive.mjs +6 -6
  38. package/esm2022/lib/components/forms/directives/static-form-field/static-form-field.directive.mjs +3 -5
  39. package/esm2022/lib/components/forms/directives/writeable-input/writeable-input.directive.mjs +2 -4
  40. package/esm2022/lib/components/forms/utils/decorated-input.base.mjs +3 -3
  41. package/esm2022/lib/components/masonry/components/masonry/masonry.component.mjs +5 -5
  42. package/esm2022/lib/components/overlay/components/bottom-sheet/partials/bottom-sheet-drag-handle/bottom-sheet-drag-handle.component.mjs +5 -5
  43. package/esm2022/lib/components/overlay/components/toggletip/directives/toggletip/toggletip.directive.mjs +41 -107
  44. package/esm2022/lib/components/overlay/components/tooltip/directives/tooltip/tooltip.directive.mjs +41 -117
  45. package/esm2022/lib/components/scrollable/components/scrollable/scrollable.component.mjs +55 -8
  46. package/esm2022/lib/components/tabs/components/nav-tabs/nav-tabs.component.mjs +1 -1
  47. package/esm2022/lib/components/tabs/partials/inline-tabs/inline-tab-header/inline-tab-header.component.mjs +1 -1
  48. package/esm2022/lib/components/tabs/utils/paginated-tab-header.directive.mjs +3 -4
  49. package/fesm2022/ethlete-cdk.mjs +290 -389
  50. package/fesm2022/ethlete-cdk.mjs.map +1 -1
  51. package/lib/components/forms/components/select/components/select/components/select/select.component.d.ts +5 -1
  52. package/lib/components/forms/components/select/components/select/partials/select-body/select-body.component.d.ts +2 -0
  53. package/lib/components/overlay/components/toggletip/directives/toggletip/toggletip.directive.d.ts +11 -18
  54. package/lib/components/overlay/components/tooltip/directives/tooltip/tooltip.directive.d.ts +6 -17
  55. package/lib/components/scrollable/components/scrollable/scrollable.component.d.ts +13 -5
  56. package/lib/components/tabs/utils/paginated-tab-header.directive.d.ts +1 -2
  57. package/package.json +3 -4
@@ -1,38 +1,14 @@
1
1
  import { AriaDescriber } from '@angular/cdk/a11y';
2
- import { Overlay } from '@angular/cdk/overlay';
3
- import { ComponentPortal } from '@angular/cdk/portal';
4
- import { Directive, ElementRef, inject, InjectionToken, Injector, Input, NgZone, ViewContainerRef, } from '@angular/core';
5
- import { FocusVisibleService, nextFrame } from '@ethlete/core';
6
- import { createPopper } from '@popperjs/core';
7
- import { debounceTime, filter, fromEvent, take, tap } from 'rxjs';
2
+ import { Directive, ElementRef, InjectionToken, Input, inject } from '@angular/core';
3
+ import { AnimatedOverlayDirective, FocusVisibleService } from '@ethlete/core';
4
+ import { debounceTime, filter, fromEvent, tap } from 'rxjs';
8
5
  import { TooltipComponent } from '../../components';
9
6
  import { TOOLTIP_CONFIG, TOOLTIP_TEMPLATE, TOOLTIP_TEXT } from '../../constants';
10
7
  import { createTooltipConfig } from '../../utils';
11
8
  import * as i0 from "@angular/core";
9
+ import * as i1 from "@ethlete/core";
12
10
  export const TOOLTIP_DIRECTIVE = new InjectionToken('TOOLTIP_DIRECTIVE');
13
11
  class TooltipDirective {
14
- constructor() {
15
- this._defaultConfig = inject(TOOLTIP_CONFIG, { optional: true }) ?? createTooltipConfig();
16
- this._tooltip = null;
17
- this.placement = this._defaultConfig.placement;
18
- this._tooltipAriaDescription = null;
19
- this._lastTooltipText = null;
20
- this._hostElementRef = inject(ElementRef);
21
- this._viewContainerRef = inject(ViewContainerRef);
22
- this._overlayService = inject(Overlay);
23
- this._ariaDescriberService = inject(AriaDescriber);
24
- this._focusVisibleService = inject(FocusVisibleService);
25
- this._injector = inject(Injector);
26
- this._zone = inject(NgZone);
27
- this._overlayRef = null;
28
- this._portal = null;
29
- this._tooltipRef = null;
30
- this._popper = null;
31
- this._willMount = true;
32
- this._hasFocus = false;
33
- this._hasHover = false;
34
- this._listenerSubscriptions = [];
35
- }
36
12
  get tooltip() {
37
13
  return this._tooltip;
38
14
  }
@@ -53,61 +29,75 @@ class TooltipDirective {
53
29
  this._tooltipAriaDescription = v;
54
30
  this._updateAriaDescription();
55
31
  }
56
- get _isMounted() {
57
- return this._overlayRef !== null;
32
+ constructor() {
33
+ this._defaultConfig = inject(TOOLTIP_CONFIG, { optional: true }) ?? createTooltipConfig();
34
+ this._animatedOverlay = inject(AnimatedOverlayDirective);
35
+ this._tooltip = null;
36
+ this._tooltipAriaDescription = null;
37
+ this._lastTooltipText = null;
38
+ this._elementRef = inject(ElementRef);
39
+ this._ariaDescriberService = inject(AriaDescriber);
40
+ this._focusVisibleService = inject(FocusVisibleService);
41
+ this._willMount = true;
42
+ this._hasFocus = false;
43
+ this._hasHover = false;
44
+ this._listenerSubscriptions = [];
45
+ this._animatedOverlay.placement = this._defaultConfig.placement;
46
+ this._animatedOverlay.offset = this._defaultConfig.offset;
47
+ this._animatedOverlay.arrowPadding = this._defaultConfig.arrowPadding;
58
48
  }
59
49
  ngOnDestroy() {
60
- this._unmountTooltip();
50
+ this._animatedOverlay._destroy();
61
51
  this._removeListeners();
62
52
  }
63
53
  _updateAriaDescription() {
64
54
  const tooltipText = this._getTooltipText();
65
55
  if (tooltipText) {
66
- this._ariaDescriberService.describe(this._hostElementRef.nativeElement, tooltipText);
56
+ this._ariaDescriberService.describe(this._elementRef.nativeElement, tooltipText);
67
57
  }
68
58
  else if (this._lastTooltipText) {
69
- this._ariaDescriberService.removeDescription(this._hostElementRef.nativeElement, this._lastTooltipText);
59
+ this._ariaDescriberService.removeDescription(this._elementRef.nativeElement, this._lastTooltipText);
70
60
  }
71
61
  this._lastTooltipText = tooltipText;
72
62
  }
73
63
  _addListeners() {
74
- const mouseEnterSub = fromEvent(this._hostElementRef.nativeElement, 'mouseenter')
64
+ const mouseEnterSub = fromEvent(this._elementRef.nativeElement, 'mouseenter')
75
65
  .pipe(tap(() => {
76
66
  this._willMount = true;
77
67
  this._hasHover = true;
78
68
  }), debounceTime(200))
79
69
  .subscribe(() => {
80
- if (!this._willMount || this._isMounted) {
70
+ if (!this._willMount || this._animatedOverlay.isMounted) {
81
71
  return;
82
72
  }
83
73
  this._mountTooltip();
84
74
  });
85
- const focusSub = fromEvent(this._hostElementRef.nativeElement, 'focus').subscribe(() => {
75
+ const focusSub = fromEvent(this._elementRef.nativeElement, 'focus').subscribe(() => {
86
76
  if (!this._focusVisibleService.isFocusVisible) {
87
77
  return;
88
78
  }
89
79
  this._hasFocus = true;
90
- if (this._isMounted) {
80
+ if (this._animatedOverlay.isMounted) {
91
81
  return;
92
82
  }
93
83
  this._mountTooltip();
94
84
  });
95
- const mouseLeaveSub = fromEvent(this._hostElementRef.nativeElement, 'mouseleave').subscribe(() => {
85
+ const mouseLeaveSub = fromEvent(this._elementRef.nativeElement, 'mouseleave').subscribe(() => {
96
86
  this._hasHover = false;
97
87
  this._willMount = false;
98
- if (this._isMounted && !this._hasFocus) {
99
- this._animateUnmount();
88
+ if (this._animatedOverlay.isMounted && !this._hasFocus) {
89
+ this._animatedOverlay.unmount();
100
90
  }
101
91
  });
102
- const blurSub = fromEvent(this._hostElementRef.nativeElement, 'blur').subscribe(() => {
92
+ const blurSub = fromEvent(this._elementRef.nativeElement, 'blur').subscribe(() => {
103
93
  this._hasFocus = false;
104
94
  this._willMount = false;
105
- if (this._isMounted && !this._hasHover) {
106
- this._animateUnmount();
95
+ if (this._animatedOverlay.isMounted && !this._hasHover) {
96
+ this._animatedOverlay.unmount();
107
97
  }
108
98
  });
109
99
  const keyupEscSub = fromEvent(document, 'keyup')
110
- .pipe(filter((e) => e.key === 'Escape'), filter(() => this._isMounted), tap(() => this._animateUnmount()))
100
+ .pipe(filter((e) => e.key === 'Escape'), filter(() => this._animatedOverlay.isMounted), tap(() => this._animatedOverlay.unmount()))
111
101
  .subscribe();
112
102
  this._listenerSubscriptions.push(mouseEnterSub, mouseLeaveSub, focusSub, blurSub, keyupEscSub);
113
103
  }
@@ -116,9 +106,8 @@ class TooltipDirective {
116
106
  this._listenerSubscriptions.length = 0;
117
107
  }
118
108
  _mountTooltip() {
119
- this._overlayRef = this._createOverlay();
120
- const injector = Injector.create({
121
- parent: this._injector,
109
+ this._animatedOverlay.mount({
110
+ component: TooltipComponent,
122
111
  providers: [
123
112
  {
124
113
  provide: TOOLTIP_CONFIG,
@@ -137,70 +126,6 @@ class TooltipDirective {
137
126
  ],
138
127
  ],
139
128
  });
140
- this._portal = this._portal ?? new ComponentPortal(TooltipComponent, this._viewContainerRef, injector);
141
- this._tooltipRef = this._overlayRef.attach(this._portal);
142
- this._tooltipRef.instance._markForCheck();
143
- this._zone.runOutsideAngular(() => {
144
- if (!this._tooltipRef) {
145
- return;
146
- }
147
- this._popper = createPopper(this._hostElementRef.nativeElement, this._tooltipRef.location.nativeElement, {
148
- placement: this.placement,
149
- modifiers: [
150
- ...(this._defaultConfig.offset
151
- ? [
152
- {
153
- name: 'offset',
154
- options: {
155
- offset: this._defaultConfig.offset,
156
- },
157
- },
158
- ]
159
- : []),
160
- ...(this._defaultConfig.arrowPadding
161
- ? [
162
- {
163
- name: 'arrow',
164
- options: {
165
- padding: this._defaultConfig.arrowPadding,
166
- },
167
- },
168
- ]
169
- : []),
170
- ],
171
- });
172
- // We need to wait for the tooltip content to be rendered
173
- nextFrame(() => {
174
- if (!this._tooltipRef) {
175
- return;
176
- }
177
- this._popper?.update();
178
- this._tooltipRef.instance._animatedLifecycle?.enter();
179
- });
180
- });
181
- }
182
- _animateUnmount() {
183
- if (!this._tooltipRef) {
184
- return;
185
- }
186
- this._tooltipRef.instance._animatedLifecycle?.leave();
187
- this._tooltipRef.instance._animatedLifecycle?.state$
188
- .pipe(filter((s) => s === 'left'), take(1))
189
- .subscribe(() => this._unmountTooltip());
190
- }
191
- _unmountTooltip() {
192
- this._zone.runOutsideAngular(() => {
193
- this._popper?.destroy();
194
- this._popper = null;
195
- });
196
- if (this._overlayRef) {
197
- this._overlayRef.dispose();
198
- this._overlayRef = null;
199
- }
200
- }
201
- _createOverlay() {
202
- const overlay = this._overlayService.create();
203
- return overlay;
204
129
  }
205
130
  _getTooltipText() {
206
131
  return this._tooltipAriaDescription
@@ -210,12 +135,12 @@ class TooltipDirective {
210
135
  : null;
211
136
  }
212
137
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: TooltipDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
213
- static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.0.0", type: TooltipDirective, isStandalone: true, selector: "[etTooltip]", inputs: { tooltip: ["etTooltip", "tooltip"], placement: "placement", tooltipAriaDescription: "tooltipAriaDescription" }, providers: [
138
+ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.0.0", type: TooltipDirective, isStandalone: true, selector: "[etTooltip]", inputs: { tooltip: ["etTooltip", "tooltip"], tooltipAriaDescription: "tooltipAriaDescription" }, providers: [
214
139
  {
215
140
  provide: TOOLTIP_DIRECTIVE,
216
141
  useExisting: TooltipDirective,
217
142
  },
218
- ], ngImport: i0 }); }
143
+ ], hostDirectives: [{ directive: i1.AnimatedOverlayDirective, inputs: ["placement", "placement"] }], ngImport: i0 }); }
219
144
  }
220
145
  export { TooltipDirective };
221
146
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: TooltipDirective, decorators: [{
@@ -229,13 +154,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImpor
229
154
  useExisting: TooltipDirective,
230
155
  },
231
156
  ],
157
+ hostDirectives: [{ directive: AnimatedOverlayDirective, inputs: ['placement'] }],
232
158
  }]
233
- }], propDecorators: { tooltip: [{
159
+ }], ctorParameters: function () { return []; }, propDecorators: { tooltip: [{
234
160
  type: Input,
235
161
  args: ['etTooltip']
236
- }], placement: [{
237
- type: Input
238
162
  }], tooltipAriaDescription: [{
239
163
  type: Input
240
164
  }] } });
241
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tooltip.directive.js","sourceRoot":"","sources":["../../../../../../../../../../../libs/cdk/src/lib/components/overlay/components/tooltip/directives/tooltip/tooltip.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { AriaDescriber } from '@angular/cdk/a11y';\nimport { Overlay, OverlayRef } from '@angular/cdk/overlay';\nimport { ComponentPortal } from '@angular/cdk/portal';\nimport {\n  ComponentRef,\n  Directive,\n  ElementRef,\n  inject,\n  InjectionToken,\n  Injector,\n  Input,\n  NgZone,\n  OnDestroy,\n  TemplateRef,\n  ViewContainerRef,\n} from '@angular/core';\nimport { FocusVisibleService, nextFrame } from '@ethlete/core';\nimport { createPopper, Instance as PopperInstance, Placement as PopperPlacement } from '@popperjs/core';\nimport { debounceTime, filter, fromEvent, Subscription, take, tap } from 'rxjs';\nimport { TooltipComponent } from '../../components';\nimport { TOOLTIP_CONFIG, TOOLTIP_TEMPLATE, TOOLTIP_TEXT } from '../../constants';\nimport { TooltipConfig } from '../../types';\nimport { createTooltipConfig } from '../../utils';\n\ntype TooltipTemplate = string | TemplateRef<unknown>;\n\nexport const TOOLTIP_DIRECTIVE = new InjectionToken<TooltipDirective>('TOOLTIP_DIRECTIVE');\n\n@Directive({\n  selector: '[etTooltip]',\n  standalone: true,\n  providers: [\n    {\n      provide: TOOLTIP_DIRECTIVE,\n      useExisting: TooltipDirective,\n    },\n  ],\n})\nexport class TooltipDirective implements OnDestroy {\n  private _defaultConfig = inject<TooltipConfig>(TOOLTIP_CONFIG, { optional: true }) ?? createTooltipConfig();\n\n  @Input('etTooltip')\n  get tooltip() {\n    return this._tooltip;\n  }\n  set tooltip(v: TooltipTemplate | null) {\n    this._tooltip = v;\n\n    this._updateAriaDescription();\n\n    if (v) {\n      this._addListeners();\n    } else {\n      this._removeListeners();\n    }\n  }\n  private _tooltip: TooltipTemplate | null = null;\n\n  @Input()\n  placement: PopperPlacement = this._defaultConfig.placement;\n\n  @Input()\n  get tooltipAriaDescription() {\n    return this._tooltipAriaDescription;\n  }\n  set tooltipAriaDescription(v: string | null) {\n    this._tooltipAriaDescription = v;\n    this._updateAriaDescription();\n  }\n  private _tooltipAriaDescription: string | null = null;\n  private _lastTooltipText: string | null = null;\n\n  private _hostElementRef = inject<ElementRef<HTMLElement>>(ElementRef);\n  private _viewContainerRef = inject(ViewContainerRef);\n  private _overlayService = inject(Overlay);\n  private _ariaDescriberService = inject(AriaDescriber);\n  private _focusVisibleService = inject(FocusVisibleService);\n  private _injector = inject(Injector);\n  private _zone = inject(NgZone);\n\n  private _overlayRef: OverlayRef | null = null;\n  private _portal: ComponentPortal<TooltipComponent> | null = null;\n  private _tooltipRef: ComponentRef<TooltipComponent> | null = null;\n  private _popper: PopperInstance | null = null;\n\n  private _willMount = true;\n  private _hasFocus = false;\n  private _hasHover = false;\n\n  private get _isMounted() {\n    return this._overlayRef !== null;\n  }\n\n  private readonly _listenerSubscriptions: Subscription[] = [];\n\n  ngOnDestroy(): void {\n    this._unmountTooltip();\n    this._removeListeners();\n  }\n\n  private _updateAriaDescription() {\n    const tooltipText = this._getTooltipText();\n\n    if (tooltipText) {\n      this._ariaDescriberService.describe(this._hostElementRef.nativeElement, tooltipText);\n    } else if (this._lastTooltipText) {\n      this._ariaDescriberService.removeDescription(this._hostElementRef.nativeElement, this._lastTooltipText);\n    }\n\n    this._lastTooltipText = tooltipText;\n  }\n\n  private _addListeners() {\n    const mouseEnterSub = fromEvent(this._hostElementRef.nativeElement, 'mouseenter')\n      .pipe(\n        tap(() => {\n          this._willMount = true;\n          this._hasHover = true;\n        }),\n        debounceTime(200),\n      )\n      .subscribe(() => {\n        if (!this._willMount || this._isMounted) {\n          return;\n        }\n\n        this._mountTooltip();\n      });\n\n    const focusSub = fromEvent(this._hostElementRef.nativeElement, 'focus').subscribe(() => {\n      if (!this._focusVisibleService.isFocusVisible) {\n        return;\n      }\n\n      this._hasFocus = true;\n\n      if (this._isMounted) {\n        return;\n      }\n\n      this._mountTooltip();\n    });\n\n    const mouseLeaveSub = fromEvent(this._hostElementRef.nativeElement, 'mouseleave').subscribe(() => {\n      this._hasHover = false;\n      this._willMount = false;\n\n      if (this._isMounted && !this._hasFocus) {\n        this._animateUnmount();\n      }\n    });\n\n    const blurSub = fromEvent(this._hostElementRef.nativeElement, 'blur').subscribe(() => {\n      this._hasFocus = false;\n      this._willMount = false;\n\n      if (this._isMounted && !this._hasHover) {\n        this._animateUnmount();\n      }\n    });\n\n    const keyupEscSub = fromEvent<KeyboardEvent>(document, 'keyup')\n      .pipe(\n        filter((e) => e.key === 'Escape'),\n        filter(() => this._isMounted),\n        tap(() => this._animateUnmount()),\n      )\n      .subscribe();\n\n    this._listenerSubscriptions.push(mouseEnterSub, mouseLeaveSub, focusSub, blurSub, keyupEscSub);\n  }\n\n  private _removeListeners() {\n    this._listenerSubscriptions.forEach((s) => s.unsubscribe());\n    this._listenerSubscriptions.length = 0;\n  }\n\n  private _mountTooltip() {\n    this._overlayRef = this._createOverlay();\n\n    const injector = Injector.create({\n      parent: this._injector,\n      providers: [\n        {\n          provide: TOOLTIP_CONFIG,\n          useValue: this._defaultConfig,\n        },\n        ...[\n          typeof this.tooltip === 'string'\n            ? {\n                provide: TOOLTIP_TEXT,\n                useValue: this.tooltip,\n              }\n            : {\n                provide: TOOLTIP_TEMPLATE,\n                useValue: this.tooltip,\n              },\n        ],\n      ],\n    });\n\n    this._portal = this._portal ?? new ComponentPortal(TooltipComponent, this._viewContainerRef, injector);\n    this._tooltipRef = this._overlayRef.attach(this._portal);\n\n    this._tooltipRef.instance._markForCheck();\n\n    this._zone.runOutsideAngular(() => {\n      if (!this._tooltipRef) {\n        return;\n      }\n      this._popper = createPopper(this._hostElementRef.nativeElement, this._tooltipRef.location.nativeElement, {\n        placement: this.placement,\n        modifiers: [\n          ...(this._defaultConfig.offset\n            ? [\n                {\n                  name: 'offset',\n                  options: {\n                    offset: this._defaultConfig.offset,\n                  },\n                },\n              ]\n            : []),\n          ...(this._defaultConfig.arrowPadding\n            ? [\n                {\n                  name: 'arrow',\n                  options: {\n                    padding: this._defaultConfig.arrowPadding,\n                  },\n                },\n              ]\n            : []),\n        ],\n      });\n\n      // We need to wait for the tooltip content to be rendered\n      nextFrame(() => {\n        if (!this._tooltipRef) {\n          return;\n        }\n\n        this._popper?.update();\n        this._tooltipRef.instance._animatedLifecycle?.enter();\n      });\n    });\n  }\n\n  private _animateUnmount() {\n    if (!this._tooltipRef) {\n      return;\n    }\n\n    this._tooltipRef.instance._animatedLifecycle?.leave();\n\n    this._tooltipRef.instance._animatedLifecycle?.state$\n      .pipe(\n        filter((s) => s === 'left'),\n        take(1),\n      )\n      .subscribe(() => this._unmountTooltip());\n  }\n\n  private _unmountTooltip() {\n    this._zone.runOutsideAngular(() => {\n      this._popper?.destroy();\n      this._popper = null;\n    });\n\n    if (this._overlayRef) {\n      this._overlayRef.dispose();\n      this._overlayRef = null;\n    }\n  }\n\n  private _createOverlay() {\n    const overlay = this._overlayService.create();\n\n    return overlay;\n  }\n\n  private _getTooltipText() {\n    return this._tooltipAriaDescription\n      ? this._tooltipAriaDescription\n      : typeof this.tooltip === 'string'\n      ? this.tooltip\n      : null;\n  }\n}\n"]}
165
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tooltip.directive.js","sourceRoot":"","sources":["../../../../../../../../../../../libs/cdk/src/lib/components/overlay/components/tooltip/directives/tooltip/tooltip.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { AriaDescriber } from '@angular/cdk/a11y';\nimport { Directive, ElementRef, InjectionToken, Input, OnDestroy, TemplateRef, inject } from '@angular/core';\nimport { AnimatedOverlayDirective, FocusVisibleService } from '@ethlete/core';\nimport { Subscription, debounceTime, filter, fromEvent, tap } from 'rxjs';\nimport { TooltipComponent } from '../../components';\nimport { TOOLTIP_CONFIG, TOOLTIP_TEMPLATE, TOOLTIP_TEXT } from '../../constants';\nimport { TooltipConfig } from '../../types';\nimport { createTooltipConfig } from '../../utils';\n\ntype TooltipTemplate = string | TemplateRef<unknown>;\n\nexport const TOOLTIP_DIRECTIVE = new InjectionToken<TooltipDirective>('TOOLTIP_DIRECTIVE');\n\n@Directive({\n  selector: '[etTooltip]',\n  standalone: true,\n  providers: [\n    {\n      provide: TOOLTIP_DIRECTIVE,\n      useExisting: TooltipDirective,\n    },\n  ],\n  hostDirectives: [{ directive: AnimatedOverlayDirective, inputs: ['placement'] }],\n})\nexport class TooltipDirective implements OnDestroy {\n  private readonly _defaultConfig = inject<TooltipConfig>(TOOLTIP_CONFIG, { optional: true }) ?? createTooltipConfig();\n  private readonly _animatedOverlay = inject<AnimatedOverlayDirective<TooltipComponent>>(AnimatedOverlayDirective);\n\n  @Input('etTooltip')\n  get tooltip() {\n    return this._tooltip;\n  }\n  set tooltip(v: TooltipTemplate | null) {\n    this._tooltip = v;\n\n    this._updateAriaDescription();\n\n    if (v) {\n      this._addListeners();\n    } else {\n      this._removeListeners();\n    }\n  }\n  private _tooltip: TooltipTemplate | null = null;\n\n  @Input()\n  get tooltipAriaDescription() {\n    return this._tooltipAriaDescription;\n  }\n  set tooltipAriaDescription(v: string | null) {\n    this._tooltipAriaDescription = v;\n    this._updateAriaDescription();\n  }\n  private _tooltipAriaDescription: string | null = null;\n  private _lastTooltipText: string | null = null;\n\n  private _elementRef = inject<ElementRef<HTMLElement>>(ElementRef);\n  private _ariaDescriberService = inject(AriaDescriber);\n  private _focusVisibleService = inject(FocusVisibleService);\n\n  private _willMount = true;\n  private _hasFocus = false;\n  private _hasHover = false;\n\n  private readonly _listenerSubscriptions: Subscription[] = [];\n\n  constructor() {\n    this._animatedOverlay.placement = this._defaultConfig.placement;\n    this._animatedOverlay.offset = this._defaultConfig.offset;\n    this._animatedOverlay.arrowPadding = this._defaultConfig.arrowPadding;\n  }\n\n  ngOnDestroy(): void {\n    this._animatedOverlay._destroy();\n    this._removeListeners();\n  }\n\n  private _updateAriaDescription() {\n    const tooltipText = this._getTooltipText();\n\n    if (tooltipText) {\n      this._ariaDescriberService.describe(this._elementRef.nativeElement, tooltipText);\n    } else if (this._lastTooltipText) {\n      this._ariaDescriberService.removeDescription(this._elementRef.nativeElement, this._lastTooltipText);\n    }\n\n    this._lastTooltipText = tooltipText;\n  }\n\n  private _addListeners() {\n    const mouseEnterSub = fromEvent(this._elementRef.nativeElement, 'mouseenter')\n      .pipe(\n        tap(() => {\n          this._willMount = true;\n          this._hasHover = true;\n        }),\n        debounceTime(200),\n      )\n      .subscribe(() => {\n        if (!this._willMount || this._animatedOverlay.isMounted) {\n          return;\n        }\n\n        this._mountTooltip();\n      });\n\n    const focusSub = fromEvent(this._elementRef.nativeElement, 'focus').subscribe(() => {\n      if (!this._focusVisibleService.isFocusVisible) {\n        return;\n      }\n\n      this._hasFocus = true;\n\n      if (this._animatedOverlay.isMounted) {\n        return;\n      }\n\n      this._mountTooltip();\n    });\n\n    const mouseLeaveSub = fromEvent(this._elementRef.nativeElement, 'mouseleave').subscribe(() => {\n      this._hasHover = false;\n      this._willMount = false;\n\n      if (this._animatedOverlay.isMounted && !this._hasFocus) {\n        this._animatedOverlay.unmount();\n      }\n    });\n\n    const blurSub = fromEvent(this._elementRef.nativeElement, 'blur').subscribe(() => {\n      this._hasFocus = false;\n      this._willMount = false;\n\n      if (this._animatedOverlay.isMounted && !this._hasHover) {\n        this._animatedOverlay.unmount();\n      }\n    });\n\n    const keyupEscSub = fromEvent<KeyboardEvent>(document, 'keyup')\n      .pipe(\n        filter((e) => e.key === 'Escape'),\n        filter(() => this._animatedOverlay.isMounted),\n        tap(() => this._animatedOverlay.unmount()),\n      )\n      .subscribe();\n\n    this._listenerSubscriptions.push(mouseEnterSub, mouseLeaveSub, focusSub, blurSub, keyupEscSub);\n  }\n\n  private _removeListeners() {\n    this._listenerSubscriptions.forEach((s) => s.unsubscribe());\n    this._listenerSubscriptions.length = 0;\n  }\n\n  private _mountTooltip() {\n    this._animatedOverlay.mount({\n      component: TooltipComponent,\n      providers: [\n        {\n          provide: TOOLTIP_CONFIG,\n          useValue: this._defaultConfig,\n        },\n        ...[\n          typeof this.tooltip === 'string'\n            ? {\n                provide: TOOLTIP_TEXT,\n                useValue: this.tooltip,\n              }\n            : {\n                provide: TOOLTIP_TEMPLATE,\n                useValue: this.tooltip,\n              },\n        ],\n      ],\n    });\n  }\n\n  private _getTooltipText() {\n    return this._tooltipAriaDescription\n      ? this._tooltipAriaDescription\n      : typeof this.tooltip === 'string'\n      ? this.tooltip\n      : null;\n  }\n}\n"]}
@@ -1,13 +1,13 @@
1
- import { coerceBooleanProperty } from '@angular/cdk/coercion';
1
+ import { coerceBooleanProperty, coerceNumberProperty } from '@angular/cdk/coercion';
2
2
  import { NgClass, NgIf } from '@angular/common';
3
- import { ChangeDetectionStrategy, Component, ElementRef, HostBinding, inject, Input, Renderer2, ViewChild, ViewEncapsulation, } from '@angular/core';
4
- import { CursorDragScrollDirective, DestroyService, equal, LetDirective, ObserveScrollStateDirective, } from '@ethlete/core';
5
- import { BehaviorSubject, takeUntil, tap } from 'rxjs';
3
+ import { ChangeDetectionStrategy, Component, ContentChildren, ElementRef, HostBinding, Input, Renderer2, ViewChild, ViewEncapsulation, inject, } from '@angular/core';
4
+ import { CursorDragScrollDirective, IS_ACTIVE_ELEMENT, LetDirective, ObserveScrollStateDirective, createDestroy, equal, scrollToElement, } from '@ethlete/core';
5
+ import { BehaviorSubject, startWith, takeUntil, tap } from 'rxjs';
6
6
  import { ChevronIconComponent } from '../../../icons';
7
7
  import * as i0 from "@angular/core";
8
8
  class ScrollableComponent {
9
9
  constructor() {
10
- this._destroy$ = inject(DestroyService, { host: true }).destroy$;
10
+ this._destroy$ = createDestroy();
11
11
  this._renderer = inject(Renderer2);
12
12
  this._elementRef = inject(ElementRef);
13
13
  this.itemSize = 'auto';
@@ -17,6 +17,9 @@ class ScrollableComponent {
17
17
  this._renderScrollbars = false;
18
18
  this._stickyButtons = false;
19
19
  this._cursorDragScroll = true;
20
+ this._activeElementScrollMargin = 40;
21
+ this._disableActiveElementScrolling = false;
22
+ this.activeElements = null;
20
23
  this.scrollState$ = new BehaviorSubject(null);
21
24
  }
22
25
  get renderMasks() {
@@ -49,6 +52,18 @@ class ScrollableComponent {
49
52
  set cursorDragScroll(value) {
50
53
  this._cursorDragScroll = coerceBooleanProperty(value);
51
54
  }
55
+ get activeElementScrollMargin() {
56
+ return this._activeElementScrollMargin;
57
+ }
58
+ set activeElementScrollMargin(value) {
59
+ this._activeElementScrollMargin = coerceNumberProperty(value);
60
+ }
61
+ get disableActiveElementScrolling() {
62
+ return this._disableActiveElementScrolling;
63
+ }
64
+ set disableActiveElementScrolling(value) {
65
+ this._disableActiveElementScrolling = coerceBooleanProperty(value);
66
+ }
52
67
  ngOnInit() {
53
68
  this.scrollState$
54
69
  .pipe(tap((state) => {
@@ -62,6 +77,31 @@ class ScrollableComponent {
62
77
  }), takeUntil(this._destroy$))
63
78
  .subscribe();
64
79
  }
80
+ ngAfterContentInit() {
81
+ if (!this.activeElements) {
82
+ return;
83
+ }
84
+ this.activeElements.changes
85
+ .pipe(startWith(this.activeElements), tap((activeElements) => {
86
+ if (this.disableActiveElementScrolling) {
87
+ return;
88
+ }
89
+ const firstActive = activeElements
90
+ .filter((a) => !!a)
91
+ .find((a) => a.isActiveElement);
92
+ if (!firstActive) {
93
+ return;
94
+ }
95
+ scrollToElement({
96
+ behavior: 'auto',
97
+ container: this.scrollable.nativeElement,
98
+ element: firstActive.elementRef.nativeElement,
99
+ scrollInlineMargin: this.direction === 'horizontal' ? this.activeElementScrollMargin : 0,
100
+ scrollBlockMargin: this.direction === 'horizontal' ? 0 : this.activeElementScrollMargin,
101
+ });
102
+ }), takeUntil(this._destroy$))
103
+ .subscribe();
104
+ }
65
105
  _scrollStateChanged(scrollState) {
66
106
  if (equal(this.scrollState$.value, scrollState)) {
67
107
  return;
@@ -85,14 +125,14 @@ class ScrollableComponent {
85
125
  });
86
126
  }
87
127
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: ScrollableComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
88
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.0.0", type: ScrollableComponent, isStandalone: true, selector: "et-scrollable", inputs: { itemSize: "itemSize", direction: "direction", scrollableRole: "scrollableRole", scrollableClass: "scrollableClass", renderMasks: "renderMasks", renderButtons: "renderButtons", renderScrollbars: "renderScrollbars", stickyButtons: "stickyButtons", cursorDragScroll: "cursorDragScroll" }, host: { properties: { "attr.item-size": "this.itemSize", "attr.direction": "this.direction", "attr.render-scrollbars": "this.renderScrollbars", "attr.sticky-buttons": "this.stickyButtons" }, classAttribute: "et-scrollable" }, providers: [DestroyService], viewQueries: [{ propertyName: "scrollable", first: true, predicate: ["scrollable"], descendants: true, static: true }], ngImport: i0, template: "<div\n #scrollable\n [attr.role]=\"scrollableRole ?? undefined\"\n [ngClass]=\"scrollableClass\"\n [etCursorDragScroll]=\"cursorDragScroll\"\n (etObserveScrollState)=\"_scrollStateChanged($event)\"\n class=\"et-scrollable-container\"\n>\n <ng-content />\n</div>\n\n<div *ngIf=\"renderMasks\" class=\"et-scrollable-masks\">\n <div class=\"et-scrollable-mask et-scrollable-mask--start\"></div>\n <div class=\"et-scrollable-mask et-scrollable-mask--end\"></div>\n</div>\n\n<div *ngIf=\"renderButtons\" class=\"et-scrollable-buttons\">\n <button\n (click)=\"scrollOneContainerSizeToStart()\"\n class=\"et-scrollable-button et-scrollable-button--start\"\n aria-hidden=\"true\"\n type=\"button\"\n tabindex=\"-1\"\n >\n <et-chevron-icon />\n </button>\n <button\n (click)=\"scrollOneContainerSizeToEnd()\"\n class=\"et-scrollable-button et-scrollable-button--end\"\n aria-hidden=\"true\"\n type=\"button\"\n tabindex=\"-1\"\n >\n <et-chevron-icon />\n </button>\n</div>\n", styles: [".et-scrollable{--mask: #121212 0, transparent 100%;--mask-size: 25px;--_auto-size: min-content;--_flow: column;display:grid;position:relative}.et-scrollable[item-size=same]{--auto-size: 1fr}.et-scrollable[render-scrollbars=false] .et-scrollable-container{scrollbar-width:none}.et-scrollable[render-scrollbars=false] .et-scrollable-container::-webkit-scrollbar{display:none}.et-scrollable[direction=horizontal] .et-scrollable-container{grid-auto-columns:var(--_auto-size)}.et-scrollable[direction=horizontal] .et-scrollable-mask,.et-scrollable[direction=horizontal] .et-scrollable-button{inline-size:var(--mask-size);block-size:100%}.et-scrollable[direction=horizontal] .et-scrollable-mask--start,.et-scrollable[direction=horizontal] .et-scrollable-button--start{inset-block-start:0;inset-inline:0 0}.et-scrollable[direction=horizontal] .et-scrollable-mask--end,.et-scrollable[direction=horizontal] .et-scrollable-button--end{inset-block-start:0;inset-inline:calc(100% - 40px) 0}.et-scrollable[direction=horizontal] .et-scrollable-mask--start{background:linear-gradient(to right,var(--mask))}.et-scrollable[direction=horizontal] .et-scrollable-mask--end{background:linear-gradient(to left,var(--mask));inset-inline:calc(100% - var(--mask-size)) 100%}.et-scrollable[direction=horizontal] .et-scrollable-button--start{inset-block-start:calc(50% - 20px);transform:rotate(-90deg)}.et-scrollable[direction=horizontal] .et-scrollable-button--end{inset-block-start:calc(50% - 20px);transform:rotate(90deg)}.et-scrollable[direction=vertical]{--_flow: row}.et-scrollable[direction=vertical] .et-scrollable-container{grid-auto-rows:var(--_auto-size)}.et-scrollable[direction=vertical] .et-scrollable-mask,.et-scrollable[direction=vertical] .et-scrollable-button{block-size:var(--mask-size);inline-size:100%}.et-scrollable[direction=vertical] .et-scrollable-mask--start,.et-scrollable[direction=vertical] .et-scrollable-button--start{inset-block-start:0;inset-inline-start:0}.et-scrollable[direction=vertical] .et-scrollable-mask--end,.et-scrollable[direction=vertical] .et-scrollable-button--end{inset-block-end:0;inset-inline-start:0}.et-scrollable[direction=vertical] .et-scrollable-mask--start{background:linear-gradient(to bottom,var(--mask))}.et-scrollable[direction=vertical] .et-scrollable-mask--end{background:linear-gradient(to top,var(--mask))}.et-scrollable[at-start=false] .et-scrollable-masks .et-scrollable-mask--start,.et-scrollable[at-start=false] .et-scrollable-buttons .et-scrollable-button--start{opacity:1}.et-scrollable[at-start=false] .et-scrollable-buttons .et-scrollable-button--start{pointer-events:all}.et-scrollable[at-end=false] .et-scrollable-masks .et-scrollable-mask--end,.et-scrollable[at-end=false] .et-scrollable-buttons .et-scrollable-button--end{opacity:1}.et-scrollable[at-end=false] .et-scrollable-buttons .et-scrollable-button--end{pointer-events:all}.et-scrollable .et-scrollable-container{display:grid;grid-auto-flow:var(--_flow);overflow:auto;grid-row:1/1;grid-column:1/1}.et-scrollable .et-scrollable-masks,.et-scrollable .et-scrollable-buttons{grid-row:1/1;grid-column:1/1;pointer-events:none}.et-scrollable .et-scrollable-masks .et-scrollable-mask,.et-scrollable .et-scrollable-masks .et-scrollable-button,.et-scrollable .et-scrollable-buttons .et-scrollable-mask,.et-scrollable .et-scrollable-buttons .et-scrollable-button{position:absolute;opacity:0;transition:opacity .3s var(--ease-5)}.et-scrollable .et-scrollable-buttons .et-scrollable-button{background:transparent;border:none;padding:12px;inline-size:40px;block-size:40px;border-radius:4px;cursor:pointer}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-chevron-icon{pointer-events:none}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-scrollable-button--start{inset-inline:0 0}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-scrollable-button--end{inset-inline:calc(100% - 40px) 0}.et-scrollable[sticky-buttons=true] .et-scrollable-buttons{margin-block:10%}.et-scrollable[sticky-buttons=true] .et-scrollable-buttons .et-scrollable-button{position:sticky}\n"], dependencies: [{ kind: "directive", type: CursorDragScrollDirective, selector: "[etCursorDragScroll]", inputs: ["etCursorDragScroll"], exportAs: ["etCursorDragScroll"] }, { kind: "directive", type: ObserveScrollStateDirective, selector: "[etObserveScrollState]", inputs: ["observerRootMargin", "observerThreshold"], outputs: ["etObserveScrollState"], exportAs: ["etObserveScrollState"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: ChevronIconComponent, selector: "et-chevron-icon" }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
128
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.0.0", type: ScrollableComponent, isStandalone: true, selector: "et-scrollable", inputs: { itemSize: "itemSize", direction: "direction", scrollableRole: "scrollableRole", scrollableClass: "scrollableClass", renderMasks: "renderMasks", renderButtons: "renderButtons", renderScrollbars: "renderScrollbars", stickyButtons: "stickyButtons", cursorDragScroll: "cursorDragScroll", activeElementScrollMargin: "activeElementScrollMargin", disableActiveElementScrolling: "disableActiveElementScrolling" }, host: { properties: { "attr.item-size": "this.itemSize", "attr.direction": "this.direction", "attr.render-scrollbars": "this.renderScrollbars", "attr.sticky-buttons": "this.stickyButtons" }, classAttribute: "et-scrollable" }, queries: [{ propertyName: "activeElements", predicate: IS_ACTIVE_ELEMENT, descendants: true }], viewQueries: [{ propertyName: "scrollable", first: true, predicate: ["scrollable"], descendants: true, static: true }], ngImport: i0, template: "<div\n #scrollable\n [attr.role]=\"scrollableRole ?? undefined\"\n [ngClass]=\"scrollableClass\"\n [etCursorDragScroll]=\"cursorDragScroll\"\n (etObserveScrollState)=\"_scrollStateChanged($event)\"\n class=\"et-scrollable-container\"\n>\n <ng-content />\n</div>\n\n<div *ngIf=\"renderMasks\" class=\"et-scrollable-masks\">\n <div class=\"et-scrollable-mask et-scrollable-mask--start\"></div>\n <div class=\"et-scrollable-mask et-scrollable-mask--end\"></div>\n</div>\n\n<div *ngIf=\"renderButtons\" class=\"et-scrollable-buttons\">\n <button\n (click)=\"scrollOneContainerSizeToStart()\"\n class=\"et-scrollable-button et-scrollable-button--start\"\n aria-hidden=\"true\"\n type=\"button\"\n tabindex=\"-1\"\n >\n <et-chevron-icon />\n </button>\n <button\n (click)=\"scrollOneContainerSizeToEnd()\"\n class=\"et-scrollable-button et-scrollable-button--end\"\n aria-hidden=\"true\"\n type=\"button\"\n tabindex=\"-1\"\n >\n <et-chevron-icon />\n </button>\n</div>\n", styles: [".et-scrollable{--mask: #121212 0, transparent 100%;--mask-size: 25px;--_auto-size: min-content;--_flow: column;display:grid;position:relative}.et-scrollable[item-size=same]{--auto-size: 1fr}.et-scrollable[render-scrollbars=false] .et-scrollable-container{scrollbar-width:none}.et-scrollable[render-scrollbars=false] .et-scrollable-container::-webkit-scrollbar{display:none}.et-scrollable[direction=horizontal] .et-scrollable-container{grid-auto-columns:var(--_auto-size)}.et-scrollable[direction=horizontal] .et-scrollable-mask,.et-scrollable[direction=horizontal] .et-scrollable-button{inline-size:var(--mask-size);block-size:100%}.et-scrollable[direction=horizontal] .et-scrollable-mask--start,.et-scrollable[direction=horizontal] .et-scrollable-button--start{inset-block-start:0;inset-inline:0 0}.et-scrollable[direction=horizontal] .et-scrollable-mask--end,.et-scrollable[direction=horizontal] .et-scrollable-button--end{inset-block-start:0;inset-inline:calc(100% - 40px) 0}.et-scrollable[direction=horizontal] .et-scrollable-mask--start{background:linear-gradient(to right,var(--mask))}.et-scrollable[direction=horizontal] .et-scrollable-mask--end{background:linear-gradient(to left,var(--mask));inset-inline:calc(100% - var(--mask-size)) 100%}.et-scrollable[direction=horizontal] .et-scrollable-button--start{inset-block-start:calc(50% - 20px);transform:rotate(-90deg)}.et-scrollable[direction=horizontal] .et-scrollable-button--end{inset-block-start:calc(50% - 20px);transform:rotate(90deg)}.et-scrollable[direction=vertical]{--_flow: row}.et-scrollable[direction=vertical] .et-scrollable-container{grid-auto-rows:var(--_auto-size)}.et-scrollable[direction=vertical] .et-scrollable-mask,.et-scrollable[direction=vertical] .et-scrollable-button{block-size:var(--mask-size);inline-size:100%}.et-scrollable[direction=vertical] .et-scrollable-mask--start,.et-scrollable[direction=vertical] .et-scrollable-button--start{inset-block-start:0;inset-inline-start:0}.et-scrollable[direction=vertical] .et-scrollable-mask--end,.et-scrollable[direction=vertical] .et-scrollable-button--end{inset-block-end:0;inset-inline-start:0}.et-scrollable[direction=vertical] .et-scrollable-mask--start{background:linear-gradient(to bottom,var(--mask))}.et-scrollable[direction=vertical] .et-scrollable-mask--end{background:linear-gradient(to top,var(--mask))}.et-scrollable[at-start=false] .et-scrollable-masks .et-scrollable-mask--start,.et-scrollable[at-start=false] .et-scrollable-buttons .et-scrollable-button--start{opacity:1}.et-scrollable[at-start=false] .et-scrollable-buttons .et-scrollable-button--start{pointer-events:all}.et-scrollable[at-end=false] .et-scrollable-masks .et-scrollable-mask--end,.et-scrollable[at-end=false] .et-scrollable-buttons .et-scrollable-button--end{opacity:1}.et-scrollable[at-end=false] .et-scrollable-buttons .et-scrollable-button--end{pointer-events:all}.et-scrollable .et-scrollable-container{display:grid;grid-auto-flow:var(--_flow);overflow:auto;grid-row:1/1;grid-column:1/1}.et-scrollable .et-scrollable-masks,.et-scrollable .et-scrollable-buttons{grid-row:1/1;grid-column:1/1;pointer-events:none}.et-scrollable .et-scrollable-masks .et-scrollable-mask,.et-scrollable .et-scrollable-masks .et-scrollable-button,.et-scrollable .et-scrollable-buttons .et-scrollable-mask,.et-scrollable .et-scrollable-buttons .et-scrollable-button{position:absolute;opacity:0;transition:opacity .3s var(--ease-5)}.et-scrollable .et-scrollable-buttons .et-scrollable-button{background:transparent;border:none;padding:12px;inline-size:40px;block-size:40px;border-radius:4px;cursor:pointer}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-chevron-icon{pointer-events:none}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-scrollable-button--start{inset-inline:0 0}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-scrollable-button--end{inset-inline:calc(100% - 40px) 0}.et-scrollable[sticky-buttons=true] .et-scrollable-buttons{margin-block:10%}.et-scrollable[sticky-buttons=true] .et-scrollable-buttons .et-scrollable-button{position:sticky}\n"], dependencies: [{ kind: "directive", type: CursorDragScrollDirective, selector: "[etCursorDragScroll]", inputs: ["etCursorDragScroll"], exportAs: ["etCursorDragScroll"] }, { kind: "directive", type: ObserveScrollStateDirective, selector: "[etObserveScrollState]", inputs: ["observerRootMargin", "observerThreshold"], outputs: ["etObserveScrollState"], exportAs: ["etObserveScrollState"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: ChevronIconComponent, selector: "et-chevron-icon" }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
89
129
  }
90
130
  export { ScrollableComponent };
91
131
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: ScrollableComponent, decorators: [{
92
132
  type: Component,
93
133
  args: [{ selector: 'et-scrollable', standalone: true, encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, imports: [CursorDragScrollDirective, ObserveScrollStateDirective, NgClass, NgIf, LetDirective, ChevronIconComponent], host: {
94
134
  class: 'et-scrollable',
95
- }, providers: [DestroyService], template: "<div\n #scrollable\n [attr.role]=\"scrollableRole ?? undefined\"\n [ngClass]=\"scrollableClass\"\n [etCursorDragScroll]=\"cursorDragScroll\"\n (etObserveScrollState)=\"_scrollStateChanged($event)\"\n class=\"et-scrollable-container\"\n>\n <ng-content />\n</div>\n\n<div *ngIf=\"renderMasks\" class=\"et-scrollable-masks\">\n <div class=\"et-scrollable-mask et-scrollable-mask--start\"></div>\n <div class=\"et-scrollable-mask et-scrollable-mask--end\"></div>\n</div>\n\n<div *ngIf=\"renderButtons\" class=\"et-scrollable-buttons\">\n <button\n (click)=\"scrollOneContainerSizeToStart()\"\n class=\"et-scrollable-button et-scrollable-button--start\"\n aria-hidden=\"true\"\n type=\"button\"\n tabindex=\"-1\"\n >\n <et-chevron-icon />\n </button>\n <button\n (click)=\"scrollOneContainerSizeToEnd()\"\n class=\"et-scrollable-button et-scrollable-button--end\"\n aria-hidden=\"true\"\n type=\"button\"\n tabindex=\"-1\"\n >\n <et-chevron-icon />\n </button>\n</div>\n", styles: [".et-scrollable{--mask: #121212 0, transparent 100%;--mask-size: 25px;--_auto-size: min-content;--_flow: column;display:grid;position:relative}.et-scrollable[item-size=same]{--auto-size: 1fr}.et-scrollable[render-scrollbars=false] .et-scrollable-container{scrollbar-width:none}.et-scrollable[render-scrollbars=false] .et-scrollable-container::-webkit-scrollbar{display:none}.et-scrollable[direction=horizontal] .et-scrollable-container{grid-auto-columns:var(--_auto-size)}.et-scrollable[direction=horizontal] .et-scrollable-mask,.et-scrollable[direction=horizontal] .et-scrollable-button{inline-size:var(--mask-size);block-size:100%}.et-scrollable[direction=horizontal] .et-scrollable-mask--start,.et-scrollable[direction=horizontal] .et-scrollable-button--start{inset-block-start:0;inset-inline:0 0}.et-scrollable[direction=horizontal] .et-scrollable-mask--end,.et-scrollable[direction=horizontal] .et-scrollable-button--end{inset-block-start:0;inset-inline:calc(100% - 40px) 0}.et-scrollable[direction=horizontal] .et-scrollable-mask--start{background:linear-gradient(to right,var(--mask))}.et-scrollable[direction=horizontal] .et-scrollable-mask--end{background:linear-gradient(to left,var(--mask));inset-inline:calc(100% - var(--mask-size)) 100%}.et-scrollable[direction=horizontal] .et-scrollable-button--start{inset-block-start:calc(50% - 20px);transform:rotate(-90deg)}.et-scrollable[direction=horizontal] .et-scrollable-button--end{inset-block-start:calc(50% - 20px);transform:rotate(90deg)}.et-scrollable[direction=vertical]{--_flow: row}.et-scrollable[direction=vertical] .et-scrollable-container{grid-auto-rows:var(--_auto-size)}.et-scrollable[direction=vertical] .et-scrollable-mask,.et-scrollable[direction=vertical] .et-scrollable-button{block-size:var(--mask-size);inline-size:100%}.et-scrollable[direction=vertical] .et-scrollable-mask--start,.et-scrollable[direction=vertical] .et-scrollable-button--start{inset-block-start:0;inset-inline-start:0}.et-scrollable[direction=vertical] .et-scrollable-mask--end,.et-scrollable[direction=vertical] .et-scrollable-button--end{inset-block-end:0;inset-inline-start:0}.et-scrollable[direction=vertical] .et-scrollable-mask--start{background:linear-gradient(to bottom,var(--mask))}.et-scrollable[direction=vertical] .et-scrollable-mask--end{background:linear-gradient(to top,var(--mask))}.et-scrollable[at-start=false] .et-scrollable-masks .et-scrollable-mask--start,.et-scrollable[at-start=false] .et-scrollable-buttons .et-scrollable-button--start{opacity:1}.et-scrollable[at-start=false] .et-scrollable-buttons .et-scrollable-button--start{pointer-events:all}.et-scrollable[at-end=false] .et-scrollable-masks .et-scrollable-mask--end,.et-scrollable[at-end=false] .et-scrollable-buttons .et-scrollable-button--end{opacity:1}.et-scrollable[at-end=false] .et-scrollable-buttons .et-scrollable-button--end{pointer-events:all}.et-scrollable .et-scrollable-container{display:grid;grid-auto-flow:var(--_flow);overflow:auto;grid-row:1/1;grid-column:1/1}.et-scrollable .et-scrollable-masks,.et-scrollable .et-scrollable-buttons{grid-row:1/1;grid-column:1/1;pointer-events:none}.et-scrollable .et-scrollable-masks .et-scrollable-mask,.et-scrollable .et-scrollable-masks .et-scrollable-button,.et-scrollable .et-scrollable-buttons .et-scrollable-mask,.et-scrollable .et-scrollable-buttons .et-scrollable-button{position:absolute;opacity:0;transition:opacity .3s var(--ease-5)}.et-scrollable .et-scrollable-buttons .et-scrollable-button{background:transparent;border:none;padding:12px;inline-size:40px;block-size:40px;border-radius:4px;cursor:pointer}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-chevron-icon{pointer-events:none}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-scrollable-button--start{inset-inline:0 0}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-scrollable-button--end{inset-inline:calc(100% - 40px) 0}.et-scrollable[sticky-buttons=true] .et-scrollable-buttons{margin-block:10%}.et-scrollable[sticky-buttons=true] .et-scrollable-buttons .et-scrollable-button{position:sticky}\n"] }]
135
+ }, template: "<div\n #scrollable\n [attr.role]=\"scrollableRole ?? undefined\"\n [ngClass]=\"scrollableClass\"\n [etCursorDragScroll]=\"cursorDragScroll\"\n (etObserveScrollState)=\"_scrollStateChanged($event)\"\n class=\"et-scrollable-container\"\n>\n <ng-content />\n</div>\n\n<div *ngIf=\"renderMasks\" class=\"et-scrollable-masks\">\n <div class=\"et-scrollable-mask et-scrollable-mask--start\"></div>\n <div class=\"et-scrollable-mask et-scrollable-mask--end\"></div>\n</div>\n\n<div *ngIf=\"renderButtons\" class=\"et-scrollable-buttons\">\n <button\n (click)=\"scrollOneContainerSizeToStart()\"\n class=\"et-scrollable-button et-scrollable-button--start\"\n aria-hidden=\"true\"\n type=\"button\"\n tabindex=\"-1\"\n >\n <et-chevron-icon />\n </button>\n <button\n (click)=\"scrollOneContainerSizeToEnd()\"\n class=\"et-scrollable-button et-scrollable-button--end\"\n aria-hidden=\"true\"\n type=\"button\"\n tabindex=\"-1\"\n >\n <et-chevron-icon />\n </button>\n</div>\n", styles: [".et-scrollable{--mask: #121212 0, transparent 100%;--mask-size: 25px;--_auto-size: min-content;--_flow: column;display:grid;position:relative}.et-scrollable[item-size=same]{--auto-size: 1fr}.et-scrollable[render-scrollbars=false] .et-scrollable-container{scrollbar-width:none}.et-scrollable[render-scrollbars=false] .et-scrollable-container::-webkit-scrollbar{display:none}.et-scrollable[direction=horizontal] .et-scrollable-container{grid-auto-columns:var(--_auto-size)}.et-scrollable[direction=horizontal] .et-scrollable-mask,.et-scrollable[direction=horizontal] .et-scrollable-button{inline-size:var(--mask-size);block-size:100%}.et-scrollable[direction=horizontal] .et-scrollable-mask--start,.et-scrollable[direction=horizontal] .et-scrollable-button--start{inset-block-start:0;inset-inline:0 0}.et-scrollable[direction=horizontal] .et-scrollable-mask--end,.et-scrollable[direction=horizontal] .et-scrollable-button--end{inset-block-start:0;inset-inline:calc(100% - 40px) 0}.et-scrollable[direction=horizontal] .et-scrollable-mask--start{background:linear-gradient(to right,var(--mask))}.et-scrollable[direction=horizontal] .et-scrollable-mask--end{background:linear-gradient(to left,var(--mask));inset-inline:calc(100% - var(--mask-size)) 100%}.et-scrollable[direction=horizontal] .et-scrollable-button--start{inset-block-start:calc(50% - 20px);transform:rotate(-90deg)}.et-scrollable[direction=horizontal] .et-scrollable-button--end{inset-block-start:calc(50% - 20px);transform:rotate(90deg)}.et-scrollable[direction=vertical]{--_flow: row}.et-scrollable[direction=vertical] .et-scrollable-container{grid-auto-rows:var(--_auto-size)}.et-scrollable[direction=vertical] .et-scrollable-mask,.et-scrollable[direction=vertical] .et-scrollable-button{block-size:var(--mask-size);inline-size:100%}.et-scrollable[direction=vertical] .et-scrollable-mask--start,.et-scrollable[direction=vertical] .et-scrollable-button--start{inset-block-start:0;inset-inline-start:0}.et-scrollable[direction=vertical] .et-scrollable-mask--end,.et-scrollable[direction=vertical] .et-scrollable-button--end{inset-block-end:0;inset-inline-start:0}.et-scrollable[direction=vertical] .et-scrollable-mask--start{background:linear-gradient(to bottom,var(--mask))}.et-scrollable[direction=vertical] .et-scrollable-mask--end{background:linear-gradient(to top,var(--mask))}.et-scrollable[at-start=false] .et-scrollable-masks .et-scrollable-mask--start,.et-scrollable[at-start=false] .et-scrollable-buttons .et-scrollable-button--start{opacity:1}.et-scrollable[at-start=false] .et-scrollable-buttons .et-scrollable-button--start{pointer-events:all}.et-scrollable[at-end=false] .et-scrollable-masks .et-scrollable-mask--end,.et-scrollable[at-end=false] .et-scrollable-buttons .et-scrollable-button--end{opacity:1}.et-scrollable[at-end=false] .et-scrollable-buttons .et-scrollable-button--end{pointer-events:all}.et-scrollable .et-scrollable-container{display:grid;grid-auto-flow:var(--_flow);overflow:auto;grid-row:1/1;grid-column:1/1}.et-scrollable .et-scrollable-masks,.et-scrollable .et-scrollable-buttons{grid-row:1/1;grid-column:1/1;pointer-events:none}.et-scrollable .et-scrollable-masks .et-scrollable-mask,.et-scrollable .et-scrollable-masks .et-scrollable-button,.et-scrollable .et-scrollable-buttons .et-scrollable-mask,.et-scrollable .et-scrollable-buttons .et-scrollable-button{position:absolute;opacity:0;transition:opacity .3s var(--ease-5)}.et-scrollable .et-scrollable-buttons .et-scrollable-button{background:transparent;border:none;padding:12px;inline-size:40px;block-size:40px;border-radius:4px;cursor:pointer}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-chevron-icon{pointer-events:none}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-scrollable-button--start{inset-inline:0 0}.et-scrollable .et-scrollable-buttons .et-scrollable-button .et-scrollable-button--end{inset-inline:calc(100% - 40px) 0}.et-scrollable[sticky-buttons=true] .et-scrollable-buttons{margin-block:10%}.et-scrollable[sticky-buttons=true] .et-scrollable-buttons .et-scrollable-button{position:sticky}\n"] }]
96
136
  }], propDecorators: { itemSize: [{
97
137
  type: Input
98
138
  }, {
@@ -123,8 +163,15 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImpor
123
163
  args: ['attr.sticky-buttons']
124
164
  }], cursorDragScroll: [{
125
165
  type: Input
166
+ }], activeElementScrollMargin: [{
167
+ type: Input
168
+ }], disableActiveElementScrolling: [{
169
+ type: Input
126
170
  }], scrollable: [{
127
171
  type: ViewChild,
128
172
  args: ['scrollable', { static: true }]
173
+ }], activeElements: [{
174
+ type: ContentChildren,
175
+ args: [IS_ACTIVE_ELEMENT, { descendants: true }]
129
176
  }] } });
130
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scrollable.component.js","sourceRoot":"","sources":["../../../../../../../../../libs/cdk/src/lib/components/scrollable/components/scrollable/scrollable.component.ts","../../../../../../../../../libs/cdk/src/lib/components/scrollable/components/scrollable/scrollable.component.html"],"names":[],"mappings":"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","sourcesContent":["import { BooleanInput, coerceBooleanProperty } from '@angular/cdk/coercion';\nimport { NgClass, NgIf } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  HostBinding,\n  inject,\n  Input,\n  OnInit,\n  Renderer2,\n  ViewChild,\n  ViewEncapsulation,\n} from '@angular/core';\nimport {\n  CursorDragScrollDirective,\n  DestroyService,\n  equal,\n  LetDirective,\n  NgClassType,\n  ObserveScrollStateDirective,\n  ScrollObserverScrollState,\n} from '@ethlete/core';\nimport { BehaviorSubject, takeUntil, tap } from 'rxjs';\nimport { ChevronIconComponent } from '../../../icons';\n\n@Component({\n  selector: 'et-scrollable',\n  templateUrl: './scrollable.component.html',\n  styleUrls: ['./scrollable.component.scss'],\n  standalone: true,\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [CursorDragScrollDirective, ObserveScrollStateDirective, NgClass, NgIf, LetDirective, ChevronIconComponent],\n  host: {\n    class: 'et-scrollable',\n  },\n  providers: [DestroyService],\n})\nexport class ScrollableComponent implements OnInit {\n  private readonly _destroy$ = inject(DestroyService, { host: true }).destroy$;\n  private readonly _renderer = inject(Renderer2);\n  private readonly _elementRef = inject<ElementRef<HTMLElement>>(ElementRef);\n\n  @Input()\n  @HostBinding('attr.item-size')\n  itemSize: 'auto' | 'same' = 'auto';\n\n  @Input()\n  @HostBinding('attr.direction')\n  direction: 'horizontal' | 'vertical' = 'horizontal';\n\n  @Input()\n  scrollableRole?: string;\n\n  @Input()\n  scrollableClass?: NgClassType;\n\n  @Input()\n  get renderMasks(): boolean {\n    return this._renderMasks;\n  }\n  set renderMasks(value: BooleanInput) {\n    this._renderMasks = coerceBooleanProperty(value);\n  }\n  private _renderMasks = true;\n\n  @Input()\n  get renderButtons(): boolean {\n    return this._renderButtons;\n  }\n  set renderButtons(value: BooleanInput) {\n    this._renderButtons = coerceBooleanProperty(value);\n  }\n  private _renderButtons = true;\n\n  @Input()\n  @HostBinding('attr.render-scrollbars')\n  get renderScrollbars(): boolean {\n    return this._renderScrollbars;\n  }\n  set renderScrollbars(value: BooleanInput) {\n    this._renderScrollbars = coerceBooleanProperty(value);\n  }\n  private _renderScrollbars = false;\n\n  @Input()\n  @HostBinding('attr.sticky-buttons')\n  get stickyButtons(): boolean {\n    return this._stickyButtons;\n  }\n  set stickyButtons(value: BooleanInput) {\n    this._stickyButtons = coerceBooleanProperty(value);\n  }\n  private _stickyButtons = false;\n\n  @Input()\n  get cursorDragScroll(): boolean {\n    return this._cursorDragScroll;\n  }\n  set cursorDragScroll(value: BooleanInput) {\n    this._cursorDragScroll = coerceBooleanProperty(value);\n  }\n  private _cursorDragScroll = true;\n\n  @ViewChild('scrollable', { static: true })\n  scrollable!: ElementRef<HTMLElement>;\n\n  protected readonly scrollState$ = new BehaviorSubject<ScrollObserverScrollState | null>(null);\n\n  ngOnInit(): void {\n    this.scrollState$\n      .pipe(\n        tap((state) => {\n          if (!state) {\n            return;\n          }\n\n          const element = this._elementRef.nativeElement;\n\n          this._renderer.setAttribute(element, 'at-start', state.isAtStart.toString());\n          this._renderer.setAttribute(element, 'at-end', state.isAtEnd.toString());\n          this._renderer.setAttribute(element, 'can-scroll', state.canScroll.toString());\n        }),\n        takeUntil(this._destroy$),\n      )\n      .subscribe();\n  }\n\n  protected _scrollStateChanged(scrollState: ScrollObserverScrollState) {\n    if (equal(this.scrollState$.value, scrollState)) {\n      return;\n    }\n\n    this.scrollState$.next(scrollState);\n  }\n\n  protected scrollOneContainerSizeToStart() {\n    this.scrollOneContainerSize('start');\n  }\n\n  protected scrollOneContainerSizeToEnd() {\n    this.scrollOneContainerSize('end');\n  }\n\n  scrollOneContainerSize(direction: 'start' | 'end') {\n    const scrollElement = this.scrollable.nativeElement;\n    const parent = this._elementRef.nativeElement;\n\n    const scrollableSize = this.direction === 'horizontal' ? parent.clientWidth : parent.clientHeight;\n    const currentScroll = this.direction === 'horizontal' ? scrollElement.scrollLeft : scrollElement.scrollTop;\n\n    scrollElement.scrollTo({\n      [this.direction === 'horizontal' ? 'left' : 'top']:\n        currentScroll + (direction === 'start' ? -scrollableSize : scrollableSize),\n      behavior: 'smooth',\n    });\n  }\n}\n","<div\n  #scrollable\n  [attr.role]=\"scrollableRole ?? undefined\"\n  [ngClass]=\"scrollableClass\"\n  [etCursorDragScroll]=\"cursorDragScroll\"\n  (etObserveScrollState)=\"_scrollStateChanged($event)\"\n  class=\"et-scrollable-container\"\n>\n  <ng-content />\n</div>\n\n<div *ngIf=\"renderMasks\" class=\"et-scrollable-masks\">\n  <div class=\"et-scrollable-mask et-scrollable-mask--start\"></div>\n  <div class=\"et-scrollable-mask et-scrollable-mask--end\"></div>\n</div>\n\n<div *ngIf=\"renderButtons\" class=\"et-scrollable-buttons\">\n  <button\n    (click)=\"scrollOneContainerSizeToStart()\"\n    class=\"et-scrollable-button et-scrollable-button--start\"\n    aria-hidden=\"true\"\n    type=\"button\"\n    tabindex=\"-1\"\n  >\n    <et-chevron-icon />\n  </button>\n  <button\n    (click)=\"scrollOneContainerSizeToEnd()\"\n    class=\"et-scrollable-button et-scrollable-button--end\"\n    aria-hidden=\"true\"\n    type=\"button\"\n    tabindex=\"-1\"\n  >\n    <et-chevron-icon />\n  </button>\n</div>\n"]}
177
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scrollable.component.js","sourceRoot":"","sources":["../../../../../../../../../libs/cdk/src/lib/components/scrollable/components/scrollable/scrollable.component.ts","../../../../../../../../../libs/cdk/src/lib/components/scrollable/components/scrollable/scrollable.component.html"],"names":[],"mappings":"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","sourcesContent":["import { BooleanInput, NumberInput, coerceBooleanProperty, coerceNumberProperty } from '@angular/cdk/coercion';\nimport { NgClass, NgIf } from '@angular/common';\nimport {\n  AfterContentInit,\n  ChangeDetectionStrategy,\n  Component,\n  ContentChildren,\n  ElementRef,\n  HostBinding,\n  Input,\n  OnInit,\n  Renderer2,\n  ViewChild,\n  ViewEncapsulation,\n  inject,\n} from '@angular/core';\nimport {\n  CursorDragScrollDirective,\n  IS_ACTIVE_ELEMENT,\n  IsActiveElementDirective,\n  LetDirective,\n  NgClassType,\n  ObserveScrollStateDirective,\n  ScrollObserverScrollState,\n  TypedQueryList,\n  createDestroy,\n  equal,\n  scrollToElement,\n} from '@ethlete/core';\nimport { BehaviorSubject, startWith, takeUntil, tap } from 'rxjs';\nimport { ChevronIconComponent } from '../../../icons';\n\n@Component({\n  selector: 'et-scrollable',\n  templateUrl: './scrollable.component.html',\n  styleUrls: ['./scrollable.component.scss'],\n  standalone: true,\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [CursorDragScrollDirective, ObserveScrollStateDirective, NgClass, NgIf, LetDirective, ChevronIconComponent],\n  host: {\n    class: 'et-scrollable',\n  },\n})\nexport class ScrollableComponent implements OnInit, AfterContentInit {\n  private readonly _destroy$ = createDestroy();\n  private readonly _renderer = inject(Renderer2);\n  private readonly _elementRef = inject<ElementRef<HTMLElement>>(ElementRef);\n\n  @Input()\n  @HostBinding('attr.item-size')\n  itemSize: 'auto' | 'same' = 'auto';\n\n  @Input()\n  @HostBinding('attr.direction')\n  direction: 'horizontal' | 'vertical' = 'horizontal';\n\n  @Input()\n  scrollableRole?: string;\n\n  @Input()\n  scrollableClass?: NgClassType;\n\n  @Input()\n  get renderMasks(): boolean {\n    return this._renderMasks;\n  }\n  set renderMasks(value: BooleanInput) {\n    this._renderMasks = coerceBooleanProperty(value);\n  }\n  private _renderMasks = true;\n\n  @Input()\n  get renderButtons(): boolean {\n    return this._renderButtons;\n  }\n  set renderButtons(value: BooleanInput) {\n    this._renderButtons = coerceBooleanProperty(value);\n  }\n  private _renderButtons = true;\n\n  @Input()\n  @HostBinding('attr.render-scrollbars')\n  get renderScrollbars(): boolean {\n    return this._renderScrollbars;\n  }\n  set renderScrollbars(value: BooleanInput) {\n    this._renderScrollbars = coerceBooleanProperty(value);\n  }\n  private _renderScrollbars = false;\n\n  @Input()\n  @HostBinding('attr.sticky-buttons')\n  get stickyButtons(): boolean {\n    return this._stickyButtons;\n  }\n  set stickyButtons(value: BooleanInput) {\n    this._stickyButtons = coerceBooleanProperty(value);\n  }\n  private _stickyButtons = false;\n\n  @Input()\n  get cursorDragScroll(): boolean {\n    return this._cursorDragScroll;\n  }\n  set cursorDragScroll(value: BooleanInput) {\n    this._cursorDragScroll = coerceBooleanProperty(value);\n  }\n  private _cursorDragScroll = true;\n\n  @Input()\n  get activeElementScrollMargin(): number {\n    return this._activeElementScrollMargin;\n  }\n  set activeElementScrollMargin(value: NumberInput) {\n    this._activeElementScrollMargin = coerceNumberProperty(value);\n  }\n  private _activeElementScrollMargin = 40;\n\n  @Input()\n  get disableActiveElementScrolling(): boolean {\n    return this._disableActiveElementScrolling;\n  }\n  set disableActiveElementScrolling(value: BooleanInput) {\n    this._disableActiveElementScrolling = coerceBooleanProperty(value);\n  }\n  private _disableActiveElementScrolling = false;\n\n  @ViewChild('scrollable', { static: true })\n  scrollable!: ElementRef<HTMLElement>;\n\n  @ContentChildren(IS_ACTIVE_ELEMENT, { descendants: true })\n  activeElements: TypedQueryList<IsActiveElementDirective> | null = null;\n\n  protected readonly scrollState$ = new BehaviorSubject<ScrollObserverScrollState | null>(null);\n\n  ngOnInit(): void {\n    this.scrollState$\n      .pipe(\n        tap((state) => {\n          if (!state) {\n            return;\n          }\n\n          const element = this._elementRef.nativeElement;\n\n          this._renderer.setAttribute(element, 'at-start', state.isAtStart.toString());\n          this._renderer.setAttribute(element, 'at-end', state.isAtEnd.toString());\n          this._renderer.setAttribute(element, 'can-scroll', state.canScroll.toString());\n        }),\n        takeUntil(this._destroy$),\n      )\n      .subscribe();\n  }\n\n  ngAfterContentInit(): void {\n    if (!this.activeElements) {\n      return;\n    }\n\n    this.activeElements.changes\n      .pipe(\n        startWith(this.activeElements),\n        tap((activeElements) => {\n          if (this.disableActiveElementScrolling) {\n            return;\n          }\n\n          const firstActive = activeElements\n            .filter((a): a is IsActiveElementDirective => !!a)\n            .find((a) => a.isActiveElement);\n\n          if (!firstActive) {\n            return;\n          }\n\n          scrollToElement({\n            behavior: 'auto',\n            container: this.scrollable.nativeElement,\n            element: firstActive.elementRef.nativeElement,\n            scrollInlineMargin: this.direction === 'horizontal' ? this.activeElementScrollMargin : 0,\n            scrollBlockMargin: this.direction === 'horizontal' ? 0 : this.activeElementScrollMargin,\n          });\n        }),\n        takeUntil(this._destroy$),\n      )\n      .subscribe();\n  }\n\n  protected _scrollStateChanged(scrollState: ScrollObserverScrollState) {\n    if (equal(this.scrollState$.value, scrollState)) {\n      return;\n    }\n\n    this.scrollState$.next(scrollState);\n  }\n\n  protected scrollOneContainerSizeToStart() {\n    this.scrollOneContainerSize('start');\n  }\n\n  protected scrollOneContainerSizeToEnd() {\n    this.scrollOneContainerSize('end');\n  }\n\n  scrollOneContainerSize(direction: 'start' | 'end') {\n    const scrollElement = this.scrollable.nativeElement;\n    const parent = this._elementRef.nativeElement;\n\n    const scrollableSize = this.direction === 'horizontal' ? parent.clientWidth : parent.clientHeight;\n    const currentScroll = this.direction === 'horizontal' ? scrollElement.scrollLeft : scrollElement.scrollTop;\n\n    scrollElement.scrollTo({\n      [this.direction === 'horizontal' ? 'left' : 'top']:\n        currentScroll + (direction === 'start' ? -scrollableSize : scrollableSize),\n      behavior: 'smooth',\n    });\n  }\n}\n","<div\n  #scrollable\n  [attr.role]=\"scrollableRole ?? undefined\"\n  [ngClass]=\"scrollableClass\"\n  [etCursorDragScroll]=\"cursorDragScroll\"\n  (etObserveScrollState)=\"_scrollStateChanged($event)\"\n  class=\"et-scrollable-container\"\n>\n  <ng-content />\n</div>\n\n<div *ngIf=\"renderMasks\" class=\"et-scrollable-masks\">\n  <div class=\"et-scrollable-mask et-scrollable-mask--start\"></div>\n  <div class=\"et-scrollable-mask et-scrollable-mask--end\"></div>\n</div>\n\n<div *ngIf=\"renderButtons\" class=\"et-scrollable-buttons\">\n  <button\n    (click)=\"scrollOneContainerSizeToStart()\"\n    class=\"et-scrollable-button et-scrollable-button--start\"\n    aria-hidden=\"true\"\n    type=\"button\"\n    tabindex=\"-1\"\n  >\n    <et-chevron-icon />\n  </button>\n  <button\n    (click)=\"scrollOneContainerSizeToEnd()\"\n    class=\"et-scrollable-button et-scrollable-button--end\"\n    aria-hidden=\"true\"\n    type=\"button\"\n    tabindex=\"-1\"\n  >\n    <et-chevron-icon />\n  </button>\n</div>\n"]}
@@ -63,7 +63,7 @@ class NavTabsComponent extends PaginatedTabHeaderDirective {
63
63
  return this.tabOutlet ? 'tablist' : this._elementRef.nativeElement.getAttribute('role');
64
64
  }
65
65
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NavTabsComponent, deps: [{ token: i0.ElementRef }, { token: i1.Directionality, optional: true }, { token: i0.NgZone }, { token: i0.ChangeDetectorRef }, { token: i2.ViewportRuler }, { token: i3.Router }], target: i0.ɵɵFactoryTarget.Component }); }
66
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.0.0", type: NavTabsComponent, isStandalone: true, selector: "[et-nav-tabs]", inputs: { tabOutlet: "tabOutlet" }, host: { properties: { "attr.role": "this._attrRole" }, classAttribute: "et-nav-tabs" }, queries: [{ propertyName: "_items", predicate: i0.forwardRef(function () { return NavTabLinkComponent; }), descendants: true }, { propertyName: "_inkBars", predicate: i0.forwardRef(function () { return ActiveTabUnderlineDirective; }), descendants: true }], viewQueries: [{ propertyName: "_scrollable", first: true, predicate: ScrollableComponent, descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: "<et-scrollable\n [itemSize]=\"itemSize\"\n [renderButtons]=\"renderButtons\"\n [scrollableClass]=\"scrollableClass\"\n [renderMasks]=\"renderMasks\"\n [renderScrollbars]=\"renderScrollbars\"\n (keydown)=\"_handleKeydown($event)\"\n (contentChanged)=\"_onContentChanges()\"\n>\n <ng-content />\n</et-scrollable>\n", dependencies: [{ kind: "component", type: ScrollableComponent, selector: "et-scrollable", inputs: ["itemSize", "direction", "scrollableRole", "scrollableClass", "renderMasks", "renderButtons", "renderScrollbars", "stickyButtons", "cursorDragScroll"] }], changeDetection: i0.ChangeDetectionStrategy.Default, encapsulation: i0.ViewEncapsulation.None }); }
66
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.0.0", type: NavTabsComponent, isStandalone: true, selector: "[et-nav-tabs]", inputs: { tabOutlet: "tabOutlet" }, host: { properties: { "attr.role": "this._attrRole" }, classAttribute: "et-nav-tabs" }, queries: [{ propertyName: "_items", predicate: i0.forwardRef(function () { return NavTabLinkComponent; }), descendants: true }, { propertyName: "_inkBars", predicate: i0.forwardRef(function () { return ActiveTabUnderlineDirective; }), descendants: true }], viewQueries: [{ propertyName: "_scrollable", first: true, predicate: ScrollableComponent, descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: "<et-scrollable\n [itemSize]=\"itemSize\"\n [renderButtons]=\"renderButtons\"\n [scrollableClass]=\"scrollableClass\"\n [renderMasks]=\"renderMasks\"\n [renderScrollbars]=\"renderScrollbars\"\n (keydown)=\"_handleKeydown($event)\"\n (contentChanged)=\"_onContentChanges()\"\n>\n <ng-content />\n</et-scrollable>\n", dependencies: [{ kind: "component", type: ScrollableComponent, selector: "et-scrollable", inputs: ["itemSize", "direction", "scrollableRole", "scrollableClass", "renderMasks", "renderButtons", "renderScrollbars", "stickyButtons", "cursorDragScroll", "activeElementScrollMargin", "disableActiveElementScrolling"] }], changeDetection: i0.ChangeDetectionStrategy.Default, encapsulation: i0.ViewEncapsulation.None }); }
67
67
  }
68
68
  export { NavTabsComponent };
69
69
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NavTabsComponent, decorators: [{
@@ -21,7 +21,7 @@ class InlineTabHeaderComponent extends PaginatedTabHeaderDirective {
21
21
  event.preventDefault();
22
22
  }
23
23
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: InlineTabHeaderComponent, deps: [{ token: i0.ElementRef }, { token: i0.ChangeDetectorRef }, { token: i1.ViewportRuler }, { token: i2.Directionality, optional: true }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component }); }
24
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.0.0", type: InlineTabHeaderComponent, isStandalone: true, selector: "et-inline-tab-header", host: { classAttribute: "et-inline-tab-header" }, queries: [{ propertyName: "_items", predicate: InlineTabLabelWrapperDirective }, { propertyName: "_inkBars", predicate: i0.forwardRef(function () { return ActiveTabUnderlineDirective; }), descendants: true }], viewQueries: [{ propertyName: "_scrollable", first: true, predicate: ScrollableComponent, descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: "<et-scrollable\n [itemSize]=\"itemSize\"\n [renderButtons]=\"renderButtons\"\n [scrollableClass]=\"scrollableClass\"\n [renderMasks]=\"renderMasks\"\n [renderScrollbars]=\"renderScrollbars\"\n (keydown)=\"_handleKeydown($event)\"\n (contentChanged)=\"_onContentChanges()\"\n>\n <ng-content />\n</et-scrollable>\n", dependencies: [{ kind: "component", type: ScrollableComponent, selector: "et-scrollable", inputs: ["itemSize", "direction", "scrollableRole", "scrollableClass", "renderMasks", "renderButtons", "renderScrollbars", "stickyButtons", "cursorDragScroll"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
24
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.0.0", type: InlineTabHeaderComponent, isStandalone: true, selector: "et-inline-tab-header", host: { classAttribute: "et-inline-tab-header" }, queries: [{ propertyName: "_items", predicate: InlineTabLabelWrapperDirective }, { propertyName: "_inkBars", predicate: i0.forwardRef(function () { return ActiveTabUnderlineDirective; }), descendants: true }], viewQueries: [{ propertyName: "_scrollable", first: true, predicate: ScrollableComponent, descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: "<et-scrollable\n [itemSize]=\"itemSize\"\n [renderButtons]=\"renderButtons\"\n [scrollableClass]=\"scrollableClass\"\n [renderMasks]=\"renderMasks\"\n [renderScrollbars]=\"renderScrollbars\"\n (keydown)=\"_handleKeydown($event)\"\n (contentChanged)=\"_onContentChanges()\"\n>\n <ng-content />\n</et-scrollable>\n", dependencies: [{ kind: "component", type: ScrollableComponent, selector: "et-scrollable", inputs: ["itemSize", "direction", "scrollableRole", "scrollableClass", "renderMasks", "renderButtons", "renderScrollbars", "stickyButtons", "cursorDragScroll", "activeElementScrollMargin", "disableActiveElementScrolling"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
25
25
  }
26
26
  export { InlineTabHeaderComponent };
27
27
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: InlineTabHeaderComponent, decorators: [{