@salt-ds/core 1.59.1 → 1.61.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 (233) hide show
  1. package/CHANGELOG.md +64 -0
  2. package/css/salt-core.css +366 -1
  3. package/dist-cjs/aria-announcer/AriaAnnounce.js +15 -3
  4. package/dist-cjs/aria-announcer/AriaAnnounce.js.map +1 -1
  5. package/dist-cjs/aria-announcer/AriaAnnouncerContext.js.map +1 -1
  6. package/dist-cjs/aria-announcer/AriaAnnouncerProvider.js +65 -43
  7. package/dist-cjs/aria-announcer/AriaAnnouncerProvider.js.map +1 -1
  8. package/dist-cjs/aria-announcer/announcementRegistry.js +31 -0
  9. package/dist-cjs/aria-announcer/announcementRegistry.js.map +1 -0
  10. package/dist-cjs/aria-announcer/useAriaAnnouncer.js +44 -16
  11. package/dist-cjs/aria-announcer/useAriaAnnouncer.js.map +1 -1
  12. package/dist-cjs/index.js +17 -1
  13. package/dist-cjs/index.js.map +1 -1
  14. package/dist-cjs/navigation-item/NavigationItem.js +2 -0
  15. package/dist-cjs/navigation-item/NavigationItem.js.map +1 -1
  16. package/dist-cjs/pagination/Pagination.js +1 -0
  17. package/dist-cjs/pagination/Pagination.js.map +1 -1
  18. package/dist-cjs/rating/Rating.css.js +6 -0
  19. package/dist-cjs/rating/Rating.css.js.map +1 -0
  20. package/dist-cjs/rating/Rating.js +140 -0
  21. package/dist-cjs/rating/Rating.js.map +1 -0
  22. package/dist-cjs/rating/RatingItem.css.js +6 -0
  23. package/dist-cjs/rating/RatingItem.css.js.map +1 -0
  24. package/dist-cjs/rating/RatingItem.js +75 -0
  25. package/dist-cjs/rating/RatingItem.js.map +1 -0
  26. package/dist-cjs/salt-provider/SaltProvider.js +3 -1
  27. package/dist-cjs/salt-provider/SaltProvider.js.map +1 -1
  28. package/dist-cjs/semantic-icon-provider/SemanticIconProvider.js +22 -20
  29. package/dist-cjs/semantic-icon-provider/SemanticIconProvider.js.map +1 -1
  30. package/dist-cjs/spinner/Spinner.js +1 -0
  31. package/dist-cjs/spinner/Spinner.js.map +1 -1
  32. package/dist-cjs/tabs/Tab.css.js +6 -0
  33. package/dist-cjs/tabs/Tab.css.js.map +1 -0
  34. package/dist-cjs/tabs/Tab.js +211 -0
  35. package/dist-cjs/tabs/Tab.js.map +1 -0
  36. package/dist-cjs/tabs/TabAction.js +63 -0
  37. package/dist-cjs/tabs/TabAction.js.map +1 -0
  38. package/dist-cjs/tabs/TabBar.css.js +6 -0
  39. package/dist-cjs/tabs/TabBar.css.js.map +1 -0
  40. package/dist-cjs/tabs/TabBar.js +45 -0
  41. package/dist-cjs/tabs/TabBar.js.map +1 -0
  42. package/dist-cjs/tabs/TabList.css.js +6 -0
  43. package/dist-cjs/tabs/TabList.css.js.map +1 -0
  44. package/dist-cjs/tabs/TabList.js +281 -0
  45. package/dist-cjs/tabs/TabList.js.map +1 -0
  46. package/dist-cjs/tabs/TabPanel.css.js +6 -0
  47. package/dist-cjs/tabs/TabPanel.css.js.map +1 -0
  48. package/dist-cjs/tabs/TabPanel.js +98 -0
  49. package/dist-cjs/tabs/TabPanel.js.map +1 -0
  50. package/dist-cjs/tabs/TabTrigger.css.js +6 -0
  51. package/dist-cjs/tabs/TabTrigger.css.js.map +1 -0
  52. package/dist-cjs/tabs/TabTrigger.js +188 -0
  53. package/dist-cjs/tabs/TabTrigger.js.map +1 -0
  54. package/dist-cjs/tabs/Tabs.css.js +6 -0
  55. package/dist-cjs/tabs/Tabs.css.js.map +1 -0
  56. package/dist-cjs/tabs/Tabs.js +200 -0
  57. package/dist-cjs/tabs/Tabs.js.map +1 -0
  58. package/dist-cjs/tabs/internal/contexts/TabContext.js +26 -0
  59. package/dist-cjs/tabs/internal/contexts/TabContext.js.map +1 -0
  60. package/dist-cjs/tabs/internal/contexts/TabListLayoutContext.js +19 -0
  61. package/dist-cjs/tabs/internal/contexts/TabListLayoutContext.js.map +1 -0
  62. package/dist-cjs/tabs/internal/contexts/TabSlotRegistryContext.js +22 -0
  63. package/dist-cjs/tabs/internal/contexts/TabSlotRegistryContext.js.map +1 -0
  64. package/dist-cjs/tabs/internal/contexts/TabsContext.js +50 -0
  65. package/dist-cjs/tabs/internal/contexts/TabsContext.js.map +1 -0
  66. package/dist-cjs/tabs/internal/hooks/useFocusWithRetry.js +64 -0
  67. package/dist-cjs/tabs/internal/hooks/useFocusWithRetry.js.map +1 -0
  68. package/dist-cjs/tabs/internal/hooks/useTabListRecovery.js +76 -0
  69. package/dist-cjs/tabs/internal/hooks/useTabListRecovery.js.map +1 -0
  70. package/dist-cjs/tabs/internal/hooks/useTabRemovalHandler.js +165 -0
  71. package/dist-cjs/tabs/internal/hooks/useTabRemovalHandler.js.map +1 -0
  72. package/dist-cjs/tabs/internal/hooks/useTabSelectionFocus.js +87 -0
  73. package/dist-cjs/tabs/internal/hooks/useTabSelectionFocus.js.map +1 -0
  74. package/dist-cjs/tabs/internal/overflow/TabOverflowList.css.js +6 -0
  75. package/dist-cjs/tabs/internal/overflow/TabOverflowList.css.js.map +1 -0
  76. package/dist-cjs/tabs/internal/overflow/TabOverflowList.js +245 -0
  77. package/dist-cjs/tabs/internal/overflow/TabOverflowList.js.map +1 -0
  78. package/dist-cjs/tabs/internal/overflow/TabSlot.js +30 -0
  79. package/dist-cjs/tabs/internal/overflow/TabSlot.js.map +1 -0
  80. package/dist-cjs/tabs/internal/overflow/overflowMath.js +86 -0
  81. package/dist-cjs/tabs/internal/overflow/overflowMath.js.map +1 -0
  82. package/dist-cjs/tabs/internal/overflow/useOverflow.js +273 -0
  83. package/dist-cjs/tabs/internal/overflow/useOverflow.js.map +1 -0
  84. package/dist-cjs/tabs/internal/overflow/useOverflowLayoutState.js +99 -0
  85. package/dist-cjs/tabs/internal/overflow/useOverflowLayoutState.js.map +1 -0
  86. package/dist-cjs/tabs/internal/overflow/useOverflowSelectionState.js +68 -0
  87. package/dist-cjs/tabs/internal/overflow/useOverflowSelectionState.js.map +1 -0
  88. package/dist-cjs/tabs/internal/overflow/useRenderedTabWidth.js +92 -0
  89. package/dist-cjs/tabs/internal/overflow/useRenderedTabWidth.js.map +1 -0
  90. package/dist-cjs/tabs/internal/overflow/widthMeasurement.js +42 -0
  91. package/dist-cjs/tabs/internal/overflow/widthMeasurement.js.map +1 -0
  92. package/dist-cjs/tabs/internal/registry/useCollection.js +197 -0
  93. package/dist-cjs/tabs/internal/registry/useCollection.js.map +1 -0
  94. package/dist-cjs/tabs/internal/registry/useRenderedTabsRegistry.js +206 -0
  95. package/dist-cjs/tabs/internal/registry/useRenderedTabsRegistry.js.map +1 -0
  96. package/dist-cjs/tabs/internal/utils/domUtils.js +13 -0
  97. package/dist-cjs/tabs/internal/utils/domUtils.js.map +1 -0
  98. package/dist-cjs/tooltip/useAriaAnnounce.js +1 -0
  99. package/dist-cjs/tooltip/useAriaAnnounce.js.map +1 -1
  100. package/dist-es/aria-announcer/AriaAnnounce.js +15 -3
  101. package/dist-es/aria-announcer/AriaAnnounce.js.map +1 -1
  102. package/dist-es/aria-announcer/AriaAnnouncerContext.js.map +1 -1
  103. package/dist-es/aria-announcer/AriaAnnouncerProvider.js +67 -45
  104. package/dist-es/aria-announcer/AriaAnnouncerProvider.js.map +1 -1
  105. package/dist-es/aria-announcer/announcementRegistry.js +28 -0
  106. package/dist-es/aria-announcer/announcementRegistry.js.map +1 -0
  107. package/dist-es/aria-announcer/useAriaAnnouncer.js +45 -17
  108. package/dist-es/aria-announcer/useAriaAnnouncer.js.map +1 -1
  109. package/dist-es/index.js +9 -1
  110. package/dist-es/index.js.map +1 -1
  111. package/dist-es/navigation-item/NavigationItem.js +2 -0
  112. package/dist-es/navigation-item/NavigationItem.js.map +1 -1
  113. package/dist-es/pagination/Pagination.js +1 -0
  114. package/dist-es/pagination/Pagination.js.map +1 -1
  115. package/dist-es/rating/Rating.css.js +4 -0
  116. package/dist-es/rating/Rating.css.js.map +1 -0
  117. package/dist-es/rating/Rating.js +138 -0
  118. package/dist-es/rating/Rating.js.map +1 -0
  119. package/dist-es/rating/RatingItem.css.js +4 -0
  120. package/dist-es/rating/RatingItem.css.js.map +1 -0
  121. package/dist-es/rating/RatingItem.js +73 -0
  122. package/dist-es/rating/RatingItem.js.map +1 -0
  123. package/dist-es/salt-provider/SaltProvider.js +3 -1
  124. package/dist-es/salt-provider/SaltProvider.js.map +1 -1
  125. package/dist-es/semantic-icon-provider/SemanticIconProvider.js +23 -21
  126. package/dist-es/semantic-icon-provider/SemanticIconProvider.js.map +1 -1
  127. package/dist-es/spinner/Spinner.js +1 -0
  128. package/dist-es/spinner/Spinner.js.map +1 -1
  129. package/dist-es/tabs/Tab.css.js +4 -0
  130. package/dist-es/tabs/Tab.css.js.map +1 -0
  131. package/dist-es/tabs/Tab.js +209 -0
  132. package/dist-es/tabs/Tab.js.map +1 -0
  133. package/dist-es/tabs/TabAction.js +61 -0
  134. package/dist-es/tabs/TabAction.js.map +1 -0
  135. package/dist-es/tabs/TabBar.css.js +4 -0
  136. package/dist-es/tabs/TabBar.css.js.map +1 -0
  137. package/dist-es/tabs/TabBar.js +43 -0
  138. package/dist-es/tabs/TabBar.js.map +1 -0
  139. package/dist-es/tabs/TabList.css.js +4 -0
  140. package/dist-es/tabs/TabList.css.js.map +1 -0
  141. package/dist-es/tabs/TabList.js +279 -0
  142. package/dist-es/tabs/TabList.js.map +1 -0
  143. package/dist-es/tabs/TabPanel.css.js +4 -0
  144. package/dist-es/tabs/TabPanel.css.js.map +1 -0
  145. package/dist-es/tabs/TabPanel.js +96 -0
  146. package/dist-es/tabs/TabPanel.js.map +1 -0
  147. package/dist-es/tabs/TabTrigger.css.js +4 -0
  148. package/dist-es/tabs/TabTrigger.css.js.map +1 -0
  149. package/dist-es/tabs/TabTrigger.js +186 -0
  150. package/dist-es/tabs/TabTrigger.js.map +1 -0
  151. package/dist-es/tabs/Tabs.css.js +4 -0
  152. package/dist-es/tabs/Tabs.css.js.map +1 -0
  153. package/dist-es/tabs/Tabs.js +198 -0
  154. package/dist-es/tabs/Tabs.js.map +1 -0
  155. package/dist-es/tabs/internal/contexts/TabContext.js +23 -0
  156. package/dist-es/tabs/internal/contexts/TabContext.js.map +1 -0
  157. package/dist-es/tabs/internal/contexts/TabListLayoutContext.js +16 -0
  158. package/dist-es/tabs/internal/contexts/TabListLayoutContext.js.map +1 -0
  159. package/dist-es/tabs/internal/contexts/TabSlotRegistryContext.js +19 -0
  160. package/dist-es/tabs/internal/contexts/TabSlotRegistryContext.js.map +1 -0
  161. package/dist-es/tabs/internal/contexts/TabsContext.js +47 -0
  162. package/dist-es/tabs/internal/contexts/TabsContext.js.map +1 -0
  163. package/dist-es/tabs/internal/hooks/useFocusWithRetry.js +62 -0
  164. package/dist-es/tabs/internal/hooks/useFocusWithRetry.js.map +1 -0
  165. package/dist-es/tabs/internal/hooks/useTabListRecovery.js +74 -0
  166. package/dist-es/tabs/internal/hooks/useTabListRecovery.js.map +1 -0
  167. package/dist-es/tabs/internal/hooks/useTabRemovalHandler.js +163 -0
  168. package/dist-es/tabs/internal/hooks/useTabRemovalHandler.js.map +1 -0
  169. package/dist-es/tabs/internal/hooks/useTabSelectionFocus.js +85 -0
  170. package/dist-es/tabs/internal/hooks/useTabSelectionFocus.js.map +1 -0
  171. package/dist-es/tabs/internal/overflow/TabOverflowList.css.js +4 -0
  172. package/dist-es/tabs/internal/overflow/TabOverflowList.css.js.map +1 -0
  173. package/dist-es/tabs/internal/overflow/TabOverflowList.js +243 -0
  174. package/dist-es/tabs/internal/overflow/TabOverflowList.js.map +1 -0
  175. package/dist-es/tabs/internal/overflow/TabSlot.js +28 -0
  176. package/dist-es/tabs/internal/overflow/TabSlot.js.map +1 -0
  177. package/dist-es/tabs/internal/overflow/overflowMath.js +82 -0
  178. package/dist-es/tabs/internal/overflow/overflowMath.js.map +1 -0
  179. package/dist-es/tabs/internal/overflow/useOverflow.js +271 -0
  180. package/dist-es/tabs/internal/overflow/useOverflow.js.map +1 -0
  181. package/dist-es/tabs/internal/overflow/useOverflowLayoutState.js +97 -0
  182. package/dist-es/tabs/internal/overflow/useOverflowLayoutState.js.map +1 -0
  183. package/dist-es/tabs/internal/overflow/useOverflowSelectionState.js +66 -0
  184. package/dist-es/tabs/internal/overflow/useOverflowSelectionState.js.map +1 -0
  185. package/dist-es/tabs/internal/overflow/useRenderedTabWidth.js +90 -0
  186. package/dist-es/tabs/internal/overflow/useRenderedTabWidth.js.map +1 -0
  187. package/dist-es/tabs/internal/overflow/widthMeasurement.js +36 -0
  188. package/dist-es/tabs/internal/overflow/widthMeasurement.js.map +1 -0
  189. package/dist-es/tabs/internal/registry/useCollection.js +195 -0
  190. package/dist-es/tabs/internal/registry/useCollection.js.map +1 -0
  191. package/dist-es/tabs/internal/registry/useRenderedTabsRegistry.js +204 -0
  192. package/dist-es/tabs/internal/registry/useRenderedTabsRegistry.js.map +1 -0
  193. package/dist-es/tabs/internal/utils/domUtils.js +11 -0
  194. package/dist-es/tabs/internal/utils/domUtils.js.map +1 -0
  195. package/dist-es/tooltip/useAriaAnnounce.js +1 -0
  196. package/dist-es/tooltip/useAriaAnnounce.js.map +1 -1
  197. package/dist-types/aria-announcer/AriaAnnounce.d.ts +9 -2
  198. package/dist-types/aria-announcer/AriaAnnouncerContext.d.ts +26 -2
  199. package/dist-types/aria-announcer/AriaAnnouncerProvider.d.ts +6 -7
  200. package/dist-types/aria-announcer/announcementRegistry.d.ts +5 -0
  201. package/dist-types/index.d.ts +2 -0
  202. package/dist-types/rating/Rating.d.ts +48 -0
  203. package/dist-types/rating/RatingItem.d.ts +47 -0
  204. package/dist-types/rating/index.d.ts +1 -0
  205. package/dist-types/semantic-icon-provider/SemanticIconProvider.d.ts +21 -19
  206. package/dist-types/tabs/Tab.d.ts +12 -0
  207. package/dist-types/tabs/TabAction.d.ts +4 -0
  208. package/dist-types/tabs/TabBar.d.ts +12 -0
  209. package/dist-types/tabs/TabList.d.ts +12 -0
  210. package/dist-types/tabs/TabPanel.d.ts +9 -0
  211. package/dist-types/tabs/TabTrigger.d.ts +4 -0
  212. package/dist-types/tabs/Tabs.d.ts +20 -0
  213. package/dist-types/tabs/index.d.ts +7 -0
  214. package/dist-types/tabs/internal/contexts/TabContext.d.ts +12 -0
  215. package/dist-types/tabs/internal/contexts/TabListLayoutContext.d.ts +9 -0
  216. package/dist-types/tabs/internal/contexts/TabSlotRegistryContext.d.ts +5 -0
  217. package/dist-types/tabs/internal/contexts/TabsContext.d.ts +43 -0
  218. package/dist-types/tabs/internal/hooks/useFocusWithRetry.d.ts +9 -0
  219. package/dist-types/tabs/internal/hooks/useTabListRecovery.d.ts +12 -0
  220. package/dist-types/tabs/internal/hooks/useTabRemovalHandler.d.ts +32 -0
  221. package/dist-types/tabs/internal/hooks/useTabSelectionFocus.d.ts +15 -0
  222. package/dist-types/tabs/internal/overflow/TabOverflowList.d.ts +10 -0
  223. package/dist-types/tabs/internal/overflow/TabSlot.d.ts +6 -0
  224. package/dist-types/tabs/internal/overflow/overflowMath.d.ts +18 -0
  225. package/dist-types/tabs/internal/overflow/useOverflow.d.ts +11 -0
  226. package/dist-types/tabs/internal/overflow/useOverflowLayoutState.d.ts +13 -0
  227. package/dist-types/tabs/internal/overflow/useOverflowSelectionState.d.ts +13 -0
  228. package/dist-types/tabs/internal/overflow/useRenderedTabWidth.d.ts +12 -0
  229. package/dist-types/tabs/internal/overflow/widthMeasurement.d.ts +5 -0
  230. package/dist-types/tabs/internal/registry/useCollection.d.ts +30 -0
  231. package/dist-types/tabs/internal/registry/useRenderedTabsRegistry.d.ts +12 -0
  232. package/dist-types/tabs/internal/utils/domUtils.d.ts +1 -0
  233. package/package.json +3 -1
