@lumx/react 3.20.1-alpha.20 → 3.20.1-alpha.22

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 (697) hide show
  1. package/_internal/Falsy.d.ts +7 -0
  2. package/_internal/index.js +236 -0
  3. package/_internal/index.js.map +1 -0
  4. package/index.d.ts +352 -7
  5. package/index.js +14486 -65
  6. package/index.js.map +1 -1
  7. package/package.json +11 -4
  8. package/src/components/alert-dialog/AlertDialog.stories.tsx +127 -0
  9. package/src/components/alert-dialog/AlertDialog.test.tsx +34 -0
  10. package/src/components/alert-dialog/AlertDialog.tsx +189 -0
  11. package/src/components/alert-dialog/index.ts +1 -0
  12. package/src/components/autocomplete/Autocomplete.stories.tsx +75 -0
  13. package/src/components/autocomplete/Autocomplete.test.tsx +99 -0
  14. package/src/components/autocomplete/Autocomplete.tsx +294 -0
  15. package/src/components/autocomplete/AutocompleteMultiple.stories.tsx +167 -0
  16. package/src/components/autocomplete/AutocompleteMultiple.test.tsx +61 -0
  17. package/src/components/autocomplete/AutocompleteMultiple.tsx +156 -0
  18. package/src/components/autocomplete/__mockData__/index.ts +39 -0
  19. package/src/components/autocomplete/index.ts +2 -0
  20. package/src/components/avatar/Avatar.stories.tsx +119 -0
  21. package/src/components/avatar/Avatar.test.tsx +32 -0
  22. package/src/components/avatar/Avatar.tsx +114 -0
  23. package/src/components/avatar/index.ts +1 -0
  24. package/src/components/badge/Badge.stories.tsx +69 -0
  25. package/src/components/badge/Badge.test.tsx +45 -0
  26. package/src/components/badge/Badge.tsx +58 -0
  27. package/src/components/badge/BadgeWrapper.stories.tsx +75 -0
  28. package/src/components/badge/BadgeWrapper.test.tsx +47 -0
  29. package/src/components/badge/BadgeWrapper.tsx +37 -0
  30. package/src/components/badge/index.ts +2 -0
  31. package/src/components/button/Button.stories.tsx +355 -0
  32. package/src/components/button/Button.test.tsx +125 -0
  33. package/src/components/button/Button.tsx +101 -0
  34. package/src/components/button/ButtonGroup.stories.tsx +52 -0
  35. package/src/components/button/ButtonGroup.test.tsx +41 -0
  36. package/src/components/button/ButtonGroup.tsx +50 -0
  37. package/src/components/button/ButtonRoot.tsx +163 -0
  38. package/src/components/button/IconButton.stories.tsx +52 -0
  39. package/src/components/button/IconButton.test.tsx +66 -0
  40. package/src/components/button/IconButton.tsx +91 -0
  41. package/src/components/button/index.ts +4 -0
  42. package/src/components/checkbox/Checkbox.stories.tsx +64 -0
  43. package/src/components/checkbox/Checkbox.test.tsx +154 -0
  44. package/src/components/checkbox/Checkbox.tsx +164 -0
  45. package/src/components/checkbox/index.ts +1 -0
  46. package/src/components/chip/Chip.stories.tsx +164 -0
  47. package/src/components/chip/Chip.test.tsx +243 -0
  48. package/src/components/chip/Chip.tsx +168 -0
  49. package/src/components/chip/ChipGroup.stories.tsx +47 -0
  50. package/src/components/chip/ChipGroup.test.tsx +37 -0
  51. package/src/components/chip/ChipGroup.tsx +60 -0
  52. package/src/components/chip/index.ts +2 -0
  53. package/src/components/comment-block/CommentBlock.stories.tsx +91 -0
  54. package/src/components/comment-block/CommentBlock.test.tsx +28 -0
  55. package/src/components/comment-block/CommentBlock.tsx +172 -0
  56. package/src/components/comment-block/index.ts +1 -0
  57. package/src/components/date-picker/DatePicker.stories.tsx +118 -0
  58. package/src/components/date-picker/DatePicker.test.tsx +34 -0
  59. package/src/components/date-picker/DatePicker.tsx +52 -0
  60. package/src/components/date-picker/DatePickerControlled.test.tsx +91 -0
  61. package/src/components/date-picker/DatePickerControlled.tsx +259 -0
  62. package/src/components/date-picker/DatePickerField.stories.tsx +100 -0
  63. package/src/components/date-picker/DatePickerField.test.tsx +70 -0
  64. package/src/components/date-picker/DatePickerField.tsx +135 -0
  65. package/src/components/date-picker/constants.ts +11 -0
  66. package/src/components/date-picker/index.ts +4 -0
  67. package/src/components/date-picker/types.ts +28 -0
  68. package/src/components/dialog/Dialog.stories.tsx +318 -0
  69. package/src/components/dialog/Dialog.test.tsx +97 -0
  70. package/src/components/dialog/Dialog.tsx +288 -0
  71. package/src/components/dialog/index.ts +1 -0
  72. package/src/components/divider/Divider.test.tsx +51 -0
  73. package/src/components/divider/Divider.tsx +50 -0
  74. package/src/components/divider/index.ts +1 -0
  75. package/src/components/drag-handle/DragHandle.test.tsx +36 -0
  76. package/src/components/drag-handle/DragHandle.tsx +51 -0
  77. package/src/components/drag-handle/index.ts +1 -0
  78. package/src/components/dropdown/Dropdown.stories.tsx +31 -0
  79. package/src/components/dropdown/Dropdown.test.tsx +66 -0
  80. package/src/components/dropdown/Dropdown.tsx +186 -0
  81. package/src/components/dropdown/index.ts +1 -0
  82. package/src/components/expansion-panel/ExpansionPanel.stories.tsx +91 -0
  83. package/src/components/expansion-panel/ExpansionPanel.test.tsx +185 -0
  84. package/src/components/expansion-panel/ExpansionPanel.tsx +195 -0
  85. package/src/components/expansion-panel/index.ts +1 -0
  86. package/src/components/flag/Flag.stories.tsx +48 -0
  87. package/src/components/flag/Flag.test.tsx +62 -0
  88. package/src/components/flag/Flag.tsx +52 -0
  89. package/src/components/flag/index.ts +1 -0
  90. package/src/components/flex-box/FlexBox.stories.tsx +156 -0
  91. package/src/components/flex-box/FlexBox.test.tsx +24 -0
  92. package/src/components/flex-box/FlexBox.tsx +100 -0
  93. package/src/components/flex-box/index.ts +1 -0
  94. package/src/components/generic-block/GenericBlock.stories.jsx +128 -0
  95. package/src/components/generic-block/GenericBlock.test.tsx +156 -0
  96. package/src/components/generic-block/GenericBlock.tsx +225 -0
  97. package/src/components/generic-block/constants.ts +9 -0
  98. package/src/components/generic-block/index.ts +2 -0
  99. package/src/components/grid/Grid.tsx +83 -0
  100. package/src/components/grid/GridItem.tsx +55 -0
  101. package/src/components/grid/index.ts +2 -0
  102. package/src/components/grid-column/GridColumn.stories.tsx +45 -0
  103. package/src/components/grid-column/GridColumn.test.jsx +56 -0
  104. package/src/components/grid-column/GridColumn.tsx +83 -0
  105. package/src/components/grid-column/index.ts +1 -0
  106. package/src/components/heading/Heading.stories.tsx +69 -0
  107. package/src/components/heading/Heading.test.tsx +82 -0
  108. package/src/components/heading/Heading.tsx +65 -0
  109. package/src/components/heading/HeadingLevelProvider.tsx +30 -0
  110. package/src/components/heading/constants.ts +16 -0
  111. package/src/components/heading/context.tsx +13 -0
  112. package/src/components/heading/index.ts +3 -0
  113. package/src/components/heading/useHeadingLevel.tsx +8 -0
  114. package/src/components/icon/Icon.stories.tsx +73 -0
  115. package/src/components/icon/Icon.test.tsx +125 -0
  116. package/src/components/icon/Icon.tsx +133 -0
  117. package/src/components/icon/index.ts +1 -0
  118. package/src/components/image-block/ImageBlock.stories.tsx +117 -0
  119. package/src/components/image-block/ImageBlock.test.tsx +58 -0
  120. package/src/components/image-block/ImageBlock.tsx +142 -0
  121. package/src/components/image-block/ImageCaption.tsx +115 -0
  122. package/src/components/image-block/index.ts +1 -0
  123. package/src/components/image-lightbox/ImageLightbox.stories.tsx +160 -0
  124. package/src/components/image-lightbox/ImageLightbox.test.tsx +254 -0
  125. package/src/components/image-lightbox/ImageLightbox.tsx +90 -0
  126. package/src/components/image-lightbox/constants.ts +11 -0
  127. package/src/components/image-lightbox/index.ts +2 -0
  128. package/src/components/image-lightbox/internal/ImageSlide.tsx +107 -0
  129. package/src/components/image-lightbox/internal/ImageSlideshow.tsx +164 -0
  130. package/src/components/image-lightbox/internal/useAnimateScroll.ts +55 -0
  131. package/src/components/image-lightbox/internal/usePointerZoom.ts +148 -0
  132. package/src/components/image-lightbox/types.ts +48 -0
  133. package/src/components/image-lightbox/useImageLightbox.tsx +141 -0
  134. package/src/components/inline-list/InlineList.stories.tsx +77 -0
  135. package/src/components/inline-list/InlineList.test.tsx +52 -0
  136. package/src/components/inline-list/InlineList.tsx +96 -0
  137. package/src/components/inline-list/index.ts +1 -0
  138. package/src/components/input-helper/InputHelper.stories.tsx +35 -0
  139. package/src/components/input-helper/InputHelper.test.tsx +55 -0
  140. package/src/components/input-helper/InputHelper.tsx +65 -0
  141. package/src/components/input-helper/constants.ts +11 -0
  142. package/src/components/input-helper/index.ts +1 -0
  143. package/src/components/input-label/InputLabel.stories.tsx +45 -0
  144. package/src/components/input-label/InputLabel.test.tsx +59 -0
  145. package/src/components/input-label/InputLabel.tsx +69 -0
  146. package/src/components/input-label/index.ts +1 -0
  147. package/src/components/lightbox/Lightbox.stories.tsx +101 -0
  148. package/src/components/lightbox/Lightbox.test.tsx +53 -0
  149. package/src/components/lightbox/Lightbox.tsx +180 -0
  150. package/src/components/lightbox/index.ts +1 -0
  151. package/src/components/link/Link.stories.tsx +195 -0
  152. package/src/components/link/Link.test.tsx +127 -0
  153. package/src/components/link/Link.tsx +106 -0
  154. package/src/components/link/index.ts +1 -0
  155. package/src/components/link-preview/LinkPreview.stories.tsx +61 -0
  156. package/src/components/link-preview/LinkPreview.test.tsx +105 -0
  157. package/src/components/link-preview/LinkPreview.tsx +158 -0
  158. package/src/components/link-preview/index.ts +1 -0
  159. package/src/components/list/List.stories.tsx +116 -0
  160. package/src/components/list/List.test.tsx +18 -0
  161. package/src/components/list/List.tsx +104 -0
  162. package/src/components/list/ListDivider.stories.tsx +12 -0
  163. package/src/components/list/ListDivider.test.tsx +22 -0
  164. package/src/components/list/ListDivider.tsx +35 -0
  165. package/src/components/list/ListItem.stories.tsx +66 -0
  166. package/src/components/list/ListItem.test.tsx +93 -0
  167. package/src/components/list/ListItem.tsx +157 -0
  168. package/src/components/list/ListSubheader.stories.tsx +11 -0
  169. package/src/components/list/ListSubheader.test.tsx +21 -0
  170. package/src/components/list/ListSubheader.tsx +44 -0
  171. package/src/components/list/index.ts +4 -0
  172. package/src/components/list/useInteractiveList.tsx +202 -0
  173. package/src/components/message/Message.stories.tsx +72 -0
  174. package/src/components/message/Message.test.tsx +76 -0
  175. package/src/components/message/Message.tsx +100 -0
  176. package/src/components/message/index.ts +1 -0
  177. package/src/components/mosaic/Mosaic.stories.tsx +89 -0
  178. package/src/components/mosaic/Mosaic.test.tsx +77 -0
  179. package/src/components/mosaic/Mosaic.tsx +98 -0
  180. package/src/components/mosaic/index.ts +1 -0
  181. package/src/components/navigation/Navigation.stories.tsx +234 -0
  182. package/src/components/navigation/Navigation.test.tsx +65 -0
  183. package/src/components/navigation/Navigation.tsx +79 -0
  184. package/src/components/navigation/NavigationItem.test.tsx +35 -0
  185. package/src/components/navigation/NavigationItem.tsx +82 -0
  186. package/src/components/navigation/NavigationSection.test.tsx +137 -0
  187. package/src/components/navigation/NavigationSection.tsx +108 -0
  188. package/src/components/navigation/context.tsx +7 -0
  189. package/src/components/navigation/index.ts +1 -0
  190. package/src/components/notification/Notification.test.tsx +94 -0
  191. package/src/components/notification/Notification.tsx +138 -0
  192. package/src/components/notification/Notifications.stories.tsx +92 -0
  193. package/src/components/notification/constants.ts +28 -0
  194. package/src/components/notification/index.ts +1 -0
  195. package/src/components/popover/Popover.stories.tsx +264 -0
  196. package/src/components/popover/Popover.test.tsx +62 -0
  197. package/src/components/popover/Popover.tsx +201 -0
  198. package/src/components/popover/constants.ts +62 -0
  199. package/src/components/popover/index.ts +3 -0
  200. package/src/components/popover/usePopoverStyle.tsx +184 -0
  201. package/src/components/popover/useRestoreFocusOnClose.tsx +47 -0
  202. package/src/components/popover-dialog/PopoverDialog.stories.tsx +64 -0
  203. package/src/components/popover-dialog/PopoverDialog.test.tsx +139 -0
  204. package/src/components/popover-dialog/PopoverDialog.tsx +74 -0
  205. package/src/components/popover-dialog/index.tsx +1 -0
  206. package/src/components/post-block/PostBlock.test.tsx +27 -0
  207. package/src/components/post-block/PostBlock.tsx +122 -0
  208. package/src/components/post-block/index.ts +1 -0
  209. package/src/components/progress/Progress.tsx +68 -0
  210. package/src/components/progress/ProgressCircular.stories.tsx +43 -0
  211. package/src/components/progress/ProgressCircular.test.tsx +46 -0
  212. package/src/components/progress/ProgressCircular.tsx +82 -0
  213. package/src/components/progress/ProgressLinear.stories.tsx +12 -0
  214. package/src/components/progress/ProgressLinear.test.tsx +30 -0
  215. package/src/components/progress/ProgressLinear.tsx +50 -0
  216. package/src/components/progress/index.ts +3 -0
  217. package/src/components/progress-tracker/ProgressTracker.stories.tsx +145 -0
  218. package/src/components/progress-tracker/ProgressTracker.test.tsx +42 -0
  219. package/src/components/progress-tracker/ProgressTracker.tsx +87 -0
  220. package/src/components/progress-tracker/ProgressTrackerProvider.test.tsx +65 -0
  221. package/src/components/progress-tracker/ProgressTrackerProvider.tsx +67 -0
  222. package/src/components/progress-tracker/ProgressTrackerStep.test.tsx +36 -0
  223. package/src/components/progress-tracker/ProgressTrackerStep.tsx +159 -0
  224. package/src/components/progress-tracker/ProgressTrackerStepPanel.test.tsx +34 -0
  225. package/src/components/progress-tracker/ProgressTrackerStepPanel.tsx +67 -0
  226. package/src/components/progress-tracker/index.ts +4 -0
  227. package/src/components/radio-button/RadioButton.stories.tsx +71 -0
  228. package/src/components/radio-button/RadioButton.test.tsx +143 -0
  229. package/src/components/radio-button/RadioButton.tsx +143 -0
  230. package/src/components/radio-button/RadioGroup.stories.tsx +39 -0
  231. package/src/components/radio-button/RadioGroup.test.tsx +29 -0
  232. package/src/components/radio-button/RadioGroup.tsx +44 -0
  233. package/src/components/radio-button/index.ts +2 -0
  234. package/src/components/select/Select.stories.tsx +385 -0
  235. package/src/components/select/Select.test.tsx +199 -0
  236. package/src/components/select/Select.tsx +199 -0
  237. package/src/components/select/SelectMultiple.stories.tsx +315 -0
  238. package/src/components/select/SelectMultiple.test.tsx +213 -0
  239. package/src/components/select/SelectMultiple.tsx +206 -0
  240. package/src/components/select/WithSelectContext.tsx +147 -0
  241. package/src/components/select/constants.ts +55 -0
  242. package/src/components/select/index.ts +2 -0
  243. package/src/components/side-navigation/SideNavigation.stories.tsx +191 -0
  244. package/src/components/side-navigation/SideNavigation.test.tsx +37 -0
  245. package/src/components/side-navigation/SideNavigation.tsx +52 -0
  246. package/src/components/side-navigation/SideNavigationItem.stories.tsx +133 -0
  247. package/src/components/side-navigation/SideNavigationItem.test.tsx +136 -0
  248. package/src/components/side-navigation/SideNavigationItem.tsx +165 -0
  249. package/src/components/side-navigation/index.ts +2 -0
  250. package/src/components/skeleton/SkeletonCircle.stories.tsx +41 -0
  251. package/src/components/skeleton/SkeletonCircle.test.tsx +27 -0
  252. package/src/components/skeleton/SkeletonCircle.tsx +52 -0
  253. package/src/components/skeleton/SkeletonRectangle.stories.tsx +82 -0
  254. package/src/components/skeleton/SkeletonRectangle.test.tsx +27 -0
  255. package/src/components/skeleton/SkeletonRectangle.tsx +88 -0
  256. package/src/components/skeleton/SkeletonTypography.stories.tsx +19 -0
  257. package/src/components/skeleton/SkeletonTypography.test.tsx +27 -0
  258. package/src/components/skeleton/SkeletonTypography.tsx +59 -0
  259. package/src/components/skeleton/index.ts +3 -0
  260. package/src/components/slider/Slider.stories.tsx +45 -0
  261. package/src/components/slider/Slider.test.tsx +29 -0
  262. package/src/components/slider/Slider.tsx +299 -0
  263. package/src/components/slider/index.ts +2 -0
  264. package/src/components/slideshow/Slides.tsx +130 -0
  265. package/src/components/slideshow/Slideshow.stories.tsx +179 -0
  266. package/src/components/slideshow/Slideshow.test.tsx +35 -0
  267. package/src/components/slideshow/Slideshow.tsx +173 -0
  268. package/src/components/slideshow/SlideshowControls.stories.tsx +100 -0
  269. package/src/components/slideshow/SlideshowControls.tsx +243 -0
  270. package/src/components/slideshow/SlideshowItem.tsx +44 -0
  271. package/src/components/slideshow/SlideshowItemGroup.tsx +60 -0
  272. package/src/components/slideshow/constants.ts +24 -0
  273. package/src/components/slideshow/index.ts +4 -0
  274. package/src/components/slideshow/useKeyNavigate.ts +28 -0
  275. package/src/components/slideshow/usePaginationVisibleRange.ts +37 -0
  276. package/src/components/slideshow/useSlideFocusManagement.tsx +92 -0
  277. package/src/components/slideshow/useSwipeNavigate.ts +18 -0
  278. package/src/components/switch/Switch.stories.tsx +49 -0
  279. package/src/components/switch/Switch.test.tsx +144 -0
  280. package/src/components/switch/Switch.tsx +145 -0
  281. package/src/components/switch/index.ts +1 -0
  282. package/src/components/table/Table.test.tsx +29 -0
  283. package/src/components/table/Table.tsx +59 -0
  284. package/src/components/table/TableBody.test.tsx +30 -0
  285. package/src/components/table/TableBody.tsx +42 -0
  286. package/src/components/table/TableCell.test.tsx +72 -0
  287. package/src/components/table/TableCell.tsx +130 -0
  288. package/src/components/table/TableHeader.test.tsx +30 -0
  289. package/src/components/table/TableHeader.tsx +48 -0
  290. package/src/components/table/TableRow.test.tsx +40 -0
  291. package/src/components/table/TableRow.tsx +71 -0
  292. package/src/components/table/index.ts +5 -0
  293. package/src/components/tabs/Tab.test.tsx +50 -0
  294. package/src/components/tabs/Tab.tsx +133 -0
  295. package/src/components/tabs/TabList.test.tsx +49 -0
  296. package/src/components/tabs/TabList.tsx +94 -0
  297. package/src/components/tabs/TabPanel.test.tsx +37 -0
  298. package/src/components/tabs/TabPanel.tsx +67 -0
  299. package/src/components/tabs/TabProvider.test.tsx +161 -0
  300. package/src/components/tabs/TabProvider.tsx +67 -0
  301. package/src/components/tabs/Tabs.stories.tsx +170 -0
  302. package/src/components/tabs/index.ts +4 -0
  303. package/src/components/tabs/state.ts +114 -0
  304. package/src/components/tabs/test-utils.ts +39 -0
  305. package/src/components/text/Text.stories.tsx +177 -0
  306. package/src/components/text/Text.test.tsx +92 -0
  307. package/src/components/text/Text.tsx +139 -0
  308. package/src/components/text/index.ts +1 -0
  309. package/src/components/text-field/TextField.stories.tsx +180 -0
  310. package/src/components/text-field/TextField.test.tsx +298 -0
  311. package/src/components/text-field/TextField.tsx +493 -0
  312. package/src/components/text-field/index.ts +1 -0
  313. package/src/components/thumbnail/Thumbnail.stories.tsx +448 -0
  314. package/src/components/thumbnail/Thumbnail.test.tsx +88 -0
  315. package/src/components/thumbnail/Thumbnail.tsx +248 -0
  316. package/src/components/thumbnail/index.ts +3 -0
  317. package/src/components/thumbnail/types.ts +48 -0
  318. package/src/components/thumbnail/useFocusPointStyle.test.ts +92 -0
  319. package/src/components/thumbnail/useFocusPointStyle.tsx +107 -0
  320. package/src/components/thumbnail/useImageLoad.ts +40 -0
  321. package/src/components/toolbar/Toolbar.tsx +68 -0
  322. package/src/components/toolbar/index.ts +1 -0
  323. package/src/components/tooltip/Tooltip.stories.tsx +118 -0
  324. package/src/components/tooltip/Tooltip.test.tsx +417 -0
  325. package/src/components/tooltip/Tooltip.tsx +165 -0
  326. package/src/components/tooltip/constants.ts +8 -0
  327. package/src/components/tooltip/context.tsx +17 -0
  328. package/src/components/tooltip/index.ts +1 -0
  329. package/src/components/tooltip/useInjectTooltipRef.tsx +55 -0
  330. package/src/components/tooltip/useTooltipOpen.tsx +143 -0
  331. package/src/components/uploader/Uploader.stories.tsx +109 -0
  332. package/src/components/uploader/Uploader.test.tsx +146 -0
  333. package/src/components/uploader/Uploader.tsx +177 -0
  334. package/src/components/uploader/index.ts +1 -0
  335. package/src/components/user-block/UserBlock.stories.tsx +133 -0
  336. package/src/components/user-block/UserBlock.test.tsx +106 -0
  337. package/src/components/user-block/UserBlock.tsx +193 -0
  338. package/src/components/user-block/index.ts +1 -0
  339. package/src/constants.ts +27 -0
  340. package/src/hooks/useBooleanState.tsx +13 -0
  341. package/src/hooks/useCallbackOnEscape.ts +34 -0
  342. package/src/hooks/useChipGroupNavigation.tsx +75 -0
  343. package/src/hooks/useClickAway.tsx +48 -0
  344. package/src/hooks/useDisableBodyScroll.ts +28 -0
  345. package/src/hooks/useEventCallback.tsx +17 -0
  346. package/src/hooks/useFocus.tsx +21 -0
  347. package/src/hooks/useFocusTrap.ts +93 -0
  348. package/src/hooks/useFocusWithin.ts +33 -0
  349. package/src/hooks/useId.test.tsx +22 -0
  350. package/src/hooks/useId.ts +15 -0
  351. package/src/hooks/useImageSize.ts +17 -0
  352. package/src/hooks/useInfiniteScroll.tsx +60 -0
  353. package/src/hooks/useIntersectionObserver.tsx +43 -0
  354. package/src/hooks/useInterval.tsx +31 -0
  355. package/src/hooks/useKeyboardListNavigation.tsx +204 -0
  356. package/src/hooks/useListenFocus.tsx +26 -0
  357. package/src/hooks/useOverflowTooltipLabel.tsx +32 -0
  358. package/src/hooks/usePopper.ts +12 -0
  359. package/src/hooks/usePreviousValue.ts +12 -0
  360. package/src/hooks/useRovingTabIndex.tsx +90 -0
  361. package/src/hooks/useSizeOnWindowResize.ts +30 -0
  362. package/src/hooks/useSlideshowControls.ts +246 -0
  363. package/src/hooks/useStopPropagation.ts +21 -0
  364. package/src/hooks/useTransitionVisibility.ts +48 -0
  365. package/src/index.ts +63 -0
  366. package/src/stories/controls/color.ts +7 -0
  367. package/src/stories/controls/element.ts +6 -0
  368. package/src/stories/controls/focusPoint.ts +1 -0
  369. package/src/stories/controls/icons.ts +126 -0
  370. package/src/stories/controls/image.ts +84 -0
  371. package/src/stories/controls/selectArgType.ts +8 -0
  372. package/src/stories/controls/theme.ts +3 -0
  373. package/src/stories/controls/typography.ts +5 -0
  374. package/src/stories/controls/withUndefined.ts +1 -0
  375. package/src/stories/decorators/withChromaticForceScreenSize.tsx +7 -0
  376. package/src/stories/decorators/withCombinations.tsx +132 -0
  377. package/src/stories/decorators/withNestedProps.tsx +22 -0
  378. package/src/stories/decorators/withResizableBox.tsx +21 -0
  379. package/src/stories/decorators/withThemedBackground.tsx +16 -0
  380. package/src/stories/decorators/withValueOnChange.tsx +18 -0
  381. package/src/stories/decorators/withWrapper.tsx +19 -0
  382. package/src/stories/generated/Autocomplete/Demos.stories.tsx +7 -0
  383. package/src/stories/generated/Avatar/Demos.stories.tsx +7 -0
  384. package/src/stories/generated/Badge/Demos.stories.tsx +9 -0
  385. package/src/stories/generated/Button/Demos.stories.tsx +11 -0
  386. package/src/stories/generated/Checkbox/Demos.stories.tsx +6 -0
  387. package/src/stories/generated/Chip/Demos.stories.tsx +11 -0
  388. package/src/stories/generated/CommentBlock/Demos.stories.tsx +8 -0
  389. package/src/stories/generated/DatePicker/Demos.stories.tsx +8 -0
  390. package/src/stories/generated/Dialog/Demos.stories.tsx +10 -0
  391. package/src/stories/generated/Divider/Demos.stories.tsx +6 -0
  392. package/src/stories/generated/Dropdown/Demos.stories.tsx +8 -0
  393. package/src/stories/generated/ExpansionPanel/Demos.stories.tsx +9 -0
  394. package/src/stories/generated/Flag/Demos.stories.tsx +6 -0
  395. package/src/stories/generated/GenericBlock/Demos.stories.tsx +8 -0
  396. package/src/stories/generated/Heading/Demos.stories.tsx +6 -0
  397. package/src/stories/generated/Icon/Demos.stories.tsx +8 -0
  398. package/src/stories/generated/ImageBlock/Demos.stories.tsx +9 -0
  399. package/src/stories/generated/ImageLightbox/Demos.stories.tsx +6 -0
  400. package/src/stories/generated/Lightbox/Demos.stories.tsx +6 -0
  401. package/src/stories/generated/LinkPreview/Demos.stories.tsx +7 -0
  402. package/src/stories/generated/List/Demos.stories.tsx +11 -0
  403. package/src/stories/generated/Message/Demos.stories.tsx +11 -0
  404. package/src/stories/generated/Mosaic/Demos.stories.tsx +10 -0
  405. package/src/stories/generated/Notification/Demos.stories.tsx +6 -0
  406. package/src/stories/generated/Popover/Demos.stories.tsx +11 -0
  407. package/src/stories/generated/PopoverDialog/Demos.stories.tsx +6 -0
  408. package/src/stories/generated/PostBlock/Demos.stories.tsx +6 -0
  409. package/src/stories/generated/Progress/Demos.stories.tsx +7 -0
  410. package/src/stories/generated/ProgressTracker/Demos.stories.tsx +9 -0
  411. package/src/stories/generated/RadioButton/Demos.stories.tsx +6 -0
  412. package/src/stories/generated/Select/Demos.stories.tsx +14 -0
  413. package/src/stories/generated/SideNavigation/Demos.stories.tsx +10 -0
  414. package/src/stories/generated/Skeleton/Demos.stories.tsx +9 -0
  415. package/src/stories/generated/Slider/Demos.stories.tsx +9 -0
  416. package/src/stories/generated/Slideshow/Demos.stories.tsx +8 -0
  417. package/src/stories/generated/Switch/Demos.stories.tsx +6 -0
  418. package/src/stories/generated/Table/Demos.stories.tsx +7 -0
  419. package/src/stories/generated/Tabs/Demos.stories.tsx +8 -0
  420. package/src/stories/generated/TextField/Demos.stories.tsx +20 -0
  421. package/src/stories/generated/Thumbnail/Demos.stories.tsx +12 -0
  422. package/src/stories/generated/Toolbar/Demos.stories.tsx +10 -0
  423. package/src/stories/generated/Tooltip/Demos.stories.tsx +8 -0
  424. package/src/stories/generated/Uploader/Demos.stories.tsx +8 -0
  425. package/src/stories/generated/UserBlock/Demos.stories.tsx +11 -0
  426. package/src/stories/utils/CustomLink.tsx +12 -0
  427. package/src/stories/utils/concatPath.tsx +17 -0
  428. package/src/stories/utils/disableArgTypes.ts +3 -0
  429. package/src/stories/utils/initDemoShadowDOMPortal.ts +10 -0
  430. package/src/stories/utils/lorem.ts +59 -0
  431. package/src/stories/utils/theming.tsx +166 -0
  432. package/src/stories/utils/toFlattenProps.ts +28 -0
  433. package/src/stories/utils/withCategory.ts +12 -0
  434. package/src/testing/utils/ThemeSentinel.tsx +10 -0
  435. package/src/testing/utils/commonTestsSuiteRTL.tsx +193 -0
  436. package/src/testing/utils/index.ts +1 -0
  437. package/src/testing/utils/queries.ts +19 -0
  438. package/src/untypped-modules.d.ts +9 -0
  439. package/src/utils/ClickAwayProvider/ClickAwayProvider.stories.jsx +70 -0
  440. package/src/utils/ClickAwayProvider/ClickAwayProvider.tsx +69 -0
  441. package/src/utils/ClickAwayProvider/index.ts +1 -0
  442. package/src/utils/MaterialThemeSwitcher/MaterialThemeSwitcher.tsx +54 -0
  443. package/src/utils/MaterialThemeSwitcher/index.ts +1 -0
  444. package/src/utils/Portal/Portal.test.tsx +31 -0
  445. package/src/utils/Portal/Portal.tsx +33 -0
  446. package/src/utils/Portal/PortalProvider.stories.jsx +22 -0
  447. package/src/utils/Portal/PortalProvider.test.tsx +72 -0
  448. package/src/utils/Portal/PortalProvider.tsx +24 -0
  449. package/src/utils/Portal/index.tsx +2 -0
  450. package/src/utils/browser/DOM/findImage.tsx +3 -0
  451. package/src/utils/browser/DOM/startViewTransition.ts +68 -0
  452. package/src/utils/browser/focus/constants.ts +7 -0
  453. package/src/utils/browser/focus/getFirstAndLastFocusable.test.ts +134 -0
  454. package/src/utils/browser/focus/getFirstAndLastFocusable.ts +21 -0
  455. package/src/utils/browser/focus/getFocusableElements.test.ts +151 -0
  456. package/src/utils/browser/focus/getFocusableElements.ts +7 -0
  457. package/src/utils/browser/isFocusVisible.ts +9 -0
  458. package/src/utils/browser/isHoverNotSupported.test.js +24 -0
  459. package/src/utils/browser/isHoverNotSupported.ts +2 -0
  460. package/src/utils/browser/isReducedMotion.ts +6 -0
  461. package/src/utils/date/addMonthResetDay.test.ts +13 -0
  462. package/src/utils/date/addMonthResetDay.ts +9 -0
  463. package/src/utils/date/formatDayNumber.test.ts +12 -0
  464. package/src/utils/date/formatDayNumber.ts +5 -0
  465. package/src/utils/date/getFirstDayOfWeek.test.ts +20 -0
  466. package/src/utils/date/getFirstDayOfWeek.ts +59 -0
  467. package/src/utils/date/getMonthCalendar.test.ts +127 -0
  468. package/src/utils/date/getMonthCalendar.ts +69 -0
  469. package/src/utils/date/getWeekDays.test.ts +48 -0
  470. package/src/utils/date/getWeekDays.ts +34 -0
  471. package/src/utils/date/getYearDisplayName.test.ts +20 -0
  472. package/src/utils/date/getYearDisplayName.ts +12 -0
  473. package/src/utils/date/isDateValid.test.ts +15 -0
  474. package/src/utils/date/isDateValid.ts +4 -0
  475. package/src/utils/date/isSameDay.test.ts +37 -0
  476. package/src/utils/date/isSameDay.ts +11 -0
  477. package/src/utils/disabled/DisabledStateContext.tsx +29 -0
  478. package/src/utils/disabled/DisabledStateProvider.stories.tsx +92 -0
  479. package/src/utils/disabled/index.ts +2 -0
  480. package/src/utils/disabled/useDisableStateProps.test.tsx +74 -0
  481. package/src/utils/disabled/useDisableStateProps.tsx +37 -0
  482. package/src/utils/function/makeListenerTowerContext.ts +32 -0
  483. package/src/utils/index.ts +7 -0
  484. package/src/utils/locale/getCurrentLocale.ts +4 -0
  485. package/src/utils/locale/parseLocale.test.ts +17 -0
  486. package/src/utils/locale/parseLocale.ts +23 -0
  487. package/src/utils/locale/types.ts +8 -0
  488. package/src/utils/number/clamp.ts +17 -0
  489. package/src/utils/object/isEqual.test.ts +25 -0
  490. package/src/utils/object/isEqual.ts +11 -0
  491. package/src/utils/partitionMulti.test.ts +27 -0
  492. package/{_internal/partitionMulti-4daccdd5.js → src/utils/partitionMulti.ts} +9 -5
  493. package/src/utils/react/OnBeforeUnmount.tsx +20 -0
  494. package/src/utils/react/RawClickable.test.tsx +153 -0
  495. package/src/utils/react/RawClickable.tsx +65 -0
  496. package/src/utils/react/flattenChildren.ts +32 -0
  497. package/src/utils/react/forwardRef.ts +11 -0
  498. package/src/utils/react/forwardRefPolymorphic.ts +9 -0
  499. package/src/utils/react/mergeRefs.ts +33 -0
  500. package/src/utils/react/renderLink.tsx +17 -0
  501. package/src/utils/react/skipRender.tsx +18 -0
  502. package/src/utils/react/unref.ts +7 -0
  503. package/src/utils/react/wrapChildrenIconWithSpaces.test.tsx +37 -0
  504. package/src/utils/react/wrapChildrenIconWithSpaces.tsx +22 -0
  505. package/src/utils/theme/ThemeContext.ts +16 -0
  506. package/src/utils/theme/invertTheme.ts +4 -0
  507. package/src/utils/type/Comp.ts +14 -0
  508. package/src/utils/type/ComponentRef.ts +16 -0
  509. package/src/utils/type/HasAriaDisabled.ts +6 -0
  510. package/src/utils/type/HasPolymorphicAs.ts +6 -0
  511. package/src/utils/type/HasRequiredLinkHref.ts +1 -0
  512. package/src/utils/type/MaybeElementOrRef.ts +6 -0
  513. package/src/utils/type/index.ts +9 -0
  514. package/src/utils/type/isComponent.ts +33 -0
  515. package/src/utils/type/isComponentType.ts +9 -0
  516. package/utils/index.d.ts +1 -1
  517. package/utils/index.js +1 -3
  518. package/utils/index.js.map +1 -1
  519. package/_internal/Button-1f227024.js +0 -98
  520. package/_internal/Button-1f227024.js.map +0 -1
  521. package/_internal/ButtonRoot-823f3e9c.js +0 -117
  522. package/_internal/ButtonRoot-823f3e9c.js.map +0 -1
  523. package/_internal/Chip-a34f6905.js +0 -146
  524. package/_internal/Chip-a34f6905.js.map +0 -1
  525. package/_internal/ClickAwayProvider-7093ba23.js +0 -95
  526. package/_internal/ClickAwayProvider-7093ba23.js.map +0 -1
  527. package/_internal/DisabledStateContext-ea04260d.js +0 -29
  528. package/_internal/DisabledStateContext-ea04260d.js.map +0 -1
  529. package/_internal/HeadingLevelProvider-ebdcb0c7.js +0 -61
  530. package/_internal/HeadingLevelProvider-ebdcb0c7.js.map +0 -1
  531. package/_internal/IconButton-f4df224c.js +0 -77
  532. package/_internal/IconButton-f4df224c.js.map +0 -1
  533. package/_internal/ImageCaption-8134a3aa.js +0 -75
  534. package/_internal/ImageCaption-8134a3aa.js.map +0 -1
  535. package/_internal/List-54237e0e.js +0 -793
  536. package/_internal/List-54237e0e.js.map +0 -1
  537. package/_internal/PopoverDialog-e0967e5f.js +0 -655
  538. package/_internal/PopoverDialog-e0967e5f.js.map +0 -1
  539. package/_internal/Portal-3f86608e.js +0 -45
  540. package/_internal/Portal-3f86608e.js.map +0 -1
  541. package/_internal/RawClickable-2c2b6a89.js +0 -52
  542. package/_internal/RawClickable-2c2b6a89.js.map +0 -1
  543. package/_internal/Slides-b7a67f32.js +0 -678
  544. package/_internal/Slides-b7a67f32.js.map +0 -1
  545. package/_internal/ThemeContext-3181f000.js +0 -14
  546. package/_internal/ThemeContext-3181f000.js.map +0 -1
  547. package/_internal/Thumbnail-b5dea0af.js +0 -313
  548. package/_internal/Thumbnail-b5dea0af.js.map +0 -1
  549. package/_internal/components/alert-dialog-b284b191.js +0 -162
  550. package/_internal/components/alert-dialog-b284b191.js.map +0 -1
  551. package/_internal/components/autocomplete-8d3f37ea.js +0 -261
  552. package/_internal/components/autocomplete-8d3f37ea.js.map +0 -1
  553. package/_internal/components/avatar-5fc70e00.js +0 -83
  554. package/_internal/components/avatar-5fc70e00.js.map +0 -1
  555. package/_internal/components/badge-8390e590.js +0 -81
  556. package/_internal/components/badge-8390e590.js.map +0 -1
  557. package/_internal/components/button-e3c7f2eb.js +0 -47
  558. package/_internal/components/button-e3c7f2eb.js.map +0 -1
  559. package/_internal/components/checkbox-d1ca9748.js +0 -140
  560. package/_internal/components/checkbox-d1ca9748.js.map +0 -1
  561. package/_internal/components/chip-e40c5521.js +0 -102
  562. package/_internal/components/chip-e40c5521.js.map +0 -1
  563. package/_internal/components/comment-block-a3cf7b9b.js +0 -138
  564. package/_internal/components/comment-block-a3cf7b9b.js.map +0 -1
  565. package/_internal/components/date-picker-6c1b14e4.js +0 -2
  566. package/_internal/components/date-picker-6c1b14e4.js.map +0 -1
  567. package/_internal/components/dialog-ebdb9500.js +0 -238
  568. package/_internal/components/dialog-ebdb9500.js.map +0 -1
  569. package/_internal/components/divider-116af6b9.js +0 -50
  570. package/_internal/components/divider-116af6b9.js.map +0 -1
  571. package/_internal/components/drag-handle-5215cd21.js +0 -51
  572. package/_internal/components/drag-handle-5215cd21.js.map +0 -1
  573. package/_internal/components/dropdown-0baed51b.js +0 -147
  574. package/_internal/components/dropdown-0baed51b.js.map +0 -1
  575. package/_internal/components/expansion-panel-5533a680.js +0 -167
  576. package/_internal/components/expansion-panel-5533a680.js.map +0 -1
  577. package/_internal/components/flag-8d4a7e72.js +0 -59
  578. package/_internal/components/flag-8d4a7e72.js.map +0 -1
  579. package/_internal/components/flex-box-55144e5f.js +0 -56
  580. package/_internal/components/flex-box-55144e5f.js.map +0 -1
  581. package/_internal/components/generic-block-ff0509ee.js +0 -124
  582. package/_internal/components/generic-block-ff0509ee.js.map +0 -1
  583. package/_internal/components/grid-6f1b5a41.js +0 -104
  584. package/_internal/components/grid-6f1b5a41.js.map +0 -1
  585. package/_internal/components/grid-column-f01df853.js +0 -59
  586. package/_internal/components/grid-column-f01df853.js.map +0 -1
  587. package/_internal/components/heading-89239843.js +0 -53
  588. package/_internal/components/heading-89239843.js.map +0 -1
  589. package/_internal/components/icon-b708cca4.js +0 -102
  590. package/_internal/components/icon-b708cca4.js.map +0 -1
  591. package/_internal/components/image-block-7938422d.js +0 -110
  592. package/_internal/components/image-block-7938422d.js.map +0 -1
  593. package/_internal/components/image-lightbox-58331704.js +0 -756
  594. package/_internal/components/image-lightbox-58331704.js.map +0 -1
  595. package/_internal/components/inline-list-e6f19a98.js +0 -74
  596. package/_internal/components/inline-list-e6f19a98.js.map +0 -1
  597. package/_internal/components/input-helper-64153099.js +0 -71
  598. package/_internal/components/input-helper-64153099.js.map +0 -1
  599. package/_internal/components/input-label-2feb0bb0.js +0 -59
  600. package/_internal/components/input-label-2feb0bb0.js.map +0 -1
  601. package/_internal/components/lightbox-570ad9e5.js +0 -155
  602. package/_internal/components/lightbox-570ad9e5.js.map +0 -1
  603. package/_internal/components/link-91f76477.js +0 -72
  604. package/_internal/components/link-91f76477.js.map +0 -1
  605. package/_internal/components/link-preview-fdd8d738.js +0 -117
  606. package/_internal/components/link-preview-fdd8d738.js.map +0 -1
  607. package/_internal/components/list-b08d8423.js +0 -71
  608. package/_internal/components/list-b08d8423.js.map +0 -1
  609. package/_internal/components/message-29cb9181.js +0 -96
  610. package/_internal/components/message-29cb9181.js.map +0 -1
  611. package/_internal/components/mosaic-cdb9f563.js +0 -95
  612. package/_internal/components/mosaic-cdb9f563.js.map +0 -1
  613. package/_internal/components/navigation-8b7d9bd8.js +0 -225
  614. package/_internal/components/navigation-8b7d9bd8.js.map +0 -1
  615. package/_internal/components/notification-a45fda96.js +0 -145
  616. package/_internal/components/notification-a45fda96.js.map +0 -1
  617. package/_internal/components/popover-65bfbc57.js +0 -3
  618. package/_internal/components/popover-65bfbc57.js.map +0 -1
  619. package/_internal/components/post-block-52e58dd5.js +0 -109
  620. package/_internal/components/post-block-52e58dd5.js.map +0 -1
  621. package/_internal/components/progress-f39c3fa2.js +0 -182
  622. package/_internal/components/progress-f39c3fa2.js.map +0 -1
  623. package/_internal/components/progress-tracker-512d7a08.js +0 -305
  624. package/_internal/components/progress-tracker-512d7a08.js.map +0 -1
  625. package/_internal/components/radio-button-559a4863.js +0 -149
  626. package/_internal/components/radio-button-559a4863.js.map +0 -1
  627. package/_internal/components/select-48d4fa8c.js +0 -454
  628. package/_internal/components/select-48d4fa8c.js.map +0 -1
  629. package/_internal/components/side-navigation-f9bc5b4e.js +0 -165
  630. package/_internal/components/side-navigation-f9bc5b4e.js.map +0 -1
  631. package/_internal/components/skeleton-c66516ee.js +0 -166
  632. package/_internal/components/skeleton-c66516ee.js.map +0 -1
  633. package/_internal/components/slider-efbfbc45.js +0 -311
  634. package/_internal/components/slider-efbfbc45.js.map +0 -1
  635. package/_internal/components/slideshow-dd312470.js +0 -151
  636. package/_internal/components/slideshow-dd312470.js.map +0 -1
  637. package/_internal/components/switch-769a2a04.js +0 -122
  638. package/_internal/components/switch-769a2a04.js.map +0 -1
  639. package/_internal/components/table-8617b1ba.js +0 -294
  640. package/_internal/components/table-8617b1ba.js.map +0 -1
  641. package/_internal/components/tabs-884c57b6.js +0 -298
  642. package/_internal/components/tabs-884c57b6.js.map +0 -1
  643. package/_internal/components/text-c7b1e079.js +0 -2
  644. package/_internal/components/text-c7b1e079.js.map +0 -1
  645. package/_internal/components/text-field-478acd86.js +0 -359
  646. package/_internal/components/text-field-478acd86.js.map +0 -1
  647. package/_internal/components/thumbnail-310a4c4b.js +0 -42
  648. package/_internal/components/thumbnail-310a4c4b.js.map +0 -1
  649. package/_internal/components/toolbar-a43533a2.js +0 -61
  650. package/_internal/components/toolbar-a43533a2.js.map +0 -1
  651. package/_internal/components/tooltip-2885ab2e.js +0 -327
  652. package/_internal/components/tooltip-2885ab2e.js.map +0 -1
  653. package/_internal/components/uploader-1e7f5bbb.js +0 -153
  654. package/_internal/components/uploader-1e7f5bbb.js.map +0 -1
  655. package/_internal/components/user-block-68a51ed0.js +0 -145
  656. package/_internal/components/user-block-68a51ed0.js.map +0 -1
  657. package/_internal/constants-b9e57936.js +0 -2155
  658. package/_internal/constants-b9e57936.js.map +0 -1
  659. package/_internal/constants-d0e3f49e.js +0 -24
  660. package/_internal/constants-d0e3f49e.js.map +0 -1
  661. package/_internal/context-9d1336a1.js +0 -19
  662. package/_internal/context-9d1336a1.js.map +0 -1
  663. package/_internal/forwardRef-15f62847.js +0 -70
  664. package/_internal/forwardRef-15f62847.js.map +0 -1
  665. package/_internal/getFocusableElements-230173a8.js +0 -13
  666. package/_internal/getFocusableElements-230173a8.js.map +0 -1
  667. package/_internal/index-9df37c0d.js +0 -436
  668. package/_internal/index-9df37c0d.js.map +0 -1
  669. package/_internal/index-a9c5cd69.js +0 -117
  670. package/_internal/index-a9c5cd69.js.map +0 -1
  671. package/_internal/isComponent-78df9309.js +0 -20
  672. package/_internal/isComponent-78df9309.js.map +0 -1
  673. package/_internal/isComponentType-e806b848.js +0 -9
  674. package/_internal/isComponentType-e806b848.js.map +0 -1
  675. package/_internal/mergeRefs-f0d7d6ea.js +0 -30
  676. package/_internal/mergeRefs-f0d7d6ea.js.map +0 -1
  677. package/_internal/partitionMulti-4daccdd5.js.map +0 -1
  678. package/_internal/state-db358714.js +0 -130
  679. package/_internal/state-db358714.js.map +0 -1
  680. package/_internal/useBooleanState-2a3d237c.js +0 -12
  681. package/_internal/useBooleanState-2a3d237c.js.map +0 -1
  682. package/_internal/useCallbackOnEscape-ea4d9eb4.js +0 -62
  683. package/_internal/useCallbackOnEscape-ea4d9eb4.js.map +0 -1
  684. package/_internal/useDisableBodyScroll-36bd7352.js +0 -219
  685. package/_internal/useDisableBodyScroll-36bd7352.js.map +0 -1
  686. package/_internal/useDisableStateProps-69e16b7c.js +0 -36
  687. package/_internal/useDisableStateProps-69e16b7c.js.map +0 -1
  688. package/_internal/useFocusTrap-c3c6378b.js +0 -112
  689. package/_internal/useFocusTrap-c3c6378b.js.map +0 -1
  690. package/_internal/useId-3a1facc0.js +0 -18
  691. package/_internal/useId-3a1facc0.js.map +0 -1
  692. package/_internal/useRovingTabIndex-7daf0f24.js +0 -77
  693. package/_internal/useRovingTabIndex-7daf0f24.js.map +0 -1
  694. package/_internal/useTransitionVisibility-321fdbfa.js +0 -50
  695. package/_internal/useTransitionVisibility-321fdbfa.js.map +0 -1
  696. package/_internal/wrapChildrenIconWithSpaces-c1faaae4.js +0 -20
  697. package/_internal/wrapChildrenIconWithSpaces-c1faaae4.js.map +0 -1
