@salt-ds/lab 1.0.0-alpha.87 → 1.0.0-alpha.89

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 (177) hide show
  1. package/CHANGELOG.md +44 -0
  2. package/css/salt-lab.css +64 -45
  3. package/dist-cjs/calendar/internal/CalendarDay.css.js +1 -1
  4. package/dist-cjs/contact-details/ContactDetails.css.js +1 -1
  5. package/dist-cjs/list-deprecated/ListStateContext.js +1 -1
  6. package/dist-cjs/list-deprecated/ListStateContext.js.map +1 -1
  7. package/dist-cjs/tabs-next/TabBar.css.js +1 -1
  8. package/dist-cjs/tabs-next/TabBar.js +1 -1
  9. package/dist-cjs/tabs-next/TabBar.js.map +1 -1
  10. package/dist-cjs/tabs-next/TabListLayoutContext.js +13 -0
  11. package/dist-cjs/tabs-next/TabListLayoutContext.js.map +1 -0
  12. package/dist-cjs/tabs-next/TabListNext.css.js +1 -1
  13. package/dist-cjs/tabs-next/TabListNext.js +179 -33
  14. package/dist-cjs/tabs-next/TabListNext.js.map +1 -1
  15. package/dist-cjs/tabs-next/TabNext.js +111 -7
  16. package/dist-cjs/tabs-next/TabNext.js.map +1 -1
  17. package/dist-cjs/tabs-next/TabNextAction.js +25 -2
  18. package/dist-cjs/tabs-next/TabNextAction.js.map +1 -1
  19. package/dist-cjs/tabs-next/TabNextPanel.js +31 -16
  20. package/dist-cjs/tabs-next/TabNextPanel.js.map +1 -1
  21. package/dist-cjs/tabs-next/TabNextTrigger.js +110 -9
  22. package/dist-cjs/tabs-next/TabNextTrigger.js.map +1 -1
  23. package/dist-cjs/tabs-next/TabOverflowList.css.js +1 -1
  24. package/dist-cjs/tabs-next/TabOverflowList.js +168 -64
  25. package/dist-cjs/tabs-next/TabOverflowList.js.map +1 -1
  26. package/dist-cjs/tabs-next/TabSlot.js +30 -0
  27. package/dist-cjs/tabs-next/TabSlot.js.map +1 -0
  28. package/dist-cjs/tabs-next/TabSlotRegistryContext.js +16 -0
  29. package/dist-cjs/tabs-next/TabSlotRegistryContext.js.map +1 -0
  30. package/dist-cjs/tabs-next/TabsNext.css.js +6 -0
  31. package/dist-cjs/tabs-next/TabsNext.css.js.map +1 -0
  32. package/dist-cjs/tabs-next/TabsNext.js +113 -47
  33. package/dist-cjs/tabs-next/TabsNext.js.map +1 -1
  34. package/dist-cjs/tabs-next/TabsNextContext.js +17 -3
  35. package/dist-cjs/tabs-next/TabsNextContext.js.map +1 -1
  36. package/dist-cjs/tabs-next/domUtils.js +13 -0
  37. package/dist-cjs/tabs-next/domUtils.js.map +1 -0
  38. package/dist-cjs/tabs-next/hooks/overflowMath.js +86 -0
  39. package/dist-cjs/tabs-next/hooks/overflowMath.js.map +1 -0
  40. package/dist-cjs/tabs-next/hooks/useCollection.js +147 -41
  41. package/dist-cjs/tabs-next/hooks/useCollection.js.map +1 -1
  42. package/dist-cjs/tabs-next/hooks/useFocusWithRetry.js +64 -0
  43. package/dist-cjs/tabs-next/hooks/useFocusWithRetry.js.map +1 -0
  44. package/dist-cjs/tabs-next/hooks/useOverflow.js +240 -156
  45. package/dist-cjs/tabs-next/hooks/useOverflow.js.map +1 -1
  46. package/dist-cjs/tabs-next/hooks/useOverflowLayoutState.js +99 -0
  47. package/dist-cjs/tabs-next/hooks/useOverflowLayoutState.js.map +1 -0
  48. package/dist-cjs/tabs-next/hooks/useOverflowSelectionState.js +60 -0
  49. package/dist-cjs/tabs-next/hooks/useOverflowSelectionState.js.map +1 -0
  50. package/dist-cjs/tabs-next/hooks/useRenderedTabWidth.js +92 -0
  51. package/dist-cjs/tabs-next/hooks/useRenderedTabWidth.js.map +1 -0
  52. package/dist-cjs/tabs-next/hooks/useRenderedTabsRegistry.js +200 -0
  53. package/dist-cjs/tabs-next/hooks/useRenderedTabsRegistry.js.map +1 -0
  54. package/dist-cjs/tabs-next/hooks/useTabListRecovery.js +76 -0
  55. package/dist-cjs/tabs-next/hooks/useTabListRecovery.js.map +1 -0
  56. package/dist-cjs/tabs-next/hooks/useTabRemovalHandler.js +165 -0
  57. package/dist-cjs/tabs-next/hooks/useTabRemovalHandler.js.map +1 -0
  58. package/dist-cjs/tabs-next/hooks/useTabSelectionFocus.js +80 -0
  59. package/dist-cjs/tabs-next/hooks/useTabSelectionFocus.js.map +1 -0
  60. package/dist-cjs/tabs-next/widthMeasurement.js +42 -0
  61. package/dist-cjs/tabs-next/widthMeasurement.js.map +1 -0
  62. package/dist-cjs/tree/Tree.css.js +1 -1
  63. package/dist-cjs/tree/TreeNode.css.js +1 -1
  64. package/dist-cjs/tree/TreeNode.js +1 -1
  65. package/dist-cjs/tree/TreeNode.js.map +1 -1
  66. package/dist-cjs/tree/TreeNodeExpansionIcon.css.js +1 -1
  67. package/dist-cjs/tree/TreeNodeTrigger.css.js +1 -1
  68. package/dist-cjs/tree/TreeNodeTrigger.js +2 -2
  69. package/dist-cjs/tree/TreeNodeTrigger.js.map +1 -1
  70. package/dist-cjs/utils/useEventCallback.js +5 -5
  71. package/dist-cjs/utils/useEventCallback.js.map +1 -1
  72. package/dist-es/calendar/internal/CalendarDay.css.js +1 -1
  73. package/dist-es/contact-details/ContactDetails.css.js +1 -1
  74. package/dist-es/list-deprecated/ListStateContext.js +1 -1
  75. package/dist-es/list-deprecated/ListStateContext.js.map +1 -1
  76. package/dist-es/tabs-next/TabBar.css.js +1 -1
  77. package/dist-es/tabs-next/TabBar.js +1 -1
  78. package/dist-es/tabs-next/TabBar.js.map +1 -1
  79. package/dist-es/tabs-next/TabListLayoutContext.js +10 -0
  80. package/dist-es/tabs-next/TabListLayoutContext.js.map +1 -0
  81. package/dist-es/tabs-next/TabListNext.css.js +1 -1
  82. package/dist-es/tabs-next/TabListNext.js +182 -36
  83. package/dist-es/tabs-next/TabListNext.js.map +1 -1
  84. package/dist-es/tabs-next/TabNext.js +113 -9
  85. package/dist-es/tabs-next/TabNext.js.map +1 -1
  86. package/dist-es/tabs-next/TabNextAction.js +25 -2
  87. package/dist-es/tabs-next/TabNextAction.js.map +1 -1
  88. package/dist-es/tabs-next/TabNextPanel.js +31 -16
  89. package/dist-es/tabs-next/TabNextPanel.js.map +1 -1
  90. package/dist-es/tabs-next/TabNextTrigger.js +110 -9
  91. package/dist-es/tabs-next/TabNextTrigger.js.map +1 -1
  92. package/dist-es/tabs-next/TabOverflowList.css.js +1 -1
  93. package/dist-es/tabs-next/TabOverflowList.js +172 -68
  94. package/dist-es/tabs-next/TabOverflowList.js.map +1 -1
  95. package/dist-es/tabs-next/TabSlot.js +28 -0
  96. package/dist-es/tabs-next/TabSlot.js.map +1 -0
  97. package/dist-es/tabs-next/TabSlotRegistryContext.js +13 -0
  98. package/dist-es/tabs-next/TabSlotRegistryContext.js.map +1 -0
  99. package/dist-es/tabs-next/TabsNext.css.js +4 -0
  100. package/dist-es/tabs-next/TabsNext.css.js.map +1 -0
  101. package/dist-es/tabs-next/TabsNext.js +114 -48
  102. package/dist-es/tabs-next/TabsNext.js.map +1 -1
  103. package/dist-es/tabs-next/TabsNextContext.js +17 -3
  104. package/dist-es/tabs-next/TabsNextContext.js.map +1 -1
  105. package/dist-es/tabs-next/domUtils.js +11 -0
  106. package/dist-es/tabs-next/domUtils.js.map +1 -0
  107. package/dist-es/tabs-next/hooks/overflowMath.js +82 -0
  108. package/dist-es/tabs-next/hooks/overflowMath.js.map +1 -0
  109. package/dist-es/tabs-next/hooks/useCollection.js +148 -42
  110. package/dist-es/tabs-next/hooks/useCollection.js.map +1 -1
  111. package/dist-es/tabs-next/hooks/useFocusWithRetry.js +62 -0
  112. package/dist-es/tabs-next/hooks/useFocusWithRetry.js.map +1 -0
  113. package/dist-es/tabs-next/hooks/useOverflow.js +242 -158
  114. package/dist-es/tabs-next/hooks/useOverflow.js.map +1 -1
  115. package/dist-es/tabs-next/hooks/useOverflowLayoutState.js +97 -0
  116. package/dist-es/tabs-next/hooks/useOverflowLayoutState.js.map +1 -0
  117. package/dist-es/tabs-next/hooks/useOverflowSelectionState.js +58 -0
  118. package/dist-es/tabs-next/hooks/useOverflowSelectionState.js.map +1 -0
  119. package/dist-es/tabs-next/hooks/useRenderedTabWidth.js +90 -0
  120. package/dist-es/tabs-next/hooks/useRenderedTabWidth.js.map +1 -0
  121. package/dist-es/tabs-next/hooks/useRenderedTabsRegistry.js +198 -0
  122. package/dist-es/tabs-next/hooks/useRenderedTabsRegistry.js.map +1 -0
  123. package/dist-es/tabs-next/hooks/useTabListRecovery.js +74 -0
  124. package/dist-es/tabs-next/hooks/useTabListRecovery.js.map +1 -0
  125. package/dist-es/tabs-next/hooks/useTabRemovalHandler.js +163 -0
  126. package/dist-es/tabs-next/hooks/useTabRemovalHandler.js.map +1 -0
  127. package/dist-es/tabs-next/hooks/useTabSelectionFocus.js +78 -0
  128. package/dist-es/tabs-next/hooks/useTabSelectionFocus.js.map +1 -0
  129. package/dist-es/tabs-next/widthMeasurement.js +36 -0
  130. package/dist-es/tabs-next/widthMeasurement.js.map +1 -0
  131. package/dist-es/tree/Tree.css.js +1 -1
  132. package/dist-es/tree/TreeNode.css.js +1 -1
  133. package/dist-es/tree/TreeNode.js +1 -1
  134. package/dist-es/tree/TreeNode.js.map +1 -1
  135. package/dist-es/tree/TreeNodeExpansionIcon.css.js +1 -1
  136. package/dist-es/tree/TreeNodeTrigger.css.js +1 -1
  137. package/dist-es/tree/TreeNodeTrigger.js +2 -2
  138. package/dist-es/tree/TreeNodeTrigger.js.map +1 -1
  139. package/dist-es/utils/useEventCallback.js +5 -5
  140. package/dist-es/utils/useEventCallback.js.map +1 -1
  141. package/dist-types/cascading-menu/internal/useMenuTriggerHandlers.d.ts +1 -1
  142. package/dist-types/list-deprecated/ListStateContext.d.ts +7 -2
  143. package/dist-types/tabs-next/TabListLayoutContext.d.ts +9 -0
  144. package/dist-types/tabs-next/TabNext.d.ts +1 -1
  145. package/dist-types/tabs-next/TabNextPanel.d.ts +2 -1
  146. package/dist-types/tabs-next/TabOverflowList.d.ts +3 -4
  147. package/dist-types/tabs-next/TabSlot.d.ts +6 -0
  148. package/dist-types/tabs-next/TabSlotRegistryContext.d.ts +5 -0
  149. package/dist-types/tabs-next/TabsNext.d.ts +2 -1
  150. package/dist-types/tabs-next/TabsNextContext.d.ts +26 -4
  151. package/dist-types/tabs-next/domUtils.d.ts +1 -0
  152. package/dist-types/tabs-next/hooks/overflowMath.d.ts +18 -0
  153. package/dist-types/tabs-next/hooks/useCollection.d.ts +15 -3
  154. package/dist-types/tabs-next/hooks/useFocusWithRetry.d.ts +9 -0
  155. package/dist-types/tabs-next/hooks/useOverflow.d.ts +5 -5
  156. package/dist-types/tabs-next/hooks/useOverflowLayoutState.d.ts +13 -0
  157. package/dist-types/tabs-next/hooks/useOverflowSelectionState.d.ts +13 -0
  158. package/dist-types/tabs-next/hooks/useRenderedTabWidth.d.ts +12 -0
  159. package/dist-types/tabs-next/hooks/useRenderedTabsRegistry.d.ts +12 -0
  160. package/dist-types/tabs-next/hooks/useTabListRecovery.d.ts +12 -0
  161. package/dist-types/tabs-next/hooks/useTabRemovalHandler.d.ts +32 -0
  162. package/dist-types/tabs-next/hooks/useTabSelectionFocus.d.ts +15 -0
  163. package/dist-types/tabs-next/widthMeasurement.d.ts +5 -0
  164. package/dist-types/tokenized-input/internal/InputPill.d.ts +1 -1
  165. package/dist-types/tokenized-input-next/internal/InputPill.d.ts +1 -1
  166. package/dist-types/utils/useEventCallback.d.ts +1 -1
  167. package/package.json +3 -3
  168. package/dist-cjs/tabs-next/hooks/useFocusOutside.js +0 -25
  169. package/dist-cjs/tabs-next/hooks/useFocusOutside.js.map +0 -1
  170. package/dist-cjs/tabs-next/hooks/useRestoreActiveTab.js +0 -93
  171. package/dist-cjs/tabs-next/hooks/useRestoreActiveTab.js.map +0 -1
  172. package/dist-es/tabs-next/hooks/useFocusOutside.js +0 -23
  173. package/dist-es/tabs-next/hooks/useFocusOutside.js.map +0 -1
  174. package/dist-es/tabs-next/hooks/useRestoreActiveTab.js +0 -91
  175. package/dist-es/tabs-next/hooks/useRestoreActiveTab.js.map +0 -1
  176. package/dist-types/tabs-next/hooks/useFocusOutside.d.ts +0 -2
  177. package/dist-types/tabs-next/hooks/useRestoreActiveTab.d.ts +0 -10
