@salt-ds/core 1.60.0 → 1.61.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (195) hide show
  1. package/CHANGELOG.md +38 -0
  2. package/css/salt-core.css +366 -1
  3. package/dist-cjs/index.js +16 -0
  4. package/dist-cjs/index.js.map +1 -1
  5. package/dist-cjs/navigation-item/NavigationItem.js +2 -0
  6. package/dist-cjs/navigation-item/NavigationItem.js.map +1 -1
  7. package/dist-cjs/rating/Rating.css.js +6 -0
  8. package/dist-cjs/rating/Rating.css.js.map +1 -0
  9. package/dist-cjs/rating/Rating.js +140 -0
  10. package/dist-cjs/rating/Rating.js.map +1 -0
  11. package/dist-cjs/rating/RatingItem.css.js +6 -0
  12. package/dist-cjs/rating/RatingItem.css.js.map +1 -0
  13. package/dist-cjs/rating/RatingItem.js +75 -0
  14. package/dist-cjs/rating/RatingItem.js.map +1 -0
  15. package/dist-cjs/semantic-icon-provider/SemanticIconProvider.js +22 -20
  16. package/dist-cjs/semantic-icon-provider/SemanticIconProvider.js.map +1 -1
  17. package/dist-cjs/tabs/Tab.css.js +6 -0
  18. package/dist-cjs/tabs/Tab.css.js.map +1 -0
  19. package/dist-cjs/tabs/Tab.js +211 -0
  20. package/dist-cjs/tabs/Tab.js.map +1 -0
  21. package/dist-cjs/tabs/TabAction.js +63 -0
  22. package/dist-cjs/tabs/TabAction.js.map +1 -0
  23. package/dist-cjs/tabs/TabBar.css.js +6 -0
  24. package/dist-cjs/tabs/TabBar.css.js.map +1 -0
  25. package/dist-cjs/tabs/TabBar.js +45 -0
  26. package/dist-cjs/tabs/TabBar.js.map +1 -0
  27. package/dist-cjs/tabs/TabList.css.js +6 -0
  28. package/dist-cjs/tabs/TabList.css.js.map +1 -0
  29. package/dist-cjs/tabs/TabList.js +281 -0
  30. package/dist-cjs/tabs/TabList.js.map +1 -0
  31. package/dist-cjs/tabs/TabPanel.css.js +6 -0
  32. package/dist-cjs/tabs/TabPanel.css.js.map +1 -0
  33. package/dist-cjs/tabs/TabPanel.js +98 -0
  34. package/dist-cjs/tabs/TabPanel.js.map +1 -0
  35. package/dist-cjs/tabs/TabTrigger.css.js +6 -0
  36. package/dist-cjs/tabs/TabTrigger.css.js.map +1 -0
  37. package/dist-cjs/tabs/TabTrigger.js +188 -0
  38. package/dist-cjs/tabs/TabTrigger.js.map +1 -0
  39. package/dist-cjs/tabs/Tabs.css.js +6 -0
  40. package/dist-cjs/tabs/Tabs.css.js.map +1 -0
  41. package/dist-cjs/tabs/Tabs.js +200 -0
  42. package/dist-cjs/tabs/Tabs.js.map +1 -0
  43. package/dist-cjs/tabs/internal/contexts/TabContext.js +26 -0
  44. package/dist-cjs/tabs/internal/contexts/TabContext.js.map +1 -0
  45. package/dist-cjs/tabs/internal/contexts/TabListLayoutContext.js +19 -0
  46. package/dist-cjs/tabs/internal/contexts/TabListLayoutContext.js.map +1 -0
  47. package/dist-cjs/tabs/internal/contexts/TabSlotRegistryContext.js +22 -0
  48. package/dist-cjs/tabs/internal/contexts/TabSlotRegistryContext.js.map +1 -0
  49. package/dist-cjs/tabs/internal/contexts/TabsContext.js +50 -0
  50. package/dist-cjs/tabs/internal/contexts/TabsContext.js.map +1 -0
  51. package/dist-cjs/tabs/internal/hooks/useFocusWithRetry.js +64 -0
  52. package/dist-cjs/tabs/internal/hooks/useFocusWithRetry.js.map +1 -0
  53. package/dist-cjs/tabs/internal/hooks/useTabListRecovery.js +76 -0
  54. package/dist-cjs/tabs/internal/hooks/useTabListRecovery.js.map +1 -0
  55. package/dist-cjs/tabs/internal/hooks/useTabRemovalHandler.js +165 -0
  56. package/dist-cjs/tabs/internal/hooks/useTabRemovalHandler.js.map +1 -0
  57. package/dist-cjs/tabs/internal/hooks/useTabSelectionFocus.js +87 -0
  58. package/dist-cjs/tabs/internal/hooks/useTabSelectionFocus.js.map +1 -0
  59. package/dist-cjs/tabs/internal/overflow/TabOverflowList.css.js +6 -0
  60. package/dist-cjs/tabs/internal/overflow/TabOverflowList.css.js.map +1 -0
  61. package/dist-cjs/tabs/internal/overflow/TabOverflowList.js +245 -0
  62. package/dist-cjs/tabs/internal/overflow/TabOverflowList.js.map +1 -0
  63. package/dist-cjs/tabs/internal/overflow/TabSlot.js +30 -0
  64. package/dist-cjs/tabs/internal/overflow/TabSlot.js.map +1 -0
  65. package/dist-cjs/tabs/internal/overflow/overflowMath.js +86 -0
  66. package/dist-cjs/tabs/internal/overflow/overflowMath.js.map +1 -0
  67. package/dist-cjs/tabs/internal/overflow/useOverflow.js +273 -0
  68. package/dist-cjs/tabs/internal/overflow/useOverflow.js.map +1 -0
  69. package/dist-cjs/tabs/internal/overflow/useOverflowLayoutState.js +99 -0
  70. package/dist-cjs/tabs/internal/overflow/useOverflowLayoutState.js.map +1 -0
  71. package/dist-cjs/tabs/internal/overflow/useOverflowSelectionState.js +68 -0
  72. package/dist-cjs/tabs/internal/overflow/useOverflowSelectionState.js.map +1 -0
  73. package/dist-cjs/tabs/internal/overflow/useRenderedTabWidth.js +92 -0
  74. package/dist-cjs/tabs/internal/overflow/useRenderedTabWidth.js.map +1 -0
  75. package/dist-cjs/tabs/internal/overflow/widthMeasurement.js +42 -0
  76. package/dist-cjs/tabs/internal/overflow/widthMeasurement.js.map +1 -0
  77. package/dist-cjs/tabs/internal/registry/useCollection.js +197 -0
  78. package/dist-cjs/tabs/internal/registry/useCollection.js.map +1 -0
  79. package/dist-cjs/tabs/internal/registry/useRenderedTabsRegistry.js +206 -0
  80. package/dist-cjs/tabs/internal/registry/useRenderedTabsRegistry.js.map +1 -0
  81. package/dist-cjs/tabs/internal/utils/domUtils.js +13 -0
  82. package/dist-cjs/tabs/internal/utils/domUtils.js.map +1 -0
  83. package/dist-es/index.js +8 -0
  84. package/dist-es/index.js.map +1 -1
  85. package/dist-es/navigation-item/NavigationItem.js +2 -0
  86. package/dist-es/navigation-item/NavigationItem.js.map +1 -1
  87. package/dist-es/rating/Rating.css.js +4 -0
  88. package/dist-es/rating/Rating.css.js.map +1 -0
  89. package/dist-es/rating/Rating.js +138 -0
  90. package/dist-es/rating/Rating.js.map +1 -0
  91. package/dist-es/rating/RatingItem.css.js +4 -0
  92. package/dist-es/rating/RatingItem.css.js.map +1 -0
  93. package/dist-es/rating/RatingItem.js +73 -0
  94. package/dist-es/rating/RatingItem.js.map +1 -0
  95. package/dist-es/semantic-icon-provider/SemanticIconProvider.js +23 -21
  96. package/dist-es/semantic-icon-provider/SemanticIconProvider.js.map +1 -1
  97. package/dist-es/tabs/Tab.css.js +4 -0
  98. package/dist-es/tabs/Tab.css.js.map +1 -0
  99. package/dist-es/tabs/Tab.js +209 -0
  100. package/dist-es/tabs/Tab.js.map +1 -0
  101. package/dist-es/tabs/TabAction.js +61 -0
  102. package/dist-es/tabs/TabAction.js.map +1 -0
  103. package/dist-es/tabs/TabBar.css.js +4 -0
  104. package/dist-es/tabs/TabBar.css.js.map +1 -0
  105. package/dist-es/tabs/TabBar.js +43 -0
  106. package/dist-es/tabs/TabBar.js.map +1 -0
  107. package/dist-es/tabs/TabList.css.js +4 -0
  108. package/dist-es/tabs/TabList.css.js.map +1 -0
  109. package/dist-es/tabs/TabList.js +279 -0
  110. package/dist-es/tabs/TabList.js.map +1 -0
  111. package/dist-es/tabs/TabPanel.css.js +4 -0
  112. package/dist-es/tabs/TabPanel.css.js.map +1 -0
  113. package/dist-es/tabs/TabPanel.js +96 -0
  114. package/dist-es/tabs/TabPanel.js.map +1 -0
  115. package/dist-es/tabs/TabTrigger.css.js +4 -0
  116. package/dist-es/tabs/TabTrigger.css.js.map +1 -0
  117. package/dist-es/tabs/TabTrigger.js +186 -0
  118. package/dist-es/tabs/TabTrigger.js.map +1 -0
  119. package/dist-es/tabs/Tabs.css.js +4 -0
  120. package/dist-es/tabs/Tabs.css.js.map +1 -0
  121. package/dist-es/tabs/Tabs.js +198 -0
  122. package/dist-es/tabs/Tabs.js.map +1 -0
  123. package/dist-es/tabs/internal/contexts/TabContext.js +23 -0
  124. package/dist-es/tabs/internal/contexts/TabContext.js.map +1 -0
  125. package/dist-es/tabs/internal/contexts/TabListLayoutContext.js +16 -0
  126. package/dist-es/tabs/internal/contexts/TabListLayoutContext.js.map +1 -0
  127. package/dist-es/tabs/internal/contexts/TabSlotRegistryContext.js +19 -0
  128. package/dist-es/tabs/internal/contexts/TabSlotRegistryContext.js.map +1 -0
  129. package/dist-es/tabs/internal/contexts/TabsContext.js +47 -0
  130. package/dist-es/tabs/internal/contexts/TabsContext.js.map +1 -0
  131. package/dist-es/tabs/internal/hooks/useFocusWithRetry.js +62 -0
  132. package/dist-es/tabs/internal/hooks/useFocusWithRetry.js.map +1 -0
  133. package/dist-es/tabs/internal/hooks/useTabListRecovery.js +74 -0
  134. package/dist-es/tabs/internal/hooks/useTabListRecovery.js.map +1 -0
  135. package/dist-es/tabs/internal/hooks/useTabRemovalHandler.js +163 -0
  136. package/dist-es/tabs/internal/hooks/useTabRemovalHandler.js.map +1 -0
  137. package/dist-es/tabs/internal/hooks/useTabSelectionFocus.js +85 -0
  138. package/dist-es/tabs/internal/hooks/useTabSelectionFocus.js.map +1 -0
  139. package/dist-es/tabs/internal/overflow/TabOverflowList.css.js +4 -0
  140. package/dist-es/tabs/internal/overflow/TabOverflowList.css.js.map +1 -0
  141. package/dist-es/tabs/internal/overflow/TabOverflowList.js +243 -0
  142. package/dist-es/tabs/internal/overflow/TabOverflowList.js.map +1 -0
  143. package/dist-es/tabs/internal/overflow/TabSlot.js +28 -0
  144. package/dist-es/tabs/internal/overflow/TabSlot.js.map +1 -0
  145. package/dist-es/tabs/internal/overflow/overflowMath.js +82 -0
  146. package/dist-es/tabs/internal/overflow/overflowMath.js.map +1 -0
  147. package/dist-es/tabs/internal/overflow/useOverflow.js +271 -0
  148. package/dist-es/tabs/internal/overflow/useOverflow.js.map +1 -0
  149. package/dist-es/tabs/internal/overflow/useOverflowLayoutState.js +97 -0
  150. package/dist-es/tabs/internal/overflow/useOverflowLayoutState.js.map +1 -0
  151. package/dist-es/tabs/internal/overflow/useOverflowSelectionState.js +66 -0
  152. package/dist-es/tabs/internal/overflow/useOverflowSelectionState.js.map +1 -0
  153. package/dist-es/tabs/internal/overflow/useRenderedTabWidth.js +90 -0
  154. package/dist-es/tabs/internal/overflow/useRenderedTabWidth.js.map +1 -0
  155. package/dist-es/tabs/internal/overflow/widthMeasurement.js +36 -0
  156. package/dist-es/tabs/internal/overflow/widthMeasurement.js.map +1 -0
  157. package/dist-es/tabs/internal/registry/useCollection.js +195 -0
  158. package/dist-es/tabs/internal/registry/useCollection.js.map +1 -0
  159. package/dist-es/tabs/internal/registry/useRenderedTabsRegistry.js +204 -0
  160. package/dist-es/tabs/internal/registry/useRenderedTabsRegistry.js.map +1 -0
  161. package/dist-es/tabs/internal/utils/domUtils.js +11 -0
  162. package/dist-es/tabs/internal/utils/domUtils.js.map +1 -0
  163. package/dist-types/index.d.ts +2 -0
  164. package/dist-types/rating/Rating.d.ts +48 -0
  165. package/dist-types/rating/RatingItem.d.ts +47 -0
  166. package/dist-types/rating/index.d.ts +1 -0
  167. package/dist-types/semantic-icon-provider/SemanticIconProvider.d.ts +21 -19
  168. package/dist-types/tabs/Tab.d.ts +12 -0
  169. package/dist-types/tabs/TabAction.d.ts +4 -0
  170. package/dist-types/tabs/TabBar.d.ts +12 -0
  171. package/dist-types/tabs/TabList.d.ts +12 -0
  172. package/dist-types/tabs/TabPanel.d.ts +9 -0
  173. package/dist-types/tabs/TabTrigger.d.ts +4 -0
  174. package/dist-types/tabs/Tabs.d.ts +20 -0
  175. package/dist-types/tabs/index.d.ts +7 -0
  176. package/dist-types/tabs/internal/contexts/TabContext.d.ts +12 -0
  177. package/dist-types/tabs/internal/contexts/TabListLayoutContext.d.ts +9 -0
  178. package/dist-types/tabs/internal/contexts/TabSlotRegistryContext.d.ts +5 -0
  179. package/dist-types/tabs/internal/contexts/TabsContext.d.ts +43 -0
  180. package/dist-types/tabs/internal/hooks/useFocusWithRetry.d.ts +9 -0
  181. package/dist-types/tabs/internal/hooks/useTabListRecovery.d.ts +12 -0
  182. package/dist-types/tabs/internal/hooks/useTabRemovalHandler.d.ts +32 -0
  183. package/dist-types/tabs/internal/hooks/useTabSelectionFocus.d.ts +15 -0
  184. package/dist-types/tabs/internal/overflow/TabOverflowList.d.ts +10 -0
  185. package/dist-types/tabs/internal/overflow/TabSlot.d.ts +6 -0
  186. package/dist-types/tabs/internal/overflow/overflowMath.d.ts +18 -0
  187. package/dist-types/tabs/internal/overflow/useOverflow.d.ts +11 -0
  188. package/dist-types/tabs/internal/overflow/useOverflowLayoutState.d.ts +13 -0
  189. package/dist-types/tabs/internal/overflow/useOverflowSelectionState.d.ts +13 -0
  190. package/dist-types/tabs/internal/overflow/useRenderedTabWidth.d.ts +12 -0
  191. package/dist-types/tabs/internal/overflow/widthMeasurement.d.ts +5 -0
  192. package/dist-types/tabs/internal/registry/useCollection.d.ts +30 -0
  193. package/dist-types/tabs/internal/registry/useRenderedTabsRegistry.d.ts +12 -0
  194. package/dist-types/tabs/internal/utils/domUtils.d.ts +1 -0
  195. package/package.json +3 -1