@@ -0,0 +1,37 @@
1
+ import { useDisabledStateContext } from './DisabledStateContext';
2
+
3
+ type GenericProps = {
4
+ disabled?: boolean;
5
+ isDisabled?: boolean;
6
+ 'aria-disabled'?: boolean | 'true' | 'false';
7
+ onClick?: any;
8
+ onChange?: any;
9
+ };
10
+
11
+ interface Output<TProps extends GenericProps> {
12
+ /** Is disabled or aria-disabled */
13
+ isAnyDisabled?: boolean;
14
+ disabledStateProps: { disabled?: boolean; 'aria-disabled'?: boolean };
15
+ otherProps: TProps & { disabled: never; 'aria-disabled': never; isDisabled: never };
16
+ }
17
+
18
+ /**
19
+ * Resolve disabled state from props.
20
+ * (handles `disabled`, `isDisabled` and `aria-disabled`)
21
+ *
22
+ * @params component props
23
+ */
24
+ export function useDisableStateProps<TProps extends GenericProps>(props: TProps): Output<TProps> {
25
+ const { disabled, isDisabled = disabled, 'aria-disabled': ariaDisabled, onClick, onChange, ...otherProps } = props;
26
+ const disabledStateContext = useDisabledStateContext();
27
+ const disabledStateProps = {
28
+ disabled: disabledStateContext?.state === 'disabled' || isDisabled,
29
+ 'aria-disabled': ariaDisabled === true || ariaDisabled === 'true',
30
+ };
31
+ const isAnyDisabled = disabledStateProps['aria-disabled'] || disabledStateProps.disabled;
32
+ if (!isAnyDisabled) {
33
+ (otherProps as any).onClick = onClick;
34
+ (otherProps as any).onChange = onChange;
35
+ }
36
+ return { disabledStateProps, otherProps: otherProps as Output<TProps>['otherProps'], isAnyDisabled };
37
+ }
@@ -0,0 +1,32 @@
1
+ import last from 'lodash/last';
2
+ import pull from 'lodash/pull';
3
+
4
+ export type Listener = { enable(): void; disable(): void };
5
+
6
+ /**
7
+ * Keep track of listeners, only the last registered listener gets activated at any point (previously registered
8
+ * listener are disabled).
9
+ * When a listener gets unregistered, the previously registered listener gets enabled again.
10
+ */
11
+ export function makeListenerTowerContext() {
12
+ const LISTENERS: Listener[] = [];
13
+
14
+ return {
15
+ register(listener: Listener) {
16
+ // Disable previous listener.
17
+ last(LISTENERS)?.disable();
18
+ // Keep track of current listener.
19
+ LISTENERS.push(listener);
20
+ // Enable current listener.
21
+ listener.enable();
22
+ },
23
+ unregister(listener: Listener) {
24
+ // Disable current listener.
25
+ listener.disable();
26
+ // Remove current listener.
27
+ pull(LISTENERS, listener);
28
+ // Enable previous listener.
29
+ last(LISTENERS)?.enable();
30
+ },
31
+ };
32
+ }
@@ -0,0 +1,7 @@
1
+ /**
2
+ * WARNING: All modules exported here are exposed to NPM in '@lumx/react/utils'.
3
+ */
4
+
5
+ export { ClickAwayProvider } from './ClickAwayProvider';
6
+ export { Portal, type PortalProps, type PortalInit, PortalProvider, type PortalProviderProps } from './Portal';
7
+ export { DisabledStateProvider, useDisabledStateContext } from './disabled';
@@ -0,0 +1,4 @@
1
+ /**
2
+ * Get current browser locale.
3
+ */
4
+ export const getCurrentLocale = (): string => navigator.languages?.[0] || navigator.language;
@@ -0,0 +1,17 @@
1
+ import { parseLocale } from '@lumx/react/utils/locale/parseLocale';
2
+
3
+ describe(parseLocale.name, () => {
4
+ it('should parse various locale formats', () => {
5
+ expect(parseLocale('en')).toEqual({ code: 'en', language: 'en' });
6
+ expect(parseLocale('EN')).toEqual({ code: 'en', language: 'en' });
7
+ expect(parseLocale('en-US')).toEqual({ code: 'en-US', language: 'en', region: 'US' });
8
+ expect(parseLocale('en-us')).toEqual({ code: 'en-US', language: 'en', region: 'US' });
9
+ expect(parseLocale('en_us')).toEqual({ code: 'en-US', language: 'en', region: 'US' });
10
+ expect(parseLocale('EN-US')).toEqual({ code: 'en-US', language: 'en', region: 'US' });
11
+ });
12
+
13
+ it('should fail on invalid locale', () => {
14
+ expect(parseLocale('-')).toBe(undefined);
15
+ expect(parseLocale('-foo')).toBe(undefined);
16
+ });
17
+ });
@@ -0,0 +1,23 @@
1
+ import { Locale } from '@lumx/react/utils/locale/types';
2
+
3
+ /**
4
+ * Parse locale code
5
+ * @example
6
+ * parseLocale('EN') // => { code: 'en', language: 'en' }
7
+ * parseLocale('en_us') // => { code: 'en-US', language: 'en', region: 'US' }
8
+ * parseLocale('EN-US') // => { code: 'en-US', language: 'en', region: 'US' }
9
+ */
10
+ export function parseLocale(locale: string): Locale | undefined {
11
+ const [rawLanguage, rawRegion] = locale.split(/[-_]/);
12
+ if (!rawLanguage) {
13
+ return undefined;
14
+ }
15
+ const language = rawLanguage.toLowerCase();
16
+ let region: string | undefined;
17
+ let code = language;
18
+ if (rawRegion) {
19
+ region = rawRegion.toUpperCase();
20
+ code += `-${region}`;
21
+ }
22
+ return { code, region, language };
23
+ }
@@ -0,0 +1,8 @@
1
+ export interface Locale {
2
+ /** ISO locale code `lang-REGION` (ex: `en-US`) */
3
+ code: string;
4
+ /** ISO locale language code (ex: `en`) */
5
+ language: string;
6
+ /** ISO locale region code (ex: `US`) */
7
+ region?: string;
8
+ }
@@ -0,0 +1,17 @@
1
+ /**
2
+ * Clamp value in range.
3
+ *
4
+ * @param value Value to clamp.
5
+ * @param min Minimum value.
6
+ * @param max Maximum value.
7
+ * @return Clamped value.
8
+ */
9
+ export const clamp = (value: number, min: number, max: number): number => {
10
+ if (value < min) {
11
+ return min;
12
+ }
13
+ if (value > max) {
14
+ return max;
15
+ }
16
+ return value;
17
+ };
@@ -0,0 +1,25 @@
1
+ import { isEqual } from './isEqual';
2
+
3
+ test(isEqual.name, () => {
4
+ expect(isEqual('', '')).toBe(true);
5
+ expect(isEqual(0, 0)).toBe(true);
6
+ expect(isEqual(Number.POSITIVE_INFINITY, Number.POSITIVE_INFINITY)).toBe(true);
7
+
8
+ expect(isEqual('', 0)).toBe(false);
9
+
10
+ expect(isEqual({}, {})).toBe(true);
11
+ expect(isEqual({ a: 1 }, { a: 1 })).toBe(true);
12
+ expect(isEqual({ a: { a: 1 } }, { a: { a: 1 } })).toBe(true);
13
+
14
+ expect(isEqual([], [])).toBe(true);
15
+
16
+ expect(isEqual([1], [2])).toBe(false);
17
+ expect(isEqual([1], [1, 2])).toBe(false);
18
+ expect(isEqual([1, 2], [2, 1])).toBe(false);
19
+
20
+ expect(isEqual({ a: 1 }, { a: 2 })).toBe(false);
21
+ expect(isEqual({ a: 1 }, {})).toBe(false);
22
+ expect(isEqual({}, { a: 1 })).toBe(false);
23
+ expect(isEqual({ a: { a: 1 } }, { a: { a: 2 } })).toBe(false);
24
+ expect(isEqual({ a: 1 }, { a: 1, b: 1 })).toBe(false);
25
+ });
@@ -0,0 +1,11 @@
1
+ /** Minimal recursive deep equal of JS values */
2
+ export function isEqual(obj1: any, obj2: any): boolean {
3
+ if (obj1 === obj2) return true;
4
+ if (typeof obj1 === 'object' && typeof obj2 === 'object') {
5
+ const keys1 = Object.keys(obj1);
6
+ const keys2 = Object.keys(obj2);
7
+ if (keys1.length !== keys2.length) return false;
8
+ return keys1.every((key1) => isEqual(obj1[key1], obj2[key1]));
9
+ }
10
+ return false;
11
+ }
@@ -0,0 +1,27 @@
1
+ import partition from 'lodash/partition';
2
+ import { partitionMulti } from './partitionMulti';
3
+
4
+ describe('partitionMulti', () => {
5
+ it('should act like partition for single predicate', () => {
6
+ const data = [0, 1, 2, 3, 4, 5];
7
+ const isEven = (n: number): boolean => n % 2 === 0;
8
+
9
+ const expected = partition(data, isEven);
10
+ const actual = partitionMulti(data, [isEven]);
11
+
12
+ expect(actual).toEqual(expected);
13
+ });
14
+
15
+ it('should partition on multiple predicates', () => {
16
+ type T = string | number | boolean;
17
+ const data: T[] = ['a', 1, 'b', false, true];
18
+ const isString = (s: T): boolean => typeof s === 'string';
19
+ const isNumber = (s: T): boolean => typeof s === 'number';
20
+
21
+ const [strings, numbers, others] = partitionMulti(data, [isString, isNumber]);
22
+
23
+ expect(strings).toEqual(['a', 'b']);
24
+ expect(numbers).toEqual([1]);
25
+ expect(others).toEqual([false, true]);
26
+ });
27
+ });
@@ -4,6 +4,8 @@ import last from 'lodash/last';
4
4
  import partition from 'lodash/partition';
