@ethlete/core 4.2.2 → 4.2.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.
@@ -57,46 +57,77 @@ export const scrollToElement = (options) => {
57
57
  }
58
58
  const elementRect = element.getBoundingClientRect();
59
59
  const containerRect = container.getBoundingClientRect();
60
- const elementInlineSize = elementRect.width;
61
- const elementBlockSize = elementRect.height;
62
- const containerInlineSize = containerRect.width;
63
- const containerBlockSize = containerRect.height;
64
- const elementInlineStart = elementRect.left;
65
- const elementBlockStart = elementRect.top;
66
- const containerInlineStart = containerRect.left;
67
- const containerBlockStart = containerRect.top;
68
- const elementInlineEnd = elementInlineStart + elementInlineSize;
69
- const elementBlockEnd = elementBlockStart + elementBlockSize;
70
- const containerInlineEnd = containerInlineStart + containerInlineSize;
71
- const containerBlockEnd = containerBlockStart + containerBlockSize;
72
- const elementInlineCenter = elementInlineStart + elementInlineSize / 2;
73
- const elementBlockCenter = elementBlockStart + elementBlockSize / 2;
74
- const containerInlineCenter = containerInlineStart + containerInlineSize / 2;
75
- const containerBlockCenter = containerBlockStart + containerBlockSize / 2;
76
- const elementInlineOrigin = origin === 'center' ? elementInlineCenter : origin === 'end' ? elementInlineEnd : elementInlineStart;
77
- const elementBlockOrigin = origin === 'center' ? elementBlockCenter : origin === 'end' ? elementBlockEnd : elementBlockStart;
78
- const containerInlineOrigin = origin === 'center' ? containerInlineCenter : origin === 'end' ? containerInlineEnd : containerInlineStart;
79
- const containerBlockOrigin = origin === 'center' ? containerBlockCenter : origin === 'end' ? containerBlockEnd : containerBlockStart;
80
- const inlineOffset = elementInlineOrigin - containerInlineOrigin - scrollInlineMargin;
81
- const blockOffset = elementBlockOrigin - containerBlockOrigin - scrollBlockMargin;
82
- let inlineScroll = direction === 'block' ? undefined : inlineOffset;
83
- let blockScroll = direction === 'inline' ? undefined : blockOffset;
84
- if (origin === 'nearest') {
85
- const elVisible = isElementVisible({ element, container, containerRect });
86
- if (elVisible?.inline && elVisible?.block) {
87
- return;
88
- }
89
- if (elVisible?.inline) {
90
- inlineScroll = undefined;
60
+ const { scrollLeft, scrollTop } = container;
61
+ const { width: elementWidth, height: elementHeight, left: elementLeft, top: elementTop } = elementRect;
62
+ const { width: containerWidth, height: containerHeight, left: containerLeft, top: containerTop } = containerRect;
63
+ const shouldScrollLeft = direction === 'inline' || direction === 'both' || !direction;
64
+ const shouldScrollTop = direction === 'block' || direction === 'both' || !direction;
65
+ let scrollLeftTo = scrollLeft;
66
+ let scrollTopTo = scrollTop;
67
+ const scrollToElementStart = () => {
68
+ const relativeTop = elementTop - containerTop;
69
+ const relativeLeft = elementLeft - containerLeft;
70
+ const amountToScrollTop = relativeTop;
71
+ const amountToScrollLeft = relativeLeft;
72
+ const scrollTopPosition = scrollTop + amountToScrollTop;
73
+ const scrollLeftPosition = scrollLeft + amountToScrollLeft;
74
+ scrollLeftTo = scrollLeftPosition - scrollInlineMargin;
75
+ scrollTopTo = scrollTopPosition - scrollBlockMargin;
76
+ };
77
+ const scrollToElementEnd = () => {
78
+ const relativeTop = elementTop - containerTop;
79
+ const relativeLeft = elementLeft - containerLeft;
80
+ const amountToScrollTop = relativeTop - containerHeight + elementHeight;
81
+ const amountToScrollLeft = relativeLeft - containerWidth + elementWidth;
82
+ const scrollTopPosition = scrollTop + amountToScrollTop;
83
+ const scrollLeftPosition = scrollLeft + amountToScrollLeft;
84
+ scrollLeftTo = scrollLeftPosition + scrollInlineMargin;
85
+ scrollTopTo = scrollTopPosition + scrollBlockMargin;
86
+ };
87
+ const scrollToElementCenter = () => {
88
+ const relativeTop = elementTop - containerTop;
89
+ const relativeLeft = elementLeft - containerLeft;
90
+ const amountToScrollTop = relativeTop - containerHeight / 2 + elementHeight / 2;
91
+ const amountToScrollLeft = relativeLeft - containerWidth / 2 + elementWidth / 2;
92
+ const scrollTopPosition = scrollTop + amountToScrollTop;
93
+ const scrollLeftPosition = scrollLeft + amountToScrollLeft;
94
+ scrollLeftTo = scrollLeftPosition;
95
+ scrollTopTo = scrollTopPosition;
96
+ };
97
+ const scrollToElementNearest = () => {
98
+ const isAbove = elementRect.bottom < containerRect.top;
99
+ const isPartialAbove = elementRect.top < containerRect.top && elementRect.bottom > containerRect.top;
100
+ const isBelow = elementRect.top > containerRect.bottom;
101
+ const isPartialBelow = elementRect.top < containerRect.bottom && elementRect.bottom > containerRect.bottom;
102
+ const isLeft = elementRect.right < containerRect.left;
103
+ const isPartialLeft = elementRect.left < containerRect.left && elementRect.right > containerRect.left;
104
+ const isRight = elementRect.left > containerRect.right;
105
+ const isPartialRight = elementRect.left < containerRect.right && elementRect.right > containerRect.right;
106
+ if (isAbove || isPartialAbove || isLeft || isPartialLeft) {
107
+ scrollToElementStart();
91
108
  }
92
- if (elVisible?.block) {
93
- blockScroll = undefined;
109
+ else if (isBelow || isPartialBelow || isRight || isPartialRight) {
110
+ scrollToElementEnd();
94
111
  }
112
+ };
113
+ switch (origin) {
114
+ case 'start':
115
+ scrollToElementStart();
116
+ break;
117
+ case 'end':
118
+ scrollToElementEnd();
119
+ break;
120
+ case 'center':
121
+ scrollToElementCenter();
122
+ break;
123
+ case 'nearest':
124
+ scrollToElementNearest();
125
+ break;
95
126
  }
96
127
  container.scrollTo({
97
- left: container.scrollLeft + (inlineScroll || 0),
98
- top: container.scrollTop + (blockScroll || 0),
99
128
  behavior,
129
+ left: shouldScrollLeft ? scrollLeftTo : undefined,
130
+ top: shouldScrollTop ? scrollTopTo : undefined,
100
131
  });
101
132
  };
102
133
  export const getElementVisibleStates = (options) => {
@@ -113,4 +144,4 @@ export const getElementVisibleStates = (options) => {
113
144
  .filter(Boolean);
114
145
  return elementVisibleStates;
115
146
  };
116
- //# 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 elementInlineEnd = elementInlineStart + elementRect.width;\n  const elementBlockEnd = elementBlockStart + elementRect.height;\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 / elementRect.width) * 100);\n  const blockIntersectionPercentage = clamp((blockIntersection / elementRect.height) * 100);\n\n  return {\n    inline: isElementInlineVisible,\n    block: isElementBlockVisible,\n    inlineIntersection: inlineIntersectionPercentage,\n    blockIntersection: blockIntersectionPercentage,\n    element,\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  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  }\n\n  container ||= document.documentElement;\n\n  const canScroll = elementCanScroll(container);\n\n  if (!canScroll) {\n    return;\n  }\n\n  const elementRect = element.getBoundingClientRect();\n  const containerRect = container.getBoundingClientRect();\n\n  const elementInlineSize = elementRect.width;\n  const elementBlockSize = elementRect.height;\n\n  const containerInlineSize = containerRect.width;\n  const containerBlockSize = containerRect.height;\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 elementInlineEnd = elementInlineStart + elementInlineSize;\n  const elementBlockEnd = elementBlockStart + elementBlockSize;\n\n  const containerInlineEnd = containerInlineStart + containerInlineSize;\n  const containerBlockEnd = containerBlockStart + containerBlockSize;\n\n  const elementInlineCenter = elementInlineStart + elementInlineSize / 2;\n  const elementBlockCenter = elementBlockStart + elementBlockSize / 2;\n\n  const containerInlineCenter = containerInlineStart + containerInlineSize / 2;\n  const containerBlockCenter = containerBlockStart + containerBlockSize / 2;\n\n  const elementInlineOrigin =\n    origin === 'center' ? elementInlineCenter : origin === 'end' ? elementInlineEnd : elementInlineStart;\n  const elementBlockOrigin =\n    origin === 'center' ? elementBlockCenter : origin === 'end' ? elementBlockEnd : elementBlockStart;\n\n  const containerInlineOrigin =\n    origin === 'center' ? containerInlineCenter : origin === 'end' ? containerInlineEnd : containerInlineStart;\n  const containerBlockOrigin =\n    origin === 'center' ? containerBlockCenter : origin === 'end' ? containerBlockEnd : containerBlockStart;\n\n  const inlineOffset = elementInlineOrigin - containerInlineOrigin - scrollInlineMargin;\n  const blockOffset = elementBlockOrigin - containerBlockOrigin - scrollBlockMargin;\n\n  let inlineScroll: number | undefined = direction === 'block' ? undefined : inlineOffset;\n  let blockScroll: number | undefined = direction === 'inline' ? undefined : blockOffset;\n\n  if (origin === 'nearest') {\n    const elVisible = isElementVisible({ element, container, containerRect });\n\n    if (elVisible?.inline && elVisible?.block) {\n      return;\n    }\n\n    if (elVisible?.inline) {\n      inlineScroll = undefined;\n    }\n\n    if (elVisible?.block) {\n      blockScroll = undefined;\n    }\n  }\n\n  container.scrollTo({\n    left: container.scrollLeft + (inlineScroll || 0),\n    top: container.scrollTop + (blockScroll || 0),\n    behavior,\n  });\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"]}
147
+ //# 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 elementInlineEnd = elementInlineStart + elementRect.width;\n  const elementBlockEnd = elementBlockStart + elementRect.height;\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 / elementRect.width) * 100);\n  const blockIntersectionPercentage = clamp((blockIntersection / elementRect.height) * 100);\n\n  return {\n    inline: isElementInlineVisible,\n    block: isElementBlockVisible,\n    inlineIntersection: inlineIntersectionPercentage,\n    blockIntersection: blockIntersectionPercentage,\n    element,\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  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  }\n\n  container ||= document.documentElement;\n\n  const canScroll = elementCanScroll(container);\n\n  if (!canScroll) {\n    return;\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  container.scrollTo({\n    behavior,\n    left: shouldScrollLeft ? scrollLeftTo : undefined,\n    top: shouldScrollTop ? scrollTopTo : undefined,\n  });\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"]}
@@ -935,46 +935,77 @@ const scrollToElement = (options) => {
935
935
  }
936
936
  const elementRect = element.getBoundingClientRect();
937
937
  const containerRect = container.getBoundingClientRect();
938
- const elementInlineSize = elementRect.width;
939
- const elementBlockSize = elementRect.height;
940
- const containerInlineSize = containerRect.width;
941
- const containerBlockSize = containerRect.height;
942
- const elementInlineStart = elementRect.left;
943
- const elementBlockStart = elementRect.top;
944
- const containerInlineStart = containerRect.left;
945
- const containerBlockStart = containerRect.top;
946
- const elementInlineEnd = elementInlineStart + elementInlineSize;
947
- const elementBlockEnd = elementBlockStart + elementBlockSize;
948
- const containerInlineEnd = containerInlineStart + containerInlineSize;
949
- const containerBlockEnd = containerBlockStart + containerBlockSize;
950
- const elementInlineCenter = elementInlineStart + elementInlineSize / 2;
951
- const elementBlockCenter = elementBlockStart + elementBlockSize / 2;
952
- const containerInlineCenter = containerInlineStart + containerInlineSize / 2;
953
- const containerBlockCenter = containerBlockStart + containerBlockSize / 2;
954
- const elementInlineOrigin = origin === 'center' ? elementInlineCenter : origin === 'end' ? elementInlineEnd : elementInlineStart;
955
- const elementBlockOrigin = origin === 'center' ? elementBlockCenter : origin === 'end' ? elementBlockEnd : elementBlockStart;
956
- const containerInlineOrigin = origin === 'center' ? containerInlineCenter : origin === 'end' ? containerInlineEnd : containerInlineStart;
957
- const containerBlockOrigin = origin === 'center' ? containerBlockCenter : origin === 'end' ? containerBlockEnd : containerBlockStart;
958
- const inlineOffset = elementInlineOrigin - containerInlineOrigin - scrollInlineMargin;
959
- const blockOffset = elementBlockOrigin - containerBlockOrigin - scrollBlockMargin;
960
- let inlineScroll = direction === 'block' ? undefined : inlineOffset;
961
- let blockScroll = direction === 'inline' ? undefined : blockOffset;
962
- if (origin === 'nearest') {
963
- const elVisible = isElementVisible({ element, container, containerRect });
964
- if (elVisible?.inline && elVisible?.block) {
965
- return;
966
- }
967
- if (elVisible?.inline) {
968
- inlineScroll = undefined;
969
- }
970
- if (elVisible?.block) {
971
- blockScroll = undefined;
938
+ const { scrollLeft, scrollTop } = container;
939
+ const { width: elementWidth, height: elementHeight, left: elementLeft, top: elementTop } = elementRect;
940
+ const { width: containerWidth, height: containerHeight, left: containerLeft, top: containerTop } = containerRect;
941
+ const shouldScrollLeft = direction === 'inline' || direction === 'both' || !direction;
942
+ const shouldScrollTop = direction === 'block' || direction === 'both' || !direction;
943
+ let scrollLeftTo = scrollLeft;
944
+ let scrollTopTo = scrollTop;
945
+ const scrollToElementStart = () => {
946
+ const relativeTop = elementTop - containerTop;
947
+ const relativeLeft = elementLeft - containerLeft;
948
+ const amountToScrollTop = relativeTop;
949
+ const amountToScrollLeft = relativeLeft;
950
+ const scrollTopPosition = scrollTop + amountToScrollTop;
951
+ const scrollLeftPosition = scrollLeft + amountToScrollLeft;
952
+ scrollLeftTo = scrollLeftPosition - scrollInlineMargin;
953
+ scrollTopTo = scrollTopPosition - scrollBlockMargin;
954
+ };
955
+ const scrollToElementEnd = () => {
956
+ const relativeTop = elementTop - containerTop;
957
+ const relativeLeft = elementLeft - containerLeft;
958
+ const amountToScrollTop = relativeTop - containerHeight + elementHeight;
959
+ const amountToScrollLeft = relativeLeft - containerWidth + elementWidth;
960
+ const scrollTopPosition = scrollTop + amountToScrollTop;
961
+ const scrollLeftPosition = scrollLeft + amountToScrollLeft;
962
+ scrollLeftTo = scrollLeftPosition + scrollInlineMargin;
963
+ scrollTopTo = scrollTopPosition + scrollBlockMargin;
964
+ };
965
+ const scrollToElementCenter = () => {
966
+ const relativeTop = elementTop - containerTop;
967
+ const relativeLeft = elementLeft - containerLeft;
968
+ const amountToScrollTop = relativeTop - containerHeight / 2 + elementHeight / 2;
969
+ const amountToScrollLeft = relativeLeft - containerWidth / 2 + elementWidth / 2;
970
+ const scrollTopPosition = scrollTop + amountToScrollTop;
971
+ const scrollLeftPosition = scrollLeft + amountToScrollLeft;
972
+ scrollLeftTo = scrollLeftPosition;
973
+ scrollTopTo = scrollTopPosition;
974
+ };
975
+ const scrollToElementNearest = () => {
976
+ const isAbove = elementRect.bottom < containerRect.top;
977
+ const isPartialAbove = elementRect.top < containerRect.top && elementRect.bottom > containerRect.top;
978
+ const isBelow = elementRect.top > containerRect.bottom;
979
+ const isPartialBelow = elementRect.top < containerRect.bottom && elementRect.bottom > containerRect.bottom;
980
+ const isLeft = elementRect.right < containerRect.left;
981
+ const isPartialLeft = elementRect.left < containerRect.left && elementRect.right > containerRect.left;
982
+ const isRight = elementRect.left > containerRect.right;
983
+ const isPartialRight = elementRect.left < containerRect.right && elementRect.right > containerRect.right;
984
+ if (isAbove || isPartialAbove || isLeft || isPartialLeft) {
985
+ scrollToElementStart();
986
+ }
987
+ else if (isBelow || isPartialBelow || isRight || isPartialRight) {
988
+ scrollToElementEnd();
972
989
  }
990
+ };
991
+ switch (origin) {
992
+ case 'start':
993
+ scrollToElementStart();
994
+ break;
995
+ case 'end':
996
+ scrollToElementEnd();
997
+ break;
998
+ case 'center':
999
+ scrollToElementCenter();
1000
+ break;
1001
+ case 'nearest':
1002
+ scrollToElementNearest();
1003
+ break;
973
1004
  }
974
1005
  container.scrollTo({
975
- left: container.scrollLeft + (inlineScroll || 0),
976
- top: container.scrollTop + (blockScroll || 0),
977
1006
  behavior,
1007
+ left: shouldScrollLeft ? scrollLeftTo : undefined,
1008
+ top: shouldScrollTop ? scrollTopTo : undefined,
978
1009
  });
979
1010
  };
980
1011
  const getElementVisibleStates = (options) => {