@@ -28,6 +28,7 @@ const NavigationItem = forwardRef(
28
28
  level = 0,
29
29
  onExpand,
30
30
  style: styleProp,
31
+ "aria-controls": ariaControls,
31
32
  ...rest
32
33
  } = props;
33
34
  const targetWindow = useWindow();
@@ -65,6 +66,7 @@ const NavigationItem = forwardRef(
65
66
  ),
66
67
  render: render ?? (isLink ? void 0 : /* @__PURE__ */ jsx("button", { type: "button" })),
67
68
  "aria-expanded": isLink ? void 0 : expanded,
69
+ "aria-controls": ariaControls,
68
70
  onClick: handleClick,
69
71
  "aria-current": isLink && active ? "page" : void 0,
70
72
  href,
@@ -1 +1 @@
1
- {"version":3,"file":"NavigationItem.js","sources":["../src/navigation-item/NavigationItem.tsx"],"sourcesContent":["import { useComponentCssInjection } from \"@salt-ds/styles\";\nimport { useWindow } from \"@salt-ds/window\";\nimport { clsx } from \"clsx\";\nimport {\n type ComponentPropsWithoutRef,\n forwardRef,\n type MouseEvent,\n type MouseEventHandler,\n} from \"react\";\nimport type { RenderPropsType } from \"../utils\";\nimport { makePrefixer } from \"../utils\";\nimport { ExpansionIcon } from \"./ExpansionIcon\";\nimport navigationItemCss from \"./NavigationItem.css\";\nimport { NavigationItemAction } from \"./NavigationItemAction\";\n\nexport interface NavigationItemProps extends ComponentPropsWithoutRef<\"div\"> {\n /**\n * Whether the navigation item is active.\n */\n active?: boolean;\n /**\n * Whether the nested group is collapsed and there is an active nested item within it.\n */\n blurActive?: boolean;\n /**\n * Whether the navigation item is expanded.\n */\n expanded?: boolean;\n /**\n * Level of nesting.\n */\n level?: number;\n /**\n * The orientation of the navigation item.\n */\n orientation?: \"horizontal\" | \"vertical\";\n /**\n * Whether the navigation item is a parent with nested items.\n */\n parent?: boolean;\n /**\n * Render prop to enable customisation of navigation item element.\n */\n render?: RenderPropsType[\"render\"];\n /**\n * Action to be triggered when the navigation item is expanded.\n */\n onExpand?: MouseEventHandler<HTMLButtonElement>;\n /**\n * Href to be passed to the Link element.\n */\n href?: string;\n}\n\nconst withBaseName = makePrefixer(\"saltNavigationItem\");\n\nexport const NavigationItem = forwardRef<HTMLDivElement, NavigationItemProps>(\n function NavigationItem(props, ref) {\n const {\n active,\n blurActive,\n render,\n children,\n className,\n expanded = false,\n href,\n orientation = \"horizontal\",\n parent,\n level = 0,\n onExpand,\n style: styleProp,\n ...rest\n } = props;\n\n const targetWindow = useWindow();\n useComponentCssInjection({\n testId: \"salt-navigation-item\",\n css: navigationItemCss,\n window: targetWindow,\n });\n\n const style = {\n ...styleProp,\n \"--saltNavigationItem-level\": `${level}`,\n };\n\n const isLink = href !== undefined;\n\n const handleClick = (event: MouseEvent<HTMLButtonElement>) => {\n onExpand?.(event);\n };\n\n return (\n <div\n ref={ref}\n className={clsx(withBaseName(), className)}\n style={style}\n {...rest}\n >\n <NavigationItemAction\n className={clsx(\n withBaseName(\"wrapper\"),\n {\n [withBaseName(\"active\")]: active || blurActive,\n [withBaseName(\"blurActive\")]: blurActive,\n [withBaseName(\"rootItem\")]: level === 0,\n },\n withBaseName(orientation),\n )}\n render={render ?? (isLink ? undefined : <button type=\"button\" />)}\n aria-expanded={isLink ? undefined : expanded}\n onClick={handleClick}\n aria-current={isLink && active ? \"page\" : undefined}\n href={href}\n >\n <span className={withBaseName(\"label\")}>{children}</span>\n {parent ? (\n <ExpansionIcon expanded={expanded} orientation={orientation} />\n ) : null}\n </NavigationItemAction>\n </div>\n );\n },\n);\n"],"names":["NavigationItem","navigationItemCss"],"mappings":";;;;;;;;;;;;;;AAsDA,MAAM,YAAA,GAAe,aAAa,oBAAoB,CAAA;AAE/C,MAAM,cAAA,GAAiB,UAAA;AAAA,EAC5B,SAASA,eAAAA,CAAe,KAAA,EAAO,GAAA,EAAK;AAClC,IAAA,MAAM;AAAA,MACJ,MAAA;AAAA,MACA,UAAA;AAAA,MACA,MAAA;AAAA,MACA,QAAA;AAAA,MACA,SAAA;AAAA,MACA,QAAA,GAAW,KAAA;AAAA,MACX,IAAA;AAAA,MACA,WAAA,GAAc,YAAA;AAAA,MACd,MAAA;AAAA,MACA,KAAA,GAAQ,CAAA;AAAA,MACR,QAAA;AAAA,MACA,KAAA,EAAO,SAAA;AAAA,MACP,GAAG;AAAA,KACL,GAAI,KAAA;AAEJ,IAAA,MAAM,eAAe,SAAA,EAAU;AAC/B,IAAA,wBAAA,CAAyB;AAAA,MACvB,MAAA,EAAQ,sBAAA;AAAA,MACR,GAAA,EAAKC,QAAA;AAAA,MACL,MAAA,EAAQ;AAAA,KACT,CAAA;AAED,IAAA,MAAM,KAAA,GAAQ;AAAA,MACZ,GAAG,SAAA;AAAA,MACH,4BAAA,EAA8B,GAAG,KAAK,CAAA;AAAA,KACxC;AAEA,IAAA,MAAM,SAAS,IAAA,KAAS,MAAA;AAExB,IAAA,MAAM,WAAA,GAAc,CAAC,KAAA,KAAyC;AAC5D,MAAA,QAAA,IAAA,IAAA,GAAA,MAAA,GAAA,QAAA,CAAW,KAAA,CAAA;AAAA,IACb,CAAA;AAEA,IAAA,uBACE,GAAA;AAAA,MAAC,KAAA;AAAA,MAAA;AAAA,QACC,GAAA;AAAA,QACA,SAAA,EAAW,IAAA,CAAK,YAAA,EAAa,EAAG,SAAS,CAAA;AAAA,QACzC,KAAA;AAAA,QACC,GAAG,IAAA;AAAA,QAEJ,QAAA,kBAAA,IAAA;AAAA,UAAC,oBAAA;AAAA,UAAA;AAAA,YACC,SAAA,EAAW,IAAA;AAAA,cACT,aAAa,SAAS,CAAA;AAAA,cACtB;AAAA,gBACE,CAAC,YAAA,CAAa,QAAQ,CAAC,GAAG,MAAA,IAAU,UAAA;AAAA,gBACpC,CAAC,YAAA,CAAa,YAAY,CAAC,GAAG,UAAA;AAAA,gBAC9B,CAAC,YAAA,CAAa,UAAU,CAAC,GAAG,KAAA,KAAU;AAAA,eACxC;AAAA,cACA,aAAa,WAAW;AAAA,aAC1B;AAAA,YACA,QAAQ,MAAA,KAAW,MAAA,GAAS,yBAAY,GAAA,CAAC,QAAA,EAAA,EAAO,MAAK,QAAA,EAAS,CAAA,CAAA;AAAA,YAC9D,eAAA,EAAe,SAAS,MAAA,GAAY,QAAA;AAAA,YACpC,OAAA,EAAS,WAAA;AAAA,YACT,cAAA,EAAc,MAAA,IAAU,MAAA,GAAS,MAAA,GAAS,MAAA;AAAA,YAC1C,IAAA;AAAA,YAEA,QAAA,EAAA;AAAA,8BAAA,GAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAW,YAAA,CAAa,OAAO,GAAI,QAAA,EAAS,CAAA;AAAA,cACjD,MAAA,mBACC,GAAA,CAAC,aAAA,EAAA,EAAc,QAAA,EAAoB,aAA0B,CAAA,GAC3D;AAAA;AAAA;AAAA;AACN;AAAA,KACF;AAAA,EAEJ;AACF;;;;"}
1
+ {"version":3,"file":"NavigationItem.js","sources":["../src/navigation-item/NavigationItem.tsx"],"sourcesContent":["import { useComponentCssInjection } from \"@salt-ds/styles\";\nimport { useWindow } from \"@salt-ds/window\";\nimport { clsx } from \"clsx\";\nimport {\n type ComponentPropsWithoutRef,\n forwardRef,\n type MouseEvent,\n type MouseEventHandler,\n} from \"react\";\nimport type { RenderPropsType } from \"../utils\";\nimport { makePrefixer } from \"../utils\";\nimport { ExpansionIcon } from \"./ExpansionIcon\";\nimport navigationItemCss from \"./NavigationItem.css\";\nimport { NavigationItemAction } from \"./NavigationItemAction\";\n\nexport interface NavigationItemProps extends ComponentPropsWithoutRef<\"div\"> {\n /**\n * Whether the navigation item is active.\n */\n active?: boolean;\n /**\n * Whether the nested group is collapsed and there is an active nested item within it.\n */\n blurActive?: boolean;\n /**\n * Whether the navigation item is expanded.\n */\n expanded?: boolean;\n /**\n * Level of nesting.\n */\n level?: number;\n /**\n * The orientation of the navigation item.\n */\n orientation?: \"horizontal\" | \"vertical\";\n /**\n * Whether the navigation item is a parent with nested items.\n */\n parent?: boolean;\n /**\n * Render prop to enable customisation of navigation item element.\n */\n render?: RenderPropsType[\"render\"];\n /**\n * Action to be triggered when the navigation item is expanded.\n */\n onExpand?: MouseEventHandler<HTMLButtonElement>;\n /**\n * Href to be passed to the Link element.\n */\n href?: string;\n}\n\nconst withBaseName = makePrefixer(\"saltNavigationItem\");\n\nexport const NavigationItem = forwardRef<HTMLDivElement, NavigationItemProps>(\n function NavigationItem(props, ref) {\n const {\n active,\n blurActive,\n render,\n children,\n className,\n expanded = false,\n href,\n orientation = \"horizontal\",\n parent,\n level = 0,\n onExpand,\n style: styleProp,\n \"aria-controls\": ariaControls,\n ...rest\n } = props;\n\n const targetWindow = useWindow();\n useComponentCssInjection({\n testId: \"salt-navigation-item\",\n css: navigationItemCss,\n window: targetWindow,\n });\n\n const style = {\n ...styleProp,\n \"--saltNavigationItem-level\": `${level}`,\n };\n\n const isLink = href !== undefined;\n\n const handleClick = (event: MouseEvent<HTMLButtonElement>) => {\n onExpand?.(event);\n };\n\n return (\n <div\n ref={ref}\n className={clsx(withBaseName(), className)}\n style={style}\n {...rest}\n >\n <NavigationItemAction\n className={clsx(\n withBaseName(\"wrapper\"),\n {\n [withBaseName(\"active\")]: active || blurActive,\n [withBaseName(\"blurActive\")]: blurActive,\n [withBaseName(\"rootItem\")]: level === 0,\n },\n withBaseName(orientation),\n )}\n render={render ?? (isLink ? undefined : <button type=\"button\" />)}\n aria-expanded={isLink ? undefined : expanded}\n aria-controls={ariaControls}\n onClick={handleClick}\n aria-current={isLink && active ? \"page\" : undefined}\n href={href}\n >\n <span className={withBaseName(\"label\")}>{children}</span>\n {parent ? (\n <ExpansionIcon expanded={expanded} orientation={orientation} />\n ) : null}\n </NavigationItemAction>\n </div>\n );\n },\n);\n"],"names":["NavigationItem","navigationItemCss"],"mappings":";;;;;;;;;;;;;;AAsDA,MAAM,YAAA,GAAe,aAAa,oBAAoB,CAAA;AAE/C,MAAM,cAAA,GAAiB,UAAA;AAAA,EAC5B,SAASA,eAAAA,CAAe,KAAA,EAAO,GAAA,EAAK;AAClC,IAAA,MAAM;AAAA,MACJ,MAAA;AAAA,MACA,UAAA;AAAA,MACA,MAAA;AAAA,MACA,QAAA;AAAA,MACA,SAAA;AAAA,MACA,QAAA,GAAW,KAAA;AAAA,MACX,IAAA;AAAA,MACA,WAAA,GAAc,YAAA;AAAA,MACd,MAAA;AAAA,MACA,KAAA,GAAQ,CAAA;AAAA,MACR,QAAA;AAAA,MACA,KAAA,EAAO,SAAA;AAAA,MACP,eAAA,EAAiB,YAAA;AAAA,MACjB,GAAG;AAAA,KACL,GAAI,KAAA;AAEJ,IAAA,MAAM,eAAe,SAAA,EAAU;AAC/B,IAAA,wBAAA,CAAyB;AAAA,MACvB,MAAA,EAAQ,sBAAA;AAAA,MACR,GAAA,EAAKC,QAAA;AAAA,MACL,MAAA,EAAQ;AAAA,KACT,CAAA;AAED,IAAA,MAAM,KAAA,GAAQ;AAAA,MACZ,GAAG,SAAA;AAAA,MACH,4BAAA,EAA8B,GAAG,KAAK,CAAA;AAAA,KACxC;AAEA,IAAA,MAAM,SAAS,IAAA,KAAS,MAAA;AAExB,IAAA,MAAM,WAAA,GAAc,CAAC,KAAA,KAAyC;AAC5D,MAAA,QAAA,IAAA,IAAA,GAAA,MAAA,GAAA,QAAA,CAAW,KAAA,CAAA;AAAA,IACb,CAAA;AAEA,IAAA,uBACE,GAAA;AAAA,MAAC,KAAA;AAAA,MAAA;AAAA,QACC,GAAA;AAAA,QACA,SAAA,EAAW,IAAA,CAAK,YAAA,EAAa,EAAG,SAAS,CAAA;AAAA,QACzC,KAAA;AAAA,QACC,GAAG,IAAA;AAAA,QAEJ,QAAA,kBAAA,IAAA;AAAA,UAAC,oBAAA;AAAA,UAAA;AAAA,YACC,SAAA,EAAW,IAAA;AAAA,cACT,aAAa,SAAS,CAAA;AAAA,cACtB;AAAA,gBACE,CAAC,YAAA,CAAa,QAAQ,CAAC,GAAG,MAAA,IAAU,UAAA;AAAA,gBACpC,CAAC,YAAA,CAAa,YAAY,CAAC,GAAG,UAAA;AAAA,gBAC9B,CAAC,YAAA,CAAa,UAAU,CAAC,GAAG,KAAA,KAAU;AAAA,eACxC;AAAA,cACA,aAAa,WAAW;AAAA,aAC1B;AAAA,YACA,QAAQ,MAAA,KAAW,MAAA,GAAS,yBAAY,GAAA,CAAC,QAAA,EAAA,EAAO,MAAK,QAAA,EAAS,CAAA,CAAA;AAAA,YAC9D,eAAA,EAAe,SAAS,MAAA,GAAY,QAAA;AAAA,YACpC,eAAA,EAAe,YAAA;AAAA,YACf,OAAA,EAAS,WAAA;AAAA,YACT,cAAA,EAAc,MAAA,IAAU,MAAA,GAAS,MAAA,GAAS,MAAA;AAAA,YAC1C,IAAA;AAAA,YAEA,QAAA,EAAA;AAAA,8BAAA,GAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAW,YAAA,CAAa,OAAO,GAAI,QAAA,EAAS,CAAA;AAAA,cACjD,MAAA,mBACC,GAAA,CAAC,aAAA,EAAA,EAAc,QAAA,EAAoB,aAA0B,CAAA,GAC3D;AAAA;AAAA;AAAA;AACN;AAAA,KACF;AAAA,EAEJ;AACF;;;;"}
@@ -0,0 +1,4 @@
1
+ var css_248z = ".saltRating {\n display: inline-flex;\n box-sizing: border-box;\n}\n\n.saltFormField .saltRating-container {\n height: var(--salt-size-base);\n padding-top: calc(var(--salt-spacing-100) + var(--salt-spacing-fixed-100));\n}\n\n.saltFormField .saltRating-label {\n min-height: var(--salt-size-base);\n padding-top: var(--salt-spacing-100);\n}\n\n.saltRating-labelBottom {\n flex-direction: column;\n align-items: flex-start;\n gap: var(--salt-spacing-75);\n}\n\n.saltRating-labelRight {\n flex-direction: row;\n gap: var(--salt-spacing-150);\n}\n\n.saltRating-container {\n display: flex;\n flex-direction: row;\n gap: var(--salt-spacing-150);\n flex-wrap: nowrap;\n justify-content: flex-start;\n padding-top: var(--salt-spacing-fixed-100);\n padding-bottom: var(--salt-spacing-fixed-100);\n box-sizing: border-box;\n}\n\n.saltRating-label {\n color: var(--salt-content-primary-foreground);\n font-family: var(--salt-text-fontFamily);\n font-size: var(--salt-text-fontSize);\n font-weight: var(--salt-text-fontWeight);\n line-height: var(--salt-text-lineHeight);\n box-sizing: border-box;\n}\n";
2
+
3
+ export { css_248z as default };
4
+ //# sourceMappingURL=Rating.css.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Rating.css.js","sources":[],"sourcesContent":[],"names":[],"mappings":";;;;"}
@@ -0,0 +1,138 @@
1
+ import { jsxs, jsx } from 'react/jsx-runtime';
2
+ import { useComponentCssInjection } from '@salt-ds/styles';
3
+ import { useWindow } from '@salt-ds/window';
4
+ import { clsx } from 'clsx';
5
+ import { forwardRef, useState } from 'react';
6
+ import '../form-field-context/FormFieldContext.js';
7
+ import { useFormFieldProps } from '../form-field-context/useFormFieldProps.js';
8
+ import { capitalize } from '../utils/capitalize.js';
9
+ import { makePrefixer } from '../utils/makePrefixer.js';
10
+ import { useControlled } from '../utils/useControlled.js';
11
+ import '../utils/useFloatingUI/useFloatingUI.js';
12
+ import { useId } from '../utils/useId.js';
13
+ import '../salt-provider/SaltProvider.js';
14
+ import '../viewport/ViewportProvider.js';
15
+ import css_248z from './Rating.css.js';
16
+ import { RatingItem } from './RatingItem.js';
17
+
18
+ const withBaseName = makePrefixer("saltRating");
19
+ const defaultGetLabel = (value) => `${value} Star${value > 1 ? "s" : ""}`;
20
+ const Rating = forwardRef(function Rating2({
21
+ value: valueProp,
22
+ defaultValue = 0,
23
+ name: nameProp,
24
+ onChange,
25
+ className,
26
+ readOnly: readOnlyProp,
27
+ disabled: disabledProp,
28
+ max = 5,
29
+ getLabel = defaultGetLabel,
30
+ getVisibleLabel,
31
+ labelPlacement = "right",
32
+ "aria-label": ariaLabel,
33
+ "aria-labelledby": ariaLabelledBy,
34
+ "aria-describedby": ariaDescribedBy,
35
+ ...restProps
36
+ }, ref) {
37
+ const targetWindow = useWindow();
38
+ useComponentCssInjection({
39
+ testId: "salt-rating",
40
+ css: css_248z,
41
+ window: targetWindow
42
+ });
43
+ const {
44
+ disabled: formFieldDisabled,
45
+ readOnly: formFieldReadOnly,
46
+ a11yProps: {
47
+ "aria-describedby": formFieldDescribedBy,
48
+ "aria-labelledby": formFieldLabelledBy
49
+ } = {}
50
+ } = useFormFieldProps();
51
+ const disabled = formFieldDisabled || disabledProp;
52
+ const readOnly = formFieldReadOnly || readOnlyProp;
53
+ const [hoveredValue, setHoveredValue] = useState(0);
54
+ const [selected, setSelected] = useControlled({
55
+ controlled: valueProp,
56
+ default: defaultValue,
57
+ name: "Rating",
58
+ state: "value"
59
+ });
60
+ const name = useId(nameProp);
61
+ const handleMouseEnter = (event) => {
62
+ if (readOnly || disabled) return;
63
+ const itemValue = Number.parseInt(event.currentTarget.value, 10);
64
+ setHoveredValue(itemValue);
65
+ };
66
+ const handleChange = (event) => {
67
+ if (readOnly) {
68
+ event.preventDefault();
69
+ return;
70
+ }
71
+ const itemValue = Number.parseInt(event.currentTarget.value, 10);
72
+ setSelected(itemValue);
73
+ onChange == null ? void 0 : onChange(event, itemValue);
74
+ };
75
+ return /* @__PURE__ */ jsxs(
76
+ "div",
77
+ {
78
+ ref,
79
+ className: clsx(
80
+ withBaseName(),
81
+ withBaseName(`label${capitalize(labelPlacement)}`),
82
+ className
83
+ ),
84
+ ...restProps,
85
+ children: [
86
+ /* @__PURE__ */ jsx(
87
+ "div",
88
+ {
89
+ role: "radiogroup",
90
+ className: withBaseName("container"),
91
+ "aria-readonly": readOnly || void 0,
92
+ "aria-label": ariaLabel,
93
+ "aria-labelledby": clsx(formFieldLabelledBy, ariaLabelledBy) || void 0,
94
+ "aria-describedby": clsx(formFieldDescribedBy, ariaDescribedBy) || void 0,
95
+ onMouseLeave: () => setHoveredValue(0),
96
+ children: Array.from({ length: max }, (_, index) => {
97
+ const itemValue = index + 1;
98
+ const isHovered = hoveredValue > 0 && itemValue <= hoveredValue;
99
+ const isSelected = hoveredValue === 0 && itemValue <= selected;
100
+ const isUnselecting = hoveredValue > 0 && itemValue > hoveredValue && itemValue <= selected;
101
+ return /* @__PURE__ */ jsx(
102
+ RatingItem,
103
+ {
104
+ currentRating: selected,
105
+ isHovered,
106
+ isSelected,
107
+ isUnselecting,
108
+ onMouseEnter: handleMouseEnter,
109
+ onChange: handleChange,
110
+ value: itemValue,
111
+ readOnly,
112
+ disabled,
113
+ name,
114
+ "aria-label": getLabel == null ? void 0 : getLabel(itemValue)
115
+ },
116
+ itemValue
117
+ );
118
+ })
119
+ }
120
+ ),
121
+ getVisibleLabel && /* @__PURE__ */ jsx(
122
+ "div",
123
+ {
124
+ className: clsx(
125
+ withBaseName("label"),
126
+ withBaseName(`label-${labelPlacement}`)
127
+ ),
128
+ "aria-hidden": true,
129
+ children: getVisibleLabel(hoveredValue || selected, max)
130
+ }
131
+ )
132
+ ]
133
+ }
134
+ );
135
+ });
136
+
137
+ export { Rating };
138
+ //# sourceMappingURL=Rating.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Rating.js","sources":["../src/rating/Rating.tsx"],"sourcesContent":["import { useComponentCssInjection } from \"@salt-ds/styles\";\nimport { useWindow } from \"@salt-ds/window\";\nimport { clsx } from \"clsx\";\nimport {\n type ChangeEvent,\n type ComponentPropsWithoutRef,\n forwardRef,\n type MouseEvent,\n useState,\n} from \"react\";\nimport { useFormFieldProps } from \"../form-field-context\";\nimport { capitalize, makePrefixer, useControlled, useId } from \"../utils\";\nimport ratingCss from \"./Rating.css\";\nimport { RatingItem } from \"./RatingItem\";\n\nconst withBaseName = makePrefixer(\"saltRating\");\n\nexport interface RatingProps\n extends Omit<ComponentPropsWithoutRef<\"div\">, \"onChange\"> {\n /**\n * When provided, the component is controlled.\n */\n value?: number;\n /**\n * Default rating value for uncontrolled mode.\n * @default 0\n */\n defaultValue?: number;\n /**\n * Callback function for rating change.\n * The first parameter is the event, and the second is the selected rating value.\n */\n onChange?: (event: ChangeEvent<HTMLInputElement>, value: number) => void;\n /**\n * If true, the rating component will be in a read-only state.\n */\n readOnly?: boolean;\n /**\n * If true, the rating component will be in a disabled state.\n */\n disabled?: boolean;\n /**\n * Total number of icons displayed.\n * @default 5\n */\n max?: number;\n /**\n * Function used to provide a user-friendly name for the current value of the rating. Primarily used by screen readers.\n */\n getLabel?: (value: number) => string;\n /**\n * Function used to provide a visible label for the rating.\n */\n getVisibleLabel?: (value: number, max: number) => string;\n /**\n * Position of the label relative to the rating component.\n * @default \"right\"\n */\n labelPlacement?: \"right\" | \"bottom\";\n /**\n * The name to be set on each radio button within the group. If not set, then one will be generated for you.\n */\n name?: string;\n}\n\nconst defaultGetLabel = (value: number) =>\n `${value} Star${value > 1 ? \"s\" : \"\"}`;\n\nexport const Rating = forwardRef<HTMLDivElement, RatingProps>(function Rating(\n {\n value: valueProp,\n defaultValue = 0,\n name: nameProp,\n onChange,\n className,\n readOnly: readOnlyProp,\n disabled: disabledProp,\n max = 5,\n getLabel = defaultGetLabel,\n getVisibleLabel,\n labelPlacement = \"right\",\n \"aria-label\": ariaLabel,\n \"aria-labelledby\": ariaLabelledBy,\n \"aria-describedby\": ariaDescribedBy,\n ...restProps\n },\n ref,\n) {\n const targetWindow = useWindow();\n useComponentCssInjection({\n testId: \"salt-rating\",\n css: ratingCss,\n window: targetWindow,\n });\n const {\n disabled: formFieldDisabled,\n readOnly: formFieldReadOnly,\n a11yProps: {\n \"aria-describedby\": formFieldDescribedBy,\n \"aria-labelledby\": formFieldLabelledBy,\n } = {},\n } = useFormFieldProps();\n\n const disabled = formFieldDisabled || disabledProp;\n const readOnly = formFieldReadOnly || readOnlyProp;\n\n const [hoveredValue, setHoveredValue] = useState(0);\n const [selected, setSelected] = useControlled({\n controlled: valueProp,\n default: defaultValue,\n name: \"Rating\",\n state: \"value\",\n });\n const name = useId(nameProp);\n\n const handleMouseEnter = (event: MouseEvent<HTMLInputElement>) => {\n if (readOnly || disabled) return;\n const itemValue = Number.parseInt(event.currentTarget.value, 10);\n setHoveredValue(itemValue);\n };\n\n const handleChange = (event: ChangeEvent<HTMLInputElement>) => {\n if (readOnly) {\n event.preventDefault();\n return;\n }\n\n const itemValue = Number.parseInt(event.currentTarget.value, 10);\n setSelected(itemValue);\n onChange?.(event, itemValue);\n };\n\n return (\n <div\n ref={ref}\n className={clsx(\n withBaseName(),\n withBaseName(`label${capitalize(labelPlacement)}`),\n className,\n )}\n {...restProps}\n >\n <div\n role=\"radiogroup\"\n className={withBaseName(\"container\")}\n aria-readonly={readOnly || undefined}\n aria-label={ariaLabel}\n aria-labelledby={clsx(formFieldLabelledBy, ariaLabelledBy) || undefined}\n aria-describedby={\n clsx(formFieldDescribedBy, ariaDescribedBy) || undefined\n }\n onMouseLeave={() => setHoveredValue(0)}\n >\n {Array.from({ length: max }, (_, index) => {\n const itemValue = index + 1;\n const isHovered = hoveredValue > 0 && itemValue <= hoveredValue;\n const isSelected = hoveredValue === 0 && itemValue <= selected;\n const isUnselecting =\n hoveredValue > 0 &&\n itemValue > hoveredValue &&\n itemValue <= selected;\n return (\n <RatingItem\n key={itemValue}\n currentRating={selected}\n isHovered={isHovered}\n isSelected={isSelected}\n isUnselecting={isUnselecting}\n onMouseEnter={handleMouseEnter}\n onChange={handleChange}\n value={itemValue}\n readOnly={readOnly}\n disabled={disabled}\n name={name}\n aria-label={getLabel?.(itemValue)}\n />\n );\n })}\n </div>\n {getVisibleLabel && (\n <div\n className={clsx(\n withBaseName(\"label\"),\n withBaseName(`label-${labelPlacement}`),\n )}\n aria-hidden\n >\n {getVisibleLabel(hoveredValue || selected, max)}\n </div>\n )}\n </div>\n );\n});\n"],"names":["Rating","ratingCss"],"mappings":";;;;;;;;;;;;;;;;;AAeA,MAAM,YAAA,GAAe,aAAa,YAAY,CAAA;AAkD9C,MAAM,eAAA,GAAkB,CAAC,KAAA,KACvB,CAAA,EAAG,KAAK,CAAA,KAAA,EAAQ,KAAA,GAAQ,CAAA,GAAI,GAAA,GAAM,EAAE,CAAA,CAAA;AAE/B,MAAM,MAAA,GAAS,UAAA,CAAwC,SAASA,OAAAA,CACrE;AAAA,EACE,KAAA,EAAO,SAAA;AAAA,EACP,YAAA,GAAe,CAAA;AAAA,EACf,IAAA,EAAM,QAAA;AAAA,EACN,QAAA;AAAA,EACA,SAAA;AAAA,EACA,QAAA,EAAU,YAAA;AAAA,EACV,QAAA,EAAU,YAAA;AAAA,EACV,GAAA,GAAM,CAAA;AAAA,EACN,QAAA,GAAW,eAAA;AAAA,EACX,eAAA;AAAA,EACA,cAAA,GAAiB,OAAA;AAAA,EACjB,YAAA,EAAc,SAAA;AAAA,EACd,iBAAA,EAAmB,cAAA;AAAA,EACnB,kBAAA,EAAoB,eAAA;AAAA,EACpB,GAAG;AACL,CAAA,EACA,GAAA,EACA;AACA,EAAA,MAAM,eAAe,SAAA,EAAU;AAC/B,EAAA,wBAAA,CAAyB;AAAA,IACvB,MAAA,EAAQ,aAAA;AAAA,IACR,GAAA,EAAKC,QAAA;AAAA,IACL,MAAA,EAAQ;AAAA,GACT,CAAA;AACD,EAAA,MAAM;AAAA,IACJ,QAAA,EAAU,iBAAA;AAAA,IACV,QAAA,EAAU,iBAAA;AAAA,IACV,SAAA,EAAW;AAAA,MACT,kBAAA,EAAoB,oBAAA;AAAA,MACpB,iBAAA,EAAmB;AAAA,QACjB;AAAC,MACH,iBAAA,EAAkB;AAEtB,EAAA,MAAM,WAAW,iBAAA,IAAqB,YAAA;AACtC,EAAA,MAAM,WAAW,iBAAA,IAAqB,YAAA;AAEtC,EAAA,MAAM,CAAC,YAAA,EAAc,eAAe,CAAA,GAAI,SAAS,CAAC,CAAA;AAClD,EAAA,MAAM,CAAC,QAAA,EAAU,WAAW,CAAA,GAAI,aAAA,CAAc;AAAA,IAC5C,UAAA,EAAY,SAAA;AAAA,IACZ,OAAA,EAAS,YAAA;AAAA,IACT,IAAA,EAAM,QAAA;AAAA,IACN,KAAA,EAAO;AAAA,GACR,CAAA;AACD,EAAA,MAAM,IAAA,GAAO,MAAM,QAAQ,CAAA;AAE3B,EAAA,MAAM,gBAAA,GAAmB,CAAC,KAAA,KAAwC;AAChE,IAAA,IAAI,YAAY,QAAA,EAAU;AAC1B,IAAA,MAAM,YAAY,MAAA,CAAO,QAAA,CAAS,KAAA,CAAM,aAAA,CAAc,OAAO,EAAE,CAAA;AAC/D,IAAA,eAAA,CAAgB,SAAS,CAAA;AAAA,EAC3B,CAAA;AAEA,EAAA,MAAM,YAAA,GAAe,CAAC,KAAA,KAAyC;AAC7D,IAAA,IAAI,QAAA,EAAU;AACZ,MAAA,KAAA,CAAM,cAAA,EAAe;AACrB,MAAA;AAAA,IACF;AAEA,IAAA,MAAM,YAAY,MAAA,CAAO,QAAA,CAAS,KAAA,CAAM,aAAA,CAAc,OAAO,EAAE,CAAA;AAC/D,IAAA,WAAA,CAAY,SAAS,CAAA;AACrB,IAAA,QAAA,IAAA,IAAA,GAAA,MAAA,GAAA,QAAA,CAAW,KAAA,EAAO,SAAA,CAAA;AAAA,EACpB,CAAA;AAEA,EAAA,uBACE,IAAA;AAAA,IAAC,KAAA;AAAA,IAAA;AAAA,MACC,GAAA;AAAA,MACA,SAAA,EAAW,IAAA;AAAA,QACT,YAAA,EAAa;AAAA,QACb,YAAA,CAAa,CAAA,KAAA,EAAQ,UAAA,CAAW,cAAc,CAAC,CAAA,CAAE,CAAA;AAAA,QACjD;AAAA,OACF;AAAA,MACC,GAAG,SAAA;AAAA,MAEJ,QAAA,EAAA;AAAA,wBAAA,GAAA;AAAA,UAAC,KAAA;AAAA,UAAA;AAAA,YACC,IAAA,EAAK,YAAA;AAAA,YACL,SAAA,EAAW,aAAa,WAAW,CAAA;AAAA,YACnC,iBAAe,QAAA,IAAY,MAAA;AAAA,YAC3B,YAAA,EAAY,SAAA;AAAA,YACZ,iBAAA,EAAiB,IAAA,CAAK,mBAAA,EAAqB,cAAc,CAAA,IAAK,MAAA;AAAA,YAC9D,kBAAA,EACE,IAAA,CAAK,oBAAA,EAAsB,eAAe,CAAA,IAAK,MAAA;AAAA,YAEjD,YAAA,EAAc,MAAM,eAAA,CAAgB,CAAC,CAAA;AAAA,YAEpC,QAAA,EAAA,KAAA,CAAM,KAAK,EAAE,MAAA,EAAQ,KAAI,EAAG,CAAC,GAAG,KAAA,KAAU;AACzC,cAAA,MAAM,YAAY,KAAA,GAAQ,CAAA;AAC1B,cAAA,MAAM,SAAA,GAAY,YAAA,GAAe,CAAA,IAAK,SAAA,IAAa,YAAA;AACnD,cAAA,MAAM,UAAA,GAAa,YAAA,KAAiB,CAAA,IAAK,SAAA,IAAa,QAAA;AACtD,cAAA,MAAM,aAAA,GACJ,YAAA,GAAe,CAAA,IACf,SAAA,GAAY,gBACZ,SAAA,IAAa,QAAA;AACf,cAAA,uBACE,GAAA;AAAA,gBAAC,UAAA;AAAA,gBAAA;AAAA,kBAEC,aAAA,EAAe,QAAA;AAAA,kBACf,SAAA;AAAA,kBACA,UAAA;AAAA,kBACA,aAAA;AAAA,kBACA,YAAA,EAAc,gBAAA;AAAA,kBACd,QAAA,EAAU,YAAA;AAAA,kBACV,KAAA,EAAO,SAAA;AAAA,kBACP,QAAA;AAAA,kBACA,QAAA;AAAA,kBACA,IAAA;AAAA,kBACA,cAAY,QAAA,IAAA,IAAA,GAAA,MAAA,GAAA,QAAA,CAAW,SAAA;AAAA,iBAAA;AAAA,gBAXlB;AAAA,eAYP;AAAA,YAEJ,CAAC;AAAA;AAAA,SACH;AAAA,QACC,eAAA,oBACC,GAAA;AAAA,UAAC,KAAA;AAAA,UAAA;AAAA,YACC,SAAA,EAAW,IAAA;AAAA,cACT,aAAa,OAAO,CAAA;AAAA,cACpB,YAAA,CAAa,CAAA,MAAA,EAAS,cAAc,CAAA,CAAE;AAAA,aACxC;AAAA,YACA,aAAA,EAAW,IAAA;AAAA,YAEV,QAAA,EAAA,eAAA,CAAgB,YAAA,IAAgB,QAAA,EAAU,GAAG;AAAA;AAAA;AAChD;AAAA;AAAA,GAEJ;AAEJ,CAAC;;;;"}
@@ -0,0 +1,4 @@
1
+ var css_248z = ".saltRatingItem {\n height: var(--salt-size-icon);\n width: var(--salt-size-icon);\n display: flex;\n position: relative;\n align-items: center;\n justify-content: center;\n cursor: var(--salt-cursor-hover);\n}\n\n.saltRatingItem-disabled,\n.saltRatingItem-readOnly {\n cursor: default;\n}\n\n.saltRatingItem-input {\n position: absolute;\n opacity: 0;\n width: 100%;\n height: 100%;\n margin: 0;\n cursor: inherit;\n z-index: 1;\n}\n\n.saltRatingItem-input::before {\n content: \"\";\n position: absolute;\n width: 100%;\n height: 100%;\n padding: var(--salt-spacing-fixed-600);\n transform: translate(-50%, -50%);\n top: 50%;\n left: 50%;\n box-sizing: content-box;\n}\n\n.saltRatingItem-icon {\n pointer-events: none;\n display: flex;\n align-items: center;\n justify-content: center;\n color: var(--salt-sentiment-neutral-dataviz);\n}\n\n.saltRatingItem-icon .saltIcon {\n --saltIcon-size: var(--salt-size-icon);\n}\n\n.saltRatingItem-input:focus-visible ~ .saltRatingItem-icon::after {\n content: \"\";\n position: absolute;\n width: 100%;\n height: 100%;\n padding: var(--salt-spacing-fixed-600);\n transform: translate(-50%, -50%);\n top: 50%;\n left: 50%;\n box-sizing: content-box;\n\n border-radius: var(--salt-palette-corner-strongest);\n outline: var(--salt-focused-outline);\n outline-offset: calc(-1 * var(--salt-spacing-fixed-200));\n}\n\n.saltRatingItem-disabled {\n cursor: var(--salt-cursor-disabled);\n opacity: 0.4;\n}\n\n.saltRatingItem-disabled .saltRatingItem-input {\n cursor: var(--salt-cursor-disabled);\n}\n\n.saltRatingItem-selected .saltRatingItem-icon,\n.saltRatingItem-hovered .saltRatingItem-icon,\n.saltRatingItem-unselecting .saltRatingItem-icon {\n color: var(--salt-sentiment-accent-dataviz);\n}\n";
2
+
3
+ export { css_248z as default };
4
+ //# sourceMappingURL=RatingItem.css.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"RatingItem.css.js","sources":[],"sourcesContent":[],"names":[],"mappings":";;;;"}
@@ -0,0 +1,73 @@
1
+ import { jsxs, jsx } from 'react/jsx-runtime';
2
+ import { useComponentCssInjection } from '@salt-ds/styles';
3
+ import { useWindow } from '@salt-ds/window';
4
+ import { clsx } from 'clsx';
5
+ import { forwardRef } from 'react';
6
+ import { useIcon } from '../semantic-icon-provider/SemanticIconProvider.js';
7
+ import { makePrefixer } from '../utils/makePrefixer.js';
8
+ import '../utils/useFloatingUI/useFloatingUI.js';
9
+ import '../utils/useId.js';
10
+ import '../salt-provider/SaltProvider.js';
11
+ import '../viewport/ViewportProvider.js';
12
+ import css_248z from './RatingItem.css.js';
13
+
14
+ const withBaseName = makePrefixer("saltRatingItem");
15
+ const RatingItem = forwardRef(
16
+ function RatingItem2(props, ref) {
17
+ const {
18
+ "aria-label": ariaLabel,
19
+ value,
20
+ currentRating,
21
+ isHovered,
22
+ isSelected,
23
+ isUnselecting,
24
+ onMouseEnter,
25
+ onChange,
26
+ readOnly = false,
27
+ disabled = false,
28
+ name
29
+ } = props;
30
+ const { RatingIcon, RatingSelectedIcon, RatingUnselectingIcon } = useIcon();
31
+ const targetWindow = useWindow();
32
+ useComponentCssInjection({
33
+ testId: "salt-rating-item",
34
+ css: css_248z,
35
+ window: targetWindow
36
+ });
37
+ const icon = isHovered || isSelected ? /* @__PURE__ */ jsx(RatingSelectedIcon, { "aria-hidden": true }) : isUnselecting ? /* @__PURE__ */ jsx(RatingUnselectingIcon, { "aria-hidden": true }) : /* @__PURE__ */ jsx(RatingIcon, { "aria-hidden": true });
38
+ return /* @__PURE__ */ jsxs(
39
+ "div",
40
+ {
41
+ className: clsx(withBaseName(), {
42
+ [withBaseName("hovered")]: isHovered,
43
+ [withBaseName("selected")]: isSelected,
44
+ [withBaseName("unselecting")]: isUnselecting,
45
+ [withBaseName("disabled")]: disabled,
46
+ [withBaseName("readOnly")]: readOnly
47
+ }),
48
+ children: [
49
+ /* @__PURE__ */ jsx(
50
+ "input",
51
+ {
52
+ ref,
53
+ type: "radio",
54
+ name,
55
+ value,
56
+ checked: currentRating === value,
57
+ onChange,
58
+ onMouseEnter,
59
+ disabled,
60
+ readOnly,
61
+ className: withBaseName("input"),
62
+ "aria-label": ariaLabel
63
+ }
64
+ ),
65
+ /* @__PURE__ */ jsx("span", { className: withBaseName("icon"), "aria-hidden": true, children: icon })
66
+ ]
67
+ }
68
+ );
69
+ }
70
+ );
71
+
72
+ export { RatingItem };
73
+ //# sourceMappingURL=RatingItem.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"RatingItem.js","sources":["../src/rating/RatingItem.tsx"],"sourcesContent":["import { useComponentCssInjection } from \"@salt-ds/styles\";\nimport { useWindow } from \"@salt-ds/window\";\nimport { clsx } from \"clsx\";\nimport {\n type ChangeEvent,\n type ComponentPropsWithoutRef,\n forwardRef,\n type MouseEvent,\n} from \"react\";\nimport { useIcon } from \"../semantic-icon-provider\";\nimport { makePrefixer } from \"../utils\";\nimport ratingItemCss from \"./RatingItem.css\";\n\nconst withBaseName = makePrefixer(\"saltRatingItem\");\n\nexport interface RatingItemProps extends ComponentPropsWithoutRef<\"div\"> {\n /**\n * specifies the value of the feedback item.\n */\n value: number;\n /**\n * To define if the current star is being hovered.\n */\n isHovered?: boolean;\n /**\n * To specify if the item is selected.\n */\n isSelected?: boolean;\n /**\n * defines the current selected rating.\n */\n currentRating?: number;\n /**\n * callback function when feedback item is hovered.\n */\n onMouseEnter: (event: MouseEvent<HTMLInputElement>) => void;\n /**\n * callback function when feedback item is clicked.\n */\n onChange: (event: ChangeEvent<HTMLInputElement>) => void;\n /**\n * If true, the rating item will be in a read-only state.\n */\n readOnly?: boolean;\n /**\n * If true, the rating item will be in a disabled state.\n */\n disabled?: boolean;\n /**\n * Indicates whether the current rating item is in an active state.\n * An active state typically means that the item is visually highlighted\n * or styled differently to indicate that it is part of the current selection\n * or interaction (e.g., hover or focus).\n */\n isUnselecting?: boolean;\n /**\n * Name of the radio group\n */\n name?: string;\n}\n\nexport const RatingItem = forwardRef<HTMLInputElement, RatingItemProps>(\n function RatingItem(props, ref) {\n const {\n \"aria-label\": ariaLabel,\n value,\n currentRating,\n isHovered,\n isSelected,\n isUnselecting,\n onMouseEnter,\n onChange,\n readOnly = false,\n disabled = false,\n name,\n } = props;\n\n const { RatingIcon, RatingSelectedIcon, RatingUnselectingIcon } = useIcon();\n\n const targetWindow = useWindow();\n useComponentCssInjection({\n testId: \"salt-rating-item\",\n css: ratingItemCss,\n window: targetWindow,\n });\n\n const icon =\n isHovered || isSelected ? (\n <RatingSelectedIcon aria-hidden />\n ) : isUnselecting ? (\n <RatingUnselectingIcon aria-hidden />\n ) : (\n <RatingIcon aria-hidden />\n );\n\n return (\n <div\n className={clsx(withBaseName(), {\n [withBaseName(\"hovered\")]: isHovered,\n [withBaseName(\"selected\")]: isSelected,\n [withBaseName(\"unselecting\")]: isUnselecting,\n [withBaseName(\"disabled\")]: disabled,\n [withBaseName(\"readOnly\")]: readOnly,\n })}\n >\n <input\n ref={ref}\n type=\"radio\"\n name={name}\n value={value}\n checked={currentRating === value}\n onChange={onChange}\n onMouseEnter={onMouseEnter}\n disabled={disabled}\n readOnly={readOnly}\n className={withBaseName(\"input\")}\n aria-label={ariaLabel}\n />\n <span className={withBaseName(\"icon\")} aria-hidden>\n {icon}\n </span>\n </div>\n );\n },\n);\n"],"names":["RatingItem","ratingItemCss"],"mappings":";;;;;;;;;;;;;AAaA,MAAM,YAAA,GAAe,aAAa,gBAAgB,CAAA;AAgD3C,MAAM,UAAA,GAAa,UAAA;AAAA,EACxB,SAASA,WAAAA,CAAW,KAAA,EAAO,GAAA,EAAK;AAC9B,IAAA,MAAM;AAAA,MACJ,YAAA,EAAc,SAAA;AAAA,MACd,KAAA;AAAA,MACA,aAAA;AAAA,MACA,SAAA;AAAA,MACA,UAAA;AAAA,MACA,aAAA;AAAA,MACA,YAAA;AAAA,MACA,QAAA;AAAA,MACA,QAAA,GAAW,KAAA;AAAA,MACX,QAAA,GAAW,KAAA;AAAA,MACX;AAAA,KACF,GAAI,KAAA;AAEJ,IAAA,MAAM,EAAE,UAAA,EAAY,kBAAA,EAAoB,qBAAA,KAA0B,OAAA,EAAQ;AAE1E,IAAA,MAAM,eAAe,SAAA,EAAU;AAC/B,IAAA,wBAAA,CAAyB;AAAA,MACvB,MAAA,EAAQ,kBAAA;AAAA,MACR,GAAA,EAAKC,QAAA;AAAA,MACL,MAAA,EAAQ;AAAA,KACT,CAAA;AAED,IAAA,MAAM,OACJ,SAAA,IAAa,UAAA,mBACX,GAAA,CAAC,kBAAA,EAAA,EAAmB,eAAW,IAAA,EAAC,CAAA,GAC9B,aAAA,mBACF,GAAA,CAAC,yBAAsB,aAAA,EAAW,IAAA,EAAC,oBAEnC,GAAA,CAAC,UAAA,EAAA,EAAW,eAAW,IAAA,EAAC,CAAA;AAG5B,IAAA,uBACE,IAAA;AAAA,MAAC,KAAA;AAAA,MAAA;AAAA,QACC,SAAA,EAAW,IAAA,CAAK,YAAA,EAAa,EAAG;AAAA,UAC9B,CAAC,YAAA,CAAa,SAAS,CAAC,GAAG,SAAA;AAAA,UAC3B,CAAC,YAAA,CAAa,UAAU,CAAC,GAAG,UAAA;AAAA,UAC5B,CAAC,YAAA,CAAa,aAAa,CAAC,GAAG,aAAA;AAAA,UAC/B,CAAC,YAAA,CAAa,UAAU,CAAC,GAAG,QAAA;AAAA,UAC5B,CAAC,YAAA,CAAa,UAAU,CAAC,GAAG;AAAA,SAC7B,CAAA;AAAA,QAED,QAAA,EAAA;AAAA,0BAAA,GAAA;AAAA,YAAC,OAAA;AAAA,YAAA;AAAA,cACC,GAAA;AAAA,cACA,IAAA,EAAK,OAAA;AAAA,cACL,IAAA;AAAA,cACA,KAAA;AAAA,cACA,SAAS,aAAA,KAAkB,KAAA;AAAA,cAC3B,QAAA;AAAA,cACA,YAAA;AAAA,cACA,QAAA;AAAA,cACA,QAAA;AAAA,cACA,SAAA,EAAW,aAAa,OAAO,CAAA;AAAA,cAC/B,YAAA,EAAY;AAAA;AAAA,WACd;AAAA,0BACA,GAAA,CAAC,UAAK,SAAA,EAAW,YAAA,CAAa,MAAM,CAAA,EAAG,aAAA,EAAW,MAC/C,QAAA,EAAA,IAAA,EACH;AAAA;AAAA;AAAA,KACF;AAAA,EAEJ;AACF;;;;"}
@@ -1,40 +1,42 @@
1
1
  import { jsx } from 'react/jsx-runtime';
