@ethlete/core 4.19.1 → 4.19.3

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,17 @@
1
1
  # @ethlete/core
2
2
 
3
+ ## 4.19.3
4
+
5
+ ### Patch Changes
6
+
7
+ - [`ba2e546`](https://github.com/ethlete-io/ethdk/commit/ba2e54669666038da926808fcfdeacac93483eb3) Thanks [@TomTomB](https://github.com/TomTomB)! - Enhance performance by caching `getBoundingClientRect()` calls
8
+
9
+ ## 4.19.2
10
+
11
+ ### Patch Changes
12
+
13
+ - [`8a08672`](https://github.com/ethlete-io/ethdk/commit/8a086726225de55877358780bfbc5a624d8f56d5) Thanks [@TomTomB](https://github.com/TomTomB)! - Fix styles not getting cleaned up inside signal styles util
14
+
3
15
  ## 4.19.1
4
16
 
5
17
  ### Patch Changes
@@ -439,11 +451,11 @@
439
451
 
440
452
  ```css
441
453
  /* before */
442
- [data-popper-placement^="top"] {
454
+ [data-popper-placement^='top'] {
443
455
  }
444
456
 
445
457
  /* after */
446
- [et-floating-placement^="top"] {
458
+ [et-floating-placement^='top'] {
447
459
  }
448
460
  ```
449
461
 
@@ -588,10 +600,10 @@
588
600
  Before:
589
601
 
590
602
  ```ts
591
- import { DestroyService } from "@ethlete/core";
603
+ import { DestroyService } from '@ethlete/core';
592
604
 
593
605
  @Component({
594
- selector: "my-component",
606
+ selector: 'my-component',
595
607
  template: `...`,
596
608
  providers: [DestroyService],
597
609
  })
@@ -603,10 +615,10 @@
603
615
  After:
604
616
 
605
617
  ```ts
606
- import { createDestroy } from "@ethlete/core";
618
+ import { createDestroy } from '@ethlete/core';
607
619
 
608
620
  @Component({
609
- selector: "my-component",
621
+ selector: 'my-component',
610
622
  template: `...`,
611
623
  })
612
624
  export class MyComponent {
@@ -18,9 +18,17 @@ export const isElementVisible = (options) => {
18
18
  container ||= document.documentElement;
19
19
  const canScroll = elementCanScroll(container);
20
20
  if (!canScroll) {
21
- return { inline: true, block: true, blockIntersection: 100, inlineIntersection: 100, element };
21
+ return {
22
+ inline: true,
23
+ block: true,
24
+ blockIntersection: 1,
25
+ inlineIntersection: 1,
26
+ intersectionRatio: 1,
27
+ isIntersecting: true,
28
+ element,
29
+ };
22
30
  }
23
- const elementRect = element.getBoundingClientRect();
31
+ const elementRect = options.elementRect || element.getBoundingClientRect();
24
32
  const containerRect = options.containerRect || container.getBoundingClientRect();
25
33
  const elementInlineStart = elementRect.left;
26
34
  const elementBlockStart = elementRect.top;
@@ -36,14 +44,17 @@ export const isElementVisible = (options) => {
36
44
  const isElementBlockVisible = elementBlockStart >= containerBlockStart && elementBlockEnd <= containerBlockEnd;
37
45
  const inlineIntersection = Math.min(elementInlineEnd, containerInlineEnd) - Math.max(elementInlineStart, containerInlineStart);
38
46
  const blockIntersection = Math.min(elementBlockEnd, containerBlockEnd) - Math.max(elementBlockStart, containerBlockStart);
39
- const inlineIntersectionPercentage = clamp((inlineIntersection / elWith) * 100);
40
- const blockIntersectionPercentage = clamp((blockIntersection / elHeight) * 100);
47
+ const inlineIntersectionPercentage = clamp(inlineIntersection / elWith, 0, 1);
48
+ const blockIntersectionPercentage = clamp(blockIntersection / elHeight, 0, 1);
41
49
  return {
42
50
  inline: isElementInlineVisible,
43
51
  block: isElementBlockVisible,
44
52
  inlineIntersection: inlineIntersectionPercentage,
45
53
  blockIntersection: blockIntersectionPercentage,
54
+ isIntersecting: isElementInlineVisible && isElementBlockVisible,
46
55
  element,
56
+ // Round the intersection ratio to the nearest 0.01 to avoid floating point errors and system scaling issues.
57
+ intersectionRatio: Math.round(Math.min(inlineIntersectionPercentage, blockIntersectionPercentage) * 100) / 100,
47
58
  };
48
59
  };
49
60
  export const getElementScrollCoordinates = (options) => {
@@ -143,18 +154,4 @@ export const getElementScrollCoordinates = (options) => {
143
154
  export const scrollToElement = (options) => {
144
155
  options.container?.scrollTo(getElementScrollCoordinates(options));
145
156
  };
146
- export const getElementVisibleStates = (options) => {
147
- let { container } = options;
148
- const { elements } = options;
149
- container ||= document.documentElement;
150
- const rect = container.getBoundingClientRect();
151
- const elementVisibleStates = elements
152
- .map((e) => {
153
- if (!e || !container)
154
- return null;
155
- return isElementVisible({ container, element: e, containerRect: rect });
156
- })
157
- .filter(Boolean);
158
- return elementVisibleStates;
159
- };
160
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scrollable.utils.js","sourceRoot":"","sources":["../../../../../../libs/core/src/lib/utils/scrollable.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { clamp } from './clamp.util';\n\nexport const elementCanScroll = (element: HTMLElement, direction?: 'x' | 'y') => {\n  const { scrollHeight, clientHeight, scrollWidth, clientWidth } = element;\n\n  if (direction === 'x') {\n    return scrollWidth > clientWidth;\n  } else if (direction === 'y') {\n    return scrollHeight > clientHeight;\n  }\n\n  return scrollHeight > clientHeight || scrollWidth > clientWidth;\n};\n\nexport interface IsElementVisibleOptions {\n  /**\n   * The element to check if it is visible inside a container.\n   */\n  element?: HTMLElement | null;\n\n  /**\n   * The container to check if the element is visible inside.\n   * @default document.documentElement\n   */\n  container?: HTMLElement | null;\n\n  /**\n   * The container's rect to check if the element is visible inside. Can be supplied to reduce the amount of DOM reads.\n   * @default container.getBoundingClientRect()\n   */\n  containerRect?: DOMRect | null;\n}\n\nexport interface CurrentElementVisibility {\n  /**\n   * Whether the element is visible in the inline direction.\n   */\n  inline: boolean;\n\n  /**\n   * Whether the element is visible in the block direction.\n   */\n  block: boolean;\n\n  /**\n   * The percentage of the element that is visible in the inline direction.\n   */\n  inlineIntersection: number;\n\n  /**\n   * The percentage of the element that is visible in the block direction.\n   */\n  blockIntersection: number;\n\n  /**\n   * The element that is being checked for visibility.\n   */\n  element: HTMLElement;\n}\n\nexport const isElementVisible = (options: IsElementVisibleOptions): CurrentElementVisibility | null => {\n  let { container } = options;\n  const { element } = options;\n\n  if (!element || container === null) {\n    return null;\n  }\n\n  container ||= document.documentElement;\n\n  const canScroll = elementCanScroll(container);\n\n  if (!canScroll) {\n    return { inline: true, block: true, blockIntersection: 100, inlineIntersection: 100, element };\n  }\n\n  const elementRect = element.getBoundingClientRect();\n  const containerRect = options.containerRect || container.getBoundingClientRect();\n\n  const elementInlineStart = elementRect.left;\n  const elementBlockStart = elementRect.top;\n\n  const containerInlineStart = containerRect.left;\n  const containerBlockStart = containerRect.top;\n\n  const elWith = elementRect.width || 1;\n  const elHeight = elementRect.height || 1;\n\n  const elementInlineEnd = elementInlineStart + elWith;\n  const elementBlockEnd = elementBlockStart + elHeight;\n\n  const containerInlineEnd = containerInlineStart + containerRect.width;\n  const containerBlockEnd = containerBlockStart + containerRect.height;\n\n  const isElementInlineVisible = elementInlineStart >= containerInlineStart && elementInlineEnd <= containerInlineEnd;\n  const isElementBlockVisible = elementBlockStart >= containerBlockStart && elementBlockEnd <= containerBlockEnd;\n\n  const inlineIntersection =\n    Math.min(elementInlineEnd, containerInlineEnd) - Math.max(elementInlineStart, containerInlineStart);\n  const blockIntersection =\n    Math.min(elementBlockEnd, containerBlockEnd) - Math.max(elementBlockStart, containerBlockStart);\n\n  const inlineIntersectionPercentage = clamp((inlineIntersection / elWith) * 100);\n  const blockIntersectionPercentage = clamp((blockIntersection / elHeight) * 100);\n\n  return {\n    inline: isElementInlineVisible,\n    block: isElementBlockVisible,\n    inlineIntersection: inlineIntersectionPercentage,\n    blockIntersection: blockIntersectionPercentage,\n    element,\n  };\n};\n\nexport const getElementScrollCoordinates = (options: ScrollToElementOptions): ScrollToOptions => {\n  let { container } = options;\n  const {\n    element,\n    direction,\n    behavior = 'smooth',\n    origin = 'nearest',\n    scrollBlockMargin = 0,\n    scrollInlineMargin = 0,\n  } = options;\n\n  if (!element || container === null) {\n    return {\n      behavior,\n      left: undefined,\n      top: undefined,\n    };\n  }\n\n  container ||= document.documentElement;\n\n  const canScroll = elementCanScroll(container);\n\n  if (!canScroll) {\n    return {\n      behavior,\n      left: undefined,\n      top: undefined,\n    };\n  }\n\n  const elementRect = element.getBoundingClientRect();\n  const containerRect = container.getBoundingClientRect();\n\n  const { scrollLeft, scrollTop } = container;\n  const { width: elementWidth, height: elementHeight, left: elementLeft, top: elementTop } = elementRect;\n  const { width: containerWidth, height: containerHeight, left: containerLeft, top: containerTop } = containerRect;\n\n  const shouldScrollLeft = direction === 'inline' || direction === 'both' || !direction;\n  const shouldScrollTop = direction === 'block' || direction === 'both' || !direction;\n\n  let scrollLeftTo = scrollLeft;\n  let scrollTopTo = scrollTop;\n\n  const scrollToElementStart = () => {\n    const relativeTop = elementTop - containerTop;\n    const relativeLeft = elementLeft - containerLeft;\n\n    const amountToScrollTop = relativeTop;\n    const amountToScrollLeft = relativeLeft;\n\n    const scrollTopPosition = scrollTop + amountToScrollTop;\n    const scrollLeftPosition = scrollLeft + amountToScrollLeft;\n\n    scrollLeftTo = scrollLeftPosition - scrollInlineMargin;\n    scrollTopTo = scrollTopPosition - scrollBlockMargin;\n  };\n\n  const scrollToElementEnd = () => {\n    const relativeTop = elementTop - containerTop;\n    const relativeLeft = elementLeft - containerLeft;\n\n    const amountToScrollTop = relativeTop - containerHeight + elementHeight;\n    const amountToScrollLeft = relativeLeft - containerWidth + elementWidth;\n\n    const scrollTopPosition = scrollTop + amountToScrollTop;\n    const scrollLeftPosition = scrollLeft + amountToScrollLeft;\n\n    scrollLeftTo = scrollLeftPosition + scrollInlineMargin;\n    scrollTopTo = scrollTopPosition + scrollBlockMargin;\n  };\n\n  const scrollToElementCenter = () => {\n    const relativeTop = elementTop - containerTop;\n    const relativeLeft = elementLeft - containerLeft;\n\n    const amountToScrollTop = relativeTop - containerHeight / 2 + elementHeight / 2;\n    const amountToScrollLeft = relativeLeft - containerWidth / 2 + elementWidth / 2;\n\n    const scrollTopPosition = scrollTop + amountToScrollTop;\n    const scrollLeftPosition = scrollLeft + amountToScrollLeft;\n\n    scrollLeftTo = scrollLeftPosition;\n    scrollTopTo = scrollTopPosition;\n  };\n\n  const scrollToElementNearest = () => {\n    const isAbove = elementRect.bottom < containerRect.top;\n    const isPartialAbove = elementRect.top < containerRect.top && elementRect.bottom > containerRect.top;\n\n    const isBelow = elementRect.top > containerRect.bottom;\n    const isPartialBelow = elementRect.top < containerRect.bottom && elementRect.bottom > containerRect.bottom;\n\n    const isLeft = elementRect.right < containerRect.left;\n    const isPartialLeft = elementRect.left < containerRect.left && elementRect.right > containerRect.left;\n\n    const isRight = elementRect.left > containerRect.right;\n    const isPartialRight = elementRect.left < containerRect.right && elementRect.right > containerRect.right;\n\n    if (isAbove || isPartialAbove || isLeft || isPartialLeft) {\n      scrollToElementStart();\n    } else if (isBelow || isPartialBelow || isRight || isPartialRight) {\n      scrollToElementEnd();\n    }\n  };\n\n  switch (origin) {\n    case 'start':\n      scrollToElementStart();\n      break;\n    case 'end':\n      scrollToElementEnd();\n      break;\n    case 'center':\n      scrollToElementCenter();\n      break;\n    case 'nearest':\n      scrollToElementNearest();\n      break;\n  }\n\n  return {\n    behavior,\n    left: shouldScrollLeft ? scrollLeftTo : undefined,\n    top: shouldScrollTop ? scrollTopTo : undefined,\n  };\n};\n\nexport interface ScrollToElementOptions {\n  /**\n   * The element to scroll to.\n   */\n  element?: HTMLElement | null;\n\n  /**\n   * The scroll container to scroll to the element in.\n   * @default document.documentElement\n   */\n  container?: HTMLElement | null;\n\n  /**\n   * The direction to scroll in.\n   * @default 'both'\n   */\n  direction?: 'inline' | 'block' | 'both';\n\n  /**\n   * The origin of the element to scroll to.\n   * @default 'nearest'\n   */\n  origin?: 'start' | 'end' | 'center' | 'nearest';\n\n  /**\n   * The scroll behavior.\n   * @default 'smooth'\n   */\n  behavior?: ScrollBehavior;\n\n  /**\n   * The scroll inline-margin\n   * @default 0\n   */\n  scrollInlineMargin?: number;\n\n  /**\n   * The scroll block-margin\n   * @default 0\n   */\n  scrollBlockMargin?: number;\n}\n\nexport const scrollToElement = (options: ScrollToElementOptions) => {\n  options.container?.scrollTo(getElementScrollCoordinates(options));\n};\n\nexport interface GetVisibleElementsOptions {\n  /**\n   * The container to check for visible elements.\n   * @default document.documentElement\n   */\n  container?: HTMLElement | null;\n\n  /**\n   * The elements to check if they are visible inside a container.\n   */\n  elements: HTMLElement[];\n}\n\nexport const getElementVisibleStates = (options: GetVisibleElementsOptions) => {\n  let { container } = options;\n  const { elements } = options;\n\n  container ||= document.documentElement;\n\n  const rect = container.getBoundingClientRect();\n\n  const elementVisibleStates = elements\n    .map((e) => {\n      if (!e || !container) return null;\n\n      return isElementVisible({ container, element: e, containerRect: rect });\n    })\n    .filter(Boolean) as CurrentElementVisibility[];\n\n  return elementVisibleStates;\n};\n"]}
157
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scrollable.utils.js","sourceRoot":"","sources":["../../../../../../libs/core/src/lib/utils/scrollable.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { clamp } from './clamp.util';\n\nexport const elementCanScroll = (element: HTMLElement, direction?: 'x' | 'y') => {\n  const { scrollHeight, clientHeight, scrollWidth, clientWidth } = element;\n\n  if (direction === 'x') {\n    return scrollWidth > clientWidth;\n  } else if (direction === 'y') {\n    return scrollHeight > clientHeight;\n  }\n\n  return scrollHeight > clientHeight || scrollWidth > clientWidth;\n};\n\nexport interface IsElementVisibleOptions {\n  /**\n   * The element to check if it is visible inside a container.\n   */\n  element?: HTMLElement | null;\n\n  /**\n   * The container to check if the element is visible inside.\n   * @default document.documentElement\n   */\n  container?: HTMLElement | null;\n\n  /**\n   * The container's rect to check if the element is visible inside. Can be supplied to reduce the amount of DOM reads.\n   * @default container.getBoundingClientRect()\n   */\n  containerRect?: DOMRect | null;\n\n  /**\n   * The element's rect. Can be supplied to reduce the amount of DOM reads.\n   * @default container.getBoundingClientRect()\n   */\n  elementRect?: DOMRect | null;\n}\n\nexport interface CurrentElementVisibility {\n  /**\n   * Whether the element is visible in the inline direction.\n   */\n  inline: boolean;\n\n  /**\n   * Whether the element is visible in the block direction.\n   */\n  block: boolean;\n\n  /**\n   * The percentage of the element that is visible in the inline direction.\n   */\n  inlineIntersection: number;\n\n  /**\n   * The percentage of the element that is visible in the block direction.\n   */\n  blockIntersection: number;\n\n  /**\n   * Whether the element is intersecting the container.\n   */\n  isIntersecting: boolean;\n\n  /**\n   * The ratio of the element that is intersecting the container.\n   */\n  intersectionRatio: number;\n\n  /**\n   * The element that is being checked for visibility.\n   */\n  element: HTMLElement;\n}\n\nexport const isElementVisible = (options: IsElementVisibleOptions): CurrentElementVisibility | null => {\n  let { container } = options;\n  const { element } = options;\n\n  if (!element || container === null) {\n    return null;\n  }\n\n  container ||= document.documentElement;\n\n  const canScroll = elementCanScroll(container);\n\n  if (!canScroll) {\n    return {\n      inline: true,\n      block: true,\n      blockIntersection: 1,\n      inlineIntersection: 1,\n      intersectionRatio: 1,\n      isIntersecting: true,\n      element,\n    };\n  }\n\n  const elementRect = options.elementRect || element.getBoundingClientRect();\n  const containerRect = options.containerRect || container.getBoundingClientRect();\n\n  const elementInlineStart = elementRect.left;\n  const elementBlockStart = elementRect.top;\n\n  const containerInlineStart = containerRect.left;\n  const containerBlockStart = containerRect.top;\n\n  const elWith = elementRect.width || 1;\n  const elHeight = elementRect.height || 1;\n\n  const elementInlineEnd = elementInlineStart + elWith;\n  const elementBlockEnd = elementBlockStart + elHeight;\n\n  const containerInlineEnd = containerInlineStart + containerRect.width;\n  const containerBlockEnd = containerBlockStart + containerRect.height;\n\n  const isElementInlineVisible = elementInlineStart >= containerInlineStart && elementInlineEnd <= containerInlineEnd;\n  const isElementBlockVisible = elementBlockStart >= containerBlockStart && elementBlockEnd <= containerBlockEnd;\n\n  const inlineIntersection =\n    Math.min(elementInlineEnd, containerInlineEnd) - Math.max(elementInlineStart, containerInlineStart);\n  const blockIntersection =\n    Math.min(elementBlockEnd, containerBlockEnd) - Math.max(elementBlockStart, containerBlockStart);\n\n  const inlineIntersectionPercentage = clamp(inlineIntersection / elWith, 0, 1);\n  const blockIntersectionPercentage = clamp(blockIntersection / elHeight, 0, 1);\n\n  return {\n    inline: isElementInlineVisible,\n    block: isElementBlockVisible,\n    inlineIntersection: inlineIntersectionPercentage,\n    blockIntersection: blockIntersectionPercentage,\n    isIntersecting: isElementInlineVisible && isElementBlockVisible,\n    element,\n\n    // Round the intersection ratio to the nearest 0.01 to avoid floating point errors and system scaling issues.\n    intersectionRatio: Math.round(Math.min(inlineIntersectionPercentage, blockIntersectionPercentage) * 100) / 100,\n  };\n};\n\nexport const getElementScrollCoordinates = (options: ScrollToElementOptions): ScrollToOptions => {\n  let { container } = options;\n  const {\n    element,\n    direction,\n    behavior = 'smooth',\n    origin = 'nearest',\n    scrollBlockMargin = 0,\n    scrollInlineMargin = 0,\n  } = options;\n\n  if (!element || container === null) {\n    return {\n      behavior,\n      left: undefined,\n      top: undefined,\n    };\n  }\n\n  container ||= document.documentElement;\n\n  const canScroll = elementCanScroll(container);\n\n  if (!canScroll) {\n    return {\n      behavior,\n      left: undefined,\n      top: undefined,\n    };\n  }\n\n  const elementRect = element.getBoundingClientRect();\n  const containerRect = container.getBoundingClientRect();\n\n  const { scrollLeft, scrollTop } = container;\n  const { width: elementWidth, height: elementHeight, left: elementLeft, top: elementTop } = elementRect;\n  const { width: containerWidth, height: containerHeight, left: containerLeft, top: containerTop } = containerRect;\n\n  const shouldScrollLeft = direction === 'inline' || direction === 'both' || !direction;\n  const shouldScrollTop = direction === 'block' || direction === 'both' || !direction;\n\n  let scrollLeftTo = scrollLeft;\n  let scrollTopTo = scrollTop;\n\n  const scrollToElementStart = () => {\n    const relativeTop = elementTop - containerTop;\n    const relativeLeft = elementLeft - containerLeft;\n\n    const amountToScrollTop = relativeTop;\n    const amountToScrollLeft = relativeLeft;\n\n    const scrollTopPosition = scrollTop + amountToScrollTop;\n    const scrollLeftPosition = scrollLeft + amountToScrollLeft;\n\n    scrollLeftTo = scrollLeftPosition - scrollInlineMargin;\n    scrollTopTo = scrollTopPosition - scrollBlockMargin;\n  };\n\n  const scrollToElementEnd = () => {\n    const relativeTop = elementTop - containerTop;\n    const relativeLeft = elementLeft - containerLeft;\n\n    const amountToScrollTop = relativeTop - containerHeight + elementHeight;\n    const amountToScrollLeft = relativeLeft - containerWidth + elementWidth;\n\n    const scrollTopPosition = scrollTop + amountToScrollTop;\n    const scrollLeftPosition = scrollLeft + amountToScrollLeft;\n\n    scrollLeftTo = scrollLeftPosition + scrollInlineMargin;\n    scrollTopTo = scrollTopPosition + scrollBlockMargin;\n  };\n\n  const scrollToElementCenter = () => {\n    const relativeTop = elementTop - containerTop;\n    const relativeLeft = elementLeft - containerLeft;\n\n    const amountToScrollTop = relativeTop - containerHeight / 2 + elementHeight / 2;\n    const amountToScrollLeft = relativeLeft - containerWidth / 2 + elementWidth / 2;\n\n    const scrollTopPosition = scrollTop + amountToScrollTop;\n    const scrollLeftPosition = scrollLeft + amountToScrollLeft;\n\n    scrollLeftTo = scrollLeftPosition;\n    scrollTopTo = scrollTopPosition;\n  };\n\n  const scrollToElementNearest = () => {\n    const isAbove = elementRect.bottom < containerRect.top;\n    const isPartialAbove = elementRect.top < containerRect.top && elementRect.bottom > containerRect.top;\n\n    const isBelow = elementRect.top > containerRect.bottom;\n    const isPartialBelow = elementRect.top < containerRect.bottom && elementRect.bottom > containerRect.bottom;\n\n    const isLeft = elementRect.right < containerRect.left;\n    const isPartialLeft = elementRect.left < containerRect.left && elementRect.right > containerRect.left;\n\n    const isRight = elementRect.left > containerRect.right;\n    const isPartialRight = elementRect.left < containerRect.right && elementRect.right > containerRect.right;\n\n    if (isAbove || isPartialAbove || isLeft || isPartialLeft) {\n      scrollToElementStart();\n    } else if (isBelow || isPartialBelow || isRight || isPartialRight) {\n      scrollToElementEnd();\n    }\n  };\n\n  switch (origin) {\n    case 'start':\n      scrollToElementStart();\n      break;\n    case 'end':\n      scrollToElementEnd();\n      break;\n    case 'center':\n      scrollToElementCenter();\n      break;\n    case 'nearest':\n      scrollToElementNearest();\n      break;\n  }\n\n  return {\n    behavior,\n    left: shouldScrollLeft ? scrollLeftTo : undefined,\n    top: shouldScrollTop ? scrollTopTo : undefined,\n  };\n};\n\nexport interface ScrollToElementOptions {\n  /**\n   * The element to scroll to.\n   */\n  element?: HTMLElement | null;\n\n  /**\n   * The scroll container to scroll to the element in.\n   * @default document.documentElement\n   */\n  container?: HTMLElement | null;\n\n  /**\n   * The direction to scroll in.\n   * @default 'both'\n   */\n  direction?: 'inline' | 'block' | 'both';\n\n  /**\n   * The origin of the element to scroll to.\n   * @default 'nearest'\n   */\n  origin?: 'start' | 'end' | 'center' | 'nearest';\n\n  /**\n   * The scroll behavior.\n   * @default 'smooth'\n   */\n  behavior?: ScrollBehavior;\n\n  /**\n   * The scroll inline-margin\n   * @default 0\n   */\n  scrollInlineMargin?: number;\n\n  /**\n   * The scroll block-margin\n   * @default 0\n   */\n  scrollBlockMargin?: number;\n}\n\nexport const scrollToElement = (options: ScrollToElementOptions) => {\n  options.container?.scrollTo(getElementScrollCoordinates(options));\n};\n"]}