@salt-ds/lab 1.0.0-alpha.10 → 1.0.0-alpha.12

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 (234) hide show
  1. package/dist-cjs/accordion/Accordion.css.js +1 -1
  2. package/dist-cjs/accordion/Accordion.js +33 -108
  3. package/dist-cjs/accordion/Accordion.js.map +1 -1
  4. package/dist-cjs/accordion/AccordionContext.js +15 -11
  5. package/dist-cjs/accordion/AccordionContext.js.map +1 -1
  6. package/dist-cjs/accordion/AccordionGroup.js +23 -0
  7. package/dist-cjs/accordion/AccordionGroup.js.map +1 -0
  8. package/dist-cjs/accordion/AccordionHeader.css.js +6 -0
  9. package/dist-cjs/accordion/AccordionHeader.css.js.map +1 -0
  10. package/dist-cjs/accordion/AccordionHeader.js +50 -0
  11. package/dist-cjs/accordion/AccordionHeader.js.map +1 -0
  12. package/dist-cjs/accordion/AccordionPanel.css.js +6 -0
  13. package/dist-cjs/accordion/AccordionPanel.css.js.map +1 -0
  14. package/dist-cjs/accordion/AccordionPanel.js +45 -0
  15. package/dist-cjs/accordion/AccordionPanel.js.map +1 -0
  16. package/dist-cjs/common-hooks/useKeyboardNavigation.js +0 -3
  17. package/dist-cjs/common-hooks/useKeyboardNavigation.js.map +1 -1
  18. package/dist-cjs/dialog/DialogTitle.css.js +1 -1
  19. package/dist-cjs/dialog/DialogTitle.js +2 -3
  20. package/dist-cjs/dialog/DialogTitle.js.map +1 -1
  21. package/dist-cjs/dropdown/Dropdown.js +5 -2
  22. package/dist-cjs/dropdown/Dropdown.js.map +1 -1
  23. package/dist-cjs/dropdown/useDropdown.js +12 -0
  24. package/dist-cjs/dropdown/useDropdown.js.map +1 -1
  25. package/dist-cjs/form-field-legacy/FormFieldLegacy.css.js +1 -1
  26. package/dist-cjs/index.js +23 -10
  27. package/dist-cjs/index.js.map +1 -1
  28. package/dist-cjs/list/useList.js +0 -1
  29. package/dist-cjs/list/useList.js.map +1 -1
  30. package/dist-cjs/list-deprecated/ListItem.css.js +1 -1
  31. package/dist-cjs/list-next/ListItemNext.css.js +6 -0
  32. package/dist-cjs/list-next/ListItemNext.css.js.map +1 -0
  33. package/dist-cjs/list-next/ListItemNext.js +74 -0
  34. package/dist-cjs/list-next/ListItemNext.js.map +1 -0
  35. package/dist-cjs/list-next/ListNext.css.js +6 -0
  36. package/dist-cjs/list-next/ListNext.css.js.map +1 -0
  37. package/dist-cjs/list-next/ListNext.js +92 -0
  38. package/dist-cjs/list-next/ListNext.js.map +1 -0
  39. package/dist-cjs/list-next/ListNextContext.js +18 -0
  40. package/dist-cjs/list-next/ListNextContext.js.map +1 -0
  41. package/dist-cjs/list-next/useList.js +188 -0
  42. package/dist-cjs/list-next/useList.js.map +1 -0
  43. package/dist-cjs/logo/Logo.css.js +1 -1
  44. package/dist-cjs/logo/Logo.js +4 -36
  45. package/dist-cjs/logo/Logo.js.map +1 -1
  46. package/dist-cjs/logo/LogoImage.js +24 -0
  47. package/dist-cjs/logo/LogoImage.js.map +1 -0
  48. package/dist-cjs/logo/LogoSeparator.css.js +6 -0
  49. package/dist-cjs/logo/LogoSeparator.css.js.map +1 -0
  50. package/dist-cjs/logo/LogoSeparator.js +32 -0
  51. package/dist-cjs/logo/LogoSeparator.js.map +1 -0
  52. package/dist-cjs/multiline-input/MultilineInput.css.js +6 -0
  53. package/dist-cjs/multiline-input/MultilineInput.css.js.map +1 -0
  54. package/dist-cjs/multiline-input/MultilineInput.js +150 -0
  55. package/dist-cjs/multiline-input/MultilineInput.js.map +1 -0
  56. package/dist-cjs/pill/Pill.css.js +1 -1
  57. package/dist-cjs/query-input/internal/QueryInputBody.js +5 -5
  58. package/dist-cjs/query-input/internal/QueryInputBody.js.map +1 -1
  59. package/dist-cjs/tabs-next/OverflowMenu.js +49 -0
  60. package/dist-cjs/tabs-next/OverflowMenu.js.map +1 -0
  61. package/dist-cjs/tabs-next/TabNext.css.js +6 -0
  62. package/dist-cjs/tabs-next/TabNext.css.js.map +1 -0
  63. package/dist-cjs/tabs-next/TabNext.js +87 -0
  64. package/dist-cjs/tabs-next/TabNext.js.map +1 -0
  65. package/dist-cjs/tabs-next/TabNextContext.js +23 -0
  66. package/dist-cjs/tabs-next/TabNextContext.js.map +1 -0
  67. package/dist-cjs/tabs-next/TabstripNext.css.js +6 -0
  68. package/dist-cjs/tabs-next/TabstripNext.css.js.map +1 -0
  69. package/dist-cjs/tabs-next/TabstripNext.js +180 -0
  70. package/dist-cjs/tabs-next/TabstripNext.js.map +1 -0
  71. package/dist-cjs/toggle-button/ToggleButton.css.js +1 -1
  72. package/dist-cjs/toggle-button/ToggleButton.js +44 -74
  73. package/dist-cjs/toggle-button/ToggleButton.js.map +1 -1
  74. package/dist-cjs/toggle-button-group/ToggleButtonGroup.css.js +6 -0
  75. package/dist-cjs/toggle-button-group/ToggleButtonGroup.js +111 -0
  76. package/dist-cjs/toggle-button-group/ToggleButtonGroup.js.map +1 -0
  77. package/dist-cjs/toggle-button-group/ToggleButtonGroupContext.js +15 -0
  78. package/dist-cjs/toggle-button-group/ToggleButtonGroupContext.js.map +1 -0
  79. package/dist-cjs/tokenized-input/TokenizedInput.css.js +1 -1
  80. package/dist-cjs/toolbar/ToolbarButton.css.js +1 -1
  81. package/dist-es/accordion/Accordion.css.js +1 -1
  82. package/dist-es/accordion/Accordion.js +35 -110
  83. package/dist-es/accordion/Accordion.js.map +1 -1
  84. package/dist-es/accordion/AccordionContext.js +15 -11
  85. package/dist-es/accordion/AccordionContext.js.map +1 -1
  86. package/dist-es/accordion/AccordionGroup.js +19 -0
  87. package/dist-es/accordion/AccordionGroup.js.map +1 -0
  88. package/dist-es/accordion/AccordionHeader.css.js +4 -0
  89. package/dist-es/accordion/AccordionHeader.css.js.map +1 -0
  90. package/dist-es/accordion/AccordionHeader.js +46 -0
  91. package/dist-es/accordion/AccordionHeader.js.map +1 -0
  92. package/dist-es/accordion/AccordionPanel.css.js +4 -0
  93. package/dist-es/accordion/AccordionPanel.css.js.map +1 -0
  94. package/dist-es/accordion/AccordionPanel.js +41 -0
  95. package/dist-es/accordion/AccordionPanel.js.map +1 -0
  96. package/dist-es/common-hooks/useKeyboardNavigation.js +0 -3
  97. package/dist-es/common-hooks/useKeyboardNavigation.js.map +1 -1
  98. package/dist-es/dialog/DialogTitle.css.js +1 -1
  99. package/dist-es/dialog/DialogTitle.js +3 -4
  100. package/dist-es/dialog/DialogTitle.js.map +1 -1
  101. package/dist-es/dropdown/Dropdown.js +5 -2
  102. package/dist-es/dropdown/Dropdown.js.map +1 -1
  103. package/dist-es/dropdown/useDropdown.js +12 -0
  104. package/dist-es/dropdown/useDropdown.js.map +1 -1
  105. package/dist-es/form-field-legacy/FormFieldLegacy.css.js +1 -1
  106. package/dist-es/index.js +12 -5
  107. package/dist-es/index.js.map +1 -1
  108. package/dist-es/list/useList.js +0 -1
  109. package/dist-es/list/useList.js.map +1 -1
  110. package/dist-es/list-deprecated/ListItem.css.js +1 -1
  111. package/dist-es/list-next/ListItemNext.css.js +4 -0
  112. package/dist-es/list-next/ListItemNext.css.js.map +1 -0
  113. package/dist-es/list-next/ListItemNext.js +70 -0
  114. package/dist-es/list-next/ListItemNext.js.map +1 -0
  115. package/dist-es/list-next/ListNext.css.js +4 -0
  116. package/dist-es/list-next/ListNext.css.js.map +1 -0
  117. package/dist-es/list-next/ListNext.js +88 -0
  118. package/dist-es/list-next/ListNext.js.map +1 -0
  119. package/dist-es/list-next/ListNextContext.js +13 -0
  120. package/dist-es/list-next/ListNextContext.js.map +1 -0
  121. package/dist-es/list-next/useList.js +184 -0
  122. package/dist-es/list-next/useList.js.map +1 -0
  123. package/dist-es/logo/Logo.css.js +1 -1
  124. package/dist-es/logo/Logo.js +5 -37
  125. package/dist-es/logo/Logo.js.map +1 -1
  126. package/dist-es/logo/LogoImage.js +20 -0
  127. package/dist-es/logo/LogoImage.js.map +1 -0
  128. package/dist-es/logo/LogoSeparator.css.js +4 -0
  129. package/dist-es/logo/LogoSeparator.css.js.map +1 -0
  130. package/dist-es/logo/LogoSeparator.js +28 -0
  131. package/dist-es/logo/LogoSeparator.js.map +1 -0
  132. package/dist-es/multiline-input/MultilineInput.css.js +4 -0
  133. package/dist-es/multiline-input/MultilineInput.css.js.map +1 -0
  134. package/dist-es/multiline-input/MultilineInput.js +146 -0
  135. package/dist-es/multiline-input/MultilineInput.js.map +1 -0
  136. package/dist-es/pill/Pill.css.js +1 -1
  137. package/dist-es/query-input/internal/QueryInputBody.js +5 -5
  138. package/dist-es/query-input/internal/QueryInputBody.js.map +1 -1
  139. package/dist-es/tabs-next/OverflowMenu.js +45 -0
  140. package/dist-es/tabs-next/OverflowMenu.js.map +1 -0
  141. package/dist-es/tabs-next/TabNext.css.js +4 -0
  142. package/dist-es/tabs-next/TabNext.css.js.map +1 -0
  143. package/dist-es/tabs-next/TabNext.js +79 -0
  144. package/dist-es/tabs-next/TabNext.js.map +1 -0
  145. package/dist-es/tabs-next/TabNextContext.js +18 -0
  146. package/dist-es/tabs-next/TabNextContext.js.map +1 -0
  147. package/dist-es/tabs-next/TabstripNext.css.js +4 -0
  148. package/dist-es/tabs-next/TabstripNext.css.js.map +1 -0
  149. package/dist-es/tabs-next/TabstripNext.js +172 -0
  150. package/dist-es/tabs-next/TabstripNext.js.map +1 -0
  151. package/dist-es/toggle-button/ToggleButton.css.js +1 -1
  152. package/dist-es/toggle-button/ToggleButton.js +45 -75
  153. package/dist-es/toggle-button/ToggleButton.js.map +1 -1
  154. package/dist-es/toggle-button-group/ToggleButtonGroup.css.js +4 -0
  155. package/dist-es/toggle-button-group/ToggleButtonGroup.js +107 -0
  156. package/dist-es/toggle-button-group/ToggleButtonGroup.js.map +1 -0
  157. package/dist-es/toggle-button-group/ToggleButtonGroupContext.js +10 -0
  158. package/dist-es/toggle-button-group/ToggleButtonGroupContext.js.map +1 -0
  159. package/dist-es/tokenized-input/TokenizedInput.css.js +1 -1
  160. package/dist-es/toolbar/ToolbarButton.css.js +1 -1
  161. package/dist-types/accordion/Accordion.d.ts +22 -6
  162. package/dist-types/accordion/AccordionContext.d.ts +8 -8
  163. package/dist-types/accordion/AccordionGroup.d.ts +4 -0
  164. package/dist-types/accordion/AccordionHeader.d.ts +4 -0
  165. package/dist-types/accordion/AccordionPanel.d.ts +4 -0
  166. package/dist-types/accordion/index.d.ts +3 -4
  167. package/dist-types/dropdown/useDropdown.d.ts +1 -1
  168. package/dist-types/index.d.ts +4 -0
  169. package/dist-types/list-next/ListItemNext.d.ts +11 -0
  170. package/dist-types/list-next/ListNext.d.ts +10 -0
  171. package/dist-types/list-next/ListNextContext.d.ts +10 -0
  172. package/dist-types/list-next/index.d.ts +2 -0
  173. package/dist-types/list-next/useList.d.ts +29 -0
  174. package/dist-types/logo/Logo.d.ts +3 -31
  175. package/dist-types/logo/LogoImage.d.ts +5 -0
  176. package/dist-types/logo/LogoSeparator.d.ts +3 -0
  177. package/dist-types/logo/index.d.ts +2 -0
  178. package/dist-types/multiline-input/MultilineInput.d.ts +40 -0
  179. package/dist-types/multiline-input/index.d.ts +1 -0
  180. package/dist-types/tabs-next/OverflowMenu.d.ts +11 -0
  181. package/dist-types/tabs-next/TabNext.d.ts +5 -0
  182. package/dist-types/tabs-next/TabNextContext.d.ts +17 -0
  183. package/dist-types/tabs-next/TabstripNext.d.ts +10 -0
  184. package/dist-types/tabs-next/index.d.ts +2 -0
  185. package/dist-types/toggle-button/ToggleButton.d.ts +6 -11
  186. package/dist-types/toggle-button/index.d.ts +0 -1
  187. package/dist-types/toggle-button-group/ToggleButtonGroup.d.ts +25 -0
  188. package/dist-types/toggle-button-group/ToggleButtonGroupContext.d.ts +10 -0
  189. package/dist-types/toggle-button-group/index.d.ts +2 -0
  190. package/package.json +7 -5
  191. package/dist-cjs/accordion/AccordionDetails.js +0 -87
  192. package/dist-cjs/accordion/AccordionDetails.js.map +0 -1
  193. package/dist-cjs/accordion/AccordionSection.js +0 -111
  194. package/dist-cjs/accordion/AccordionSection.js.map +0 -1
  195. package/dist-cjs/accordion/AccordionSectionContext.js +0 -24
  196. package/dist-cjs/accordion/AccordionSectionContext.js.map +0 -1
  197. package/dist-cjs/accordion/AccordionSummary.js +0 -64
  198. package/dist-cjs/accordion/AccordionSummary.js.map +0 -1
  199. package/dist-cjs/accordion/utils.js +0 -8
  200. package/dist-cjs/accordion/utils.js.map +0 -1
  201. package/dist-cjs/logo/internal/LogoTitle.js +0 -28
  202. package/dist-cjs/logo/internal/LogoTitle.js.map +0 -1
  203. package/dist-cjs/toggle-button/ToggleButtonGroup.css.js +0 -6
  204. package/dist-cjs/toggle-button/ToggleButtonGroup.js +0 -147
  205. package/dist-cjs/toggle-button/ToggleButtonGroup.js.map +0 -1
  206. package/dist-cjs/toggle-button/internal/ToggleButtonGroupContext.js +0 -10
  207. package/dist-cjs/toggle-button/internal/ToggleButtonGroupContext.js.map +0 -1
  208. package/dist-es/accordion/AccordionDetails.js +0 -83
  209. package/dist-es/accordion/AccordionDetails.js.map +0 -1
  210. package/dist-es/accordion/AccordionSection.js +0 -107
  211. package/dist-es/accordion/AccordionSection.js.map +0 -1
  212. package/dist-es/accordion/AccordionSectionContext.js +0 -19
  213. package/dist-es/accordion/AccordionSectionContext.js.map +0 -1
  214. package/dist-es/accordion/AccordionSummary.js +0 -60
  215. package/dist-es/accordion/AccordionSummary.js.map +0 -1
  216. package/dist-es/accordion/utils.js +0 -4
  217. package/dist-es/accordion/utils.js.map +0 -1
  218. package/dist-es/logo/internal/LogoTitle.js +0 -24
  219. package/dist-es/logo/internal/LogoTitle.js.map +0 -1
  220. package/dist-es/toggle-button/ToggleButtonGroup.css.js +0 -4
  221. package/dist-es/toggle-button/ToggleButtonGroup.js +0 -143
  222. package/dist-es/toggle-button/ToggleButtonGroup.js.map +0 -1
  223. package/dist-es/toggle-button/internal/ToggleButtonGroupContext.js +0 -6
  224. package/dist-es/toggle-button/internal/ToggleButtonGroupContext.js.map +0 -1
  225. package/dist-types/accordion/AccordionDetails.d.ts +0 -8
  226. package/dist-types/accordion/AccordionSection.d.ts +0 -9
  227. package/dist-types/accordion/AccordionSectionContext.d.ts +0 -8
  228. package/dist-types/accordion/AccordionSummary.d.ts +0 -5
  229. package/dist-types/accordion/utils.d.ts +0 -1
  230. package/dist-types/logo/internal/LogoTitle.d.ts +0 -6
  231. package/dist-types/toggle-button/ToggleButtonGroup.d.ts +0 -18
  232. package/dist-types/toggle-button/internal/ToggleButtonGroupContext.d.ts +0 -14
  233. /package/dist-cjs/{toggle-button → toggle-button-group}/ToggleButtonGroup.css.js.map +0 -0
  234. /package/dist-es/{toggle-button → toggle-button-group}/ToggleButtonGroup.css.js.map +0 -0