2
- import { FavoriteStrongIcon, FavoriteSolidIcon, FavoriteIcon, ProgressInprogressIcon, LockedIcon, SuccessCircleSolidIcon, StepActiveIcon, StepDefaultIcon, TearOutIcon, CloseIcon, CalendarIcon, UserSolidIcon, OverflowMenuIcon, WarningSolidIcon, InfoSolidIcon, ErrorSolidIcon, UploadIcon, TriangleDownIcon, TriangleUpIcon, ChevronLeftIcon, ChevronRightIcon, ChevronDownIcon, ChevronUpIcon } from '@salt-ds/icons';
2
+ import { WarningSolidIcon, UserSolidIcon, UploadIcon, SuccessCircleSolidIcon, FavoriteStrongIcon, FavoriteSolidIcon, FavoriteIcon, ChevronLeftIcon, StepDefaultIcon, OverflowMenuIcon, ChevronRightIcon, LockedIcon, ProgressInprogressIcon, InfoSolidIcon, TriangleUpIcon, TearOutIcon, ChevronDownIcon, ErrorSolidIcon, TriangleDownIcon, DoubleChevronRightIcon, DoubleChevronLeftIcon, ChevronUpIcon, CloseIcon, CalendarIcon, StepActiveIcon } from '@salt-ds/icons';
3
3
  import { createContext, useContext } from 'react';