5
5
  import reduce from 'lodash/reduce';
6
6
 
7
+ import { Predicate } from '@lumx/core/js/types';
8
+
7
9
  /**
8
10
  * Similar to lodash `partition` function but working with multiple predicates.
9
11
  *
@@ -17,9 +19,11 @@ import reduce from 'lodash/reduce';
17
19
  * @param predicates array of predicates to apply on elements
18
20
  * @return partitioned elements by the given predicates
19
21
  */
20
- function partitionMulti(elements, predicates) {
21
- return reduce(predicates, (partitioned, predicate) => concat(dropRight(partitioned), partition(last(partitioned), predicate)), [elements]);
22
+ export function partitionMulti<T>(elements: T[], predicates: Array<Predicate<T>>): T[][] {
23
+ return reduce(
24
+ predicates,
25
+ (partitioned: T[][], predicate: Predicate<T>) =>
26
+ concat(dropRight(partitioned), partition(last(partitioned), predicate)),
27
+ [elements],
28
+ );
22
29
  }
23
-
24
- export { partitionMulti as p };
25
- //# sourceMappingURL=partitionMulti-4daccdd5.js.map
@@ -0,0 +1,20 @@
1
+ import { useLayoutEffect } from 'react';
2
+
3
+ /**
4
+ * Helper component using useLayoutEffect to trigger a callback on before unmount.
5
+ *
6
+ * The callback must be wrapped in a React ref to avoid updating the `useLayoutEffect` before the un-mount
7
+ */
8
+ export const OnBeforeUnmount = ({ callbackRef }: { callbackRef: React.RefObject<(() => void) | undefined> }) => {
9
+ useLayoutEffect(
10
+ () => {
11
+ return () => {
12
+ // On unmount
13
+ // eslint-disable-next-line react-hooks/exhaustive-deps
14
+ callbackRef.current?.();
15
+ };
16
+ }, // eslint-disable-next-line react-hooks/exhaustive-deps
17
+ [],
18
+ );
19
+ return null;
20
+ };
@@ -0,0 +1,153 @@
1
+ import React from 'react';
2
+ import { render, screen } from '@testing-library/react';
3
+ import userEvent from '@testing-library/user-event';
4
+ import { RawClickable, RawClickableProps } from './RawClickable';
5
+ import { CustomLink } from '../../stories/utils/CustomLink';
6
+
7
+ /**
8
+ * Mounts the component and returns common DOM elements / data needed in multiple tests.
9
+ */
10
+ const setup = (props: RawClickableProps<any>) => {
11
+ render(<RawClickable {...props} data-testid="raw-element" />);
12
+ const element = screen.getByTestId('raw-element');
13
+ return { props, element };
14
+ };
15
+
16
+ describe(`<RawClickable>`, () => {
17
+ describe('as a button', () => {
18
+ it('should render a button by default', () => {
19
+ const { element } = setup({ as: 'button', children: 'Click me' });
20
+ expect(element.tagName).toBe('BUTTON');
21
+ expect(element).toHaveAttribute('type', 'button');
22
+ expect(screen.getByRole('button', { name: 'Click me' })).toBe(element);
23
+ });
24
+
25
+ it('should trigger onClick', async () => {
26
+ const onClick = vi.fn();
27
+ const { element } = setup({ as: 'button', children: 'Click me', onClick });
28
+ await userEvent.click(element);
29
+ expect(onClick).toHaveBeenCalledTimes(1);
30
+ });
31
+
32
+ it('should be disabled with `disabled` prop', async () => {
33
+ const onClick = vi.fn();
34
+ const { element } = setup({ as: 'button', children: 'Click me', onClick, disabled: true });
35
+ expect(element).toBeDisabled();
36
+ await userEvent.click(element);
37
+ expect(onClick).not.toHaveBeenCalled();
38
+ });
39
+
40
+ it('should be disabled with `isDisabled` prop', async () => {
41
+ const onClick = vi.fn();
42
+ const { element } = setup({ as: 'button', children: 'Click me', onClick, isDisabled: true });
43
+ expect(element).toBeDisabled();
44
+ await userEvent.click(element);
45
+ expect(onClick).not.toHaveBeenCalled();
46
+ });
47
+
48
+ it('should be aria-disabled with `aria-disabled` prop', async () => {
49
+ const onClick = vi.fn();
50
+ const { element } = setup({ as: 'button', children: 'Click me', onClick, 'aria-disabled': true });
51
+ expect(element).not.toBeDisabled();
52
+ expect(element).toHaveAttribute('aria-disabled', 'true');
53
+ await userEvent.click(element);
54
+ expect(onClick).not.toHaveBeenCalled();
55
+ });
56
+ });
57
+
58
+ describe('as a link', () => {
59
+ const href = 'https://example.com';
60
+
61
+ it('should render a link with `href` prop', () => {
62
+ const { element } = setup({ as: 'a', children: 'Click me', href });
63
+ expect(element.tagName).toBe('A');
64
+ expect(element).toHaveAttribute('href', href);
65
+ expect(screen.getByRole('link', { name: 'Click me' })).toBe(element);
66
+ });
67
+
68
+ it('should trigger onClick', async () => {
69
+ const onClick = vi.fn((evt: any) => evt.preventDefault());
70
+ const { element } = setup({ as: 'a', children: 'Click me', href, onClick });
71
+ await userEvent.click(element);
72
+ expect(onClick).toHaveBeenCalledTimes(1);
73
+ });
74
+
75
+ it('should be disabled with `disabled` prop', async () => {
76
+ const onClick = vi.fn();
77
+ const { element } = setup({ as: 'a', children: 'Click me', href, onClick, disabled: true });
78
+ expect(element).toHaveAttribute('aria-disabled', 'true');
79
+ expect(element).toHaveAttribute('tabindex', '-1');
80
+ await userEvent.click(element);
81
+ expect(onClick).not.toHaveBeenCalled();
82
+ });
83
+
84
+ it('should be disabled with `isDisabled` prop', async () => {
85
+ const onClick = vi.fn();
86
+ const { element } = setup({ as: 'a', children: 'Click me', href, onClick, isDisabled: true });
87
+ expect(element).toHaveAttribute('aria-disabled', 'true');
88
+ expect(element).toHaveAttribute('tabindex', '-1');
89
+ await userEvent.click(element);
90
+ expect(onClick).not.toHaveBeenCalled();
91
+ });
92
+
93
+ it('should be aria-disabled with `aria-disabled` prop', async () => {
94
+ const onClick = vi.fn();
95
+ const { element } = setup({ as: 'a', children: 'Click me', href, onClick, 'aria-disabled': true });
96
+ expect(element).toHaveAttribute('aria-disabled', 'true');
97
+ await userEvent.click(element);
98
+ expect(onClick).not.toHaveBeenCalled();
99
+ });
100
+ });
101
+
102
+ describe('as a custom component', () => {
103
+ it('should render a custom component with `linkAs` prop', () => {
104
+ const { element } = setup({ as: CustomLink, children: 'Click me' });
105
+ expect(element).toHaveAttribute('data-custom-link');
106
+ });
107
+
108
+ it('should trigger onClick', async () => {
109
+ const onClick = vi.fn();
110
+ const { element } = setup({ as: CustomLink, children: 'Click me', onClick });
111
+ expect(element).toHaveAttribute('data-custom-link');
112
+ await userEvent.click(element);
113
+ expect(onClick).toHaveBeenCalledTimes(1);
114
+ });
115
+
116
+ it('should be disabled with `disabled` prop', async () => {
117
+ const onClick = vi.fn();
118
+ const { element } = setup({ as: CustomLink, children: 'Click me', onClick, disabled: true });
119
+ expect(element).toHaveAttribute('data-custom-link');
120
+ expect(element).toHaveAttribute('aria-disabled', 'true');
121
+ expect(element).toHaveAttribute('tabindex', '-1');
122
+ await userEvent.click(element);
123
+ expect(onClick).not.toHaveBeenCalled();
124
+ });
125
+ });
126
+
127
+ describe('prop forwarding', () => {
128
+ it('should forward className', () => {
129
+ const { element } = setup({ as: 'button', className: 'foo bar' });
130
+ expect(element).toHaveClass('foo bar');
131
+ });
132
+
133
+ it('should forward ref and override type in button', () => {
134
+ const ref = React.createRef<HTMLButtonElement>();
135
+ const { element } = setup({ as: 'button', ref, type: 'submit' });
136
+ expect(element).toHaveAttribute('type', 'submit');
137
+ expect(ref.current).toBeInstanceOf(HTMLButtonElement);
138
+ });
139
+
140
+ it('should forward ref and override tabindex in link', () => {
141
+ const ref = React.createRef<HTMLAnchorElement>();
142
+ const { element } = setup({ as: 'a', ref, href: '#', tabIndex: -1 });
143
+ expect(ref.current).toBeInstanceOf(HTMLAnchorElement);
144
+ expect(element).toHaveAttribute('tabindex', '-1');
145
+ });
146
+
147
+ it('should forward ref to custom component', () => {
148
+ const ref = React.createRef<HTMLAnchorElement>();
149
+ setup({ as: CustomLink, ref });
150
+ expect(ref.current).toBeInstanceOf(HTMLAnchorElement);
151
+ });
152
+ });
153
+ });
@@ -0,0 +1,65 @@
1
+ import { AriaAttributes, ElementType } from 'react';
2
+ import { forwardRefPolymorphic } from '@lumx/react/utils/react/forwardRefPolymorphic';
3
+ import { ComponentRef, HasPolymorphicAs } from '@lumx/react/utils/type';
4
+ import { HasRequiredLinkHref } from '@lumx/react/utils/type/HasRequiredLinkHref';
5
+
6
+ type ClickableElement = 'a' | 'button' | ElementType;
7
+
8
+ type BaseClickableProps<E extends ClickableElement> = {
9
+ children?: React.ReactNode;
10
+ isDisabled?: boolean;
11
+ disabled?: boolean;
12
+ 'aria-disabled'?: AriaAttributes['aria-disabled'];
13
+ onClick?: React.MouseEventHandler<E>;
14
+ };
15
+
16
+ export type RawClickableProps<E extends ClickableElement> = HasPolymorphicAs<E> &
17
+ HasRequiredLinkHref<E> &
18
+ BaseClickableProps<E>;
19
+
20
+ /**
21
+ * Render clickable element (link, button or custom element)
22
+ * (also does some basic disabled state handling)
23
+ */
24
+ export const RawClickable = forwardRefPolymorphic(
25
+ <E extends ClickableElement>(props: RawClickableProps<E>, ref: ComponentRef<E>) => {
26
+ const {
27
+ children,
28
+ onClick,
29
+ disabled,
30
+ isDisabled = disabled,
31
+ 'aria-disabled': ariaDisabled,
32
+ as,
33
+ ...forwardedProps
34
+ } = props;
35
+
36
+ const isAnyDisabled = isDisabled || ariaDisabled === 'true' || ariaDisabled === true;
37
+
38
+ const Component = as as any;
39
+ let clickableProps;
40
+ if (Component === 'button') {
41
+ clickableProps = { type: forwardedProps.type || 'button', disabled: isDisabled };
42
+ } else {
43
+ clickableProps = { tabIndex: isDisabled ? '-1' : forwardedProps.tabIndex };
44
+ }
45
+
46
+ return (
47
+ <Component
48
+ ref={ref}
49
+ aria-disabled={isAnyDisabled || undefined}
50
+ {...forwardedProps}
51
+ {...clickableProps}
52
+ onClick={(event: any) => {
53
+ if (isAnyDisabled) {
54
+ event.stopPropagation();
55
+ event.preventDefault();
56
+ return;
57
+ }
58
+ onClick?.(event);
59
+ }}
60
+ >
61
+ {children}
62
+ </Component>
63
+ );
64
+ },
65
+ );
@@ -0,0 +1,32 @@
1
+ import get from 'lodash/get';
2
+ import { Children, Key, ReactChild, ReactNode, cloneElement, isValidElement } from 'react';
3
+ import { isFragment } from 'react-is';
4
+
5
+ /**
6
+ * Flatten list of react nodes removing fragments and adding keys.
7
+ * based on: https://github.com/grrowl/react-keyed-flatten-children/blob/5d421644a449765ddd62b659946196b4b5d7b135/index.ts
8
+ *
9
+ * @param children React nodes to flatten.
10
+ * @return Flattened react nodes.
11
+ * @deprecated This function was used to develop keyboard navigation in the List component,
12
+ * but we will replace this with a better method that does not require to hack the children inspection
13
+ *
14
+ * TODO: remove List keyboard navigation once we have provided alternative component (Menu, Combobox, etc.),
15
+ * remove this function and then remove react-is from the dependencies
16
+ */
17
+ export function flattenChildren(children: ReactNode) {
18
+ function recur(nodes: ReactNode, keys: Key[] = []) {
19
+ return Children.toArray(nodes).reduce((acc: ReactChild[], node, index) => {
20
+ const nodeKeys = keys.concat(get(node, 'key') ?? index);
21
+ if (isFragment(node)) {
22
+ acc.push(...recur(node.props.children, nodeKeys));
23
+ } else if (isValidElement(node)) {
24
+ acc.push(cloneElement(node, { key: nodeKeys.join('.') }));
25
+ } else if (typeof node === 'string' || typeof node === 'number') {
26
+ acc.push(node);
27
+ }
28
+ return acc;
29
+ }, []);
30
+ }
31
+ return recur(children);
32
+ }
@@ -0,0 +1,11 @@
1
+ import React, { type ReactNode, type ForwardedRef } from 'react';
2
+ import type { Comp } from '../type';
3
+
4
+ type ForwardRef = <P, T = HTMLElement, A = unknown>(
5
+ render: (props: P, ref: ForwardedRef<T>) => ReactNode,
6
+ ) => Comp<P, T> & A;
7
+
8
+ /**
9
+ * React.forwardRef but re-typed to attach some custom metadata on our components.
10
+ */
11
+ export const forwardRef = React.forwardRef as ForwardRef;
@@ -0,0 +1,9 @@
1
+ import React, { type ElementType } from 'react';
2
+ import type { ComponentRef } from '@lumx/react/utils/type';
3
+
4
+ type ForwardRefPolymorphic = <E extends ElementType, P extends { as?: E }>(
5
+ render: (props: P, ref: ComponentRef<E>) => React.ReactNode,
6
+ ) => (props: P & React.ComponentProps<E> & { ref?: ComponentRef<E> }) => React.JSX.Element;
7
+
8
+ /** Same as `React.forwardRef` but inferring Ref type from the `as` prop. */
9
+ export const forwardRefPolymorphic = React.forwardRef as ForwardRefPolymorphic;
@@ -0,0 +1,33 @@
1
+ import { Falsy } from '@lumx/react/utils/type';
2
+ import { MutableRefObject, useMemo } from 'react';
3
+
4
+ type FnRef<T> = (value: T) => void;
5
+
6
+ /**
7
+ * Merge refs into a single function ref.
8
+ *
9
+ * @param refs React references to merge.
10
+ * @return the merged ref.
11
+ */
12
+ export function mergeRefs<T>(...refs: Array<MutableRefObject<T | null> | FnRef<T> | Falsy>): FnRef<T> {
13
+ return (value) =>
14
+ refs.forEach((ref) => {
15
+ if (typeof ref === 'function') {
16
+ ref(value);
17
+ } else if (ref) {
18
+ // eslint-disable-next-line no-param-reassign
19
+ (ref as MutableRefObject<T>).current = value;
20
+ }
21
+ });
22
+ }
23
+
24
+ /**
25
+ * Same as `mergeRefs` but memoized
26
+ */
27
+ export const useMergeRefs = <T>(...refs: Array<MutableRefObject<T | null> | FnRef<T> | Falsy>) => {
28
+ return useMemo(
29
+ () => mergeRefs(...refs),
30
+ // eslint-disable-next-line react-hooks/exhaustive-deps
31
+ refs,
32
+ );
33
+ };
@@ -0,0 +1,17 @@
1
+ import React, { ReactElement, ReactNode } from 'react';
2
+
3
+ interface Props {
4
+ linkAs?: any;
5
+ }
6
+
7
+ /**
8
+ * Render link with default <a> HTML component or a custom one provided by `linkAs`.
9
+ *
10
+ * Can be used to inject the `Link` component from `react-router` and provide better a11y on LumX components.
11
+ *
12
+ * @param linkAs Custom link component.
13
+ * @param children Link children.
14
+ * @return A link.
15
+ */
16
+ export const renderLink = <P extends Props>({ linkAs, ...forwardedProps }: P, ...children: ReactNode[]): ReactElement =>
17
+ React.createElement(linkAs || 'a', forwardedProps, ...children);
@@ -0,0 +1,18 @@
1
+ import React from 'react';
2
+
3
+ import { DOCUMENT } from '@lumx/react/constants';
4
+ import type { Comp } from '@lumx/react/utils/type';
5
+ /**
6
+ * HOC component wrapping a component to skip render if predicate return falsy
7
+ */
8
+ export const skipRender = <P, T>(predicate: (props: P) => any, Component: Comp<P, T>) => {
9
+ const Wrapper = React.forwardRef<T, P>((props, ref) => {
10
+ if (!DOCUMENT) {
11
+ // Can't render in SSR.
12
+ return null;
13
+ }
14
+ return <Component ref={ref} {...props} />;
15
+ });
16
+ Wrapper.displayName = Component.displayName;
17
+ return Wrapper;
18
+ };
@@ -0,0 +1,7 @@
1
+ import { MaybeElementOrRef } from '@lumx/react/utils/type';
2
+
3
+ /** Unref a react ref or element */
4
+ export function unref(maybeElement: MaybeElementOrRef<HTMLElement>) {
5
+ if (maybeElement instanceof HTMLElement) return maybeElement;
6
+ return maybeElement?.current;
7
+ }
@@ -0,0 +1,37 @@
1
+ import { Fragment } from 'react';
2
+
3
+ import { Icon } from '@lumx/react';
4
+ import { mdiEarth, mdiFoodApple, mdiPencil } from '@lumx/icons';
5
+ import { wrapChildrenIconWithSpaces } from './wrapChildrenIconWithSpaces';
6
+
7
+ describe(wrapChildrenIconWithSpaces, () => {
8
+ it('should ignore null or undefined children', () => {
9
+ expect(wrapChildrenIconWithSpaces(undefined)).toBeUndefined();
10
+ expect(wrapChildrenIconWithSpaces(null)).toBeUndefined();
11
+ });
12
+
13
+ it('should wrap icons with spaces', () => {
14
+ expect(
15
+ wrapChildrenIconWithSpaces(
16
+ <>
17
+ <Icon icon={mdiEarth} />a string
18
+ <>
19
+ some more string with
20
+ <Icon icon={mdiFoodApple} />
21
+ </>
22
+ {['array with', <Icon key="custom-key" icon={mdiPencil} />]}
23
+ </>,
24
+ ),
25
+ ).toEqual([
26
+ // prettier-ignore
27
+ <Fragment key=".0">
28
+ {' '}
29
+ <Icon key=".0" icon={mdiEarth} />{' '}a string
30
+ <Fragment key=".2">
31
+ some more string with{' '}<Icon key=".1" icon={mdiFoodApple} />{' '}
32
+ </Fragment>
33
+ array with{' '}<Icon key=".3:$custom-key" icon={mdiPencil} />{' '}
34
+ </Fragment>,
35
+ ]);
36
+ });
37
+ });