@@ -0,0 +1,243 @@
1
+ import { jsxs, Fragment, jsx } from 'react/jsx-runtime';
2
+ import { offset, size, flip, shift, useInteractions, useClick, useDismiss } from '@floating-ui/react';
3
+ import { useComponentCssInjection } from '@salt-ds/styles';
4
+ import { useWindow } from '@salt-ds/window';
5
+ import { clsx } from 'clsx';
6
+ import { forwardRef, useRef, useEffect } from 'react';
7
+ import { Button } from '../../../button/Button.js';
8
+ import { useIcon } from '../../../semantic-icon-provider/SemanticIconProvider.js';
9
+ import { makePrefixer } from '../../../utils/makePrefixer.js';
10
+ import { useIsomorphicLayoutEffect } from '../../../utils/useIsomorphicLayoutEffect.js';
11
+ import { useFloatingUI, useFloatingComponent } from '../../../utils/useFloatingUI/useFloatingUI.js';
12
+ import { useForkRef } from '../../../utils/useForkRef.js';
13
+ import { useId } from '../../../utils/useId.js';
14
+ import '../../../salt-provider/SaltProvider.js';
15
+ import '../../../viewport/ViewportProvider.js';
16
+ import { useTabs } from '../contexts/TabsContext.js';
17
+ import { isHTMLElement } from '../utils/domUtils.js';
18
+ import css_248z from './TabOverflowList.css.js';
19
+ import { TabSlot } from './TabSlot.js';
20
+ import { seedWidthMap, getMeasuredWidth, updateWidthMap } from './widthMeasurement.js';
21
+
22
+ const withBaseName = makePrefixer("saltTabOverflowList");
23
+ const TabOverflowList = forwardRef(
24
+ function TabOverflowList2(props, ref) {
25
+ var _a, _b;
26
+ const {
27
+ buttonRef,
28
+ className,
29
+ hiddenValues,
30
+ order,
31
+ open,
32
+ setOpen,
33
+ ...rest
34
+ } = props;
35
+ const targetWindow = useWindow();
36
+ useComponentCssInjection({
37
+ testId: "salt-tab-overflow-list",
38
+ css: css_248z,
39
+ window: targetWindow
40
+ });
41
+ const overflowRef = useRef(null);
42
+ const hadOverflowItemsRef = useRef(false);
43
+ const { OverflowIcon } = useIcon();
44
+ const { registerTab, updateTab, activeTab } = useTabs();
45
+ const { refs, x, y, strategy, context, elements } = useFloatingUI({
46
+ open,
47
+ onOpenChange(open2, _, reason) {
48
+ var _a2;
49
+ setOpen(open2);
50
+ if (reason === "escape-key") {
51
+ (_a2 = overflowRef.current) == null ? void 0 : _a2.focus();
52
+ }
53
+ },
54
+ placement: "bottom-start",
55
+ middleware: [
56
+ offset(1),
57
+ size({
58
+ apply({ elements: elements2, availableHeight }) {
59
+ Object.assign(elements2.floating.style, {
60
+ maxHeight: `max(calc((var(--salt-size-base) + var(--salt-spacing-100)) * 5), calc(${availableHeight}px - var(--salt-spacing-100)))`
61
+ });
62
+ }
63
+ }),
64
+ flip(),
65
+ shift({
66
+ padding: 8
67
+ })
68
+ ]
69
+ });
70
+ const { getFloatingProps, getReferenceProps } = useInteractions([
71
+ useClick(context),
72
+ useDismiss(context)
73
+ ]);
74
+ const handleListRef = useForkRef(ref, refs.setFloating);
75
+ const handleButtonRef = useForkRef(
76
+ buttonRef,
77
+ refs.setReference
78
+ );
79
+ const handleRef = useForkRef(handleButtonRef, overflowRef);
80
+ const { Component: FloatingComponent } = useFloatingComponent();
81
+ const overflowId = useId();
82
+ const overlayId = useId();
83
+ const handleFocus = () => {
84
+ if (overflowId) {
85
+ activeTab.current = { value: overflowId, id: overflowId };
86
+ }
87
+ };
88
+ const handleFloatingKeyDown = (event) => {
89
+ if (event.key !== "Tab") {
90
+ return;
91
+ }
92
+ if (event.shiftKey) {
93
+ event.preventDefault();
94
+ setOpen(false);
95
+ const scheduleFocus = targetWindow == null ? void 0 : targetWindow.requestAnimationFrame;
96
+ if (scheduleFocus) {
97
+ scheduleFocus(
98
+ () => {
99
+ var _a2;
100
+ return (_a2 = overflowRef.current) == null ? void 0 : _a2.focus({ preventScroll: true });
101
+ }
102
+ );
103
+ return;
104
+ }
105
+ queueMicrotask(
106
+ () => {
107
+ var _a2;
108
+ return (_a2 = overflowRef.current) == null ? void 0 : _a2.focus({ preventScroll: true });
109
+ }
110
+ );
111
+ return;
112
+ }
113
+ };
114
+ const overflowItemCount = hiddenValues.length;
115
+ const hasOverflowItems = overflowItemCount > 0;
116
+ const initialOrderRef = useRef(order);
117
+ useEffect(() => {
118
+ var _a2;
119
+ const tabList = (_a2 = overflowRef.current) == null ? void 0 : _a2.parentElement;
120
+ const resizeObserverCtor = targetWindow == null ? void 0 : targetWindow.ResizeObserver;
121
+ if (!open || !tabList || !resizeObserverCtor) {
122
+ return;
123
+ }
124
+ const observedElements = [tabList, tabList.parentElement].filter(
125
+ (element) => element != null
126
+ );
127
+ const widths = seedWidthMap(observedElements);
128
+ const resizeObserver = new resizeObserverCtor(
129
+ (entries) => {
130
+ for (const entry of entries) {
131
+ if (!isHTMLElement(entry.target)) {
132
+ continue;
133
+ }
134
+ const nextWidth = entry.contentRect.width || getMeasuredWidth(entry.target);
135
+ if (updateWidthMap(widths, entry.target, nextWidth)) {
136
+ setOpen(false);
137
+ return;
138
+ }
139
+ }
140
+ }
141
+ );
142
+ for (const element of observedElements) {
143
+ resizeObserver.observe(element);
144
+ }
145
+ return () => {
146
+ resizeObserver.disconnect();
147
+ };
148
+ }, [open, setOpen, targetWindow]);
149
+ useIsomorphicLayoutEffect(() => {
150
+ if (open && !hasOverflowItems) {
151
+ setOpen(false);
152
+ }
153
+ }, [hasOverflowItems, open, setOpen]);
154
+ useIsomorphicLayoutEffect(() => {
155
+ if (hasOverflowItems && !hadOverflowItemsRef.current) {
156
+ setOpen(false);
157
+ }
158
+ hadOverflowItemsRef.current = hasOverflowItems;
159
+ }, [hasOverflowItems, setOpen]);
160
+ useIsomorphicLayoutEffect(() => {
161
+ if (overflowId && overflowRef.current && hasOverflowItems) {
162
+ const item = {
163
+ id: overflowId,
164
+ value: overflowId,
165
+ element: overflowRef.current,
166
+ location: "main",
167
+ order: initialOrderRef.current
168
+ };
169
+ return registerTab(item);
170
+ }
171
+ }, [hasOverflowItems, overflowId, registerTab]);
172
+ useIsomorphicLayoutEffect(() => {
173
+ if (!overflowId || !hasOverflowItems) {
174
+ return;
175
+ }
176
+ updateTab(overflowId, {
177
+ element: overflowRef.current,
178
+ location: "main",
179
+ order
180
+ });
181
+ }, [hasOverflowItems, order, overflowId, updateTab]);
182
+ if (!hasOverflowItems) return null;
183
+ return /* @__PURE__ */ jsxs(Fragment, { children: [
184
+ /* @__PURE__ */ jsx(
185
+ Button,
186
+ {
187
+ className: clsx(withBaseName(), className),
188
+ "data-overflowbutton": true,
189
+ appearance: "transparent",
190
+ sentiment: "neutral",
191
+ ...getReferenceProps({
192
+ onFocus: handleFocus
193
+ }),
194
+ ref: handleRef,
195
+ "aria-label": "Overflow",
196
+ "aria-expanded": open,
197
+ "aria-controls": overlayId,
198
+ role: "tab",
199
+ tabIndex: -1,
200
+ ...rest,
201
+ children: /* @__PURE__ */ jsx(OverflowIcon, { "aria-hidden": true })
202
+ }
203
+ ),
204
+ /* @__PURE__ */ jsx(
205
+ FloatingComponent,
206
+ {
207
+ ref: handleListRef,
208
+ ...getFloatingProps({
209
+ id: overlayId,
210
+ onKeyDown: handleFloatingKeyDown
211
+ }),
212
+ focusManagerProps: context ? {
213
+ context,
214
+ initialFocus: 0,
215
+ returnFocus: false,
216
+ modal: false,
217
+ closeOnFocusOut: true
218
+ } : void 0,
219
+ className: withBaseName("list"),
220
+ open,
221
+ left: x ?? 0,
222
+ top: y ?? 0,
223
+ position: strategy,
224
+ width: (_a = elements.floating) == null ? void 0 : _a.offsetWidth,
225
+ height: (_b = elements.floating) == null ? void 0 : _b.offsetHeight,
226
+ children: /* @__PURE__ */ jsx(
227
+ "div",
228
+ {
229
+ role: "tablist",
230
+ "aria-orientation": "vertical",
231
+ className: withBaseName("listContainer"),
232
+ "aria-label": "Overflow tab options",
233
+ children: hiddenValues.map((value) => /* @__PURE__ */ jsx(TabSlot, { slotId: `overflow:${value}`, value }, value))
234
+ }
235
+ )
236
+ }
237
+ )
238
+ ] });
239
+ }
240
+ );
241
+
242
+ export { TabOverflowList };
243
+ //# sourceMappingURL=TabOverflowList.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TabOverflowList.js","sources":["../src/tabs/internal/overflow/TabOverflowList.tsx"],"sourcesContent":["import {\n flip,\n offset,\n shift,\n size,\n useClick,\n useDismiss,\n useInteractions,\n} from \"@floating-ui/react\";\nimport { useComponentCssInjection } from \"@salt-ds/styles\";\nimport { useWindow } from \"@salt-ds/window\";\nimport { clsx } from \"clsx\";\nimport {\n type ComponentPropsWithoutRef,\n type Dispatch,\n forwardRef,\n type KeyboardEvent,\n type Ref,\n type SetStateAction,\n useEffect,\n useRef,\n} from \"react\";\nimport { Button } from \"../../../button\";\nimport { useIcon } from \"../../../semantic-icon-provider\";\nimport {\n makePrefixer,\n useFloatingComponent,\n useFloatingUI,\n useForkRef,\n useId,\n useIsomorphicLayoutEffect,\n} from \"../../../utils\";\nimport { useTabs } from \"../contexts/TabsContext\";\nimport { isHTMLElement } from \"../utils/domUtils\";\nimport tabOverflowListCss from \"./TabOverflowList.css\";\nimport { TabSlot } from \"./TabSlot\";\nimport {\n getMeasuredWidth,\n seedWidthMap,\n updateWidthMap,\n} from \"./widthMeasurement\";\n\ninterface TabOverflowListProps extends ComponentPropsWithoutRef<\"button\"> {\n buttonRef?: Ref<HTMLButtonElement>;\n hiddenValues: string[];\n open: boolean;\n setOpen: Dispatch<SetStateAction<boolean>>;\n order: number;\n}\n\nconst withBaseName = makePrefixer(\"saltTabOverflowList\");\n\nexport const TabOverflowList = forwardRef<HTMLDivElement, TabOverflowListProps>(\n function TabOverflowList(props, ref) {\n const {\n buttonRef,\n className,\n hiddenValues,\n order,\n open,\n setOpen,\n ...rest\n } = props;\n\n const targetWindow = useWindow();\n useComponentCssInjection({\n testId: \"salt-tab-overflow-list\",\n css: tabOverflowListCss,\n window: targetWindow,\n });\n\n const overflowRef = useRef<HTMLButtonElement>(null);\n const hadOverflowItemsRef = useRef(false);\n\n const { OverflowIcon } = useIcon();\n const { registerTab, updateTab, activeTab } = useTabs();\n\n const { refs, x, y, strategy, context, elements } = useFloatingUI({\n open: open,\n onOpenChange(open, _, reason) {\n setOpen(open);\n\n if (reason === \"escape-key\") {\n overflowRef.current?.focus();\n }\n },\n placement: \"bottom-start\",\n middleware: [\n offset(1),\n size({\n apply({ elements, availableHeight }) {\n Object.assign(elements.floating.style, {\n maxHeight: `max(calc((var(--salt-size-base) + var(--salt-spacing-100)) * 5), calc(${availableHeight}px - var(--salt-spacing-100)))`,\n });\n },\n }),\n flip(),\n shift({\n padding: 8,\n }),\n ],\n });\n\n const { getFloatingProps, getReferenceProps } = useInteractions([\n useClick(context),\n useDismiss(context),\n ]);\n\n const handleListRef = useForkRef<HTMLDivElement>(ref, refs.setFloating);\n\n const handleButtonRef = useForkRef<HTMLButtonElement>(\n buttonRef,\n refs.setReference,\n );\n const handleRef = useForkRef(handleButtonRef, overflowRef);\n\n const { Component: FloatingComponent } = useFloatingComponent();\n\n const overflowId = useId();\n const overlayId = useId();\n\n const handleFocus = () => {\n if (overflowId) {\n activeTab.current = { value: overflowId, id: overflowId };\n }\n };\n\n const handleFloatingKeyDown = (event: KeyboardEvent<HTMLDivElement>) => {\n if (event.key !== \"Tab\") {\n return;\n }\n\n if (event.shiftKey) {\n event.preventDefault();\n setOpen(false);\n const scheduleFocus = targetWindow?.requestAnimationFrame;\n if (scheduleFocus) {\n scheduleFocus(() =>\n overflowRef.current?.focus({ preventScroll: true }),\n );\n return;\n }\n\n queueMicrotask(() =>\n overflowRef.current?.focus({ preventScroll: true }),\n );\n return;\n }\n };\n\n const overflowItemCount = hiddenValues.length;\n const hasOverflowItems = overflowItemCount > 0;\n const initialOrderRef = useRef(order);\n\n useEffect(() => {\n const tabList = overflowRef.current?.parentElement;\n const resizeObserverCtor = (\n targetWindow as\n | (Window & { ResizeObserver?: typeof ResizeObserver })\n | undefined\n )?.ResizeObserver;\n if (!open || !tabList || !resizeObserverCtor) {\n return;\n }\n\n const observedElements = [tabList, tabList.parentElement].filter(\n (element): element is HTMLElement => element != null,\n );\n const widths = seedWidthMap(observedElements);\n\n const resizeObserver = new resizeObserverCtor(\n (entries: ResizeObserverEntry[]) => {\n for (const entry of entries) {\n if (!isHTMLElement(entry.target)) {\n continue;\n }\n\n const nextWidth =\n entry.contentRect.width || getMeasuredWidth(entry.target);\n if (updateWidthMap(widths, entry.target, nextWidth)) {\n setOpen(false);\n return;\n }\n }\n },\n );\n\n for (const element of observedElements) {\n resizeObserver.observe(element);\n }\n\n return () => {\n resizeObserver.disconnect();\n };\n }, [open, setOpen, targetWindow]);\n\n useIsomorphicLayoutEffect(() => {\n if (open && !hasOverflowItems) {\n setOpen(false);\n }\n }, [hasOverflowItems, open, setOpen]);\n\n useIsomorphicLayoutEffect(() => {\n if (hasOverflowItems && !hadOverflowItemsRef.current) {\n setOpen(false);\n }\n\n hadOverflowItemsRef.current = hasOverflowItems;\n }, [hasOverflowItems, setOpen]);\n\n useIsomorphicLayoutEffect(() => {\n if (overflowId && overflowRef.current && hasOverflowItems) {\n const item = {\n id: overflowId,\n value: overflowId,\n element: overflowRef.current,\n location: \"main\" as const,\n order: initialOrderRef.current,\n };\n\n return registerTab(item);\n }\n }, [hasOverflowItems, overflowId, registerTab]);\n\n useIsomorphicLayoutEffect(() => {\n if (!overflowId || !hasOverflowItems) {\n return;\n }\n\n updateTab(overflowId, {\n element: overflowRef.current,\n location: \"main\",\n order,\n });\n }, [hasOverflowItems, order, overflowId, updateTab]);\n\n if (!hasOverflowItems) return null;\n\n return (\n <>\n <Button\n className={clsx(withBaseName(), className)}\n data-overflowbutton\n appearance=\"transparent\"\n sentiment=\"neutral\"\n {...getReferenceProps({\n onFocus: handleFocus,\n })}\n ref={handleRef}\n aria-label=\"Overflow\"\n aria-expanded={open}\n aria-controls={overlayId}\n role=\"tab\"\n tabIndex={-1}\n {...rest}\n >\n <OverflowIcon aria-hidden />\n </Button>\n <FloatingComponent\n ref={handleListRef}\n {...getFloatingProps({\n id: overlayId,\n onKeyDown: handleFloatingKeyDown,\n })}\n focusManagerProps={\n context\n ? {\n context,\n initialFocus: 0,\n returnFocus: false,\n modal: false,\n closeOnFocusOut: true,\n }\n : undefined\n }\n className={withBaseName(\"list\")}\n open={open}\n left={x ?? 0}\n top={y ?? 0}\n position={strategy}\n width={elements.floating?.offsetWidth}\n height={elements.floating?.offsetHeight}\n >\n <div\n role=\"tablist\"\n aria-orientation=\"vertical\"\n className={withBaseName(\"listContainer\")}\n aria-label=\"Overflow tab options\"\n >\n {hiddenValues.map((value) => (\n <TabSlot key={value} slotId={`overflow:${value}`} value={value} />\n ))}\n </div>\n </FloatingComponent>\n </>\n );\n },\n);\n"],"names":["TabOverflowList","tabOverflowListCss","open","_a","elements"],"mappings":"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"}
@@ -0,0 +1,28 @@
1
+ import { jsx } from 'react/jsx-runtime';
2
+ import { useCallback } from 'react';
3
+ import { useTabSlotRegistry } from '../contexts/TabSlotRegistryContext.js';
4
+
5
+ function TabSlot({ slotId, value, ...rest }) {
6
+ const slotRegistry = useTabSlotRegistry();
7
+ const handleRef = useCallback(
8
+ (element) => {
9
+ slotRegistry == null ? void 0 : slotRegistry.registerSlot(slotId, element);
10
+ },
11
+ [slotId, slotRegistry]
12
+ );
13
+ return /* @__PURE__ */ jsx(
14
+ "div",
15
+ {
16
+ role: "presentation",
17
+ "data-tabslot": "",
18
+ "data-slotid": slotId,
19
+ "data-value": value,
20
+ ref: handleRef,
21
+ style: { display: "contents" },
22
+ ...rest
23
+ }
24
+ );
25
+ }
26
+
27
+ export { TabSlot };
28
+ //# sourceMappingURL=TabSlot.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TabSlot.js","sources":["../src/tabs/internal/overflow/TabSlot.tsx"],"sourcesContent":["import { type ComponentPropsWithoutRef, useCallback } from \"react\";\nimport { useTabSlotRegistry } from \"../contexts/TabSlotRegistryContext\";\n\nexport interface TabSlotProps extends ComponentPropsWithoutRef<\"div\"> {\n slotId: string;\n value: string;\n}\n\nexport function TabSlot({ slotId, value, ...rest }: TabSlotProps) {\n const slotRegistry = useTabSlotRegistry();\n const handleRef = useCallback(\n (element: HTMLDivElement | null) => {\n slotRegistry?.registerSlot(slotId, element);\n },\n [slotId, slotRegistry],\n );\n\n return (\n <div\n role=\"presentation\"\n data-tabslot=\"\"\n data-slotid={slotId}\n data-value={value}\n ref={handleRef}\n style={{ display: \"contents\" }}\n {...rest}\n />\n );\n}\n"],"names":[],"mappings":";;;;AAQO,SAAS,QAAQ,EAAE,MAAA,EAAQ,KAAA,EAAO,GAAG,MAAK,EAAiB;AAChE,EAAA,MAAM,eAAe,kBAAA,EAAmB;AACxC,EAAA,MAAM,SAAA,GAAY,WAAA;AAAA,IAChB,CAAC,OAAA,KAAmC;AAClC,MAAA,YAAA,IAAA,IAAA,GAAA,MAAA,GAAA,YAAA,CAAc,aAAa,MAAA,EAAQ,OAAA,CAAA;AAAA,IACrC,CAAA;AAAA,IACA,CAAC,QAAQ,YAAY;AAAA,GACvB;AAEA,EAAA,uBACE,GAAA;AAAA,IAAC,KAAA;AAAA,IAAA;AAAA,MACC,IAAA,EAAK,cAAA;AAAA,MACL,cAAA,EAAa,EAAA;AAAA,MACb,aAAA,EAAa,MAAA;AAAA,MACb,YAAA,EAAY,KAAA;AAAA,MACZ,GAAA,EAAK,SAAA;AAAA,MACL,KAAA,EAAO,EAAE,OAAA,EAAS,UAAA,EAAW;AAAA,MAC5B,GAAG;AAAA;AAAA,GACN;AAEJ;;;;"}
@@ -0,0 +1,82 @@
1
+ const MIN_TRUSTED_TAB_WIDTH = 0.5;
2
+ function calculateVisibleCount({
3
+ gap,
4
+ maxWidth: initialMaxWidth,
5
+ overflowWidth,
6
+ pinnedValue,
7
+ tabs
8
+ }) {
9
+ let maxWidth = initialMaxWidth;
10
+ let currentWidth = 0;
11
+ let nextVisibleCount = 0;
12
+ const visibleItems = [];
13
+ while (nextVisibleCount < tabs.length) {
14
+ const item = tabs[nextVisibleCount];
15
+ if (!item) {
16
+ break;
17
+ }
18
+ if (item.width == null) {
19
+ return null;
20
+ }
21
+ const itemWidth = item.width + gap;
22
+ if (currentWidth + itemWidth > maxWidth) {
23
+ break;
24
+ }
25
+ currentWidth += itemWidth;
26
+ visibleItems.push(item);
27
+ nextVisibleCount += 1;
28
+ }
29
+ const allTabsFit = nextVisibleCount >= tabs.length;
30
+ if (allTabsFit) {
31
+ return nextVisibleCount;
32
+ }
33
+ maxWidth -= overflowWidth;
34
+ while (currentWidth > maxWidth) {
35
+ const removed = visibleItems.pop();
36
+ if (!removed) {
37
+ break;
38
+ }
39
+ if (removed.width == null) {
40
+ return null;
41
+ }
42
+ currentWidth -= removed.width + gap;
43
+ nextVisibleCount -= 1;
44
+ }
45
+ const pinnedItem = pinnedValue == null ? null : tabs.find((item) => item.value === pinnedValue) ?? null;
46
+ if (pinnedItem && !visibleItems.includes(pinnedItem)) {
47
+ if (pinnedItem.width == null) {
48
+ return null;
49
+ }
50
+ const pinnedWidth = pinnedItem.width + gap;
51
+ while (currentWidth + pinnedWidth > maxWidth) {
52
+ const removed = visibleItems.pop();
53
+ if (!removed) {
54
+ break;
55
+ }
56
+ if (removed.width == null) {
57
+ return null;
58
+ }
59
+ currentWidth -= removed.width + gap;
60
+ nextVisibleCount -= 1;
61
+ }
62
+ }
63
+ return Math.max(0, nextVisibleCount);
64
+ }
65
+ function partitionVisibleValues(orderedValues, visibleCount, pinnedValue) {
66
+ let visibleValues = orderedValues.slice(0, visibleCount);
67
+ let hiddenValues = orderedValues.slice(visibleCount);
68
+ const hiddenPinnedIndex = pinnedValue != null ? hiddenValues.indexOf(pinnedValue) : -1;
69
+ if (hiddenPinnedIndex !== -1) {
70
+ const pinnedHiddenValue = hiddenValues[hiddenPinnedIndex];
71
+ hiddenValues = hiddenValues.filter(
72
+ (_, index) => index !== hiddenPinnedIndex
73
+ );
74
+ if (pinnedHiddenValue !== void 0) {
75
+ visibleValues = [...visibleValues, pinnedHiddenValue];
76
+ }
77
+ }
78
+ return { visibleValues, hiddenValues };
79
+ }
80
+
81
+ export { MIN_TRUSTED_TAB_WIDTH, calculateVisibleCount, partitionVisibleValues };
82
+ //# sourceMappingURL=overflowMath.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"overflowMath.js","sources":["../src/tabs/internal/overflow/overflowMath.ts"],"sourcesContent":["export const MIN_TRUSTED_TAB_WIDTH = 0.5;\n\nexport interface MeasuredOverflowTab {\n value: string;\n width: number | null;\n}\n\ninterface CalculateVisibleCountProps {\n gap: number;\n maxWidth: number;\n overflowWidth: number;\n pinnedValue?: string;\n tabs: MeasuredOverflowTab[];\n}\n\nexport function calculateVisibleCount({\n gap,\n maxWidth: initialMaxWidth,\n overflowWidth,\n pinnedValue,\n tabs,\n}: CalculateVisibleCountProps) {\n let maxWidth = initialMaxWidth;\n let currentWidth = 0;\n let nextVisibleCount = 0;\n const visibleItems: MeasuredOverflowTab[] = [];\n\n while (nextVisibleCount < tabs.length) {\n const item = tabs[nextVisibleCount];\n if (!item) {\n break;\n }\n\n if (item.width == null) {\n return null;\n }\n\n const itemWidth = item.width + gap;\n if (currentWidth + itemWidth > maxWidth) {\n break;\n }\n\n currentWidth += itemWidth;\n visibleItems.push(item);\n nextVisibleCount += 1;\n }\n\n const allTabsFit = nextVisibleCount >= tabs.length;\n if (allTabsFit) {\n return nextVisibleCount;\n }\n\n maxWidth -= overflowWidth;\n\n while (currentWidth > maxWidth) {\n const removed = visibleItems.pop();\n if (!removed) {\n break;\n }\n if (removed.width == null) {\n return null;\n }\n currentWidth -= removed.width + gap;\n nextVisibleCount -= 1;\n }\n\n const pinnedItem =\n pinnedValue == null\n ? null\n : (tabs.find((item) => item.value === pinnedValue) ?? null);\n\n if (pinnedItem && !visibleItems.includes(pinnedItem)) {\n if (pinnedItem.width == null) {\n return null;\n }\n\n const pinnedWidth = pinnedItem.width + gap;\n while (currentWidth + pinnedWidth > maxWidth) {\n const removed = visibleItems.pop();\n if (!removed) {\n break;\n }\n if (removed.width == null) {\n return null;\n }\n currentWidth -= removed.width + gap;\n nextVisibleCount -= 1;\n }\n }\n\n return Math.max(0, nextVisibleCount);\n}\n\nexport function partitionVisibleValues(\n orderedValues: string[],\n visibleCount: number,\n pinnedValue?: string,\n) {\n let visibleValues = orderedValues.slice(0, visibleCount);\n let hiddenValues = orderedValues.slice(visibleCount);\n\n const hiddenPinnedIndex =\n pinnedValue != null ? hiddenValues.indexOf(pinnedValue) : -1;\n\n if (hiddenPinnedIndex !== -1) {\n const pinnedHiddenValue = hiddenValues[hiddenPinnedIndex];\n hiddenValues = hiddenValues.filter(\n (_, index) => index !== hiddenPinnedIndex,\n );\n if (pinnedHiddenValue !== undefined) {\n visibleValues = [...visibleValues, pinnedHiddenValue];\n }\n }\n\n return { visibleValues, hiddenValues };\n}\n"],"names":[],"mappings":"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"}
@@ -0,0 +1,271 @@
1
+ import { useMemo, useState, useRef, useCallback, useEffect } from 'react';
2
+ import 'clsx';
3
+ import { ownerWindow } from '../../../utils/ownerWindow.js';
4
+ import 'react/jsx-runtime';
5
+ import { useIsomorphicLayoutEffect } from '../../../utils/useIsomorphicLayoutEffect.js';
6
+ import '../../../utils/useFloatingUI/useFloatingUI.js';
7
+ import '../../../utils/useId.js';
8
+ import '../../../salt-provider/SaltProvider.js';
9
+ import '../../../viewport/ViewportProvider.js';
10
+ import { isHTMLElement } from '../utils/domUtils.js';
11
+ import { calculateVisibleCount, MIN_TRUSTED_TAB_WIDTH, partitionVisibleValues } from './overflowMath.js';
12
+ import { seedWidthMap, updateWidthMap, getMeasuredWidth, getGapValue } from './widthMeasurement.js';
13
+
14
+ function getTabWidth(tab) {
15
+ const width = tab.width || getMeasuredWidth(tab.root);
16
+ return width > MIN_TRUSTED_TAB_WIDTH ? width : null;
17
+ }
18
+ function getAvailableWidth(element) {
19
+ const parent = element.parentElement;
20
+ if (!parent) {
21
+ return getMeasuredWidth(element);
22
+ }
23
+ const parentWidth = getMeasuredWidth(parent);
24
+ const parentStyles = ownerWindow(parent).getComputedStyle(parent);
25
+ const parentGap = getGapValue(parentStyles);
26
+ const siblings = Array.from(parent.children).filter(
27
+ (child) => {
28
+ if (!isHTMLElement(child) || child === element) {
29
+ return false;
30
+ }
31
+ return ownerWindow(child).getComputedStyle(child).display !== "none";
32
+ }
33
+ );
34
+ const siblingWidth = siblings.reduce((width, sibling) => {
35
+ return width + getMeasuredWidth(sibling);
36
+ }, 0);
37
+ const gapCount = siblings.length > 0 ? siblings.length : 0;
38
+ const availableWidth = Math.max(
39
+ 0,
40
+ parentWidth - siblingWidth - gapCount * parentGap
41
+ );
42
+ return availableWidth;
43
+ }
44
+ function isSelectedValueHidden(selected, hiddenValues) {
45
+ return selected !== void 0 && hiddenValues.includes(selected);
46
+ }
47
+ function getPinnedSelectionValue(selected, selectedIsHidden, pinnedSelectionRef) {
48
+ return selectedIsHidden ? selected : pinnedSelectionRef.current;
49
+ }
50
+ function useOverflow({
51
+ container,
52
+ overflowButton,
53
+ tabs,
54
+ selected,
55
+ menuOpen
56
+ }) {
57
+ const orderedValues = useMemo(() => tabs.map((tab) => tab.value), [tabs]);
58
+ const measurementInputKey = useMemo(() => {
59
+ return tabs.map((tab) => `${tab.value}:${tab.width.toFixed(2)}`).join("\0");
60
+ }, [tabs]);
61
+ const [visibleCount, setVisibleCount] = useState(0);
62
+ const [isMeasuring, setIsMeasuring] = useState(true);
63
+ const [measureRetryVersion, setMeasureRetryVersion] = useState(0);
64
+ const pinnedSelectionRef = useRef(selected);
65
+ const previousOverflowButtonWidthRef = useRef(0);
66
+ const previousMeasurementInputKeyRef = useRef(measurementInputKey);
67
+ const previousMenuOpenRef = useRef(menuOpen);
68
+ const measureRetryFrameRef = useRef(null);
69
+ const measureRetryCountRef = useRef(0);
70
+ const baseHiddenValues = orderedValues.slice(visibleCount);
71
+ const selectedIsHidden = isSelectedValueHidden(selected, baseHiddenValues);
72
+ const pinnedValue = getPinnedSelectionValue(
73
+ selected,
74
+ selectedIsHidden,
75
+ pinnedSelectionRef
76
+ );
77
+ const getCurrentPinnedValue = useCallback(() => {
78
+ return getPinnedSelectionValue(
79
+ selected,
80
+ selectedIsHidden,
81
+ pinnedSelectionRef
82
+ );
83
+ }, [selected, selectedIsHidden]);
84
+ const markMeasurementStale = useCallback(() => {
85
+ setIsMeasuring(true);
86
+ }, []);
87
+ const measureVisibleCount = useCallback(
88
+ (pinnedValue2) => {
89
+ const element = container.current;
90
+ if (!element) {
91
+ return null;
92
+ }
93
+ const maxWidth = getAvailableWidth(element);
94
+ const styles = ownerWindow(element).getComputedStyle(element);
95
+ const gap = getGapValue(styles);
96
+ const overflowWidth = overflowButton.current ? overflowButton.current.offsetWidth + gap : 0;
97
+ const measuredTabs = tabs.map((tab) => ({
98
+ value: tab.value,
99
+ width: getTabWidth(tab)
100
+ }));
101
+ return calculateVisibleCount({
102
+ gap,
103
+ maxWidth,
104
+ overflowWidth,
105
+ pinnedValue: pinnedValue2,
106
+ tabs: measuredTabs
107
+ });
108
+ },
109
+ [container, overflowButton, tabs]
110
+ );
111
+ const clearMeasureRetry = useCallback(() => {
112
+ const element = container.current;
113
+ const frame = measureRetryFrameRef.current;
114
+ if (element && frame != null) {
115
+ ownerWindow(element).cancelAnimationFrame(frame);
116
+ }
117
+ measureRetryFrameRef.current = null;
118
+ measureRetryCountRef.current = 0;
119
+ }, [container]);
120
+ useEffect(() => {
121
+ return clearMeasureRetry;
122
+ }, [clearMeasureRetry]);
123
+ useIsomorphicLayoutEffect(() => {
124
+ if (selected !== void 0 && selectedIsHidden) {
125
+ pinnedSelectionRef.current = selected;
126
+ const nextVisibleCount = measureVisibleCount(selected);
127
+ if (nextVisibleCount == null) {
128
+ markMeasurementStale();
129
+ return;
130
+ }
131
+ if (nextVisibleCount !== visibleCount) {
132
+ setVisibleCount(nextVisibleCount);
133
+ }
134
+ if (isMeasuring) {
135
+ setIsMeasuring(false);
136
+ }
137
+ }
138
+ }, [
139
+ isMeasuring,
140
+ markMeasurementStale,
141
+ measureVisibleCount,
142
+ selected,
143
+ selectedIsHidden,
144
+ visibleCount
145
+ ]);
146
+ useEffect(() => {
147
+ const element = container.current;
148
+ if (!element || menuOpen || isMeasuring) {
149
+ return;
150
+ }
151
+ const observedElements = [element];
152
+ const parent = element.parentElement;
153
+ if (parent) {
154
+ observedElements.push(parent);
155
+ for (const child of Array.from(parent.children)) {
156
+ if (isHTMLElement(child) && child !== element) {
157
+ observedElements.push(child);
158
+ }
159
+ }
160
+ }
161
+ const widths = seedWidthMap(observedElements);
162
+ const resizeObserverCtor = ownerWindow(element).ResizeObserver;
163
+ if (!resizeObserverCtor) {
164
+ return;
165
+ }
166
+ const resizeObserver = new resizeObserverCtor(
167
+ (entries) => {
168
+ for (const entry of entries) {
169
+ if (!isHTMLElement(entry.target)) {
170
+ continue;
171
+ }
172
+ const nextWidth = entry.contentRect.width;
173
+ if (updateWidthMap(widths, entry.target, nextWidth)) {
174
+ const nextVisibleCount = measureVisibleCount(
175
+ getCurrentPinnedValue()
176
+ );
177
+ if (nextVisibleCount != null && nextVisibleCount === visibleCount) {
178
+ continue;
179
+ }
180
+ markMeasurementStale();
181
+ return;
182
+ }
183
+ }
184
+ }
185
+ );
186
+ for (const observedElement of observedElements) {
187
+ resizeObserver.observe(observedElement);
188
+ }
189
+ return () => {
190
+ resizeObserver.disconnect();
191
+ };
192
+ }, [
193
+ container,
194
+ getCurrentPinnedValue,
195
+ isMeasuring,
196
+ markMeasurementStale,
197
+ measureVisibleCount,
198
+ menuOpen,
199
+ visibleCount
200
+ ]);
201
+ useIsomorphicLayoutEffect(() => {
202
+ if (previousMenuOpenRef.current && !menuOpen) {
203
+ markMeasurementStale();
204
+ }
205
+ previousMenuOpenRef.current = menuOpen;
206
+ }, [markMeasurementStale, menuOpen]);
207
+ useIsomorphicLayoutEffect(() => {
208
+ var _a;
209
+ const nextOverflowButtonWidth = ((_a = overflowButton.current) == null ? void 0 : _a.offsetWidth) ?? 0;
210
+ if (previousOverflowButtonWidthRef.current === nextOverflowButtonWidth) {
211
+ return;
212
+ }
213
+ previousOverflowButtonWidthRef.current = nextOverflowButtonWidth;
214
+ if (visibleCount < tabs.length) {
215
+ markMeasurementStale();
216
+ }
217
+ });
218
+ useIsomorphicLayoutEffect(() => {
219
+ if (previousMeasurementInputKeyRef.current !== measurementInputKey) {
220
+ previousMeasurementInputKeyRef.current = measurementInputKey;
221
+ markMeasurementStale();
222
+ }
223
+ }, [markMeasurementStale, measurementInputKey]);
224
+ useIsomorphicLayoutEffect(() => {
225
+ if (!isMeasuring || menuOpen) {
226
+ return;
227
+ }
228
+ const nextVisibleCount = measureVisibleCount(getCurrentPinnedValue());
229
+ if (nextVisibleCount == null) {
230
+ if (measureRetryFrameRef.current != null) {
231
+ return;
232
+ }
233
+ const element = container.current;
234
+ if (!element || getMeasuredWidth(element) <= MIN_TRUSTED_TAB_WIDTH) {
235
+ measureRetryCountRef.current = 0;
236
+ return;
237
+ }
238
+ if (measureRetryCountRef.current >= 5) {
239
+ return;
240
+ }
241
+ measureRetryCountRef.current += 1;
242
+ measureRetryFrameRef.current = ownerWindow(element).requestAnimationFrame(
243
+ () => {
244
+ measureRetryFrameRef.current = null;
245
+ setMeasureRetryVersion((currentVersion) => currentVersion + 1);
246
+ }
247
+ );
248
+ return;
249
+ }
250
+ clearMeasureRetry();
251
+ setVisibleCount(nextVisibleCount);
252
+ setIsMeasuring(false);
253
+ }, [
254
+ clearMeasureRetry,
255
+ container.current,
256
+ getCurrentPinnedValue,
257
+ isMeasuring,
258
+ measureRetryVersion,
259
+ measureVisibleCount,
260
+ menuOpen
261
+ ]);
262
+ const { visibleValues, hiddenValues } = partitionVisibleValues(
263
+ orderedValues,
264
+ visibleCount,
265
+ pinnedValue
266
+ );
267
+ return [visibleValues, hiddenValues, isMeasuring];
268
+ }
269
+
270
+ export { useOverflow };
271
+ //# sourceMappingURL=useOverflow.js.map