4
4
  import { ErrorAdornmentIcon } from '../status-adornment/ErrorAdornment.js';
5
5
  import { SuccessAdornmentIcon } from '../status-adornment/SuccessAdornment.js';
6
6
  import { WarningAdornmentIcon } from '../status-adornment/WarningAdornment.js';
7
7
 
8
8
  const defaultIconMap = {
9
- ExpandIcon: ChevronDownIcon,
10
- CollapseIcon: ChevronUpIcon,
11
- ExpandGroupIcon: ChevronRightIcon,
9
+ ActiveIcon: StepActiveIcon,
10
+ CalendarIcon,
11
+ CloseIcon,
12
12
  CollapseGroupIcon: ChevronDownIcon,
13
- NextIcon: ChevronRightIcon,
14
- PreviousIcon: ChevronLeftIcon,
15
- IncreaseIcon: TriangleUpIcon,
13
+ CollapseIcon: ChevronUpIcon,
14
+ CollapseLeftIcon: DoubleChevronLeftIcon,
15
+ CollapseRightIcon: DoubleChevronRightIcon,
16
+ CompletedIcon: SuccessCircleSolidIcon,
16
17
  DecreaseIcon: TriangleDownIcon,
17
- UploadIcon,
18
18
  ErrorIcon: ErrorSolidIcon,
19
- SuccessIcon: SuccessCircleSolidIcon,
19
+ ErrorStatusAdornment: ErrorAdornmentIcon,
20
+ ExpandGroupIcon: ChevronRightIcon,
21
+ ExpandIcon: ChevronDownIcon,
22
+ ExternalIcon: TearOutIcon,
23
+ IncreaseIcon: TriangleUpIcon,
20
24
  InfoIcon: InfoSolidIcon,
21
- WarningIcon: WarningSolidIcon,
25
+ InProgressIcon: ProgressInprogressIcon,
26
+ LockedIcon,
27
+ NextIcon: ChevronRightIcon,
22
28
  OverflowIcon: OverflowMenuIcon,
23
- UserIcon: UserSolidIcon,
24
- CalendarIcon,
25
- CloseIcon,
26
- ExternalIcon: TearOutIcon,
27
29
  PendingIcon: StepDefaultIcon,
28
- ActiveIcon: StepActiveIcon,
29
- CompletedIcon: SuccessCircleSolidIcon,
30
- LockedIcon,
31
- InProgressIcon: ProgressInprogressIcon,
30
+ PreviousIcon: ChevronLeftIcon,
32
31
  RatingIcon: FavoriteIcon,
33
32
  RatingSelectedIcon: FavoriteSolidIcon,
34
33
  RatingUnselectingIcon: FavoriteStrongIcon,
35
- ErrorStatusAdornment: ErrorAdornmentIcon,
36
- WarningStatusAdornment: WarningAdornmentIcon,
37
- SuccessStatusAdornment: SuccessAdornmentIcon
34
+ SuccessIcon: SuccessCircleSolidIcon,
35
+ SuccessStatusAdornment: SuccessAdornmentIcon,
36
+ UploadIcon,
37
+ UserIcon: UserSolidIcon,
38
+ WarningIcon: WarningSolidIcon,
39
+ WarningStatusAdornment: WarningAdornmentIcon
38
40
  };