@@ -22,39 +22,54 @@ const TabNextPanel = react.forwardRef(
22
22
  });
23
23
  const id = core.useId(idProp);
24
24
  const { registerPanel, getTabId, selected } = TabsNextContext.useTabsNext();
25
+ const hidden = selected !== value;
25
26
  const panelRef = react.useRef(null);
26
27
  const handleRef = core.useForkRef(panelRef, ref);
27
28
  core.useIsomorphicLayoutEffect(() => {
28
- if (value && id) {
29
+ if (id) {
29
30
  return registerPanel(id, value);
30
31
  }
31
32
  }, [value, id, registerPanel]);
32
33
  const [hasFocusableChildren, setHasFocusableChildren] = react.useState(false);
33
34
  react.useEffect(() => {
34
- if (!panelRef.current) return;
35
+ const element = panelRef.current;
36
+ const mutationObserverCtor = targetWindow == null ? void 0 : targetWindow.MutationObserver;
37
+ if (!element || hidden) return;
38
+ let rafId = null;
35
39
  const detectFocusableChildren = () => {
36
- requestAnimationFrame(() => {
37
- if (!panelRef.current) return;
38
- const elements = tabbable.tabbable(panelRef.current);
39
- setHasFocusableChildren(elements.length > 0);
40
+ rafId = null;
41
+ const elements = tabbable.tabbable(element);
42
+ const nextHasFocusableChildren = elements.length > 0;
43
+ setHasFocusableChildren((prev) => {
44
+ return prev === nextHasFocusableChildren ? prev : nextHasFocusableChildren;
40
45
  });
41
46
  };
42
- const observer = new MutationObserver(() => {
43
- detectFocusableChildren();
44
- });
45
- requestAnimationFrame(() => {
46
- detectFocusableChildren();
47
- });
48
- observer.observe(panelRef.current, {
47
+ const scheduleDetectFocusableChildren = () => {
48
+ if (rafId != null && targetWindow) {
49
+ targetWindow.cancelAnimationFrame(rafId);
50
+ }
51
+ if (!(targetWindow == null ? void 0 : targetWindow.requestAnimationFrame)) {
52
+ detectFocusableChildren();
53
+ return;
54
+ }
55
+ rafId = targetWindow.requestAnimationFrame(detectFocusableChildren);
56
+ };
57
+ const observer = mutationObserverCtor ? new mutationObserverCtor(() => {
58
+ scheduleDetectFocusableChildren();
59
+ }) : null;
60
+ scheduleDetectFocusableChildren();
61
+ observer == null ? void 0 : observer.observe(element, {
49
62
  childList: true,
50
63
  subtree: true,
51
64
  attributes: true
52
65
  });
53
66
  return () => {
54
- observer.disconnect();
67
+ observer == null ? void 0 : observer.disconnect();
68
+ if (rafId != null && targetWindow) {
69
+ targetWindow.cancelAnimationFrame(rafId);
70
+ }
55
71
  };
56
- }, []);
57
- const hidden = selected !== value;
72
+ }, [hidden, targetWindow]);
58
73
  const tabId = getTabId(value);
59
74
  return /* @__PURE__ */ jsxRuntime.jsx(
60
75
  "div",
@@ -1 +1 @@
1
- {"version":3,"file":"TabNextPanel.js","sources":["../src/tabs-next/TabNextPanel.tsx"],"sourcesContent":["import {\n makePrefixer,\n useForkRef,\n useId,\n useIsomorphicLayoutEffect,\n} from \"@salt-ds/core\";\nimport { useComponentCssInjection } from \"@salt-ds/styles\";\nimport { useWindow } from \"@salt-ds/window\";\nimport { clsx } from \"clsx\";\nimport {\n type ComponentPropsWithoutRef,\n forwardRef,\n useEffect,\n useRef,\n useState,\n} from \"react\";\nimport { tabbable } from \"tabbable\";\nimport tabPanelCss from \"./TabNextPanel.css\";\nimport { useTabsNext } from \"./TabsNextContext\";\n\nexport interface TabNextPanelProps extends ComponentPropsWithoutRef<\"div\"> {\n /**\n * The value of the panel, this should map to the corresponding tab.\n */\n value: string;\n}\n\nconst withBaseName = makePrefixer(\"saltTabNextPanel\");\n\nexport const TabNextPanel = forwardRef<HTMLDivElement, TabNextPanelProps>(\n function TabNextPanel(props, ref) {\n const { className, children, id: idProp, value, ...rest } = props;\n const targetWindow = useWindow();\n useComponentCssInjection({\n testId: \"salt-tab-next-panel\",\n css: tabPanelCss,\n window: targetWindow,\n });\n const id = useId(idProp);\n const { registerPanel, getTabId, selected } = useTabsNext();\n\n const panelRef = useRef<HTMLDivElement>(null);\n const handleRef = useForkRef(panelRef, ref);\n\n useIsomorphicLayoutEffect(() => {\n if (value && id) {\n return registerPanel(id, value);\n }\n }, [value, id, registerPanel]);\n\n const [hasFocusableChildren, setHasFocusableChildren] = useState(false);\n useEffect(() => {\n if (!panelRef.current) return;\n\n const detectFocusableChildren = () => {\n requestAnimationFrame(() => {\n if (!panelRef.current) return;\n const elements = tabbable(panelRef.current);\n setHasFocusableChildren(elements.length > 0);\n });\n };\n\n const observer = new MutationObserver(() => {\n detectFocusableChildren();\n });\n\n requestAnimationFrame(() => {\n detectFocusableChildren();\n });\n\n observer.observe(panelRef.current, {\n childList: true,\n subtree: true,\n attributes: true,\n });\n\n return () => {\n observer.disconnect();\n };\n }, []);\n\n const hidden = selected !== value;\n const tabId = getTabId(value);\n\n return (\n <div\n id={id}\n ref={handleRef}\n role=\"tabpanel\"\n aria-labelledby={tabId}\n className={clsx(withBaseName(), className)}\n hidden={hidden || undefined}\n tabIndex={hidden || hasFocusableChildren ? undefined : 0}\n {...rest}\n >\n {children}\n </div>\n );\n },\n);\n"],"names":["makePrefixer","forwardRef","TabNextPanel","useWindow","useComponentCssInjection","tabPanelCss","useId","useTabsNext","useRef","useForkRef","useIsomorphicLayoutEffect","useState","useEffect","tabbable","jsx","clsx"],"mappings":";;;;;;;;;;;;AA2BA,MAAM,YAAA,GAAeA,kBAAa,kBAAkB,CAAA;AAE7C,MAAM,YAAA,GAAeC,gBAAA;AAAA,EAC1B,SAASC,aAAAA,CAAa,KAAA,EAAO,GAAA,EAAK;AAChC,IAAA,MAAM,EAAE,WAAW,QAAA,EAAU,EAAA,EAAI,QAAQ,KAAA,EAAO,GAAG,MAAK,GAAI,KAAA;AAC5D,IAAA,MAAM,eAAeC,gBAAA,EAAU;AAC/B,IAAAC,+BAAA,CAAyB;AAAA,MACvB,MAAA,EAAQ,qBAAA;AAAA,MACR,GAAA,EAAKC,cAAA;AAAA,MACL,MAAA,EAAQ;AAAA,KACT,CAAA;AACD,IAAA,MAAM,EAAA,GAAKC,WAAM,MAAM,CAAA;AACvB,IAAA,MAAM,EAAE,aAAA,EAAe,QAAA,EAAU,QAAA,KAAaC,2BAAA,EAAY;AAE1D,IAAA,MAAM,QAAA,GAAWC,aAAuB,IAAI,CAAA;AAC5C,IAAA,MAAM,SAAA,GAAYC,eAAA,CAAW,QAAA,EAAU,GAAG,CAAA;AAE1C,IAAAC,8BAAA,CAA0B,MAAM;AAC9B,MAAA,IAAI,SAAS,EAAA,EAAI;AACf,QAAA,OAAO,aAAA,CAAc,IAAI,KAAK,CAAA;AAAA,MAChC;AAAA,IACF,CAAA,EAAG,CAAC,KAAA,EAAO,EAAA,EAAI,aAAa,CAAC,CAAA;AAE7B,IAAA,MAAM,CAAC,oBAAA,EAAsB,uBAAuB,CAAA,GAAIC,eAAS,KAAK,CAAA;AACtE,IAAAC,eAAA,CAAU,MAAM;AACd,MAAA,IAAI,CAAC,SAAS,OAAA,EAAS;AAEvB,MAAA,MAAM,0BAA0B,MAAM;AACpC,QAAA,qBAAA,CAAsB,MAAM;AAC1B,UAAA,IAAI,CAAC,SAAS,OAAA,EAAS;AACvB,UAAA,MAAM,QAAA,GAAWC,iBAAA,CAAS,QAAA,CAAS,OAAO,CAAA;AAC1C,UAAA,uBAAA,CAAwB,QAAA,CAAS,SAAS,CAAC,CAAA;AAAA,QAC7C,CAAC,CAAA;AAAA,MACH,CAAA;AAEA,MAAA,MAAM,QAAA,GAAW,IAAI,gBAAA,CAAiB,MAAM;AAC1C,QAAA,uBAAA,EAAwB;AAAA,MAC1B,CAAC,CAAA;AAED,MAAA,qBAAA,CAAsB,MAAM;AAC1B,QAAA,uBAAA,EAAwB;AAAA,MAC1B,CAAC,CAAA;AAED,MAAA,QAAA,CAAS,OAAA,CAAQ,SAAS,OAAA,EAAS;AAAA,QACjC,SAAA,EAAW,IAAA;AAAA,QACX,OAAA,EAAS,IAAA;AAAA,QACT,UAAA,EAAY;AAAA,OACb,CAAA;AAED,MAAA,OAAO,MAAM;AACX,QAAA,QAAA,CAAS,UAAA,EAAW;AAAA,MACtB,CAAA;AAAA,IACF,CAAA,EAAG,EAAE,CAAA;AAEL,IAAA,MAAM,SAAS,QAAA,KAAa,KAAA;AAC5B,IAAA,MAAM,KAAA,GAAQ,SAAS,KAAK,CAAA;AAE5B,IAAA,uBACEC,cAAA;AAAA,MAAC,KAAA;AAAA,MAAA;AAAA,QACC,EAAA;AAAA,QACA,GAAA,EAAK,SAAA;AAAA,QACL,IAAA,EAAK,UAAA;AAAA,QACL,iBAAA,EAAiB,KAAA;AAAA,QACjB,SAAA,EAAWC,SAAA,CAAK,YAAA,EAAa,EAAG,SAAS,CAAA;AAAA,QACzC,QAAQ,MAAA,IAAU,MAAA;AAAA,QAClB,QAAA,EAAU,MAAA,IAAU,oBAAA,GAAuB,MAAA,GAAY,CAAA;AAAA,QACtD,GAAG,IAAA;AAAA,QAEH;AAAA;AAAA,KACH;AAAA,EAEJ;AACF;;;;"}
1
+ {"version":3,"file":"TabNextPanel.js","sources":["../src/tabs-next/TabNextPanel.tsx"],"sourcesContent":["import {\n makePrefixer,\n useForkRef,\n useId,\n useIsomorphicLayoutEffect,\n} from \"@salt-ds/core\";\nimport { useComponentCssInjection } from \"@salt-ds/styles\";\nimport { useWindow } from \"@salt-ds/window\";\nimport { clsx } from \"clsx\";\nimport {\n type ComponentPropsWithoutRef,\n forwardRef,\n useEffect,\n useRef,\n useState,\n} from \"react\";\nimport { tabbable } from \"tabbable\";\nimport tabPanelCss from \"./TabNextPanel.css\";\nimport { useTabsNext } from \"./TabsNextContext\";\n\nexport interface TabNextPanelProps extends ComponentPropsWithoutRef<\"div\"> {\n /**\n * The value of the panel. This should map to the corresponding tab and must\n * be unique within a `TabsNext` instance.\n */\n value: string;\n}\n\nconst withBaseName = makePrefixer(\"saltTabNextPanel\");\n\nexport const TabNextPanel = forwardRef<HTMLDivElement, TabNextPanelProps>(\n function TabNextPanel(props, ref) {\n const { className, children, id: idProp, value, ...rest } = props;\n const targetWindow = useWindow();\n useComponentCssInjection({\n testId: \"salt-tab-next-panel\",\n css: tabPanelCss,\n window: targetWindow,\n });\n const id = useId(idProp);\n const { registerPanel, getTabId, selected } = useTabsNext();\n const hidden = selected !== value;\n\n const panelRef = useRef<HTMLDivElement>(null);\n const handleRef = useForkRef(panelRef, ref);\n\n useIsomorphicLayoutEffect(() => {\n if (id) {\n return registerPanel(id, value);\n }\n }, [value, id, registerPanel]);\n\n const [hasFocusableChildren, setHasFocusableChildren] = useState(false);\n useEffect(() => {\n const element = panelRef.current;\n const mutationObserverCtor = (\n targetWindow as\n | (Window & { MutationObserver?: typeof MutationObserver })\n | undefined\n )?.MutationObserver;\n if (!element || hidden) return;\n\n let rafId: number | null = null;\n\n const detectFocusableChildren = () => {\n rafId = null;\n const elements = tabbable(element);\n const nextHasFocusableChildren = elements.length > 0;\n setHasFocusableChildren((prev) => {\n return prev === nextHasFocusableChildren\n ? prev\n : nextHasFocusableChildren;\n });\n };\n\n const scheduleDetectFocusableChildren = () => {\n if (rafId != null && targetWindow) {\n targetWindow.cancelAnimationFrame(rafId);\n }\n\n if (!targetWindow?.requestAnimationFrame) {\n detectFocusableChildren();\n return;\n }\n\n rafId = targetWindow.requestAnimationFrame(detectFocusableChildren);\n };\n\n const observer = mutationObserverCtor\n ? new mutationObserverCtor(() => {\n scheduleDetectFocusableChildren();\n })\n : null;\n\n scheduleDetectFocusableChildren();\n\n observer?.observe(element, {\n childList: true,\n subtree: true,\n attributes: true,\n });\n\n return () => {\n observer?.disconnect();\n if (rafId != null && targetWindow) {\n targetWindow.cancelAnimationFrame(rafId);\n }\n };\n }, [hidden, targetWindow]);\n\n const tabId = getTabId(value);\n\n return (\n <div\n id={id}\n ref={handleRef}\n role=\"tabpanel\"\n aria-labelledby={tabId}\n className={clsx(withBaseName(), className)}\n hidden={hidden || undefined}\n tabIndex={hidden || hasFocusableChildren ? undefined : 0}\n {...rest}\n >\n {children}\n </div>\n );\n },\n);\n"],"names":["makePrefixer","forwardRef","TabNextPanel","useWindow","useComponentCssInjection","tabPanelCss","useId","useTabsNext","useRef","useForkRef","useIsomorphicLayoutEffect","useState","useEffect","tabbable","jsx","clsx"],"mappings":";;;;;;;;;;;;AA4BA,MAAM,YAAA,GAAeA,kBAAa,kBAAkB,CAAA;AAE7C,MAAM,YAAA,GAAeC,gBAAA;AAAA,EAC1B,SAASC,aAAAA,CAAa,KAAA,EAAO,GAAA,EAAK;AAChC,IAAA,MAAM,EAAE,WAAW,QAAA,EAAU,EAAA,EAAI,QAAQ,KAAA,EAAO,GAAG,MAAK,GAAI,KAAA;AAC5D,IAAA,MAAM,eAAeC,gBAAA,EAAU;AAC/B,IAAAC,+BAAA,CAAyB;AAAA,MACvB,MAAA,EAAQ,qBAAA;AAAA,MACR,GAAA,EAAKC,cAAA;AAAA,MACL,MAAA,EAAQ;AAAA,KACT,CAAA;AACD,IAAA,MAAM,EAAA,GAAKC,WAAM,MAAM,CAAA;AACvB,IAAA,MAAM,EAAE,aAAA,EAAe,QAAA,EAAU,QAAA,KAAaC,2BAAA,EAAY;AAC1D,IAAA,MAAM,SAAS,QAAA,KAAa,KAAA;AAE5B,IAAA,MAAM,QAAA,GAAWC,aAAuB,IAAI,CAAA;AAC5C,IAAA,MAAM,SAAA,GAAYC,eAAA,CAAW,QAAA,EAAU,GAAG,CAAA;AAE1C,IAAAC,8BAAA,CAA0B,MAAM;AAC9B,MAAA,IAAI,EAAA,EAAI;AACN,QAAA,OAAO,aAAA,CAAc,IAAI,KAAK,CAAA;AAAA,MAChC;AAAA,IACF,CAAA,EAAG,CAAC,KAAA,EAAO,EAAA,EAAI,aAAa,CAAC,CAAA;AAE7B,IAAA,MAAM,CAAC,oBAAA,EAAsB,uBAAuB,CAAA,GAAIC,eAAS,KAAK,CAAA;AACtE,IAAAC,eAAA,CAAU,MAAM;AACd,MAAA,MAAM,UAAU,QAAA,CAAS,OAAA;AACzB,MAAA,MAAM,uBACJ,YAAA,IAAA,IAAA,GAAA,MAAA,GAAA,YAAA,CAGC,gBAAA;AACH,MAAA,IAAI,CAAC,WAAW,MAAA,EAAQ;AAExB,MAAA,IAAI,KAAA,GAAuB,IAAA;AAE3B,MAAA,MAAM,0BAA0B,MAAM;AACpC,QAAA,KAAA,GAAQ,IAAA;AACR,QAAA,MAAM,QAAA,GAAWC,kBAAS,OAAO,CAAA;AACjC,QAAA,MAAM,wBAAA,GAA2B,SAAS,MAAA,GAAS,CAAA;AACnD,QAAA,uBAAA,CAAwB,CAAC,IAAA,KAAS;AAChC,UAAA,OAAO,IAAA,KAAS,2BACZ,IAAA,GACA,wBAAA;AAAA,QACN,CAAC,CAAA;AAAA,MACH,CAAA;AAEA,MAAA,MAAM,kCAAkC,MAAM;AAC5C,QAAA,IAAI,KAAA,IAAS,QAAQ,YAAA,EAAc;AACjC,UAAA,YAAA,CAAa,qBAAqB,KAAK,CAAA;AAAA,QACzC;AAEA,QAAA,IAAI,EAAC,6CAAc,qBAAA,CAAA,EAAuB;AACxC,UAAA,uBAAA,EAAwB;AACxB,UAAA;AAAA,QACF;AAEA,QAAA,KAAA,GAAQ,YAAA,CAAa,sBAAsB,uBAAuB,CAAA;AAAA,MACpE,CAAA;AAEA,MAAA,MAAM,QAAA,GAAW,oBAAA,GACb,IAAI,oBAAA,CAAqB,MAAM;AAC7B,QAAA,+BAAA,EAAgC;AAAA,MAClC,CAAC,CAAA,GACD,IAAA;AAEJ,MAAA,+BAAA,EAAgC;AAEhC,MAAA,QAAA,IAAA,IAAA,GAAA,MAAA,GAAA,QAAA,CAAU,QAAQ,OAAA,EAAS;AAAA,QACzB,SAAA,EAAW,IAAA;AAAA,QACX,OAAA,EAAS,IAAA;AAAA,QACT,UAAA,EAAY;AAAA,OACd,CAAA;AAEA,MAAA,OAAO,MAAM;AACX,QAAA,QAAA,IAAA,IAAA,GAAA,MAAA,GAAA,QAAA,CAAU,UAAA,EAAA;AACV,QAAA,IAAI,KAAA,IAAS,QAAQ,YAAA,EAAc;AACjC,UAAA,YAAA,CAAa,qBAAqB,KAAK,CAAA;AAAA,QACzC;AAAA,MACF,CAAA;AAAA,IACF,CAAA,EAAG,CAAC,MAAA,EAAQ,YAAY,CAAC,CAAA;AAEzB,IAAA,MAAM,KAAA,GAAQ,SAAS,KAAK,CAAA;AAE5B,IAAA,uBACEC,cAAA;AAAA,MAAC,KAAA;AAAA,MAAA;AAAA,QACC,EAAA;AAAA,QACA,GAAA,EAAK,SAAA;AAAA,QACL,IAAA,EAAK,UAAA;AAAA,QACL,iBAAA,EAAiB,KAAA;AAAA,QACjB,SAAA,EAAWC,SAAA,CAAK,YAAA,EAAa,EAAG,SAAS,CAAA;AAAA,QACzC,QAAQ,MAAA,IAAU,MAAA;AAAA,QAClB,QAAA,EAAU,MAAA,IAAU,oBAAA,GAAuB,MAAA,GAAY,CAAA;AAAA,QACtD,GAAG,IAAA;AAAA,QAEH;AAAA;AAAA,KACH;AAAA,EAEJ;AACF;;;;"}
@@ -6,6 +6,7 @@ var styles = require('@salt-ds/styles');
6
6
  var window = require('@salt-ds/window');
7
7
  var clsx = require('clsx');
8
8
  var react = require('react');
9
+ var TabListLayoutContext = require('./TabListLayoutContext.js');
9
10
  var TabNextContext = require('./TabNextContext.js');
10
11
  var TabNextTrigger$1 = require('./TabNextTrigger.css.js');
11
12
  var TabsNextContext = require('./TabsNextContext.js');
@@ -22,37 +23,135 @@ function getAriaDescription(count) {
22
23
  }
23
24
  const ariaActionSupported = typeof HTMLElement !== "undefined" && "ariaActions" in HTMLElement.prototype;
24
25
  const TabNextTrigger = react.forwardRef(function TabNextTrigger2(props, ref) {
25
- const { children, onClick, onKeyDown, ...rest } = props;
26
+ const { children, onClick, onKeyDown, onFocus, ...rest } = props;
26
27
  const targetWindow = window.useWindow();
27
28
  styles.useComponentCssInjection({
28
29
  testId: "salt-tab-next-trigger",
29
30
  css: TabNextTrigger$1,
30
31
  window: targetWindow
31
32
  });
32
- const { setSelected, registerTab, getPanelId } = TabsNextContext.useTabsNext();
33
+ const {
34
+ setSelected,
35
+ registerTab,
36
+ updateTab,
37
+ updateRenderedTab,
38
+ getRenderedTabOrder,
39
+ getPanelId,
40
+ getTabId,
41
+ selected: selectedValue,
42
+ activeTab,
43
+ menuOpen,
44
+ setMenuOpen
45
+ } = TabsNextContext.useTabsNext();
33
46
  const { selected, value, focused, disabled, tabId, actions } = TabNextContext.useTabNext();
47
+ const tabListLayout = TabListLayoutContext.useTabListLayout();
34
48
  const tabRef = react.useRef(null);
49
+ const location = (tabListLayout == null ? void 0 : tabListLayout.getLocation(value)) ?? "main";
50
+ const selectionSource = location === "overflow" ? "overflow" : "main";
51
+ const hidden = location === "hidden";
52
+ const overflowOpen = location === "overflow" && menuOpen;
53
+ const renderOrder = getRenderedTabOrder(value);
54
+ const order = renderOrder >= 0 ? renderOrder : void 0;
55
+ const initialLocationRef = react.useRef(location);
56
+ const initialOrderRef = react.useRef(order);
35
57
  const id = tabId;
36
58
  core.useIsomorphicLayoutEffect(() => {
37
- if (value && id && tabRef.current) {
38
- return registerTab({ id, value, element: tabRef.current });
59
+ if (id && tabRef.current) {
60
+ const item = {
61
+ id,
62
+ value,
63
+ element: tabRef.current,
64
+ location: initialLocationRef.current,
65
+ order: initialOrderRef.current
66
+ };
67
+ return registerTab(item);
39
68
  }
40
- }, [value, id, registerTab]);
69
+ }, [id, registerTab, value]);
70
+ core.useIsomorphicLayoutEffect(() => {
71
+ if (!id) {
72
+ return;
73
+ }
74
+ updateTab(id, {
75
+ element: tabRef.current,
76
+ location,
77
+ order
78
+ });
79
+ }, [id, location, order, updateTab]);
80
+ core.useIsomorphicLayoutEffect(() => {
81
+ updateRenderedTab(value, {
82
+ trigger: tabRef.current
83
+ });
84
+ }, [updateRenderedTab, value]);
85
+ core.useIsomorphicLayoutEffect(() => {
86
+ var _a;
87
+ if (!overflowOpen || (tabListLayout == null ? void 0 : tabListLayout.overflowActiveValue) !== value) {
88
+ return;
89
+ }
90
+ (_a = tabRef.current) == null ? void 0 : _a.focus({ preventScroll: true });
91
+ }, [overflowOpen, tabListLayout == null ? void 0 : tabListLayout.overflowActiveValue, value]);
41
92
  const handleClick = (event) => {
42
93
  onClick == null ? void 0 : onClick(event);
43
- setSelected(event, value);
94
+ setSelected(event, value, selectionSource);
44
95
  };
45
96
  const handleKeyDown = (event) => {
46
97
  onKeyDown == null ? void 0 : onKeyDown(event);
98
+ if (location === "overflow" && event.key === "Tab" && event.shiftKey) {
99
+ event.preventDefault();
100
+ setMenuOpen(false);
101
+ const doc = event.currentTarget.ownerDocument;
102
+ const overflowTrigger = doc.querySelector(
103
+ "[data-overflowbutton]"
104
+ );
105
+ const scheduleFocus = targetWindow == null ? void 0 : targetWindow.requestAnimationFrame;
106
+ if (scheduleFocus) {
107
+ scheduleFocus(() => overflowTrigger == null ? void 0 : overflowTrigger.focus({ preventScroll: true }));
108
+ } else {
109
+ queueMicrotask(() => overflowTrigger == null ? void 0 : overflowTrigger.focus({ preventScroll: true }));
110
+ }
111
+ return;
112
+ }
113
+ if (location === "overflow" && tabListLayout && (event.key === "ArrowDown" || event.key === "ArrowUp" || event.key === "Home" || event.key === "End")) {
114
+ const moved = tabListLayout.moveOverflowFocus(event.key, value);
115
+ if (moved) {
116
+ event.preventDefault();
117
+ return;
118
+ }
119
+ }
120
+ if (disabled) {
121
+ if (event.key === "Enter" || event.key === " ") {
122
+ event.preventDefault();
123
+ }
124
+ return;
125
+ }
47
126
  if (event.key === "Enter" || event.key === " ") {
48
- setSelected(event, value);
127
+ event.preventDefault();
128
+ setSelected(event, value, selectionSource);
129
+ }
130
+ };
131
+ const handleFocus = (event) => {
132
+ var _a;
133
+ onFocus == null ? void 0 : onFocus(event);
134
+ if (id) {
135
+ activeTab.current = { value, id };
136
+ }
137
+ if (location === "overflow") {
138
+ tabListLayout == null ? void 0 : tabListLayout.setOverflowActiveValue(value);
49
139
  }
140
+ (_a = event.currentTarget.parentElement) == null ? void 0 : _a.scrollIntoView({
141
+ block: "nearest",
142
+ inline: "nearest"
143
+ });
50
144
  };
51
145
  const handleRef = core.useForkRef(tabRef, ref);
52
146
  const panelId = getPanelId(value);
53
147
  const ariaActionsProps = ariaActionSupported ? {
54
148
  "aria-actions": clsx.clsx(actions) || void 0
55
149
  } : {};
150
+ const active = location === "overflow" && (tabListLayout == null ? void 0 : tabListLayout.overflowActiveValue) === value;
151
+ const hasSelectedTab = selectedValue !== void 0 && getTabId(selectedValue) != null;
152
+ const fallbackTabStop = !hasSelectedTab && location === "main" && order === 0;
153
+ const isTabStop = !hidden && (focused || selected || active || fallbackTabStop);
154
+ const shouldHandleKeyDown = location === "overflow" || !disabled;
56
155
  return /* @__PURE__ */ jsxRuntime.jsx(
57
156
  "button",
58
157
  {
@@ -61,14 +160,16 @@ const TabNextTrigger = react.forwardRef(function TabNextTrigger2(props, ref) {
61
160
  "aria-controls": panelId,
62
161
  ...ariaActionsProps,
63
162
  "aria-description": getAriaDescription(actions.length),
64
- tabIndex: focused || selected ? void 0 : -1,
163
+ tabIndex: isTabStop ? 0 : -1,
65
164
  role: "tab",
66
165
  type: "button",
67
166
  onClick: !disabled ? handleClick : void 0,
68
- onKeyDown: !disabled ? handleKeyDown : void 0,
167
+ onKeyDown: shouldHandleKeyDown ? handleKeyDown : void 0,
168
+ onFocus: handleFocus,
69
169
  className: withBaseName(),
70
170
  id,
71
171
  ref: handleRef,
172
+ "data-value": value,
72
173
  ...rest,
73
174
  children
74
175
  }
@@ -1 +1 @@
1
- {"version":3,"file":"TabNextTrigger.js","sources":["../src/tabs-next/TabNextTrigger.tsx"],"sourcesContent":["import {\n makePrefixer,\n useForkRef,\n useIsomorphicLayoutEffect,\n} from \"@salt-ds/core\";\nimport { useComponentCssInjection } from \"@salt-ds/styles\";\nimport { useWindow } from \"@salt-ds/window\";\nimport { clsx } from \"clsx\";\nimport {\n type ComponentPropsWithoutRef,\n forwardRef,\n type KeyboardEvent,\n type MouseEvent,\n useRef,\n} from \"react\";\nimport { useTabNext } from \"./TabNextContext\";\nimport tabTriggerCss from \"./TabNextTrigger.css\";\nimport { useTabsNext } from \"./TabsNextContext\";\n\nexport interface TabNextTriggerProps\n extends Omit<ComponentPropsWithoutRef<\"button\">, \"id\"> {}\n\nconst withBaseName = makePrefixer(\"saltTabNextTrigger\");\n\nfunction getAriaDescription(count: number) {\n if (count < 1) {\n return undefined;\n }\n\n if (count === 1) {\n return \"1 action available\";\n }\n\n return `${count} actions available`;\n}\n\nconst ariaActionSupported =\n typeof HTMLElement !== \"undefined\" && \"ariaActions\" in HTMLElement.prototype;\n\nexport const TabNextTrigger = forwardRef<\n HTMLButtonElement,\n TabNextTriggerProps\n>(function TabNextTrigger(props, ref) {\n const { children, onClick, onKeyDown, ...rest } = props;\n\n const targetWindow = useWindow();\n useComponentCssInjection({\n testId: \"salt-tab-next-trigger\",\n css: tabTriggerCss,\n window: targetWindow,\n });\n\n const { setSelected, registerTab, getPanelId } = useTabsNext();\n const { selected, value, focused, disabled, tabId, actions } = useTabNext();\n\n const tabRef = useRef<HTMLButtonElement>(null);\n\n const id = tabId;\n\n useIsomorphicLayoutEffect(() => {\n if (value && id && tabRef.current) {\n return registerTab({ id, value, element: tabRef.current });\n }\n }, [value, id, registerTab]);\n\n const handleClick = (event: MouseEvent<HTMLButtonElement>) => {\n onClick?.(event);\n\n setSelected(event, value);\n };\n\n const handleKeyDown = (event: KeyboardEvent<HTMLButtonElement>) => {\n onKeyDown?.(event);\n\n if (event.key === \"Enter\" || event.key === \" \") {\n setSelected(event, value);\n }\n };\n\n const handleRef = useForkRef<HTMLButtonElement>(tabRef, ref);\n const panelId = getPanelId(value);\n\n // Applying aria-actions this way avoid React warnings about unknown props\n const ariaActionsProps = ariaActionSupported\n ? {\n \"aria-actions\": clsx(actions) || undefined,\n }\n : {};\n\n return (\n <button\n aria-selected={selected}\n aria-disabled={disabled}\n aria-controls={panelId}\n {...ariaActionsProps}\n aria-description={getAriaDescription(actions.length)}\n tabIndex={focused || selected ? undefined : -1}\n role=\"tab\"\n type=\"button\"\n onClick={!disabled ? handleClick : undefined}\n onKeyDown={!disabled ? handleKeyDown : undefined}\n className={withBaseName()}\n id={id}\n ref={handleRef}\n {...rest}\n >\n {children}\n </button>\n );\n});\n"],"names":["makePrefixer","forwardRef","TabNextTrigger","useWindow","useComponentCssInjection","tabTriggerCss","useTabsNext","useTabNext","useRef","useIsomorphicLayoutEffect","useForkRef","clsx","jsx"],"mappings":"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"}
1
+ {"version":3,"file":"TabNextTrigger.js","sources":["../src/tabs-next/TabNextTrigger.tsx"],"sourcesContent":["import {\n makePrefixer,\n useForkRef,\n useIsomorphicLayoutEffect,\n} from \"@salt-ds/core\";\nimport { useComponentCssInjection } from \"@salt-ds/styles\";\nimport { useWindow } from \"@salt-ds/window\";\nimport { clsx } from \"clsx\";\nimport {\n type ComponentPropsWithoutRef,\n type FocusEvent,\n forwardRef,\n type KeyboardEvent,\n type MouseEvent,\n useRef,\n} from \"react\";\nimport { useTabListLayout } from \"./TabListLayoutContext\";\nimport { useTabNext } from \"./TabNextContext\";\nimport tabTriggerCss from \"./TabNextTrigger.css\";\nimport { useTabsNext } from \"./TabsNextContext\";\n\nexport interface TabNextTriggerProps\n extends Omit<ComponentPropsWithoutRef<\"button\">, \"id\"> {}\n\nconst withBaseName = makePrefixer(\"saltTabNextTrigger\");\n\nfunction getAriaDescription(count: number) {\n if (count < 1) {\n return undefined;\n }\n\n if (count === 1) {\n return \"1 action available\";\n }\n\n return `${count} actions available`;\n}\n\nconst ariaActionSupported =\n typeof HTMLElement !== \"undefined\" && \"ariaActions\" in HTMLElement.prototype;\n\nexport const TabNextTrigger = forwardRef<\n HTMLButtonElement,\n TabNextTriggerProps\n>(function TabNextTrigger(props, ref) {\n const { children, onClick, onKeyDown, onFocus, ...rest } = props;\n\n const targetWindow = useWindow();\n useComponentCssInjection({\n testId: \"salt-tab-next-trigger\",\n css: tabTriggerCss,\n window: targetWindow,\n });\n\n const {\n setSelected,\n registerTab,\n updateTab,\n updateRenderedTab,\n getRenderedTabOrder,\n getPanelId,\n getTabId,\n selected: selectedValue,\n activeTab,\n menuOpen,\n setMenuOpen,\n } = useTabsNext();\n const { selected, value, focused, disabled, tabId, actions } = useTabNext();\n const tabListLayout = useTabListLayout();\n\n const tabRef = useRef<HTMLButtonElement>(null);\n\n const location = tabListLayout?.getLocation(value) ?? \"main\";\n const selectionSource = location === \"overflow\" ? \"overflow\" : \"main\";\n const hidden = location === \"hidden\";\n const overflowOpen = location === \"overflow\" && menuOpen;\n const renderOrder = getRenderedTabOrder(value);\n const order = renderOrder >= 0 ? renderOrder : undefined;\n const initialLocationRef = useRef(location);\n const initialOrderRef = useRef(order);\n\n const id = tabId;\n\n useIsomorphicLayoutEffect(() => {\n if (id && tabRef.current) {\n const item = {\n id,\n value,\n element: tabRef.current,\n location: initialLocationRef.current,\n order: initialOrderRef.current,\n };\n\n return registerTab(item);\n }\n }, [id, registerTab, value]);\n\n useIsomorphicLayoutEffect(() => {\n if (!id) {\n return;\n }\n\n updateTab(id, {\n element: tabRef.current,\n location,\n order,\n });\n }, [id, location, order, updateTab]);\n\n useIsomorphicLayoutEffect(() => {\n updateRenderedTab(value, {\n trigger: tabRef.current,\n });\n }, [updateRenderedTab, value]);\n\n useIsomorphicLayoutEffect(() => {\n if (!overflowOpen || tabListLayout?.overflowActiveValue !== value) {\n return;\n }\n\n tabRef.current?.focus({ preventScroll: true });\n }, [overflowOpen, tabListLayout?.overflowActiveValue, value]);\n\n const handleClick = (event: MouseEvent<HTMLButtonElement>) => {\n onClick?.(event);\n\n setSelected(event, value, selectionSource);\n };\n\n const handleKeyDown = (event: KeyboardEvent<HTMLButtonElement>) => {\n onKeyDown?.(event);\n\n if (location === \"overflow\" && event.key === \"Tab\" && event.shiftKey) {\n event.preventDefault();\n setMenuOpen(false);\n\n const doc = event.currentTarget.ownerDocument;\n const overflowTrigger = doc.querySelector<HTMLElement>(\n \"[data-overflowbutton]\",\n );\n const scheduleFocus = targetWindow?.requestAnimationFrame;\n\n if (scheduleFocus) {\n scheduleFocus(() => overflowTrigger?.focus({ preventScroll: true }));\n } else {\n queueMicrotask(() => overflowTrigger?.focus({ preventScroll: true }));\n }\n\n return;\n }\n\n if (\n location === \"overflow\" &&\n tabListLayout &&\n (event.key === \"ArrowDown\" ||\n event.key === \"ArrowUp\" ||\n event.key === \"Home\" ||\n event.key === \"End\")\n ) {\n const moved = tabListLayout.moveOverflowFocus(event.key, value);\n if (moved) {\n event.preventDefault();\n return;\n }\n }\n\n if (disabled) {\n if (event.key === \"Enter\" || event.key === \" \") {\n event.preventDefault();\n }\n return;\n }\n\n if (event.key === \"Enter\" || event.key === \" \") {\n event.preventDefault();\n setSelected(event, value, selectionSource);\n }\n };\n\n const handleFocus = (event: FocusEvent<HTMLButtonElement>) => {\n onFocus?.(event);\n\n if (id) {\n activeTab.current = { value, id };\n }\n\n if (location === \"overflow\") {\n tabListLayout?.setOverflowActiveValue(value);\n }\n\n // Ensures the associated tab is in view.\n event.currentTarget.parentElement?.scrollIntoView({\n block: \"nearest\",\n inline: \"nearest\",\n });\n };\n\n const handleRef = useForkRef<HTMLButtonElement>(tabRef, ref);\n const panelId = getPanelId(value);\n\n // Applying aria-actions this way avoids React warnings about unknown props\n const ariaActionsProps = ariaActionSupported\n ? {\n \"aria-actions\": clsx(actions) || undefined,\n }\n : {};\n\n const active =\n location === \"overflow\" && tabListLayout?.overflowActiveValue === value;\n const hasSelectedTab =\n selectedValue !== undefined && getTabId(selectedValue) != null;\n const fallbackTabStop = !hasSelectedTab && location === \"main\" && order === 0;\n const isTabStop =\n !hidden && (focused || selected || active || fallbackTabStop);\n const shouldHandleKeyDown = location === \"overflow\" || !disabled;\n\n return (\n <button\n aria-selected={selected}\n aria-disabled={disabled}\n aria-controls={panelId}\n {...ariaActionsProps}\n aria-description={getAriaDescription(actions.length)}\n tabIndex={isTabStop ? 0 : -1}\n role=\"tab\"\n type=\"button\"\n onClick={!disabled ? handleClick : undefined}\n onKeyDown={shouldHandleKeyDown ? handleKeyDown : undefined}\n onFocus={handleFocus}\n className={withBaseName()}\n id={id}\n ref={handleRef}\n data-value={value}\n {...rest}\n >\n {children}\n </button>\n );\n});\n"],"names":["makePrefixer","forwardRef","TabNextTrigger","useWindow","useComponentCssInjection","tabTriggerCss","useTabsNext","useTabNext","useTabListLayout","useRef","useIsomorphicLayoutEffect","useForkRef","clsx","jsx"],"mappings":"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"}
@@ -1,6 +1,6 @@
1
1
  'use strict';
2
2
 
3
- var css_248z = ".saltTabOverflow {\n position: relative;\n}\n\n.saltTabOverflow-list {\n background: var(--salt-container-primary-background);\n border: var(--salt-size-fixed-100) var(--salt-borderStyle-solid) var(--salt-selectable-borderColor-selected);\n overflow: hidden;\n overflow-y: auto;\n position: absolute;\n z-index: var(--salt-zIndex-flyover);\n box-shadow: var(--salt-overlayable-shadow-popout);\n box-sizing: border-box;\n border-radius: var(--salt-palette-corner, 0);\n}\n\n.saltTabOverflow-listContainer {\n display: flex;\n flex-direction: column;\n gap: var(--salt-spacing-fixed-100);\n max-height: inherit;\n min-height: inherit;\n}\n\n.saltTabOverflow-list[data-hidden=\"true\"] {\n opacity: 0;\n pointer-events: none;\n /* Avoid causing page to overflow with the hidden elements */\n width: 1px;\n height: 1px;\n}\n\n.saltTabOverflow-list .saltTabNext {\n color: var(--salt-content-primary-foreground);\n background: var(--salt-selectable-background);\n font-size: var(--salt-text-fontSize);\n font-weight: var(--salt-text-fontWeight);\n min-height: calc(var(--salt-size-base) + var(--salt-spacing-100));\n padding-left: var(--salt-spacing-100);\n padding-right: var(--salt-spacing-100);\n display: flex;\n gap: var(--salt-spacing-100);\n position: relative;\n align-items: center;\n cursor: var(--salt-cursor-hover);\n box-sizing: border-box;\n flex-shrink: 0;\n justify-content: flex-start;\n}\n\n@supports selector(:has(*)) {\n .saltTabListNext:has([data-overflowbutton]) .saltTabOverflow-list .saltTabNext {\n max-width: unset;\n }\n}\n\n.saltTabOverflow-list .saltTabNext .saltTabNextTrigger {\n justify-content: start;\n}\n\n.saltTabOverflow-list .saltTabNext::after {\n display: none;\n}\n\n.saltTabOverflow-list .saltTabNext[aria-disabled=\"true\"] {\n color: var(--salt-content-primary-foreground-disabled);\n cursor: var(--salt-cursor-disabled);\n}\n\n.saltTabOverflow-list .saltTabNext-focusVisible {\n outline: var(--salt-focused-outline);\n outline-offset: calc(var(--salt-size-fixed-100) * -2);\n}\n\n.saltTabOverflow-list .saltTabNext:hover {\n background: var(--salt-selectable-background-hover);\n}\n\n.saltTabOverflow-list .saltTabNext:active {\n background: var(--salt-selectable-background-selected);\n box-shadow:\n 0 calc(var(--salt-size-fixed-100) * -1) 0 0 var(--salt-selectable-borderColor-selected),\n 0 var(--salt-size-fixed-100) 0 0 var(--salt-selectable-borderColor-selected);\n}\n";
3
+ var css_248z = ".saltTabOverflow {\n position: relative;\n}\n\n.saltTabOverflow-list {\n background: var(--salt-container-primary-background);\n border: var(--salt-size-fixed-100) var(--salt-borderStyle-solid) var(--salt-selectable-borderColor-selected);\n overflow: hidden;\n overflow-y: auto;\n position: absolute;\n box-shadow: var(--salt-overlayable-shadow-popout);\n box-sizing: border-box;\n border-radius: var(--salt-palette-corner, 0);\n z-index: var(--salt-zIndex-flyover);\n}\n\n.saltTabOverflow-listContainer {\n display: flex;\n flex-direction: column;\n gap: var(--salt-spacing-fixed-100);\n max-height: inherit;\n min-height: inherit;\n}\n\n.saltTabOverflow-list .saltTabNext {\n color: var(--salt-content-primary-foreground);\n background: var(--salt-selectable-background);\n font-size: var(--salt-text-fontSize);\n font-weight: var(--salt-text-fontWeight);\n min-height: calc(var(--salt-size-base) + var(--salt-spacing-100));\n padding-left: var(--salt-spacing-100);\n padding-right: var(--salt-spacing-100);\n display: flex;\n gap: var(--salt-spacing-100);\n position: relative;\n align-items: center;\n cursor: var(--salt-cursor-hover);\n box-sizing: border-box;\n flex-shrink: 0;\n justify-content: flex-start;\n}\n\n@supports selector(:has(*)) {\n .saltTabListNext:has([data-overflowbutton]) .saltTabOverflow-list .saltTabNext {\n max-width: unset;\n }\n}\n\n.saltTabOverflow-list .saltTabNext .saltTabNextTrigger {\n justify-content: start;\n}\n\n.saltTabOverflow-list .saltTabNext::after {\n display: none;\n}\n\n.saltTabOverflow-list .saltTabNext[aria-disabled=\"true\"] {\n color: var(--salt-content-primary-foreground-disabled);\n cursor: var(--salt-cursor-disabled);\n}\n\n.saltTabOverflow-list .saltTabNext-focusVisible {\n outline: var(--salt-focused-outline);\n outline-offset: calc(var(--salt-size-fixed-100) * -2);\n}\n\n.saltTabOverflow-list .saltTabNext:hover {\n background: var(--salt-selectable-background-hover);\n}\n\n.saltTabOverflow-list .saltTabNext:active {\n background: var(--salt-selectable-background-selected);\n box-shadow:\n 0 calc(var(--salt-size-fixed-100) * -1) 0 0 var(--salt-selectable-borderColor-selected),\n 0 var(--salt-size-fixed-100) 0 0 var(--salt-selectable-borderColor-selected);\n}\n";
4
4
 
5
5
  module.exports = css_248z;
6
6
  //# sourceMappingURL=TabOverflowList.css.js.map
@@ -5,18 +5,23 @@ var react$1 = require('@floating-ui/react');
5
5
  var core = require('@salt-ds/core');
6
6
  var styles = require('@salt-ds/styles');
7
7
  var window = require('@salt-ds/window');
8
+ var clsx = require('clsx');
8
9
  var react = require('react');
9
- var useFocusOutside = require('./hooks/useFocusOutside.js');
10
+ var domUtils = require('./domUtils.js');
10
11
  var TabOverflowList$1 = require('./TabOverflowList.css.js');
12
+ var TabSlot = require('./TabSlot.js');
13
+ var TabsNextContext = require('./TabsNextContext.js');
14
+ var widthMeasurement = require('./widthMeasurement.js');
11
15
 
12
16
  const withBaseName = core.makePrefixer("saltTabOverflow");
13
17
  const TabOverflowList = react.forwardRef(
14
18
  function TabOverflowList2(props, ref) {
19
+ var _a, _b;
15
20
  const {
16
21
  buttonRef,
17
- tabstripRef,
18
- children,
19
- isMeasuring,
22
+ className,
23
+ hiddenValues,
24
+ order,
20
25
  open,
21
26
  setOpen,
22
27
  ...rest
@@ -27,104 +32,203 @@ const TabOverflowList = react.forwardRef(
27
32
  css: TabOverflowList$1,
28
33
  window: targetWindow
29
34
  });
35
+ const overflowRef = react.useRef(null);
36
+ const hadOverflowItemsRef = react.useRef(false);
30
37
  const { OverflowIcon } = core.useIcon();
31
- const { refs, x, y, strategy, context } = core.useFloatingUI({
38
+ const { registerTab, updateTab, activeTab } = TabsNextContext.useTabsNext();
39
+ const { refs, x, y, strategy, context, elements } = core.useFloatingUI({
32
40
  open,
33
41
  onOpenChange(open2, _, reason) {
42
+ var _a2;
43
+ setOpen(open2);
34
44
  if (reason === "escape-key") {
35
- queueMicrotask(() => {
36
- var _a, _b, _c;
37
- const allTabs = ((_a = tabstripRef.current) == null ? void 0 : _a.querySelectorAll(
38
- '[role="tab"]:not([aria-hidden])'
39
- )) ?? [];
40
- const numberOfTabsInOverflow = ((_b = listRef.current) == null ? void 0 : _b.querySelectorAll('[role="tab"]').length) ?? 0;
41
- (_c = allTabs[allTabs.length - numberOfTabsInOverflow - 1]) == null ? void 0 : _c.focus({
42
- preventScroll: true
43
- });
44
- });
45
+ (_a2 = overflowRef.current) == null ? void 0 : _a2.focus();
45
46
  }
46
- setOpen(open2);
47
47
  },
48
48
  placement: "bottom-start",
49
49
  middleware: [
50
50
  react$1.offset(1),
51
51
  react$1.size({
52
- apply({ elements, availableHeight }) {
53
- Object.assign(elements.floating.style, {
52
+ apply({ elements: elements2, availableHeight }) {
53
+ Object.assign(elements2.floating.style, {
54
54
  maxHeight: `max(calc((var(--salt-size-base) + var(--salt-spacing-100)) * 5), calc(${availableHeight}px - var(--salt-spacing-100)))`
55
55
  });
56
56
  }
57
57
  }),
58
- react$1.flip()
58
+ react$1.flip(),
59
+ react$1.shift({
60
+ padding: 8
61
+ })
59
62
  ]
60
63
  });
61
- const { getFloatingProps } = react$1.useInteractions([react$1.useDismiss(context)]);
62
- const rootRef = react.useRef(null);
63
- const handleRootRef = core.useForkRef(rootRef, ref);
64
- const listRef = react.useRef(null);
65
- const handleListRef = core.useForkRef(listRef, refs.setFloating);
66
- const handleFocusOutside = react.useCallback(() => {
67
- setOpen(false);
68
- }, [setOpen]);
69
- useFocusOutside.useFocusOutside(
70
- rootRef,
71
- handleFocusOutside,
72
- open,
73
- "[data-floating-ui-portal]"
74
- );
75
- const handleClick = () => {
76
- var _a, _b;
77
- if (!open) {
78
- (_b = (_a = listRef.current) == null ? void 0 : _a.querySelectorAll('[role="tab"]')[0]) == null ? void 0 : _b.focus({ preventScroll: true });
79
- } else {
80
- setOpen(false);
81
- }
82
- };
83
- const handleFocus = () => {
84
- setOpen(true);
85
- };
64
+ const { getFloatingProps, getReferenceProps } = react$1.useInteractions([
65
+ react$1.useClick(context),
66
+ react$1.useDismiss(context)
67
+ ]);
68
+ const handleListRef = core.useForkRef(ref, refs.setFloating);
86
69
  const handleButtonRef = core.useForkRef(
87
70
  buttonRef,
88
71
  refs.setReference
89
72
  );
90
- const listId = core.useId();
91
- const childCount = react.Children.count(children);
92
- if (childCount === 0 && !isMeasuring) return null;
93
- return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: withBaseName(), ref: handleRootRef, "data-overflow": true, children: [
73
+ const handleRef = core.useForkRef(handleButtonRef, overflowRef);
74
+ const { Component: FloatingComponent } = core.useFloatingComponent();
75
+ const overflowId = core.useId();
76
+ const overlayId = core.useId();
77
+ const handleFocus = () => {
78
+ if (overflowId) {
79
+ activeTab.current = { value: overflowId, id: overflowId };
80
+ }
81
+ };
82
+ const handleFloatingKeyDown = (event) => {
83
+ if (event.key !== "Tab") {
84
+ return;
85
+ }
86
+ if (event.shiftKey) {
87
+ event.preventDefault();
88
+ setOpen(false);
89
+ const scheduleFocus = targetWindow == null ? void 0 : targetWindow.requestAnimationFrame;
90
+ if (scheduleFocus) {
91
+ scheduleFocus(
92
+ () => {
93
+ var _a2;
94
+ return (_a2 = overflowRef.current) == null ? void 0 : _a2.focus({ preventScroll: true });
95
+ }
96
+ );
97
+ return;
98
+ }
99
+ queueMicrotask(
100
+ () => {
101
+ var _a2;
102
+ return (_a2 = overflowRef.current) == null ? void 0 : _a2.focus({ preventScroll: true });
103
+ }
104
+ );
105
+ return;
106
+ }
107
+ };
108
+ const overflowItemCount = hiddenValues.length;
109
+ const hasOverflowItems = overflowItemCount > 0;
110
+ const initialOrderRef = react.useRef(order);
111
+ react.useEffect(() => {
112
+ var _a2;
113
+ const tabList = (_a2 = overflowRef.current) == null ? void 0 : _a2.parentElement;
114
+ const resizeObserverCtor = targetWindow == null ? void 0 : targetWindow.ResizeObserver;
115
+ if (!open || !tabList || !resizeObserverCtor) {
116
+ return;
117
+ }
118
+ const observedElements = [tabList, tabList.parentElement].filter(
119
+ (element) => element != null
120
+ );
121
+ const widths = widthMeasurement.seedWidthMap(observedElements);
122
+ const resizeObserver = new resizeObserverCtor(
123
+ (entries) => {
124
+ for (const entry of entries) {
125
+ if (!domUtils.isHTMLElement(entry.target)) {
126
+ continue;
127
+ }
128
+ const nextWidth = entry.contentRect.width || widthMeasurement.getMeasuredWidth(entry.target);
129
+ if (widthMeasurement.updateWidthMap(widths, entry.target, nextWidth)) {
130
+ setOpen(false);
131
+ return;
132
+ }
133
+ }
134
+ }
135
+ );
136
+ for (const element of observedElements) {
137
+ resizeObserver.observe(element);
138
+ }
139
+ return () => {
140
+ resizeObserver.disconnect();
141
+ };
142
+ }, [open, setOpen, targetWindow]);
143
+ core.useIsomorphicLayoutEffect(() => {
144
+ if (open && !hasOverflowItems) {
145
+ setOpen(false);
146
+ }
147
+ }, [hasOverflowItems, open, setOpen]);
148
+ core.useIsomorphicLayoutEffect(() => {
149
+ if (hasOverflowItems && !hadOverflowItemsRef.current) {
150
+ setOpen(false);
151
+ }
152
+ hadOverflowItemsRef.current = hasOverflowItems;
153
+ }, [hasOverflowItems, setOpen]);
154
+ core.useIsomorphicLayoutEffect(() => {
155
+ if (overflowId && overflowRef.current && hasOverflowItems) {
156
+ const item = {
157
+ id: overflowId,
158
+ value: overflowId,
159
+ element: overflowRef.current,
160
+ location: "main",
161
+ order: initialOrderRef.current
162
+ };
163
+ return registerTab(item);
164
+ }
165
+ }, [hasOverflowItems, overflowId, registerTab]);
166
+ core.useIsomorphicLayoutEffect(() => {
167
+ if (!overflowId || !hasOverflowItems) {
168
+ return;
169
+ }
170
+ updateTab(overflowId, {
171
+ element: overflowRef.current,
172
+ location: "main",
173
+ order
174
+ });
175
+ }, [hasOverflowItems, order, overflowId, updateTab]);
176
+ if (!hasOverflowItems) return null;
177
+ return /* @__PURE__ */ jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [
94
178
  /* @__PURE__ */ jsxRuntime.jsx(
95
179
  core.Button,
96
180
  {
181
+ className: clsx.clsx(withBaseName(), className),
97
182
  "data-overflowbutton": true,
98
- tabIndex: -1,
99
183
  appearance: "transparent",
100
184
  sentiment: "neutral",
101
- onClick: handleClick,
102
- ref: handleButtonRef,
103
- "aria-label": `Overflow menu. ${childCount} tabs hidden`,
185
+ ...getReferenceProps({
186
+ onFocus: handleFocus
187
+ }),
188
+ ref: handleRef,
189
+ "aria-label": "Overflow",
104
190
  "aria-expanded": open,
105
- "aria-controls": listId,
106
- "aria-hidden": "true",
191
+ "aria-controls": overlayId,
107
192
  role: "tab",
108
- "aria-haspopup": true,
193
+ tabIndex: -1,
109
194
  ...rest,
110
195
  children: /* @__PURE__ */ jsxRuntime.jsx(OverflowIcon, { "aria-hidden": true })
111
196
  }
112
197
  ),
113
- /* @__PURE__ */ jsxRuntime.jsx(react$1.FloatingTree, { children: /* @__PURE__ */ jsxRuntime.jsx(
114
- "div",
198
+ /* @__PURE__ */ jsxRuntime.jsx(
199
+ FloatingComponent,
115
200
  {
116
201
  ref: handleListRef,
117
202
  ...getFloatingProps({
118
- onFocus: handleFocus,
119
- role: "presentation"
203
+ id: overlayId,
204
+ onKeyDown: handleFloatingKeyDown
120
205
  }),
206
+ focusManagerProps: context ? {
207
+ context,
208
+ initialFocus: 0,
209
+ returnFocus: false,
210
+ modal: false,
211
+ closeOnFocusOut: true
212
+ } : void 0,
121
213
  className: withBaseName("list"),
122
- "data-hidden": !open,
123
- style: open ? { left: x ?? 0, top: y ?? 0, position: strategy } : void 0,
124
- id: listId,
125
- children: /* @__PURE__ */ jsxRuntime.jsx("div", { className: withBaseName("listContainer"), children })
214
+ open,
215
+ left: x ?? 0,
216
+ top: y ?? 0,
217
+ position: strategy,
218
+ width: (_a = elements.floating) == null ? void 0 : _a.offsetWidth,
219
+ height: (_b = elements.floating) == null ? void 0 : _b.offsetHeight,
220
+ children: /* @__PURE__ */ jsxRuntime.jsx(
221
+ "div",
222
+ {
223
+ role: "tablist",
224
+ "aria-orientation": "vertical",
225
+ className: withBaseName("listContainer"),
226
+ "aria-label": "Overflow tab options",
227
+ children: hiddenValues.map((value) => /* @__PURE__ */ jsxRuntime.jsx(TabSlot.TabSlot, { slotId: `overflow:${value}`, value }, value))
228
+ }
229
+ )
126
230
  }
127
- ) })
231
+ )
128
232
  ] });
129
233
  }
130
234
  );