@@ -0,0 +1,18 @@
1
+ import { createContext } from '@salt-ds/core';
2
+ import { useContext } from 'react';
3
+
4
+ const TabsContext = createContext("TabsContext", {
5
+ disabled: false,
6
+ select: () => void 0,
7
+ isSelected: () => false,
8
+ setFocusable: () => void 0,
9
+ isFocusable: () => false,
10
+ registerTab: () => void 0,
11
+ unregisterTab: () => void 0
12
+ });
13
+ function useTabs() {
14
+ return useContext(TabsContext);
15
+ }
16
+
17
+ export { TabsContext, useTabs };
18
+ //# sourceMappingURL=TabNextContext.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TabNextContext.js","sources":["../src/tabs-next/TabNextContext.tsx"],"sourcesContent":["import { createContext } from \"@salt-ds/core\";\nimport { ReactNode, SyntheticEvent, useContext } from \"react\";\n\ntype TabValue = {\n value: string;\n label: ReactNode;\n};\n\nexport interface TabsContextValue {\n disabled?: boolean;\n select: (event: SyntheticEvent<HTMLButtonElement>) => void;\n isSelected: (id: string) => boolean;\n setFocusable: (id: string) => void;\n isFocusable: (id: string) => boolean;\n registerTab: (tab: TabValue) => void;\n unregisterTab: (id: string) => void;\n}\n\nexport const TabsContext = createContext<TabsContextValue>(\"TabsContext\", {\n disabled: false,\n select: () => undefined,\n isSelected: () => false,\n setFocusable: () => undefined,\n isFocusable: () => false,\n registerTab: () => undefined,\n unregisterTab: () => undefined,\n});\n\nexport function useTabs() {\n return useContext(TabsContext);\n}\n"],"names":[],"mappings":";;;AAkBa,MAAA,WAAA,GAAc,cAAgC,aAAe,EAAA;AAAA,EACxE,QAAU,EAAA,KAAA;AAAA,EACV,QAAQ,MAAM,KAAA,CAAA;AAAA,EACd,YAAY,MAAM,KAAA;AAAA,EAClB,cAAc,MAAM,KAAA,CAAA;AAAA,EACpB,aAAa,MAAM,KAAA;AAAA,EACnB,aAAa,MAAM,KAAA,CAAA;AAAA,EACnB,eAAe,MAAM,KAAA,CAAA;AACvB,CAAC,EAAA;AAEM,SAAS,OAAU,GAAA;AACxB,EAAA,OAAO,WAAW,WAAW,CAAA,CAAA;AAC/B;;;;"}
@@ -0,0 +1,4 @@
1
+ var css_248z = "/* Component class applied to the root element */\n.saltTabstripNext {\n display: flex;\n flex-wrap: nowrap;\n align-items: center;\n position: relative;\n background: transparent;\n width: 100%;\n}\n\n.saltTabstripNext::before {\n content: \"\";\n position: absolute;\n inset: auto 0 0 0;\n height: var(--salt-size-border);\n background: var(--salt-container-primary-borderColor);\n}\n\n.saltTabstripNext .saltTabNext-wrapper {\n padding-right: var(--salt-spacing-50);\n}\n";
2
+
3
+ export { css_248z as default };
4
+ //# sourceMappingURL=TabstripNext.css.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TabstripNext.css.js","sources":[],"sourcesContent":[],"names":[],"mappings":";;;;"}
@@ -0,0 +1,172 @@
1
+ import { jsx, jsxs } from 'react/jsx-runtime';
2
+ import { makePrefixer, useForkRef, useControlled } from '@salt-ds/core';
3
+ import clsx from 'clsx';
4
+ import { forwardRef, useRef, useState, useCallback, useMemo } from 'react';
5
+ import { useComponentCssInjection } from '@salt-ds/styles';
6
+ import { useWindow } from '@salt-ds/window';
7
+ import { Overflow } from '@fluentui/react-overflow';
8
+ import { OverflowMenu } from './OverflowMenu.js';
9
+ import css_248z from './TabstripNext.css.js';
10
+ import { TabsContext } from './TabNextContext.js';
11
+
12
+ const withBaseName = makePrefixer("saltTabstripNext");
13
+ const TabstripNext = forwardRef(
14
+ function TabstripNext2(props, ref) {
15
+ const {
16
+ children,
17
+ className,
18
+ disabled,
19
+ selected: selectedProp,
20
+ defaultSelected,
21
+ onChange,
22
+ onKeyDown,
23
+ ...rest
24
+ } = props;
25
+ const targetWindow = useWindow();
26
+ useComponentCssInjection({
27
+ testId: "salt-tabstrip-next",
28
+ css: css_248z,
29
+ window: targetWindow
30
+ });
31
+ const tabstripRef = useRef(null);
32
+ const handleRef = useForkRef(tabstripRef, ref);
33
+ const [selected, setSelected] = useControlled({
34
+ controlled: selectedProp,
35
+ default: defaultSelected,
36
+ name: "TabstripNext",
37
+ state: "selected"
38
+ });
39
+ const [focusable, setFocusableState] = useState(
40
+ selected
41
+ );
42
+ const [overflowOpen, setOverflowOpen] = useState(false);
43
+ const select = useCallback(
44
+ (event) => {
45
+ const newValue = event.currentTarget.value;
46
+ setSelected(newValue);
47
+ if (selected !== newValue) {
48
+ onChange == null ? void 0 : onChange(event, { value: newValue });
49
+ }
50
+ },
51
+ [onChange, selected, setSelected]
52
+ );
53
+ const isSelected = useCallback(
54
+ (id) => {
55
+ return selected === id;
56
+ },
57
+ [selected]
58
+ );
59
+ const setFocusable = useCallback((id) => {
60
+ setFocusableState(id);
61
+ }, []);
62
+ const isFocusable = useCallback(
63
+ (id) => {
64
+ return focusable === id || !focusable;
65
+ },
66
+ [focusable]
67
+ );
68
+ const [tabList, setTabList] = useState([]);
69
+ const registerTab = useCallback((tab) => {
70
+ setTabList((list) => list.concat([tab]));
71
+ }, []);
72
+ const unregisterTab = useCallback((id) => {
73
+ setTabList((list) => list.filter((item) => item.value !== id));
74
+ }, []);
75
+ const handleKeyDown = (event) => {
76
+ var _a, _b, _c, _d, _e, _f;
77
+ if (overflowOpen)
78
+ return;
79
+ const elements = Array.from(
80
+ (_b = (_a = tabstripRef.current) == null ? void 0 : _a.querySelectorAll(
81
+ `div:not([data-overflowing]) > [role="tab"]:not([disabled])`
82
+ )) != null ? _b : []
83
+ );
84
+ const currentIndex = elements.findIndex(
85
+ (element) => element === (targetWindow == null ? void 0 : targetWindow.document.activeElement)
86
+ );
87
+ if (currentIndex < 0)
88
+ return;
89
+ switch (event.key) {
90
+ case "ArrowDown":
91
+ case "ArrowRight":
92
+ (_c = elements[Math.min(currentIndex + 1, elements.length)]) == null ? void 0 : _c.focus();
93
+ break;
94
+ case "ArrowUp":
95
+ case "ArrowLeft":
96
+ (_d = elements[Math.max(0, currentIndex - 1)]) == null ? void 0 : _d.focus();
97
+ break;
98
+ case "Home":
99
+ (_e = elements[0]) == null ? void 0 : _e.focus();
100
+ break;
101
+ case "End":
102
+ (_f = elements[elements.length - 1]) == null ? void 0 : _f.focus();
103
+ }
104
+ onKeyDown == null ? void 0 : onKeyDown(event);
105
+ };
106
+ const handleOverflowItemClick = (event, item) => {
107
+ if (item) {
108
+ setSelected(item.value);
109
+ requestAnimationFrame(() => {
110
+ var _a;
111
+ const element = (_a = tabstripRef.current) == null ? void 0 : _a.querySelector(
112
+ `[value="${item.value}"]`
113
+ );
114
+ if (element instanceof HTMLElement) {
115
+ element == null ? void 0 : element.focus();
116
+ }
117
+ });
118
+ if (selected !== item.value) {
119
+ onChange == null ? void 0 : onChange(event, { value: item.value });
120
+ }
121
+ }
122
+ };
123
+ const handleOverflowOpenChange = (isOpen) => {
124
+ setOverflowOpen(isOpen);
125
+ };
126
+ const value = useMemo(
127
+ () => ({
128
+ select,
129
+ isSelected,
130
+ setFocusable,
131
+ isFocusable,
132
+ registerTab,
133
+ unregisterTab
134
+ }),
135
+ [
136
+ select,
137
+ isSelected,
138
+ setFocusable,
139
+ isFocusable,
140
+ registerTab,
141
+ unregisterTab
142
+ ]
143
+ );
144
+ return /* @__PURE__ */ jsx(TabsContext.Provider, {
145
+ value,
146
+ children: /* @__PURE__ */ jsx(Overflow, {
147
+ ref: handleRef,
148
+ children: /* @__PURE__ */ jsxs("div", {
149
+ role: "tablist",
150
+ className: clsx(
151
+ withBaseName(),
152
+ withBaseName("horizontal"),
153
+ className
154
+ ),
155
+ onKeyDown: handleKeyDown,
156
+ ...rest,
157
+ children: [
158
+ children,
159
+ /* @__PURE__ */ jsx(OverflowMenu, {
160
+ tabs: tabList,
161
+ onOpenChange: handleOverflowOpenChange,
162
+ onSelectionChange: handleOverflowItemClick
163
+ })
164
+ ]
165
+ })
166
+ })
167
+ });
168
+ }
169
+ );
170
+
171
+ export { TabstripNext };
172
+ //# sourceMappingURL=TabstripNext.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"TabstripNext.js","sources":["../src/tabs-next/TabstripNext.tsx"],"sourcesContent":["import { makePrefixer, useControlled, useForkRef } from \"@salt-ds/core\";\nimport clsx from \"clsx\";\nimport {\n ComponentPropsWithoutRef,\n forwardRef,\n ReactNode,\n SyntheticEvent,\n KeyboardEvent,\n useCallback,\n useMemo,\n useRef,\n useState,\n} from \"react\";\nimport { useComponentCssInjection } from \"@salt-ds/styles\";\nimport { useWindow } from \"@salt-ds/window\";\nimport { Overflow } from \"@fluentui/react-overflow\";\nimport { OverflowMenu } from \"./OverflowMenu\";\nimport tabstripCss from \"./TabstripNext.css\";\nimport { TabsContext } from \"./TabNextContext\";\nimport { SelectionChangeHandler } from \"../common-hooks\";\n\nconst withBaseName = makePrefixer(\"saltTabstripNext\");\n\nexport interface TabstripNextProps\n extends Omit<ComponentPropsWithoutRef<\"div\">, \"onChange\"> {\n disabled?: boolean;\n /* Value for the uncontrolled version. */\n selected?: string;\n /* Callback for the controlled version. */\n onChange?: (e: SyntheticEvent, data: { value: string }) => void;\n /* Initial value for the uncontrolled version. */\n defaultSelected?: string;\n}\n\ntype TabValue = {\n value: string;\n label: ReactNode;\n};\n\nexport const TabstripNext = forwardRef<HTMLDivElement, TabstripNextProps>(\n function TabstripNext(props, ref) {\n const {\n children,\n className,\n disabled,\n selected: selectedProp,\n defaultSelected,\n onChange,\n onKeyDown,\n ...rest\n } = props;\n const targetWindow = useWindow();\n useComponentCssInjection({\n testId: \"salt-tabstrip-next\",\n css: tabstripCss,\n window: targetWindow,\n });\n\n const tabstripRef = useRef<HTMLDivElement>(null);\n const handleRef = useForkRef(tabstripRef, ref);\n\n const [selected, setSelected] = useControlled({\n controlled: selectedProp,\n default: defaultSelected,\n name: \"TabstripNext\",\n state: \"selected\",\n });\n const [focusable, setFocusableState] = useState<string | undefined>(\n selected\n );\n const [overflowOpen, setOverflowOpen] = useState(false);\n\n const select = useCallback(\n (event: SyntheticEvent<HTMLButtonElement>) => {\n const newValue = event.currentTarget.value;\n setSelected(newValue);\n if (selected !== newValue) {\n onChange?.(event, { value: newValue });\n }\n },\n [onChange, selected, setSelected]\n );\n\n const isSelected = useCallback(\n (id: string | undefined) => {\n return selected === id;\n },\n [selected]\n );\n\n const setFocusable = useCallback((id: string | undefined) => {\n setFocusableState(id);\n }, []);\n\n const isFocusable = useCallback(\n (id: string | undefined) => {\n return focusable === id || !focusable;\n },\n [focusable]\n );\n\n const [tabList, setTabList] = useState<TabValue[]>([]);\n const registerTab = useCallback((tab: TabValue) => {\n setTabList((list) => list.concat([tab]));\n }, []);\n\n const unregisterTab = useCallback((id: string) => {\n setTabList((list) => list.filter((item) => item.value !== id));\n }, []);\n\n const handleKeyDown = (event: KeyboardEvent<HTMLDivElement>) => {\n if (overflowOpen) return;\n\n const elements: HTMLElement[] = Array.from(\n tabstripRef.current?.querySelectorAll(\n `div:not([data-overflowing]) > [role=\"tab\"]:not([disabled])`\n ) ?? []\n );\n\n const currentIndex = elements.findIndex(\n (element) => element === targetWindow?.document.activeElement\n );\n\n if (currentIndex < 0) return;\n\n switch (event.key) {\n case \"ArrowDown\":\n case \"ArrowRight\":\n elements[Math.min(currentIndex + 1, elements.length)]?.focus();\n break;\n case \"ArrowUp\":\n case \"ArrowLeft\":\n elements[Math.max(0, currentIndex - 1)]?.focus();\n break;\n case \"Home\":\n elements[0]?.focus();\n break;\n case \"End\":\n elements[elements.length - 1]?.focus();\n }\n\n onKeyDown?.(event);\n };\n\n const handleOverflowItemClick: SelectionChangeHandler<TabValue> = (\n event,\n item\n ) => {\n if (item) {\n setSelected(item.value);\n requestAnimationFrame(() => {\n const element = tabstripRef.current?.querySelector(\n `[value=\"${item.value}\"]`\n );\n if (element instanceof HTMLElement) {\n element?.focus();\n }\n });\n if (selected !== item.value) {\n onChange?.(event, { value: item.value });\n }\n }\n };\n\n const handleOverflowOpenChange = (isOpen: boolean) => {\n setOverflowOpen(isOpen);\n };\n\n const value = useMemo(\n () => ({\n select,\n isSelected,\n setFocusable,\n isFocusable,\n registerTab,\n unregisterTab,\n }),\n [\n select,\n isSelected,\n setFocusable,\n isFocusable,\n registerTab,\n unregisterTab,\n ]\n );\n\n return (\n <TabsContext.Provider value={value}>\n <Overflow ref={handleRef}>\n <div\n role=\"tablist\"\n className={clsx(\n withBaseName(),\n withBaseName(\"horizontal\"),\n className\n )}\n onKeyDown={handleKeyDown}\n {...rest}\n >\n {children}\n <OverflowMenu\n tabs={tabList}\n onOpenChange={handleOverflowOpenChange}\n onSelectionChange={handleOverflowItemClick}\n />\n </div>\n </Overflow>\n </TabsContext.Provider>\n );\n }\n);\n"],"names":["TabstripNext","tabstripCss"],"mappings":"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"}
@@ -1,4 +1,4 @@
1
- var css_248z = ".salt-density-touch {\n --toggleButton-icon-padding: 11px;\n}\n\n.salt-density-low {\n --toggleButton-icon-padding: 7px;\n}\n\n.salt-density-medium {\n --toggleButton-icon-padding: 5px;\n}\n\n.salt-density-high {\n --toggleButton-icon-padding: 1px;\n}\n\n.saltToggleButton-horizontal.saltToggleButton-iconOnly {\n padding: var(--saltToggleButton-icon-padding, var(--toggleButton-icon-padding));\n}\n\n.saltToggleButton {\n --saltButton-text-color: var(--salt-text-primary-foreground);\n --saltButton-text-color-disabled: var(--salt-text-primary-foreground-disabled);\n\n --saltIcon-color: var(--salt-text-primary-foreground);\n}\n\n.saltToggleButton-disabled.saltButton-disabled {\n --saltButton-text-color-active: var(--salt-text-primary-foreground-disabled);\n --saltButton-text-color-hover: var(--salt-text-primary-foreground-disabled);\n\n --saltIcon-color: var(--salt-text-primary-foreground-disabled);\n}\n\n/* CTA */\n.saltToggleButton-cta {\n --saltButton-background: var(--salt-selectable-background);\n --saltButton-background-hover: var(--salt-selectable-cta-background-hover);\n --saltButton-background-active: var(--salt-selectable-cta-background-selected);\n --saltButton-background-disabled: var(--salt-selectable-background-disabled);\n --saltButton-text-color-hover: var(--salt-selectable-cta-foreground-hover);\n --saltButton-text-color-active: var(--salt-selectable-cta-foreground-selected);\n}\n.saltToggleButton-cta.saltToggleButton-disabled.saltButton-disabled {\n --saltButton-background-hover: var(--salt-selectable-background-disabled);\n --saltButton-background-active: var(--salt-selectable-cta-background-disabled);\n}\n.saltToggleButton-cta.saltToggleButton-disabled {\n --saltButton-background-active: var(--salt-selectable-cta-background-selectedDisabled);\n --saltButton-text-color-active: var(--salt-selectable-cta-foreground-selectedDisabled);\n\n --saltIcon-color-active: var(--salt-selectable-cta-foreground-selectedDisabled);\n}\n.saltToggleButton-cta.saltToggleButton-toggled {\n --saltButton-background: var(--salt-selectable-cta-background-selected);\n --saltButton-background-hover: var(--salt-selectable-cta-background-selected);\n --saltButton-text-color: var(--salt-selectable-cta-foreground-selected);\n --saltButton-text-color-hover: var(--salt-selectable-cta-foreground-selected);\n\n --saltIcon-color: var(--salt-selectable-cta-foreground-selected);\n}\n.saltToggleButton-cta.saltToggleButton-toggled.saltToggleButton-disabled {\n --saltButton-background-disabled: var(--salt-selectable-cta-background-selectedDisabled);\n --saltButton-background-hover: var(--salt-selectable-cta-background-selectedDisabled);\n --saltButton-text-color-disabled: var(--salt-selectable-cta-foreground-selectedDisabled);\n --saltButton-text-color-hover: var(--salt-selectable-cta-foreground-selectedDisabled);\n\n --saltIcon-color: var(--salt-selectable-cta-foreground-selectedDisabled);\n}\n\n/* Primary */\n.saltToggleButton-primary {\n --saltButton-background: var(--salt-selectable-background);\n --saltButton-background-hover: var(--salt-selectable-primary-background-hover);\n --saltButton-background-active: var(--salt-selectable-primary-background-selected);\n --saltButton-background-disabled: var(--salt-selectable-primary-background-disabled);\n --saltButton-text-color-hover: var(--salt-selectable-primary-foreground-hover);\n --saltButton-text-color-active: var(--salt-selectable-primary-foreground-selected);\n}\n.saltToggleButton-primary.saltToggleButton-disabled.saltButton-disabled {\n --saltButton-background-active: var(--salt-selectable-primary-background-disabled);\n --saltButton-background-hover: var(--salt-selectable-background-disabled);\n}\n.saltToggleButton-primary.saltToggleButton-disabled.saltButton-active {\n --saltButton-background-active: var(--salt-selectable-primary-background-selectedDisabled);\n --saltButton-text-color-active: var(--salt-selectable-primary-foreground-selectedDisabled);\n}\n.saltToggleButton-primary.saltToggleButton-toggled {\n --saltButton-background: var(--salt-selectable-primary-background-selected);\n --saltButton-background-hover: var(--salt-selectable-primary-background-selected);\n --saltButton-text-color: var(--salt-selectable-primary-foreground-selected);\n --saltButton-text-color-hover: var(--salt-selectable-primary-foreground-selected);\n\n --saltIcon-color: var(--salt-selectable-primary-foreground-selected);\n}\n.saltToggleButton-primary.saltToggleButton-toggled.saltToggleButton-disabled {\n --saltButton-background-disabled: var(--salt-selectable-primary-background-selectedDisabled);\n --saltButton-background-hover: var(--salt-selectable-primary-background-selectedDisabled);\n --saltButton-text-color-disabled: var(--salt-selectable-primary-foreground-selectedDisabled);\n --saltButton-text-color-hover: var(--salt-selectable-primary-foreground-selectedDisabled);\n\n --saltIcon-color: var(--salt-selectable-primary-foreground-selectedDisabled);\n}\n\n/* Secondary */\n.saltToggleButton-secondary {\n --saltButton-background: var(--salt-selectable-background);\n --saltButton-background-hover: var(--salt-selectable-secondary-background-hover);\n --saltButton-background-active: var(--salt-selectable-secondary-background-selected);\n --saltButton-background-disabled: var(--salt-selectable-background-disabled);\n --saltButton-text-color-hover: var(--salt-selectable-secondary-foreground-hover);\n --saltButton-text-color-active: var(--salt-selectable-secondary-foreground-selected);\n}\n.saltToggleButton-secondary.saltToggleButton-disabled.saltButton-disabled {\n --saltButton-background-active: var(--salt-selectable-secondary-background-disabled);\n --saltButton-background-hover: var(--salt-selectable-background-disabled);\n}\n.saltToggleButton-secondary.saltToggleButton-disabled.saltButton-active {\n --saltButton-background-active: var(--salt-selectable-secondary-background-selectedDisabled);\n --saltButton-text-color-active: var(--salt-selectable-secondary-foreground-selectedDisabled);\n\n --saltIcon-color-active: var(--salt-selectable-secondary-foreground-selectedDisabled);\n}\n.saltToggleButton-secondary.saltToggleButton-toggled {\n --saltButton-background: var(--salt-selectable-secondary-background-selected);\n --saltButton-background-hover: var(--salt-selectable-secondary-background-selected);\n --saltButton-text-color: var(--salt-selectable-secondary-foreground-selected);\n --saltButton-text-color-hover: var(--salt-selectable-secondary-foreground-selected);\n\n --saltIcon-color: var(--salt-selectable-secondary-foreground-selected);\n}\n.saltToggleButton-secondary.saltToggleButton-toggled.saltToggleButton-disabled {\n --saltButton-background-disabled: var(--salt-selectable-secondary-background-selectedDisabled);\n --saltButton-background-hover: var(--salt-selectable-secondary-background-selectedDisabled);\n --saltButton-text-color-disabled: var(--salt-selectable-secondary-foreground-selectedDisabled);\n --saltButton-text-color-hover: var(--salt-selectable-secondary-foreground-selectedDisabled);\n\n --saltIcon-color: var(--salt-selectable-secondary-foreground-selectedDisabled);\n}\n";
1
+ var css_248z = ".saltToggleButton {\n align-items: center;\n justify-content: center;\n appearance: none;\n display: inline-flex;\n background: var(--salt-actionable-secondary-background);\n border: 0 solid transparent;\n border-radius: 0;\n height: var(--salt-size-base);\n color: var(--salt-actionable-secondary-foreground);\n text-transform: var(--salt-actionable-textTransform);\n font-weight: var(--salt-actionable-secondary-fontWeight);\n font-family: var(--salt-text-fontFamily);\n text-align: var(--salt-actionable-textAlign);\n letter-spacing: var(--salt-actionable-letterSpacing);\n line-height: var(--salt-text-lineHeight);\n font-size: var(--salt-text-fontSize);\n padding-inline: var(--salt-spacing-100);\n gap: var(--salt-spacing-50);\n --saltIcon-color: var(--salt-actionable-secondary-foreground);\n}\n\n.saltToggleButton:hover {\n background: var(--salt-actionable-secondary-background-hover);\n color: var(--salt-actionable-secondary-foreground-hover);\n cursor: var(--salt-actionable-cursor-hover);\n --saltIcon-color: var(--salt-actionable-secondary-foreground-hover);\n}\n\n.saltToggleButton:focus-visible {\n outline: var(--salt-focused-outline);\n background: var(--salt-actionable-secondary-background-hover);\n color: var(--salt-actionable-secondary-foreground-hover);\n cursor: var(--salt-actionable-cursor-hover);\n --saltIcon-color: var(--salt-actionable-secondary-foreground-hover);\n}\n\n.saltToggleButton[aria-checked=\"true\"]:focus-visible {\n background: var(--salt-actionable-secondary-background-active);\n color: var(--salt-actionable-secondary-foreground-active);\n cursor: var(--salt-actionable-cursor-active);\n --saltIcon-color: var(--salt-actionable-secondary-foreground-active);\n}\n\n.saltToggleButton[aria-checked=\"true\"] {\n background: var(--salt-actionable-secondary-background-active);\n color: var(--salt-actionable-secondary-foreground-active);\n cursor: var(--salt-actionable-cursor-active);\n --saltIcon-color: var(--salt-actionable-secondary-foreground-active);\n}\n\n.saltToggleButton:disabled {\n background: var(--salt-actionable-secondary-background-disabled);\n cursor: var(--salt-actionable-cursor-disabled);\n color: var(--salt-actionable-secondary-foreground-disabled);\n --saltIcon-color: var(--salt-actionable-secondary-foreground-disabled);\n}\n";
2
2
 