39
41
  const SemanticIconContext = createContext(defaultIconMap);
40
42
  const SemanticIconProvider = ({
@@ -1 +1 @@
1
- {"version":3,"file":"SemanticIconProvider.js","sources":["../src/semantic-icon-provider/SemanticIconProvider.tsx"],"sourcesContent":["import {\n CalendarIcon,\n ChevronDownIcon,\n ChevronLeftIcon,\n ChevronRightIcon,\n ChevronUpIcon,\n CloseIcon,\n ErrorSolidIcon,\n FavoriteIcon,\n FavoriteSolidIcon,\n FavoriteStrongIcon,\n InfoSolidIcon,\n LockedIcon,\n OverflowMenuIcon,\n ProgressInprogressIcon,\n StepActiveIcon,\n StepDefaultIcon,\n SuccessCircleSolidIcon,\n TearOutIcon,\n TriangleDownIcon,\n TriangleUpIcon,\n UploadIcon,\n UserSolidIcon,\n WarningSolidIcon,\n} from \"@salt-ds/icons\";\nimport {\n createContext,\n type ElementType,\n type ReactNode,\n useContext,\n} from \"react\";\nimport { ErrorAdornmentIcon } from \"../status-adornment/ErrorAdornment\";\nimport { SuccessAdornmentIcon } from \"../status-adornment/SuccessAdornment\";\nimport { WarningAdornmentIcon } from \"../status-adornment/WarningAdornment\";\n\nexport interface SemanticIconMap {\n ExpandIcon: ElementType;\n CollapseIcon: ElementType;\n ExpandGroupIcon: ElementType;\n CollapseGroupIcon: ElementType;\n NextIcon: ElementType;\n PreviousIcon: ElementType;\n IncreaseIcon: ElementType;\n DecreaseIcon: ElementType;\n UploadIcon: ElementType;\n ErrorIcon: ElementType;\n SuccessIcon: ElementType;\n InfoIcon: ElementType;\n WarningIcon: ElementType;\n OverflowIcon: ElementType;\n UserIcon: ElementType;\n CalendarIcon: ElementType;\n CloseIcon: ElementType;\n ExternalIcon: ElementType;\n PendingIcon: ElementType;\n ActiveIcon: ElementType;\n CompletedIcon: ElementType;\n LockedIcon: ElementType;\n InProgressIcon: ElementType;\n RatingIcon: ElementType;\n RatingSelectedIcon: ElementType;\n RatingUnselectingIcon: ElementType;\n ErrorStatusAdornment: ElementType;\n WarningStatusAdornment: ElementType;\n SuccessStatusAdornment: ElementType;\n}\n\nexport interface SemanticIconProviderProps {\n /**\n * Custom mapping of icon names to components. Overrides default icons if provided.\n */\n iconMap?: Partial<SemanticIconMap>;\n\n /**\n * Child elements that will use the provided icons.\n */\n children: ReactNode;\n}\n\nconst defaultIconMap: SemanticIconMap = {\n ExpandIcon: ChevronDownIcon,\n CollapseIcon: ChevronUpIcon,\n ExpandGroupIcon: ChevronRightIcon,\n CollapseGroupIcon: ChevronDownIcon,\n NextIcon: ChevronRightIcon,\n PreviousIcon: ChevronLeftIcon,\n IncreaseIcon: TriangleUpIcon,\n DecreaseIcon: TriangleDownIcon,\n UploadIcon,\n ErrorIcon: ErrorSolidIcon,\n SuccessIcon: SuccessCircleSolidIcon,\n InfoIcon: InfoSolidIcon,\n WarningIcon: WarningSolidIcon,\n OverflowIcon: OverflowMenuIcon,\n UserIcon: UserSolidIcon,\n CalendarIcon: CalendarIcon,\n CloseIcon: CloseIcon,\n ExternalIcon: TearOutIcon,\n PendingIcon: StepDefaultIcon,\n ActiveIcon: StepActiveIcon,\n CompletedIcon: SuccessCircleSolidIcon,\n LockedIcon: LockedIcon,\n InProgressIcon: ProgressInprogressIcon,\n RatingIcon: FavoriteIcon,\n RatingSelectedIcon: FavoriteSolidIcon,\n RatingUnselectingIcon: FavoriteStrongIcon,\n ErrorStatusAdornment: ErrorAdornmentIcon,\n WarningStatusAdornment: WarningAdornmentIcon,\n SuccessStatusAdornment: SuccessAdornmentIcon,\n};\n\nconst SemanticIconContext = createContext<SemanticIconMap>(defaultIconMap);\n\nexport const SemanticIconProvider = ({\n iconMap = {},\n children,\n}: SemanticIconProviderProps) => (\n <SemanticIconContext.Provider value={{ ...defaultIconMap, ...iconMap }}>\n {children}\n </SemanticIconContext.Provider>\n);\n\nexport const useIcon = () => {\n const context = useContext(SemanticIconContext);\n return context || defaultIconMap;\n};\n"],"names":[],"mappings":";;;;;;;AA+EA,MAAM,cAAA,GAAkC;AAAA,EACtC,UAAA,EAAY,eAAA;AAAA,EACZ,YAAA,EAAc,aAAA;AAAA,EACd,eAAA,EAAiB,gBAAA;AAAA,EACjB,iBAAA,EAAmB,eAAA;AAAA,EACnB,QAAA,EAAU,gBAAA;AAAA,EACV,YAAA,EAAc,eAAA;AAAA,EACd,YAAA,EAAc,cAAA;AAAA,EACd,YAAA,EAAc,gBAAA;AAAA,EACd,UAAA;AAAA,EACA,SAAA,EAAW,cAAA;AAAA,EACX,WAAA,EAAa,sBAAA;AAAA,EACb,QAAA,EAAU,aAAA;AAAA,EACV,WAAA,EAAa,gBAAA;AAAA,EACb,YAAA,EAAc,gBAAA;AAAA,EACd,QAAA,EAAU,aAAA;AAAA,EACV,YAAA;AAAA,EACA,SAAA;AAAA,EACA,YAAA,EAAc,WAAA;AAAA,EACd,WAAA,EAAa,eAAA;AAAA,EACb,UAAA,EAAY,cAAA;AAAA,EACZ,aAAA,EAAe,sBAAA;AAAA,EACf,UAAA;AAAA,EACA,cAAA,EAAgB,sBAAA;AAAA,EAChB,UAAA,EAAY,YAAA;AAAA,EACZ,kBAAA,EAAoB,iBAAA;AAAA,EACpB,qBAAA,EAAuB,kBAAA;AAAA,EACvB,oBAAA,EAAsB,kBAAA;AAAA,EACtB,sBAAA,EAAwB,oBAAA;AAAA,EACxB,sBAAA,EAAwB;AAC1B,CAAA;AAEA,MAAM,mBAAA,GAAsB,cAA+B,cAAc,CAAA;AAElE,MAAM,uBAAuB,CAAC;AAAA,EACnC,UAAU,EAAC;AAAA,EACX;AACF,CAAA,qBACE,GAAA,CAAC,mBAAA,CAAoB,QAAA,EAApB,EAA6B,KAAA,EAAO,EAAE,GAAG,cAAA,EAAgB,GAAG,OAAA,EAAQ,EAClE,QAAA,EACH;AAGK,MAAM,UAAU,MAAM;AAC3B,EAAA,MAAM,OAAA,GAAU,WAAW,mBAAmB,CAAA;AAC9C,EAAA,OAAO,OAAA,IAAW,cAAA;AACpB;;;;"}
1
+ {"version":3,"file":"SemanticIconProvider.js","sources":["../src/semantic-icon-provider/SemanticIconProvider.tsx"],"sourcesContent":["import {\n CalendarIcon,\n ChevronDownIcon,\n ChevronLeftIcon,\n ChevronRightIcon,\n ChevronUpIcon,\n CloseIcon,\n DoubleChevronLeftIcon,\n DoubleChevronRightIcon,\n ErrorSolidIcon,\n FavoriteIcon,\n FavoriteSolidIcon,\n FavoriteStrongIcon,\n InfoSolidIcon,\n LockedIcon,\n OverflowMenuIcon,\n ProgressInprogressIcon,\n StepActiveIcon,\n StepDefaultIcon,\n SuccessCircleSolidIcon,\n TearOutIcon,\n TriangleDownIcon,\n TriangleUpIcon,\n UploadIcon,\n UserSolidIcon,\n WarningSolidIcon,\n} from \"@salt-ds/icons\";\nimport {\n createContext,\n type ElementType,\n type ReactNode,\n useContext,\n} from \"react\";\nimport { ErrorAdornmentIcon } from \"../status-adornment/ErrorAdornment\";\nimport { SuccessAdornmentIcon } from \"../status-adornment/SuccessAdornment\";\nimport { WarningAdornmentIcon } from \"../status-adornment/WarningAdornment\";\n\nexport interface SemanticIconMap {\n ActiveIcon: ElementType;\n CalendarIcon: ElementType;\n CloseIcon: ElementType;\n CollapseGroupIcon: ElementType;\n CollapseIcon: ElementType;\n CollapseLeftIcon: ElementType;\n CollapseRightIcon: ElementType;\n CompletedIcon: ElementType;\n DecreaseIcon: ElementType;\n ErrorIcon: ElementType;\n ErrorStatusAdornment: ElementType;\n ExpandGroupIcon: ElementType;\n ExpandIcon: ElementType;\n ExternalIcon: ElementType;\n IncreaseIcon: ElementType;\n InfoIcon: ElementType;\n InProgressIcon: ElementType;\n LockedIcon: ElementType;\n NextIcon: ElementType;\n OverflowIcon: ElementType;\n PendingIcon: ElementType;\n PreviousIcon: ElementType;\n RatingIcon: ElementType;\n RatingSelectedIcon: ElementType;\n RatingUnselectingIcon: ElementType;\n SuccessIcon: ElementType;\n SuccessStatusAdornment: ElementType;\n UploadIcon: ElementType;\n UserIcon: ElementType;\n WarningIcon: ElementType;\n WarningStatusAdornment: ElementType;\n}\n\nexport interface SemanticIconProviderProps {\n /**\n * Custom mapping of icon names to components. Overrides default icons if provided.\n */\n iconMap?: Partial<SemanticIconMap>;\n\n /**\n * Child elements that will use the provided icons.\n */\n children: ReactNode;\n}\n\nconst defaultIconMap: SemanticIconMap = {\n ActiveIcon: StepActiveIcon,\n CalendarIcon: CalendarIcon,\n CloseIcon: CloseIcon,\n CollapseGroupIcon: ChevronDownIcon,\n CollapseIcon: ChevronUpIcon,\n CollapseLeftIcon: DoubleChevronLeftIcon,\n CollapseRightIcon: DoubleChevronRightIcon,\n CompletedIcon: SuccessCircleSolidIcon,\n DecreaseIcon: TriangleDownIcon,\n ErrorIcon: ErrorSolidIcon,\n ErrorStatusAdornment: ErrorAdornmentIcon,\n ExpandGroupIcon: ChevronRightIcon,\n ExpandIcon: ChevronDownIcon,\n ExternalIcon: TearOutIcon,\n IncreaseIcon: TriangleUpIcon,\n InfoIcon: InfoSolidIcon,\n InProgressIcon: ProgressInprogressIcon,\n LockedIcon: LockedIcon,\n NextIcon: ChevronRightIcon,\n OverflowIcon: OverflowMenuIcon,\n PendingIcon: StepDefaultIcon,\n PreviousIcon: ChevronLeftIcon,\n RatingIcon: FavoriteIcon,\n RatingSelectedIcon: FavoriteSolidIcon,\n RatingUnselectingIcon: FavoriteStrongIcon,\n SuccessIcon: SuccessCircleSolidIcon,\n SuccessStatusAdornment: SuccessAdornmentIcon,\n UploadIcon,\n UserIcon: UserSolidIcon,\n WarningIcon: WarningSolidIcon,\n WarningStatusAdornment: WarningAdornmentIcon,\n};\n\nconst SemanticIconContext = createContext<SemanticIconMap>(defaultIconMap);\n\nexport const SemanticIconProvider = ({\n iconMap = {},\n children,\n}: SemanticIconProviderProps) => (\n <SemanticIconContext.Provider value={{ ...defaultIconMap, ...iconMap }}>\n {children}\n </SemanticIconContext.Provider>\n);\n\nexport const useIcon = () => {\n const context = useContext(SemanticIconContext);\n return context || defaultIconMap;\n};\n"],"names":[],"mappings":";;;;;;;AAmFA,MAAM,cAAA,GAAkC;AAAA,EACtC,UAAA,EAAY,cAAA;AAAA,EACZ,YAAA;AAAA,EACA,SAAA;AAAA,EACA,iBAAA,EAAmB,eAAA;AAAA,EACnB,YAAA,EAAc,aAAA;AAAA,EACd,gBAAA,EAAkB,qBAAA;AAAA,EAClB,iBAAA,EAAmB,sBAAA;AAAA,EACnB,aAAA,EAAe,sBAAA;AAAA,EACf,YAAA,EAAc,gBAAA;AAAA,EACd,SAAA,EAAW,cAAA;AAAA,EACX,oBAAA,EAAsB,kBAAA;AAAA,EACtB,eAAA,EAAiB,gBAAA;AAAA,EACjB,UAAA,EAAY,eAAA;AAAA,EACZ,YAAA,EAAc,WAAA;AAAA,EACd,YAAA,EAAc,cAAA;AAAA,EACd,QAAA,EAAU,aAAA;AAAA,EACV,cAAA,EAAgB,sBAAA;AAAA,EAChB,UAAA;AAAA,EACA,QAAA,EAAU,gBAAA;AAAA,EACV,YAAA,EAAc,gBAAA;AAAA,EACd,WAAA,EAAa,eAAA;AAAA,EACb,YAAA,EAAc,eAAA;AAAA,EACd,UAAA,EAAY,YAAA;AAAA,EACZ,kBAAA,EAAoB,iBAAA;AAAA,EACpB,qBAAA,EAAuB,kBAAA;AAAA,EACvB,WAAA,EAAa,sBAAA;AAAA,EACb,sBAAA,EAAwB,oBAAA;AAAA,EACxB,UAAA;AAAA,EACA,QAAA,EAAU,aAAA;AAAA,EACV,WAAA,EAAa,gBAAA;AAAA,EACb,sBAAA,EAAwB;AAC1B,CAAA;AAEA,MAAM,mBAAA,GAAsB,cAA+B,cAAc,CAAA;AAElE,MAAM,uBAAuB,CAAC;AAAA,EACnC,UAAU,EAAC;AAAA,EACX;AACF,CAAA,qBACE,GAAA,CAAC,mBAAA,CAAoB,QAAA,EAApB,EAA6B,KAAA,EAAO,EAAE,GAAG,cAAA,EAAgB,GAAG,OAAA,EAAQ,EAClE,QAAA,EACH;AAGK,MAAM,UAAU,MAAM;AAC3B,EAAA,MAAM,OAAA,GAAU,WAAW,mBAAmB,CAAA;AAC9C,EAAA,OAAO,OAAA,IAAW,cAAA;AACpB;;;;"}
@@ -0,0 +1,4 @@
1
+ var css_248z = "/* Class applied to root Tab element */\n.saltTab {\n display: inline-flex;\n align-items: center;\n justify-content: center;\n appearance: none;\n -webkit-appearance: none;\n gap: var(--salt-spacing-100);\n border: none;\n border-left: var(--salt-size-fixed-100) var(--salt-borderStyle-solid) transparent;\n border-right: var(--salt-size-fixed-100) var(--salt-borderStyle-solid) transparent;\n white-space: pre;\n min-height: calc(var(--salt-size-base) + var(--salt-spacing-100));\n min-width: 4em;\n padding: var(--salt-spacing-50) var(--salt-spacing-100);\n position: relative;\n flex-shrink: 0;\n box-sizing: border-box;\n overflow: hidden;\n\n cursor: var(--salt-cursor-hover);\n color: var(--salt-content-primary-foreground);\n font-weight: var(--salt-text-fontWeight);\n font-family: var(--salt-text-fontFamily);\n text-align: var(--salt-text-textAlign);\n letter-spacing: var(--salt-text-letterSpacing);\n line-height: var(--salt-text-lineHeight);\n font-size: var(--salt-text-fontSize);\n max-width: 100%;\n}\n\n@supports selector(:has(*)) {\n .saltTabList:has([data-overflowbutton]) .saltTab {\n /* 100% of the tab list but reserving space for the overflow button */\n max-width: calc(100% - var(--salt-spacing-100) - var(--salt-size-base));\n }\n}\n\n.saltTab::after {\n content: \"\";\n position: absolute;\n left: calc(var(--salt-size-fixed-100) * -1);\n right: calc(var(--salt-size-fixed-100) * -1);\n height: var(--salt-size-indicator);\n}\n\n.saltTabList-bordered .saltTab::after {\n top: 0;\n}\n\n.saltTabList-transparent .saltTab::after {\n bottom: 0;\n}\n\n.saltTab:hover::after,\n.saltTab-focusVisible::after {\n background: var(--salt-navigable-indicator-hover);\n}\n\n.saltTab-disabled:hover::after,\n.saltTab-disabled.saltTab-focusVisible::after {\n background: none;\n}\n\n.saltTab-focusVisible {\n outline: var(--salt-focused-outline);\n}\n\n.saltTabList-bordered .saltTab.saltTab-selected {\n background: var(--saltTabList-activeColor);\n border-left: var(--salt-size-fixed-100) var(--salt-borderStyle-solid) var(--salt-separable-secondary-borderColor);\n border-right: var(--salt-size-fixed-100) var(--salt-borderStyle-solid) var(--salt-separable-secondary-borderColor);\n}\n\n.saltTab.saltTab-selected::after {\n background: var(--salt-navigable-accent-indicator-active);\n}\n\n.saltTab.saltTab-disabled {\n cursor: var(--salt-cursor-disabled);\n color: var(--salt-content-primary-foreground-disabled);\n}\n";
2
+
3
+ export { css_248z as default };
4
+ //# sourceMappingURL=Tab.css.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Tab.css.js","sources":[],"sourcesContent":[],"names":[],"mappings":";;;;"}
@@ -0,0 +1,209 @@
1
+ import { jsx, jsxs, Fragment } from 'react/jsx-runtime';
2
+ import { useComponentCssInjection } from '@salt-ds/styles';
3
+ import { useWindow } from '@salt-ds/window';
4
+ import { clsx } from 'clsx';
5
+ import { forwardRef, useRef, useState, useCallback, useMemo } from 'react';
6
+ import { createPortal } from 'react-dom';
7
+ import { makePrefixer } from '../utils/makePrefixer.js';
8
+ import { useIsomorphicLayoutEffect } from '../utils/useIsomorphicLayoutEffect.js';
9
+ import '../utils/useFloatingUI/useFloatingUI.js';
10
+ import { useForkRef } from '../utils/useForkRef.js';
11
+ import { useId } from '../utils/useId.js';
12
+ import '../salt-provider/SaltProvider.js';
13
+ import '../viewport/ViewportProvider.js';
14
+ import { TabContext } from './internal/contexts/TabContext.js';
15
+ import { useTabs } from './internal/contexts/TabsContext.js';
16
+ import { useRenderedTabWidth } from './internal/overflow/useRenderedTabWidth.js';
17
+ import { getIntrinsicMeasuredWidth } from './internal/overflow/widthMeasurement.js';
18
+ import css_248z from './Tab.css.js';
19
+
20
+ const withBaseName = makePrefixer("saltTab");
21
+ const Tab = forwardRef(
22
+ function Tab2(props, ref) {
23
+ const {
24
+ children,
25
+ className,
26
+ disabled: disabledProp,
27
+ onBlur,
28
+ onMouseDown,
29
+ onFocus,
30
+ onFocusCapture,
31
+ value,
32
+ id: idProp,
33
+ ...rest
34
+ } = props;
35
+ const targetWindow = useWindow();
36
+ useComponentCssInjection({
37
+ testId: "salt-tab",
38
+ css: css_248z,
39
+ window: targetWindow
40
+ });
41
+ const {
42
+ selected,
43
+ activeTab,
44
+ renderMode,
45
+ registerBootstrapTab,
46
+ setBootstrapTabReady,
47
+ registerRenderedTab,
48
+ updateRenderedTab
49
+ } = useTabs();
50
+ const disabled = !!disabledProp;
51
+ const id = useId(idProp);
52
+ const wasMouseDown = useRef(false);
53
+ const [focusVisible, setFocusVisible] = useState(false);
54
+ const [focused, setFocused] = useState(false);
55
+ const [hostElement, setHostElement] = useState(null);
56
+ const markerRef = useRef(null);
57
+ const tabRootRef = useRef(null);
58
+ const handleFocusCapture = (event) => {
59
+ onFocusCapture == null ? void 0 : onFocusCapture(event);
60
+ if (id) {
61
+ activeTab.current = { value, id };
62
+ }
63
+ };
64
+ const handleFocus = (event) => {
65
+ onFocus == null ? void 0 : onFocus(event);
66
+ setFocused(true);
67
+ if (!wasMouseDown.current && event.target.getAttribute("role") === "tab") {
68
+ setFocusVisible(true);
69
+ }
70
+ wasMouseDown.current = false;
71
+ };
72
+ const handleBlur = (event) => {
73
+ onBlur == null ? void 0 : onBlur(event);
74
+ setFocused(false);
75
+ setFocusVisible(false);
76
+ };
77
+ const handleMouseDown = (event) => {
78
+ onMouseDown == null ? void 0 : onMouseDown(event);
79
+ if (id) {
80
+ activeTab.current = { value, id };
81
+ }
82
+ wasMouseDown.current = true;
83
+ };
84
+ const [actionIds, setActionIds] = useState(() => /* @__PURE__ */ new Set());
85
+ const registerAction = useCallback((id2) => {
86
+ setActionIds((old) => {
87
+ if (old.has(id2)) {
88
+ return old;
89
+ }
90
+ const next = new Set(old);
91
+ next.add(id2);
92
+ return next;
93
+ });
94
+ return () => {
95
+ setActionIds((old) => {
96
+ if (!old.has(id2)) {
97
+ return old;
98
+ }
99
+ const next = new Set(old);
100
+ next.delete(id2);
101
+ return next;
102
+ });
103
+ };
104
+ }, []);
105
+ const actions = useMemo(() => Array.from(actionIds), [actionIds]);
106
+ const context = useMemo(
107
+ () => ({
108
+ tabId: id,
109
+ selected: selected === value,
110
+ focused,
111
+ value,
112
+ disabled,
113
+ actions,
114
+ registerAction
115
+ }),
116
+ [id, selected, value, focused, disabled, actions, registerAction]
117
+ );
118
+ useIsomorphicLayoutEffect(() => {
119
+ const doc = targetWindow == null ? void 0 : targetWindow.document;
120
+ if (!doc) {
121
+ return;
122
+ }
123
+ const host = doc.createElement("div");
124
+ host.dataset.tabHost = value;
125
+ host.role = "presentation";
126
+ host.style.display = "contents";
127
+ setHostElement(host);
128
+ return () => {
129
+ host.remove();
130
+ };
131
+ }, [targetWindow, value]);
132
+ useIsomorphicLayoutEffect(() => {
133
+ if (renderMode !== "inline") {
134
+ return;
135
+ }
136
+ return registerBootstrapTab(value);
137
+ }, [registerBootstrapTab, renderMode, value]);
138
+ useIsomorphicLayoutEffect(() => {
139
+ setBootstrapTabReady(value, hostElement != null);
140
+ return () => {
141
+ setBootstrapTabReady(value, false);
142
+ };
143
+ }, [hostElement, setBootstrapTabReady, value]);
144
+ useIsomorphicLayoutEffect(() => {
145
+ if (!hostElement || !id) {
146
+ return;
147
+ }
148
+ return registerRenderedTab({
149
+ host: hostElement,
150
+ id,
151
+ marker: markerRef.current,
152
+ root: tabRootRef.current,
153
+ trigger: null,
154
+ value,
155
+ width: getIntrinsicMeasuredWidth(tabRootRef.current)
156
+ });
157
+ }, [hostElement, id, registerRenderedTab, value]);
158
+ useIsomorphicLayoutEffect(() => {
159
+ const updates = {
160
+ marker: markerRef.current,
161
+ root: tabRootRef.current
162
+ };
163
+ if (renderMode === "inline") {
164
+ updates.width = getIntrinsicMeasuredWidth(tabRootRef.current);
165
+ }
166
+ updateRenderedTab(value, updates);
167
+ }, [renderMode, updateRenderedTab, value]);
168
+ useRenderedTabWidth({
169
+ hostElement,
170
+ renderMode,
171
+ tabRootRef,
172
+ targetWindow,
173
+ updateRenderedTab,
174
+ value
175
+ });
176
+ const handleTabRootRef = useForkRef(tabRootRef, ref);
177
+ const tabMarkup = /* @__PURE__ */ jsx(TabContext.Provider, { value: context, children: /* @__PURE__ */ jsx(
178
+ "div",
179
+ {
180
+ className: clsx(
181
+ withBaseName(),
182
+ {
183
+ [withBaseName("selected")]: selected === value,
184
+ [withBaseName("disabled")]: disabled,
185
+ [withBaseName("focusVisible")]: focusVisible
186
+ },
187
+ className
188
+ ),
189
+ "data-overflowitem": "true",
190
+ "data-value": value,
191
+ ref: handleTabRootRef,
192
+ onMouseDown: handleMouseDown,
193
+ onFocusCapture: handleFocusCapture,
194
+ onFocus: handleFocus,
195
+ onBlur: handleBlur,
196
+ role: "presentation",
197
+ ...rest,
198
+ children
199
+ }
200
+ ) });
201
+ return /* @__PURE__ */ jsxs(Fragment, { children: [
202
+ /* @__PURE__ */ jsx("span", { role: "presentation", "data-tab-marker": "", hidden: true, ref: markerRef }),
203
+ renderMode === "inline" ? tabMarkup : hostElement ? createPortal(tabMarkup, hostElement) : null
204
+ ] });
205
+ }
206
+ );
207
+
208
+ export { Tab };
209
+ //# sourceMappingURL=Tab.js.map