3
3
  export { css_248z as default };
4
4
  //# sourceMappingURL=ToggleButton.css.js.map
@@ -1,27 +1,26 @@
1
1
  import { jsx } from 'react/jsx-runtime';
2
- import { makePrefixer, Tooltip, Button } from '@salt-ds/core';
2
+ import { forwardRef, useRef } from 'react';
3
+ import { makePrefixer, useForkRef, useControlled } from '@salt-ds/core';
3
4
  import { clsx } from 'clsx';
4
- import { forwardRef, useState, useRef, useContext, useCallback, useEffect } from 'react';
5
- import { ToggleButtonGroupContext } from './internal/ToggleButtonGroupContext.js';
5
+ import '../toggle-button-group/ToggleButtonGroup.js';
6
+ import { useToggleButtonGroup } from '../toggle-button-group/ToggleButtonGroupContext.js';
6
7
  import { useWindow } from '@salt-ds/window';
7
8
  import { useComponentCssInjection } from '@salt-ds/styles';
8
9
  import css_248z from './ToggleButton.css.js';
9
10
 
10
11
  const withBaseName = makePrefixer("saltToggleButton");
11
12
  const ToggleButton = forwardRef(
12
- (props, ref) => {
13
+ function ToggleButton2(props, ref) {
13
14
  const {
14
- "aria-label": ariaLabel,
15
+ children,
15
16
  className,
16
- onToggle,
17
- toggled = false,
18
- tooltipText = ariaLabel,
19
- variant: variantProp = "primary",
20
17
  disabled: disabledProp,
21
- disableTooltip: disableTooltipProp,
22
- focusableWhenDisabled: focusableWhenDisabledProp,
23
- "data-button-index": index,
24
- ...restProps
18
+ value,
19
+ onClick,
20
+ onFocus,
21
+ onChange,
22
+ selected: selectedProp,
23
+ ...rest
25
24
  } = props;
26
25
  const targetWindow = useWindow();
27
26
  useComponentCssInjection({
@@ -29,70 +28,41 @@ const ToggleButton = forwardRef(
29
28
  css: css_248z,
30
29
  window: targetWindow
31
30
  });
32
- const [iconOnly, setIconOnly] = useState(false);
33
31
  const buttonRef = useRef(null);
34
- const groupContext = useContext(ToggleButtonGroupContext);
35
- const handleIconOnlyButton = useCallback(
36
- (button) => {
37
- setIconOnly(
38
- (button == null ? void 0 : button.querySelector(".saltIcon")) != null && (button == null ? void 0 : button.childElementCount) === 1
39
- );
40
- },
41
- [setIconOnly]
42
- );
43
- const {
44
- register,
45
- unregister,
46
- disabled = disabledProp,
47
- disableTooltip = disableTooltipProp,
48
- focusableWhenDisabled = focusableWhenDisabledProp,
49
- orientation = "horizontal",
50
- variant = variantProp
51
- } = groupContext || {};
52
- useEffect(() => {
53
- if (index !== void 0 && register && unregister && (!disabled || focusableWhenDisabled)) {
54
- register(buttonRef.current, index);
55
- return function cleanup() {
56
- unregister(index);
57
- };
58
- }
59
- }, [index, disabled, focusableWhenDisabled, register, unregister]);
60
- const handleToggle = (event) => {
61
- if (!disabled) {
62
- onToggle == null ? void 0 : onToggle(event, !toggled);
63
- }
32
+ const handleRef = useForkRef(ref, buttonRef);
33
+ const toggleButtonGroup = useToggleButtonGroup();
34
+ const toggleButtonGroupSelected = toggleButtonGroup ? toggleButtonGroup.isSelected(value) : selectedProp;
35
+ const focusable = toggleButtonGroup ? toggleButtonGroup == null ? void 0 : toggleButtonGroup.isFocused(value) : true;
36
+ const disabled = (toggleButtonGroup == null ? void 0 : toggleButtonGroup.disabled) || disabledProp;
37
+ const [selected, setSelected] = useControlled({
38
+ controlled: toggleButtonGroupSelected,
39
+ default: Boolean(selectedProp),
40
+ name: "ToggleButton",
41
+ state: "selected"
42
+ });
43
+ const handleClick = (event) => {
44
+ toggleButtonGroup == null ? void 0 : toggleButtonGroup.select(event);
45
+ setSelected(!selected);
46
+ onChange == null ? void 0 : onChange(event);
47
+ onClick == null ? void 0 : onClick(event);
48
+ };
49
+ const handleFocus = (event) => {
50
+ toggleButtonGroup == null ? void 0 : toggleButtonGroup.focus(value);
51
+ onFocus == null ? void 0 : onFocus(event);
64
52
  };
65
- const tabIndex = toggled && !disabled ? 0 : -1;
66
- return /* @__PURE__ */ jsx(Tooltip, {
67
- content: tooltipText,
68
- disabled: disableTooltip,
69
- placement: orientation === "horizontal" ? "bottom" : "right",
70
- children: /* @__PURE__ */ jsx(Button, {
71
- ref: handleIconOnlyButton,
72
- "aria-checked": toggled,
73
- "aria-label": ariaLabel,
74
- "aria-posinset": index !== void 0 ? index + 1 : void 0,
75
- className: clsx(
76
- withBaseName(),
77
- withBaseName(orientation),
78
- {
79
- [withBaseName("primary")]: variant === "primary",
80
- [withBaseName("cta")]: variant === "cta",
81
- [withBaseName("secondary")]: variant === "secondary",
82
- [withBaseName("toggled")]: toggled,
83
- [withBaseName("disabled")]: disabled,
84
- [withBaseName("iconOnly")]: iconOnly
85
- },
86
- className
87
- ),
88
- onClick: handleToggle,
89
- disabled,
90
- focusableWhenDisabled,
91
- role: groupContext ? "radio" : "checkbox",
92
- tabIndex: groupContext ? tabIndex : void 0,
93
- variant,
94
- ...restProps
95
- })
53
+ const ariaChecked = selected && !disabled;
54
+ return /* @__PURE__ */ jsx("button", {
55
+ "aria-checked": ariaChecked,
56
+ className: clsx(withBaseName(), className),
57
+ disabled,
58
+ role: toggleButtonGroup ? "radio" : "checkbox",
59
+ ref: handleRef,
60
+ onClick: handleClick,
61
+ onFocus: handleFocus,
62
+ tabIndex: focusable && !disabled ? 0 : -1,
63
+ value,
64
+ ...rest,
65
+ children
96
66
  });
97
67
  }
98
68
  );
@@ -1 +1 @@
1
- {"version":3,"file":"ToggleButton.js","sources":["../src/toggle-button/ToggleButton.tsx"],"sourcesContent":["import { Button, ButtonProps, makePrefixer, Tooltip } from \"@salt-ds/core\";\nimport { clsx } from \"clsx\";\nimport {\n forwardRef,\n SyntheticEvent,\n useCallback,\n useContext,\n useEffect,\n useRef,\n useState,\n} from \"react\";\nimport { ToggleButtonGroupContext } from \"./internal/ToggleButtonGroupContext\";\n\nimport { useWindow } from \"@salt-ds/window\";\nimport { useComponentCssInjection } from \"@salt-ds/styles\";\n\nimport toggleButtonCss from \"./ToggleButton.css\";\n\nconst withBaseName = makePrefixer(\"saltToggleButton\");\n\nexport type ToggleButtonToggleEventHandler = (\n event: SyntheticEvent<HTMLButtonElement>,\n toggled: boolean\n) => void;\n\nexport interface ToggleButtonProps extends ButtonProps {\n \"aria-label\"?: string;\n \"data-button-index\"?: number;\n toggled?: boolean;\n tooltipText?: string;\n disableTooltip?: boolean;\n onToggle?: ToggleButtonToggleEventHandler;\n}\n\nexport const ToggleButton = forwardRef<HTMLButtonElement, ToggleButtonProps>(\n (props, ref) => {\n const {\n \"aria-label\": ariaLabel,\n className,\n onToggle,\n toggled = false,\n tooltipText = ariaLabel,\n variant: variantProp = \"primary\",\n disabled: disabledProp,\n disableTooltip: disableTooltipProp,\n focusableWhenDisabled: focusableWhenDisabledProp,\n \"data-button-index\": index,\n ...restProps\n } = props;\n\n const targetWindow = useWindow();\n useComponentCssInjection({\n testId: \"salt-toggle-button\",\n css: toggleButtonCss,\n window: targetWindow,\n });\n\n const [iconOnly, setIconOnly] = useState(false);\n const buttonRef = useRef<HTMLButtonElement>(null);\n const groupContext = useContext(ToggleButtonGroupContext);\n\n const handleIconOnlyButton = useCallback(\n (button: HTMLButtonElement | null) => {\n setIconOnly(\n button?.querySelector(\".saltIcon\") != null &&\n button?.childElementCount === 1\n );\n },\n [setIconOnly]\n );\n\n const {\n register,\n unregister,\n disabled = disabledProp,\n disableTooltip = disableTooltipProp,\n focusableWhenDisabled = focusableWhenDisabledProp,\n orientation = \"horizontal\",\n variant = variantProp,\n } = groupContext || {};\n\n useEffect(() => {\n if (\n index !== undefined &&\n register &&\n unregister &&\n (!disabled || focusableWhenDisabled)\n ) {\n register(buttonRef.current, index);\n\n return function cleanup() {\n unregister(index);\n };\n }\n }, [index, disabled, focusableWhenDisabled, register, unregister]);\n\n const handleToggle = (event: SyntheticEvent<HTMLButtonElement>) => {\n if (!disabled) {\n onToggle?.(event, !toggled);\n }\n };\n\n const tabIndex = toggled && !disabled ? 0 : -1;\n\n return (\n <Tooltip\n content={tooltipText}\n disabled={disableTooltip}\n placement={orientation === \"horizontal\" ? \"bottom\" : \"right\"}\n >\n <Button\n ref={handleIconOnlyButton}\n aria-checked={toggled}\n aria-label={ariaLabel}\n aria-posinset={index !== undefined ? index + 1 : undefined}\n className={clsx(\n withBaseName(),\n withBaseName(orientation),\n {\n [withBaseName(\"primary\")]: variant === \"primary\",\n [withBaseName(\"cta\")]: variant === \"cta\",\n [withBaseName(\"secondary\")]: variant === \"secondary\",\n [withBaseName(\"toggled\")]: toggled,\n [withBaseName(\"disabled\")]: disabled,\n [withBaseName(\"iconOnly\")]: iconOnly,\n },\n className\n )}\n onClick={handleToggle}\n disabled={disabled}\n focusableWhenDisabled={focusableWhenDisabled}\n role={groupContext ? \"radio\" : \"checkbox\"}\n tabIndex={groupContext ? tabIndex : undefined}\n variant={variant}\n {...restProps}\n />\n </Tooltip>\n );\n }\n);\n"],"names":["toggleButtonCss"],"mappings":"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"}
1
+ {"version":3,"file":"ToggleButton.js","sources":["../src/toggle-button/ToggleButton.tsx"],"sourcesContent":["import {\n ComponentProps,\n forwardRef,\n useRef,\n MouseEvent,\n FocusEvent,\n} from \"react\";\nimport { makePrefixer, useControlled, useForkRef } from \"@salt-ds/core\";\nimport { clsx } from \"clsx\";\nimport { useToggleButtonGroup } from \"../toggle-button-group\";\nimport { useWindow } from \"@salt-ds/window\";\nimport { useComponentCssInjection } from \"@salt-ds/styles\";\n\nimport toggleButtonCss from \"./ToggleButton.css\";\n\nexport interface ToggleButtonProps extends ComponentProps<\"button\"> {\n selected?: boolean;\n onChange?: (event: MouseEvent<HTMLButtonElement>) => void;\n value: string | ReadonlyArray<string> | number | undefined;\n}\n\nconst withBaseName = makePrefixer(\"saltToggleButton\");\n\nexport const ToggleButton = forwardRef<HTMLButtonElement, ToggleButtonProps>(\n function ToggleButton(props, ref) {\n const {\n children,\n className,\n disabled: disabledProp,\n value,\n onClick,\n onFocus,\n onChange,\n selected: selectedProp,\n ...rest\n } = props;\n\n const targetWindow = useWindow();\n useComponentCssInjection({\n testId: \"salt-toggle-button\",\n css: toggleButtonCss,\n window: targetWindow,\n });\n\n const buttonRef = useRef<HTMLButtonElement>(null);\n const handleRef = useForkRef(ref, buttonRef);\n\n const toggleButtonGroup = useToggleButtonGroup();\n\n const toggleButtonGroupSelected = toggleButtonGroup\n ? toggleButtonGroup.isSelected(value)\n : selectedProp;\n const focusable = toggleButtonGroup\n ? toggleButtonGroup?.isFocused(value)\n : true;\n const disabled = toggleButtonGroup?.disabled || disabledProp;\n\n const [selected, setSelected] = useControlled({\n controlled: toggleButtonGroupSelected,\n default: Boolean(selectedProp),\n name: \"ToggleButton\",\n state: \"selected\",\n });\n\n const handleClick = (event: MouseEvent<HTMLButtonElement>) => {\n toggleButtonGroup?.select(event);\n setSelected(!selected);\n onChange?.(event);\n onClick?.(event);\n };\n\n const handleFocus = (event: FocusEvent<HTMLButtonElement>) => {\n toggleButtonGroup?.focus(value);\n onFocus?.(event);\n };\n\n const ariaChecked = selected && !disabled;\n\n return (\n <button\n aria-checked={ariaChecked}\n className={clsx(withBaseName(), className)}\n disabled={disabled}\n role={toggleButtonGroup ? \"radio\" : \"checkbox\"}\n ref={handleRef}\n onClick={handleClick}\n onFocus={handleFocus}\n tabIndex={focusable && !disabled ? 0 : -1}\n value={value}\n {...rest}\n >\n {children}\n </button>\n );\n }\n);\n"],"names":["ToggleButton","toggleButtonCss"],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ var css_248z = ".saltToggleButtonGroup {\n display: flex;\n background: var(--salt-container-primary-background);\n border: var(--salt-size-border) var(--salt-container-borderStyle) var(--salt-container-primary-borderColor);\n width: fit-content;\n gap: var(--salt-spacing-50);\n padding: calc(var(--salt-spacing-50) - var(--salt-size-border));\n flex-direction: row;\n}\n\n.saltToggleButtonGroup-horizontal .saltToggleButton {\n height: calc(var(--salt-size-base) - var(--salt-spacing-100));\n}\n\n.saltToggleButtonGroup-vertical {\n flex-direction: column;\n}\n\n.saltToggleButtonGroup-vertical .saltToggleButton {\n justify-content: start;\n}\n";
2
+
3
+ export { css_248z as default };
4
+ //# sourceMappingURL=ToggleButtonGroup.css.js.map
@@ -0,0 +1,107 @@
1
+ import { jsx } from 'react/jsx-runtime';
2
+ import { forwardRef, useRef, useState, useCallback, useMemo } from 'react';
3
+ import { makePrefixer, useForkRef, useControlled } from '@salt-ds/core';
4
+ import { clsx } from 'clsx';
5
+ import { useWindow } from '@salt-ds/window';
6
+ import { useComponentCssInjection } from '@salt-ds/styles';
7
+ import { ToggleButtonGroupContext } from './ToggleButtonGroupContext.js';
8
+ import css_248z from './ToggleButtonGroup.css.js';
9
+
10
+ const withBaseName = makePrefixer("saltToggleButtonGroup");
11
+ const ToggleButtonGroup = forwardRef(function ToggleButtonGroup2(props, ref) {
12
+ const {
13
+ children,
14
+ className,
15
+ value: valueProp,
16
+ defaultValue,
17
+ disabled,
18
+ onChange,
19
+ onKeyDown,
20
+ orientation = "horizontal",
21
+ ...rest
22
+ } = props;
23
+ const targetWindow = useWindow();
24
+ useComponentCssInjection({
25
+ testId: "salt-toggle-button-group",
26
+ css: css_248z,
27
+ window: targetWindow
28
+ });
29
+ const groupRef = useRef(null);
30
+ const handleRef = useForkRef(ref, groupRef);
31
+ const [value, setValue] = useControlled({
32
+ default: defaultValue,
33
+ controlled: valueProp,
34
+ name: "ToggleButtonGroup",
35
+ state: "value"
36
+ });
37
+ const [focused, setFocused] = useState(value);
38
+ const select = useCallback(
39
+ (event) => {
40
+ const newValue = event.currentTarget.value;
41
+ setValue(newValue);
42
+ if (value !== newValue) {
43
+ onChange == null ? void 0 : onChange(event);
44
+ }
45
+ },
46
+ [onChange, value, setValue]
47
+ );
48
+ const isSelected = useCallback(
49
+ (id) => {
50
+ return value === id;
51
+ },
52
+ [value]
53
+ );
54
+ const focus = (id) => {
55
+ setFocused(id);
56
+ };
57
+ const isFocused = useCallback(
58
+ (id) => {
59
+ return focused === id || !focused;
60
+ },
61
+ [focused]
62
+ );
63
+ const contextValue = useMemo(
64
+ () => ({
65
+ select,
66
+ isSelected,
67
+ focus,
68
+ isFocused,
69
+ disabled
70
+ }),
71
+ [select, isSelected, isFocused, disabled]
72
+ );
73
+ const handleKeyDown = (event) => {
74
+ var _a, _b, _c, _d;
75
+ const elements = Array.from(
76
+ (_b = (_a = groupRef.current) == null ? void 0 : _a.querySelectorAll("button:not([disabled])")) != null ? _b : []
77
+ );
78
+ const currentIndex = elements.findIndex(
79
+ (element) => element === document.activeElement
80
+ );
81
+ switch (event.key) {
82
+ case "ArrowDown":
83
+ case "ArrowRight":
84
+ (_c = elements[(currentIndex + 1) % elements.length]) == null ? void 0 : _c.focus();
85
+ break;
86
+ case "ArrowUp":
87
+ case "ArrowLeft":
88
+ (_d = elements[(currentIndex - 1 + elements.length) % elements.length]) == null ? void 0 : _d.focus();
89
+ break;
90
+ }
91
+ onKeyDown == null ? void 0 : onKeyDown(event);
92
+ };
93
+ return /* @__PURE__ */ jsx(ToggleButtonGroupContext.Provider, {
94
+ value: contextValue,
95
+ children: /* @__PURE__ */ jsx("div", {
96
+ className: clsx(withBaseName(), withBaseName(orientation), className),
97
+ role: "radiogroup",
98
+ ref: handleRef,
99
+ onKeyDown: handleKeyDown,
100
+ ...rest,
101
+ children
102
+ })
103
+ });
104
+ });
105
+
106
+ export { ToggleButtonGroup };
107
+ //# sourceMappingURL=ToggleButtonGroup.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ToggleButtonGroup.js","sources":["../src/toggle-button-group/ToggleButtonGroup.tsx"],"sourcesContent":["import {\n ComponentPropsWithoutRef,\n forwardRef,\n KeyboardEvent,\n SyntheticEvent,\n useCallback,\n useMemo,\n useRef,\n useState,\n} from \"react\";\nimport { makePrefixer, useControlled, useForkRef } from \"@salt-ds/core\";\nimport { clsx } from \"clsx\";\nimport { useWindow } from \"@salt-ds/window\";\nimport { useComponentCssInjection } from \"@salt-ds/styles\";\n\nimport { ToggleButtonGroupContext } from \"./ToggleButtonGroupContext\";\nimport toggleButtonGroupCss from \"./ToggleButtonGroup.css\";\n\nexport interface ToggleButtonGroupProps\n extends Omit<ComponentPropsWithoutRef<\"div\">, \"onChange\"> {\n /**\n * The default value. Use when the component is not controlled.\n */\n defaultValue?: string | ReadonlyArray<string> | number | undefined;\n /**\n * If `true`, the Toggle Button Group will be disabled.\n */\n disabled?: boolean;\n /**\n * The value. Use when the component is controlled.\n */\n value?: string | ReadonlyArray<string> | number | undefined;\n /**\n * Callback fired when the selection changes.\n * @param event\n */\n onChange?: (event: SyntheticEvent<HTMLButtonElement>) => void;\n /**\n * The orientation of the toggle buttons.\n */\n orientation?: \"horizontal\" | \"vertical\";\n}\n\nconst withBaseName = makePrefixer(\"saltToggleButtonGroup\");\n\nexport const ToggleButtonGroup = forwardRef<\n HTMLDivElement,\n ToggleButtonGroupProps\n>(function ToggleButtonGroup(props, ref) {\n const {\n children,\n className,\n value: valueProp,\n defaultValue,\n disabled,\n onChange,\n onKeyDown,\n orientation = \"horizontal\",\n ...rest\n } = props;\n\n const targetWindow = useWindow();\n useComponentCssInjection({\n testId: \"salt-toggle-button-group\",\n css: toggleButtonGroupCss,\n window: targetWindow,\n });\n\n const groupRef = useRef<HTMLDivElement>(null);\n const handleRef = useForkRef(ref, groupRef);\n\n const [value, setValue] = useControlled({\n default: defaultValue,\n controlled: valueProp,\n name: \"ToggleButtonGroup\",\n state: \"value\",\n });\n const [focused, setFocused] = useState<\n string | ReadonlyArray<string> | number | undefined\n >(value);\n\n const select = useCallback(\n (event: SyntheticEvent<HTMLButtonElement>) => {\n const newValue = event.currentTarget.value;\n setValue(newValue);\n if (value !== newValue) {\n onChange?.(event);\n }\n },\n [onChange, value, setValue]\n );\n\n const isSelected = useCallback(\n (id: string | ReadonlyArray<string> | number | undefined) => {\n return value === id;\n },\n [value]\n );\n\n const focus = (id: string | ReadonlyArray<string> | number | undefined) => {\n setFocused(id);\n };\n\n const isFocused = useCallback(\n (id: string | ReadonlyArray<string> | number | undefined) => {\n return focused === id || !focused;\n },\n [focused]\n );\n\n const contextValue = useMemo(\n () => ({\n select,\n isSelected,\n focus,\n isFocused,\n disabled,\n }),\n [select, isSelected, isFocused, disabled]\n );\n\n const handleKeyDown = (event: KeyboardEvent<HTMLDivElement>) => {\n const elements: HTMLElement[] = Array.from(\n groupRef.current?.querySelectorAll(\"button:not([disabled])\") ?? []\n );\n const currentIndex = elements.findIndex(\n (element) => element === document.activeElement\n );\n switch (event.key) {\n case \"ArrowDown\":\n case \"ArrowRight\":\n elements[(currentIndex + 1) % elements.length]?.focus();\n break;\n case \"ArrowUp\":\n case \"ArrowLeft\":\n elements[\n (currentIndex - 1 + elements.length) % elements.length\n ]?.focus();\n break;\n }\n\n onKeyDown?.(event);\n };\n\n return (\n <ToggleButtonGroupContext.Provider value={contextValue}>\n <div\n className={clsx(withBaseName(), withBaseName(orientation), className)}\n role=\"radiogroup\"\n ref={handleRef}\n onKeyDown={handleKeyDown}\n {...rest}\n >\n {children}\n </div>\n </ToggleButtonGroupContext.Provider>\n );\n});\n"],"names":["ToggleButtonGroup","toggleButtonGroupCss"],"mappings":";;;;;;;;;AA2CA,MAAM,YAAA,GAAe,aAAa,uBAAuB,CAAA,CAAA;AAElD,MAAM,iBAAoB,GAAA,UAAA,CAG/B,SAASA,kBAAAA,CAAkB,OAAO,GAAK,EAAA;AACvC,EAAM,MAAA;AAAA,IACJ,QAAA;AAAA,IACA,SAAA;AAAA,IACA,KAAO,EAAA,SAAA;AAAA,IACP,YAAA;AAAA,IACA,QAAA;AAAA,IACA,QAAA;AAAA,IACA,SAAA;AAAA,IACA,WAAc,GAAA,YAAA;AAAA,IACX,GAAA,IAAA;AAAA,GACD,GAAA,KAAA,CAAA;AAEJ,EAAA,MAAM,eAAe,SAAU,EAAA,CAAA;AAC/B,EAAyB,wBAAA,CAAA;AAAA,IACvB,MAAQ,EAAA,0BAAA;AAAA,IACR,GAAK,EAAAC,QAAA;AAAA,IACL,MAAQ,EAAA,YAAA;AAAA,GACT,CAAA,CAAA;AAED,EAAM,MAAA,QAAA,GAAW,OAAuB,IAAI,CAAA,CAAA;AAC5C,EAAM,MAAA,SAAA,GAAY,UAAW,CAAA,GAAA,EAAK,QAAQ,CAAA,CAAA;AAE1C,EAAA,MAAM,CAAC,KAAA,EAAO,QAAQ,CAAA,GAAI,aAAc,CAAA;AAAA,IACtC,OAAS,EAAA,YAAA;AAAA,IACT,UAAY,EAAA,SAAA;AAAA,IACZ,IAAM,EAAA,mBAAA;AAAA,IACN,KAAO,EAAA,OAAA;AAAA,GACR,CAAA,CAAA;AACD,EAAA,MAAM,CAAC,OAAA,EAAS,UAAU,CAAA,GAAI,SAE5B,KAAK,CAAA,CAAA;AAEP,EAAA,MAAM,MAAS,GAAA,WAAA;AAAA,IACb,CAAC,KAA6C,KAAA;AAC5C,MAAM,MAAA,QAAA,GAAW,MAAM,aAAc,CAAA,KAAA,CAAA;AACrC,MAAA,QAAA,CAAS,QAAQ,CAAA,CAAA;AACjB,MAAA,IAAI,UAAU,QAAU,EAAA;AACtB,QAAW,QAAA,IAAA,IAAA,GAAA,KAAA,CAAA,GAAA,QAAA,CAAA,KAAA,CAAA,CAAA;AAAA,OACb;AAAA,KACF;AAAA,IACA,CAAC,QAAU,EAAA,KAAA,EAAO,QAAQ,CAAA;AAAA,GAC5B,CAAA;AAEA,EAAA,MAAM,UAAa,GAAA,WAAA;AAAA,IACjB,CAAC,EAA4D,KAAA;AAC3D,MAAA,OAAO,KAAU,KAAA,EAAA,CAAA;AAAA,KACnB;AAAA,IACA,CAAC,KAAK,CAAA;AAAA,GACR,CAAA;AAEA,EAAM,MAAA,KAAA,GAAQ,CAAC,EAA4D,KAAA;AACzE,IAAA,UAAA,CAAW,EAAE,CAAA,CAAA;AAAA,GACf,CAAA;AAEA,EAAA,MAAM,SAAY,GAAA,WAAA;AAAA,IAChB,CAAC,EAA4D,KAAA;AAC3D,MAAO,OAAA,OAAA,KAAY,MAAM,CAAC,OAAA,CAAA;AAAA,KAC5B;AAAA,IACA,CAAC,OAAO,CAAA;AAAA,GACV,CAAA;AAEA,EAAA,MAAM,YAAe,GAAA,OAAA;AAAA,IACnB,OAAO;AAAA,MACL,MAAA;AAAA,MACA,UAAA;AAAA,MACA,KAAA;AAAA,MACA,SAAA;AAAA,MACA,QAAA;AAAA,KACF,CAAA;AAAA,IACA,CAAC,MAAA,EAAQ,UAAY,EAAA,SAAA,EAAW,QAAQ,CAAA;AAAA,GAC1C,CAAA;AAEA,EAAM,MAAA,aAAA,GAAgB,CAAC,KAAyC,KAAA;AAzHlE,IAAA,IAAA,EAAA,EAAA,EAAA,EAAA,EAAA,EAAA,EAAA,CAAA;AA0HI,IAAA,MAAM,WAA0B,KAAM,CAAA,IAAA;AAAA,MAAA,CACpC,oBAAS,OAAT,KAAA,IAAA,GAAA,KAAA,CAAA,GAAA,EAAA,CAAkB,gBAAiB,CAAA,wBAAA,CAAA,KAAnC,YAAgE,EAAC;AAAA,KACnE,CAAA;AACA,IAAA,MAAM,eAAe,QAAS,CAAA,SAAA;AAAA,MAC5B,CAAC,OAAY,KAAA,OAAA,KAAY,QAAS,CAAA,aAAA;AAAA,KACpC,CAAA;AACA,IAAA,QAAQ,KAAM,CAAA,GAAA;AAAA,MACP,KAAA,WAAA,CAAA;AAAA,MACA,KAAA,YAAA;AACH,QAAA,CAAA,EAAA,GAAA,QAAA,CAAA,CAAU,YAAe,GAAA,CAAA,IAAK,QAAS,CAAA,MAAA,CAAA,KAAvC,IAAgD,GAAA,KAAA,CAAA,GAAA,EAAA,CAAA,KAAA,EAAA,CAAA;AAChD,QAAA,MAAA;AAAA,MACG,KAAA,SAAA,CAAA;AAAA,MACA,KAAA,WAAA;AACH,QAAA,CAAA,EAAA,GAAA,QAAA,CAAA,CACG,eAAe,CAAI,GAAA,QAAA,CAAS,MAAU,IAAA,QAAA,CAAS,YADlD,IAEG,GAAA,KAAA,CAAA,GAAA,EAAA,CAAA,KAAA,EAAA,CAAA;AACH,QAAA,MAAA;AAAA,KAAA;AAGJ,IAAY,SAAA,IAAA,IAAA,GAAA,KAAA,CAAA,GAAA,SAAA,CAAA,KAAA,CAAA,CAAA;AAAA,GACd,CAAA;AAEA,EACE,uBAAA,GAAA,CAAC,yBAAyB,QAAzB,EAAA;AAAA,IAAkC,KAAO,EAAA,YAAA;AAAA,IACxC,QAAC,kBAAA,GAAA,CAAA,KAAA,EAAA;AAAA,MACC,WAAW,IAAK,CAAA,YAAA,IAAgB,YAAa,CAAA,WAAW,GAAG,SAAS,CAAA;AAAA,MACpE,IAAK,EAAA,YAAA;AAAA,MACL,GAAK,EAAA,SAAA;AAAA,MACL,SAAW,EAAA,aAAA;AAAA,MACV,GAAG,IAAA;AAAA,MAEH,QAAA;AAAA,KACH,CAAA;AAAA,GACF,CAAA,CAAA;AAEJ,CAAC;;;;"}
@@ -0,0 +1,10 @@
1
+ import { createContext } from '@salt-ds/core';
2
+ import { useContext } from 'react';
3
+
4
+ const ToggleButtonGroupContext = createContext("ToggleButtonGroupContext", void 0);
5
+ function useToggleButtonGroup() {
6
+ return useContext(ToggleButtonGroupContext);
7
+ }
8
+
9
+ export { ToggleButtonGroupContext, useToggleButtonGroup };
10
+ //# sourceMappingURL=ToggleButtonGroupContext.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ToggleButtonGroupContext.js","sources":["../src/toggle-button-group/ToggleButtonGroupContext.ts"],"sourcesContent":["import { createContext } from \"@salt-ds/core\";\nimport { SyntheticEvent, useContext } from \"react\";\n\nexport interface ToggleButtonGroupContextValue {\n disabled?: boolean;\n select: (event: SyntheticEvent<HTMLButtonElement>) => void;\n isSelected: (\n id: string | ReadonlyArray<string> | number | undefined\n ) => boolean;\n focus: (id: string | ReadonlyArray<string> | number | undefined) => void;\n isFocused: (\n id: string | ReadonlyArray<string> | number | undefined\n ) => boolean;\n}\n\nexport const ToggleButtonGroupContext = createContext<\n ToggleButtonGroupContextValue | undefined\n>(\"ToggleButtonGroupContext\", undefined);\n\nexport function useToggleButtonGroup() {\n return useContext(ToggleButtonGroupContext);\n}\n"],"names":[],"mappings":";;;AAea,MAAA,wBAAA,GAA2B,aAEtC,CAAA,0BAAA,EAA4B,KAAS,CAAA,EAAA;AAEhC,SAAS,oBAAuB,GAAA;AACrC,EAAA,OAAO,WAAW,wBAAwB,CAAA,CAAA;AAC5C;;;;"}