@spaced-out/ui-design-system 0.3.38 → 0.3.39

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 (307) hide show
  1. package/.cspell/custom-words.txt +4 -0
  2. package/.github/workflows/publish_to_npm.yml +32 -4
  3. package/CHANGELOG.md +21 -0
  4. package/dts-generator/.editorconfig +10 -0
  5. package/dts-generator/.gitattributes +4 -0
  6. package/dts-generator/README.md +63 -0
  7. package/dts-generator/convert.js +101 -0
  8. package/dts-generator/package.json +16 -0
  9. package/gulpfile.js +20 -1
  10. package/lib/components/Avatar/Avatar.d.ts +264 -0
  11. package/lib/components/Avatar/index.d.ts +1 -0
  12. package/lib/components/AvatarGroup/AvatarGroup.d.ts +161 -0
  13. package/lib/components/AvatarGroup/index.d.ts +1 -0
  14. package/lib/components/Badge/Badge.d.ts +138 -0
  15. package/lib/components/Badge/index.d.ts +1 -0
  16. package/lib/components/Banner/Banner.d.ts +52 -0
  17. package/lib/components/Banner/index.d.ts +1 -0
  18. package/lib/components/Breadcrumbs/BreadcrumbLink/BreadcrumbLink.d.ts +50 -0
  19. package/lib/components/Breadcrumbs/BreadcrumbLink/index.d.ts +1 -0
  20. package/lib/components/Breadcrumbs/Breadcrumbs.d.ts +64 -0
  21. package/lib/components/Breadcrumbs/index.d.ts +2 -0
  22. package/lib/components/Button/Button.d.ts +288 -0
  23. package/lib/components/Button/index.d.ts +14 -0
  24. package/lib/components/ButtonDropdown/ButtonDropdown.d.ts +192 -0
  25. package/lib/components/ButtonDropdown/SimpleButtonDropdown.d.ts +148 -0
  26. package/lib/components/ButtonDropdown/index.d.ts +2 -0
  27. package/lib/components/ButtonTabs/ButtonTab/ButtonTab.d.ts +82 -0
  28. package/lib/components/ButtonTabs/ButtonTab/index.d.ts +1 -0
  29. package/lib/components/ButtonTabs/ButtonTabDropdown.d.ts +118 -0
  30. package/lib/components/ButtonTabs/ButtonTabs.d.ts +113 -0
  31. package/lib/components/ButtonTabs/index.d.ts +2 -0
  32. package/lib/components/Card/Card.d.ts +163 -0
  33. package/lib/components/Card/index.d.ts +1 -0
  34. package/lib/components/Charts/ChartTooltip/index.d.ts +10 -0
  35. package/lib/components/Charts/ChartWrapper/ChartWrapper.d.ts +184 -0
  36. package/lib/components/Charts/ChartWrapper/index.d.ts +1 -0
  37. package/lib/components/Charts/ColumnChart/ColumnChart.d.ts +111 -0
  38. package/lib/components/Charts/ColumnChart/index.d.ts +1 -0
  39. package/lib/components/Charts/DonutChart/DonutChart.d.ts +142 -0
  40. package/lib/components/Charts/DonutChart/index.d.ts +1 -0
  41. package/lib/components/Charts/FunnelChart/FunnelChart.d.ts +99 -0
  42. package/lib/components/Charts/FunnelChart/index.d.ts +1 -0
  43. package/lib/components/Charts/LineChart/LineChart.d.ts +97 -0
  44. package/lib/components/Charts/LineChart/index.d.ts +1 -0
  45. package/lib/components/Charts/SpiderChart/SpiderChart.d.ts +95 -0
  46. package/lib/components/Charts/SpiderChart/index.d.ts +1 -0
  47. package/lib/components/Charts/index.d.ts +7 -0
  48. package/lib/components/ChatBubble/ChatBubble.d.ts +203 -0
  49. package/lib/components/ChatBubble/index.d.ts +1 -0
  50. package/lib/components/Checkbox/Checkbox.d.ts +165 -0
  51. package/lib/components/Checkbox/CheckboxGroup.d.ts +113 -0
  52. package/lib/components/Checkbox/index.d.ts +2 -0
  53. package/lib/components/Chip/Chip.d.ts +181 -0
  54. package/lib/components/Chip/index.d.ts +2 -0
  55. package/lib/components/CircularLoader/CircularLoader.d.ts +56 -0
  56. package/lib/components/CircularLoader/index.d.ts +1 -0
  57. package/lib/components/CollapsibleCard/CollapsibleCard.d.ts +134 -0
  58. package/lib/components/CollapsibleCard/index.d.ts +1 -0
  59. package/lib/components/Combobox/Combobox.d.ts +292 -0
  60. package/lib/components/Combobox/helper.d.ts +205 -0
  61. package/lib/components/Combobox/index.d.ts +1 -0
  62. package/lib/components/ConditionalWrapper/ConditionalWrapper.d.ts +12 -0
  63. package/lib/components/ConditionalWrapper/index.d.ts +1 -0
  64. package/lib/components/DateRangePicker/Calendar.d.ts +119 -0
  65. package/lib/components/DateRangePicker/DateRangePicker.d.ts +196 -0
  66. package/lib/components/DateRangePicker/DateRangeWrapper.d.ts +311 -0
  67. package/lib/components/DateRangePicker/Day.d.ts +70 -0
  68. package/lib/components/DateRangePicker/index.d.ts +1 -0
  69. package/lib/components/Dialog/Dialog.d.ts +208 -0
  70. package/lib/components/Dialog/index.d.ts +1 -0
  71. package/lib/components/Disclaimer/Disclaimer.d.ts +32 -0
  72. package/lib/components/Disclaimer/index.d.ts +1 -0
  73. package/lib/components/Dropdown/Dropdown.d.ts +145 -0
  74. package/lib/components/Dropdown/SimpleDropdown.d.ts +142 -0
  75. package/lib/components/Dropdown/index.d.ts +2 -0
  76. package/lib/components/EmptyState/EmptyImages/CalendarEmptyImage.d.ts +108 -0
  77. package/lib/components/EmptyState/EmptyImages/ChartEmptyImage.d.ts +194 -0
  78. package/lib/components/EmptyState/EmptyImages/DataEmptyImage.d.ts +116 -0
  79. package/lib/components/EmptyState/EmptyImages/FileEmptyImage.d.ts +133 -0
  80. package/lib/components/EmptyState/EmptyImages/MessageEmptyImage.d.ts +64 -0
  81. package/lib/components/EmptyState/EmptyImages/UploadEmptyImage.d.ts +67 -0
  82. package/lib/components/EmptyState/EmptyImages/index.d.ts +6 -0
  83. package/lib/components/EmptyState/EmptyState.d.ts +82 -0
  84. package/lib/components/EmptyState/index.d.ts +1 -0
  85. package/lib/components/ErrorMessage/ErrorImages/ForbiddenImage.d.ts +85 -0
  86. package/lib/components/ErrorMessage/ErrorImages/NotFoundImage.d.ts +117 -0
  87. package/lib/components/ErrorMessage/ErrorImages/ServerErrorImage.d.ts +237 -0
  88. package/lib/components/ErrorMessage/ErrorImages/UnauthorizedImage.d.ts +73 -0
  89. package/lib/components/ErrorMessage/ErrorImages/index.d.ts +4 -0
  90. package/lib/components/ErrorMessage/ErrorMessage.d.ts +86 -0
  91. package/lib/components/ErrorMessage/index.d.ts +1 -0
  92. package/lib/components/FileUpload/FileBlock/FileBlock.d.ts +137 -0
  93. package/lib/components/FileUpload/FileBlock/index.d.ts +1 -0
  94. package/lib/components/FileUpload/FileUpload.d.ts +190 -0
  95. package/lib/components/FileUpload/index.d.ts +2 -0
  96. package/lib/components/FocusManager/FocusManager.d.ts +52 -0
  97. package/lib/components/FocusManager/index.d.ts +1 -0
  98. package/lib/components/FocusManagerWithArrowKeyNavigation/FocusManagerWithArrowKeyNavigation.d.ts +137 -0
  99. package/lib/components/FocusManagerWithArrowKeyNavigation/index.d.ts +1 -0
  100. package/lib/components/FormTitleWrapper/FormTitleWrapper.d.ts +62 -0
  101. package/lib/components/FormTitleWrapper/index.d.ts +1 -0
  102. package/lib/components/Grid/Grid.d.ts +104 -0
  103. package/lib/components/Grid/index.d.ts +1 -0
  104. package/lib/components/Icon/ClickableIcon.d.ts +101 -0
  105. package/lib/components/Icon/Icon.d.ts +71 -0
  106. package/lib/components/Icon/Icon.docs.d.ts +138 -0
  107. package/lib/components/Icon/SemanticIcon.d.ts +42 -0
  108. package/lib/components/Icon/index.d.ts +3 -0
  109. package/lib/components/InContextAlert/InContextAlert.d.ts +210 -0
  110. package/lib/components/InContextAlert/index.d.ts +1 -0
  111. package/lib/components/InlineDropdown/InlineDropdown.d.ts +146 -0
  112. package/lib/components/InlineDropdown/SimpleInlineDropdown.d.ts +145 -0
  113. package/lib/components/InlineDropdown/index.d.ts +2 -0
  114. package/lib/components/Input/Input.d.ts +327 -0
  115. package/lib/components/Input/index.d.ts +1 -0
  116. package/lib/components/KPIBox/KPIBox.d.ts +89 -0
  117. package/lib/components/KPIBox/index.d.ts +1 -0
  118. package/lib/components/LinearLoader/LinearLoader.d.ts +44 -0
  119. package/lib/components/LinearLoader/index.d.ts +1 -0
  120. package/lib/components/Link/Link.d.ts +225 -0
  121. package/lib/components/Link/index.d.ts +1 -0
  122. package/lib/components/Menu/Menu.d.ts +379 -0
  123. package/lib/components/Menu/MenuOptionButton.d.ts +199 -0
  124. package/lib/components/Menu/index.d.ts +2 -0
  125. package/lib/components/Modal/Modal.d.ts +354 -0
  126. package/lib/components/Modal/index.d.ts +14 -0
  127. package/lib/components/Notification/Notification.d.ts +147 -0
  128. package/lib/components/Notification/index.d.ts +1 -0
  129. package/lib/components/OptionButton/OptionButton.d.ts +150 -0
  130. package/lib/components/OptionButton/SimpleOptionButton.d.ts +150 -0
  131. package/lib/components/OptionButton/index.d.ts +2 -0
  132. package/lib/components/PageTitle/PageTitle.d.ts +256 -0
  133. package/lib/components/PageTitle/index.d.ts +1 -0
  134. package/lib/components/Pagination/Pagination.d.ts +144 -0
  135. package/lib/components/Pagination/PaginationItem.d.ts +136 -0
  136. package/lib/components/Pagination/index.d.ts +1 -0
  137. package/lib/components/Panel/Panel.d.ts +168 -0
  138. package/lib/components/Panel/index.d.ts +9 -0
  139. package/lib/components/ProgressDonut/ProgressDonut.d.ts +102 -0
  140. package/lib/components/ProgressDonut/index.d.ts +1 -0
  141. package/lib/components/PromptChip/PromptChip.d.ts +167 -0
  142. package/lib/components/PromptChip/index.d.ts +1 -0
  143. package/lib/components/PromptInput/PromptInput.d.ts +189 -0
  144. package/lib/components/PromptInput/index.d.ts +1 -0
  145. package/lib/components/RadioButton/RadioButton.d.ts +132 -0
  146. package/lib/components/RadioButton/RadioGroup.d.ts +88 -0
  147. package/lib/components/RadioButton/index.d.ts +2 -0
  148. package/lib/components/RadioTile/RadioTile.d.ts +109 -0
  149. package/lib/components/RadioTile/index.d.ts +1 -0
  150. package/lib/components/RangeSlider/RangeSlider.d.ts +207 -0
  151. package/lib/components/RangeSlider/index.d.ts +1 -0
  152. package/lib/components/Rating/Rating.d.ts +131 -0
  153. package/lib/components/Rating/index.d.ts +1 -0
  154. package/lib/components/ScoreBar/ScoreBar.d.ts +122 -0
  155. package/lib/components/ScoreBar/index.d.ts +1 -0
  156. package/lib/components/SearchInput/SearchInput.d.ts +79 -0
  157. package/lib/components/SearchInput/index.d.ts +2 -0
  158. package/lib/components/Separator/Separator.d.ts +51 -0
  159. package/lib/components/Separator/index.d.ts +1 -0
  160. package/lib/components/Shimmer/Shimmer.d.ts +131 -0
  161. package/lib/components/Shimmer/index.d.ts +1 -0
  162. package/lib/components/SideMenuLink/SideMenuLink.d.ts +336 -0
  163. package/lib/components/SideMenuLink/index.d.ts +1 -0
  164. package/lib/components/StatusIndicator/StatusIndicator.d.ts +60 -0
  165. package/lib/components/StatusIndicator/index.d.ts +1 -0
  166. package/lib/components/Stepper/Step/Step.d.ts +119 -0
  167. package/lib/components/Stepper/Step/StepContent.d.ts +35 -0
  168. package/lib/components/Stepper/Step/StepLabel.d.ts +35 -0
  169. package/lib/components/Stepper/Step/index.d.ts +3 -0
  170. package/lib/components/Stepper/Stepper.d.ts +60 -0
  171. package/lib/components/Stepper/index.d.ts +2 -0
  172. package/lib/components/StickyBar/StickyBar.d.ts +55 -0
  173. package/lib/components/StickyBar/index.d.ts +1 -0
  174. package/lib/components/SubMenu/SubMenu.d.ts +89 -0
  175. package/lib/components/SubMenu/SubMenuGroup.d.ts +156 -0
  176. package/lib/components/SubMenu/SubMenuItem.d.ts +161 -0
  177. package/lib/components/SubMenu/SubMenuLink.d.ts +81 -0
  178. package/lib/components/SubMenu/index.d.ts +4 -0
  179. package/lib/components/Table/Cell.d.ts +107 -0
  180. package/lib/components/Table/DefaultRow.d.ts +144 -0
  181. package/lib/components/Table/DefaultTableHeader.d.ts +236 -0
  182. package/lib/components/Table/StaticTable.d.ts +198 -0
  183. package/lib/components/Table/Table.d.ts +113 -0
  184. package/lib/components/Table/Table.docs.d.ts +532 -0
  185. package/lib/components/Table/TableActionBar.d.ts +48 -0
  186. package/lib/components/Table/TableBottomBar.d.ts +20 -0
  187. package/lib/components/Table/TableTopBar.d.ts +20 -0
  188. package/lib/components/Table/dummyTableData.d.ts +2189 -0
  189. package/lib/components/Table/hooks.d.ts +98 -0
  190. package/lib/components/Table/index.d.ts +8 -0
  191. package/lib/components/Tabs/Tab/Tab.d.ts +146 -0
  192. package/lib/components/Tabs/Tab/index.d.ts +1 -0
  193. package/lib/components/Tabs/TabList/TabDropdown.d.ts +100 -0
  194. package/lib/components/Tabs/TabList/TabList.d.ts +157 -0
  195. package/lib/components/Tabs/TabList/index.d.ts +1 -0
  196. package/lib/components/Tabs/index.d.ts +2 -0
  197. package/lib/components/Text/Text.d.ts +1021 -0
  198. package/lib/components/Text/index.d.ts +30 -0
  199. package/lib/components/TextTile/TextTile.d.ts +40 -0
  200. package/lib/components/TextTile/index.d.ts +1 -0
  201. package/lib/components/Textarea/Textarea.d.ts +132 -0
  202. package/lib/components/Textarea/index.d.ts +2 -0
  203. package/lib/components/Timeline/Timeline.d.ts +40 -0
  204. package/lib/components/Timeline/TimelineItem/TimelineItem.d.ts +121 -0
  205. package/lib/components/Timeline/TimelineItem/index.d.ts +1 -0
  206. package/lib/components/Timeline/index.d.ts +2 -0
  207. package/lib/components/Toast/Toast.d.ts +248 -0
  208. package/lib/components/Toast/ToastContainer.d.ts +140 -0
  209. package/lib/components/Toast/ToastManager.d.ts +58 -0
  210. package/lib/components/Toast/index.d.ts +10 -0
  211. package/lib/components/Toggle/Toggle.d.ts +111 -0
  212. package/lib/components/Toggle/index.d.ts +1 -0
  213. package/lib/components/TokenListInput/TokenListInput.d.ts +333 -0
  214. package/lib/components/TokenListInput/TokenValueChips.d.ts +64 -0
  215. package/lib/components/TokenListInput/index.d.ts +1 -0
  216. package/lib/components/Tooltip/Tooltip.d.ts +186 -0
  217. package/lib/components/Tooltip/index.d.ts +1 -0
  218. package/lib/components/Truncate/Truncate.d.ts +87 -0
  219. package/lib/components/Truncate/index.d.ts +2 -0
  220. package/lib/components/TruncatedTextWithTooltip/TruncatedTextWithTooltip.d.ts +95 -0
  221. package/lib/components/TruncatedTextWithTooltip/index.d.ts +1 -0
  222. package/lib/components/Typeahead/SimpleTypeahead.d.ts +141 -0
  223. package/lib/components/Typeahead/Typeahead.d.ts +217 -0
  224. package/lib/components/Typeahead/index.d.ts +2 -0
  225. package/lib/components/WeekdayPicker/WeekdayPicker.d.ts +242 -0
  226. package/lib/components/WeekdayPicker/index.d.ts +1 -0
  227. package/lib/components/index.d.ts +71 -0
  228. package/lib/hooks/index.d.ts +12 -0
  229. package/lib/hooks/useArbitraryOptionAddition/index.d.ts +1 -0
  230. package/lib/hooks/useArbitraryOptionAddition/useArbitraryOptionAddition.d.ts +124 -0
  231. package/lib/hooks/useCopyToClipboard/index.d.ts +1 -0
  232. package/lib/hooks/useCopyToClipboard/useCopyToClipboard.d.ts +27 -0
  233. package/lib/hooks/useFileUpload/index.d.ts +1 -0
  234. package/lib/hooks/useFileUpload/useFileUpload.d.ts +342 -0
  235. package/lib/hooks/useFilteredOptions/index.d.ts +1 -0
  236. package/lib/hooks/useFilteredOptions/useFilteredOptions.d.ts +87 -0
  237. package/lib/hooks/useInputState/index.d.ts +1 -0
  238. package/lib/hooks/useInputState/useInputState.d.ts +22 -0
  239. package/lib/hooks/useLockedBody/index.d.ts +1 -0
  240. package/lib/hooks/useLockedBody/useLockedBody.d.ts +46 -0
  241. package/lib/hooks/useModal/index.d.ts +1 -0
  242. package/lib/hooks/useModal/useModal.d.ts +32 -0
  243. package/lib/hooks/useMountTransition/index.d.ts +24 -0
  244. package/lib/hooks/usePagination/index.d.ts +1 -0
  245. package/lib/hooks/usePagination/usePagination.d.ts +140 -0
  246. package/lib/hooks/useToastPortal/index.d.ts +1 -0
  247. package/lib/hooks/useToastPortal/useToastPortal.d.ts +31 -0
  248. package/lib/hooks/useToggle/index.d.ts +1 -0
  249. package/lib/hooks/useToggle/useToggle.d.ts +11 -0
  250. package/lib/hooks/useWindowSize/index.d.ts +1 -0
  251. package/lib/hooks/useWindowSize/useWindowSize.d.ts +28 -0
  252. package/lib/index.d.ts +5 -0
  253. package/lib/styles/index.d.ts +267 -0
  254. package/lib/styles/variables/_border.d.ts +12 -0
  255. package/lib/styles/variables/_color.d.ts +104 -0
  256. package/lib/styles/variables/_elevation.d.ts +7 -0
  257. package/lib/styles/variables/_font.d.ts +30 -0
  258. package/lib/styles/variables/_motion.d.ts +6 -0
  259. package/lib/styles/variables/_opacity.d.ts +15 -0
  260. package/lib/styles/variables/_shadow.d.ts +24 -0
  261. package/lib/styles/variables/_size.d.ts +57 -0
  262. package/lib/styles/variables/_space.d.ts +12 -0
  263. package/lib/types/charts.d.ts +225 -0
  264. package/lib/types/common.d.ts +10 -0
  265. package/lib/types/date-range-picker.d.ts +17 -0
  266. package/lib/types/date-range-picker.js.flow +2 -3
  267. package/lib/types/index.d.ts +6 -0
  268. package/lib/types/menu.d.ts +13 -0
  269. package/lib/types/toast.d.ts +32 -0
  270. package/lib/types/typography.d.ts +17 -0
  271. package/lib/utils/array/are-arrays-equal.d.ts +11 -0
  272. package/lib/utils/array/index.d.ts +1 -0
  273. package/lib/utils/charts/charts.d.ts +102 -0
  274. package/lib/utils/charts/columnChart.d.ts +51 -0
  275. package/lib/utils/charts/donutChart.d.ts +116 -0
  276. package/lib/utils/charts/funnelChart.d.ts +94 -0
  277. package/lib/utils/charts/helpers.d.ts +44 -0
  278. package/lib/utils/charts/index.d.ts +7 -0
  279. package/lib/utils/charts/lineChart.d.ts +44 -0
  280. package/lib/utils/charts/spiderChart.d.ts +46 -0
  281. package/lib/utils/charts/typography.d.ts +42 -0
  282. package/lib/utils/classify/index.d.ts +27 -0
  283. package/lib/utils/click-away/click-away.d.ts +194 -0
  284. package/lib/utils/click-away/index.d.ts +1 -0
  285. package/lib/utils/date-range-picker/date-range-picker.d.ts +391 -0
  286. package/lib/utils/date-range-picker/index.d.ts +2 -0
  287. package/lib/utils/date-range-picker/timezones.d.ts +262 -0
  288. package/lib/utils/dom/dom.d.ts +245 -0
  289. package/lib/utils/dom/index.d.ts +1 -0
  290. package/lib/utils/helpers/helpers.d.ts +48 -0
  291. package/lib/utils/helpers/index.d.ts +1 -0
  292. package/lib/utils/index.d.ts +14 -0
  293. package/lib/utils/makeClassNameComponent/index.d.ts +1 -0
  294. package/lib/utils/makeClassNameComponent/makeClassNameComponent.d.ts +73 -0
  295. package/lib/utils/menu/index.d.ts +1 -0
  296. package/lib/utils/merge-refs/index.d.ts +1 -0
  297. package/lib/utils/merge-refs/merge-refs.d.ts +15 -0
  298. package/lib/utils/rating/index.d.ts +1 -0
  299. package/lib/utils/rating/rating.d.ts +31 -0
  300. package/lib/utils/score-bar/index.d.ts +1 -0
  301. package/lib/utils/score-bar/score-bar.d.ts +54 -0
  302. package/lib/utils/string/index.d.ts +1 -0
  303. package/lib/utils/string/string.d.ts +30 -0
  304. package/lib/utils/token-list-input/token-list-input.d.ts +32 -0
  305. package/lib/utils/tokens/index.d.ts +1 -0
  306. package/lib/utils/tokens/tokens.d.ts +230 -0
  307. package/package.json +1 -1
@@ -0,0 +1,199 @@
1
+ import * as React from 'react';
2
+ import {classify} from '../../utils/classify';
3
+ import {UnstyledButton} from '../Button';
4
+ import {Checkbox} from '../Checkbox';
5
+ import {Icon} from '../Icon';
6
+ import {RadioButton} from '../RadioButton';
7
+ import {Truncate} from '../Truncate';
8
+ import {TruncatedTextWithTooltip} from '../TruncatedTextWithTooltip';
9
+ import type {BaseMenuProps, MenuOption} from './Menu';
10
+ import css from './Menu.module.css';
11
+ export type MenuOptionProps = BaseMenuProps & {
12
+ option: MenuOption;
13
+ isLastItem?: boolean;
14
+ style?: unknown;
15
+ };
16
+ export const MenuOptionButton = (props: MenuOptionProps): React.ReactNode => {
17
+ const lastMenuItemRef: {
18
+ current: HTMLButtonElement | null;
19
+ } = React.useRef(null);
20
+ const {
21
+ option,
22
+ size = 'medium',
23
+ onSelect,
24
+ selectedOption,
25
+ menuDisabled,
26
+ classNames,
27
+ optionsVariant = 'normal',
28
+ selectedKeys,
29
+ isLastItem,
30
+ onTabOut,
31
+ resolveLabel,
32
+ resolveSecondaryLabel,
33
+ style,
34
+ showLabelTooltip,
35
+ allowWrap = false,
36
+ } = props;
37
+ const {
38
+ key,
39
+ label,
40
+ secondaryLabel,
41
+ customComponent,
42
+ iconLeft,
43
+ iconLeftType,
44
+ classNames: optionClassNames,
45
+ iconRight,
46
+ iconRightType,
47
+ disabled,
48
+ optionSize,
49
+ optionVariant = optionsVariant,
50
+ indeterminate = false,
51
+ } = option;
52
+ const [buttonSize, setButtonSize] = React.useState(optionSize || size);
53
+ const resolvedLabel = resolveLabel ? resolveLabel(option) : label;
54
+ const resolvedSecondaryLabel = resolveSecondaryLabel
55
+ ? resolveSecondaryLabel(option)
56
+ : secondaryLabel;
57
+
58
+ const isSelected = () => {
59
+ if (!selectedKeys || !Array.isArray(selectedKeys) || !selectedKeys.length) {
60
+ return false;
61
+ }
62
+
63
+ return selectedKeys.includes(option.key);
64
+ };
65
+
66
+ React.useEffect(() => {
67
+ setButtonSize(optionSize || size);
68
+ }, [optionSize, size]);
69
+ React.useEffect(() => {
70
+ const handleKeyDown = (event: KeyboardEvent) => {
71
+ if (event.key === 'Tab' && !event.shiftKey) {
72
+ // Tab pressed without shift key, calling tab out callback
73
+ onTabOut?.();
74
+ }
75
+ };
76
+
77
+ lastMenuItemRef.current?.addEventListener('keydown', handleKeyDown);
78
+ return () => {
79
+ lastMenuItemRef.current?.removeEventListener('keydown', handleKeyDown);
80
+ };
81
+ }, [isLastItem]);
82
+ return (
83
+ <UnstyledButton
84
+ className={classify(
85
+ css.option,
86
+ {
87
+ [css.selected]: isSelected() || key === selectedOption?.key,
88
+ [css.optionSmall]: buttonSize === 'small',
89
+ [css.optionMedium]: buttonSize === 'medium',
90
+ [css.disabled]: menuDisabled || disabled,
91
+ [css.withIconLeft]: !!iconLeft,
92
+ [css.withIconRight]: !!iconRight,
93
+ },
94
+ classNames?.option,
95
+ optionClassNames?.wrapper,
96
+ )}
97
+ style={style}
98
+ disabled={menuDisabled || disabled}
99
+ onClick={(e) => onSelect && onSelect(option, e)}
100
+ autoFocus={selectedOption?.key === key}
101
+ {...(isLastItem
102
+ ? {
103
+ ref: lastMenuItemRef,
104
+ }
105
+ : {})}
106
+ >
107
+ {optionVariant === 'checkbox' && (
108
+ <Checkbox
109
+ tabIndex={-1}
110
+ disabled={menuDisabled || disabled}
111
+ checked={isSelected()}
112
+ indeterminate={indeterminate}
113
+ />
114
+ )}
115
+ {optionVariant === 'radio' && (
116
+ <RadioButton
117
+ disabled={menuDisabled || disabled}
118
+ value={option.key}
119
+ selectedValue={selectedKeys?.[0]}
120
+ tabIndex={-1}
121
+ />
122
+ )}
123
+ {!!iconLeft && (
124
+ <Icon
125
+ name={iconLeft}
126
+ type={iconLeftType}
127
+ size="small"
128
+ className={css.icon}
129
+ />
130
+ )}
131
+ <div
132
+ className={classify(
133
+ css.optionTextContainer,
134
+ classNames?.optionTextContainer,
135
+ )}
136
+ >
137
+ {React.isValidElement(customComponent) ? (
138
+ customComponent
139
+ ) : (
140
+ <div
141
+ className={classify(
142
+ css.optionTextLabel,
143
+ classNames?.optionTextLabel,
144
+ )}
145
+ >
146
+ {renderLabel(resolvedLabel, allowWrap, showLabelTooltip)}
147
+ </div>
148
+ )}
149
+
150
+ {!!secondaryLabel && (
151
+ <div className={css.optionTextSecondaryLabel}>
152
+ <Truncate>{resolvedSecondaryLabel}</Truncate>
153
+ </div>
154
+ )}
155
+ </div>
156
+
157
+ {!!iconRight && (
158
+ <Icon
159
+ name={iconRight}
160
+ type={iconRightType}
161
+ size="small"
162
+ className={classify(css.icon, css.rightIcon)}
163
+ />
164
+ )}
165
+ </UnstyledButton>
166
+ );
167
+ };
168
+
169
+ const renderLabel = (label, allowWrap, showLabelTooltip) => {
170
+ if (showLabelTooltip) {
171
+ return (
172
+ <TruncatedTextWithTooltip
173
+ tooltip={{
174
+ bodyMaxLines: showLabelTooltip.maxLines ?? 10,
175
+ delayMotionDuration: 'normal',
176
+ }}
177
+ line={3}
178
+ >
179
+ {label}
180
+ </TruncatedTextWithTooltip>
181
+ );
182
+ }
183
+
184
+ if (allowWrap) {
185
+ return (
186
+ <TruncatedTextWithTooltip
187
+ tooltip={{
188
+ bodyMaxLines: 10,
189
+ delayMotionDuration: 'normal',
190
+ }}
191
+ line={3}
192
+ >
193
+ {label}
194
+ </TruncatedTextWithTooltip>
195
+ );
196
+ }
197
+
198
+ return <Truncate>{label}</Truncate>;
199
+ };
@@ -0,0 +1,2 @@
1
+ export * from './Menu';
2
+ export * from './MenuOptionButton';
@@ -0,0 +1,354 @@
1
+ import {$ReadOnly} from 'utility-types';
2
+ import * as React from 'react';
3
+ // @ts-expect-error[untyped-import]
4
+ import {createPortal} from 'react-dom';
5
+ import {
6
+ // $FlowFixMe[untyped-import]
7
+ FloatingFocusManager, // $FlowFixMe[untyped-import]
8
+ useFloating, // $FlowFixMe[untyped-import]
9
+ useTransitionStyles,
10
+ } from '@floating-ui/react';
11
+ import useMountTransition from '../../hooks/useMountTransition';
12
+ import {
13
+ motionDurationNormal,
14
+ motionDurationSlow,
15
+ } from '../../styles/variables/_motion';
16
+ import {spaceNegHalfFluid} from '../../styles/variables/_space';
17
+ import classify from '../../utils/classify';
18
+ import {uuid} from '../../utils/helpers';
19
+ import {Button} from '../Button/Button';
20
+ import {Truncate} from '../Truncate/Truncate';
21
+ import css from './Modal.module.css';
22
+ type ClassNames = $ReadOnly<{
23
+ container?: string;
24
+ content?: string;
25
+ backdrop?: string;
26
+ }>;
27
+ export type UseTransitionStylesProps = {
28
+ duration?:
29
+ | number
30
+ | {
31
+ open: number;
32
+ close: number;
33
+ };
34
+ initial?: Readonly<Record<string, unknown>>;
35
+ open?: Readonly<Record<string, unknown>>;
36
+ close?: Readonly<Record<string, unknown>>;
37
+ common?: Readonly<Record<string, unknown>>;
38
+ };
39
+ type FooterClassNames = $ReadOnly<{
40
+ wrapper?: string;
41
+ actions?: string;
42
+ }>;
43
+ export type ModalProps = {
44
+ classNames?: ClassNames;
45
+ children?: React.ReactNode;
46
+ isOpen?: boolean;
47
+ onClose?:
48
+ | ((arg0: React.SyntheticEvent<HTMLElement>) => unknown)
49
+ | null
50
+ | undefined;
51
+ hideBackdrop?: boolean;
52
+ tapOutsideToClose?: boolean;
53
+ allowBackgroundInteraction?: boolean;
54
+ initialFocus?: number;
55
+ customAnimation?: UseTransitionStylesProps;
56
+ };
57
+ export type ModalSize = 'small' | 'medium' | 'large';
58
+ export type BaseModalProps = ModalProps & {
59
+ size?: ModalSize;
60
+ };
61
+ export type ModalHeaderProps = {
62
+ children?: React.ReactNode;
63
+ hideCloseBtn?: boolean;
64
+ onCloseButtonClick?:
65
+ | ((arg0: React.SyntheticEvent<HTMLElement>) => unknown)
66
+ | null
67
+ | undefined;
68
+ className?: string;
69
+ };
70
+ export type ModalFooterProps = {
71
+ children?: React.ReactNode;
72
+ classNames?: FooterClassNames;
73
+ };
74
+ export type ModalBodyProps = {
75
+ children?: React.ReactNode;
76
+ className?: string;
77
+ };
78
+ const DEFAULT_MODAL_ANIMATION = {
79
+ duration: {
80
+ open: parseInt(motionDurationSlow),
81
+ close: parseInt(motionDurationNormal),
82
+ },
83
+ initial: {
84
+ transform: `translate(${spaceNegHalfFluid}, ${spaceNegHalfFluid}) scale(0.95)`,
85
+ },
86
+ open: {
87
+ transform: `translate(${spaceNegHalfFluid}, ${spaceNegHalfFluid}) scale(1)`,
88
+ },
89
+ close: {
90
+ transform: `translate(${spaceNegHalfFluid}, ${spaceNegHalfFluid}) scale(0.95)`,
91
+ },
92
+ };
93
+ export const ModalHeader = ({
94
+ children,
95
+ hideCloseBtn,
96
+ onCloseButtonClick,
97
+ className,
98
+ }: ModalHeaderProps): React.ReactNode => (
99
+ <>
100
+ {React.Children.count(children) > 0 && (
101
+ <div className={classify(css.modalHeader, className)}>
102
+ <div className={css.headerContent}>
103
+ <Truncate>{children}</Truncate>
104
+ </div>
105
+ {!hideCloseBtn && (
106
+ <Button
107
+ iconLeftName="xmark"
108
+ type="ghost"
109
+ onClick={onCloseButtonClick}
110
+ ariaLabel="Close Button"
111
+ ></Button>
112
+ )}
113
+ </div>
114
+ )}
115
+ </>
116
+ );
117
+ export const ModalBody = ({
118
+ children,
119
+ className,
120
+ }: ModalBodyProps): React.ReactNode => (
121
+ <div className={classify(css.modalBody, className)}>{children}</div>
122
+ );
123
+ export const ModalFooter = ({
124
+ children,
125
+ classNames,
126
+ }: ModalFooterProps): React.ReactNode => (
127
+ <>
128
+ {React.Children.count(children) > 0 && (
129
+ <div className={classify(css.modalFooter, classNames?.wrapper)}>
130
+ <div className={classify(css.modalFooterActions, classNames?.actions)}>
131
+ {children}
132
+ </div>
133
+ </div>
134
+ )}
135
+ </>
136
+ );
137
+
138
+ const createPortalRoot = (id: string) => {
139
+ const modalRoot = document.createElement('div');
140
+ modalRoot.setAttribute('id', `modal-root-${id}`);
141
+ return modalRoot;
142
+ };
143
+
144
+ const getModalRoot = (id: string) =>
145
+ document.getElementById(`modal-root-${id}`);
146
+
147
+ function hasChildNode(nodeList) {
148
+ for (let i = 0, len = nodeList.length; i < len; i++) {
149
+ if (nodeList[i].firstChild !== null) {
150
+ return true;
151
+ }
152
+ }
153
+
154
+ return false;
155
+ }
156
+
157
+ const fixBody = (bodyEl: HTMLBodyElement) => {
158
+ const body = document.getElementsByTagName('body')[0];
159
+
160
+ if (body && body.classList) {
161
+ body.classList.add('fixed');
162
+ }
163
+
164
+ if (bodyEl) {
165
+ bodyEl.style.overflow = 'hidden';
166
+ }
167
+ };
168
+
169
+ const unfixBody = (bodyEl: HTMLBodyElement) => {
170
+ const body = document.getElementsByTagName('body')[0];
171
+
172
+ if (body && body.classList) {
173
+ body.classList.remove('fixed');
174
+ }
175
+
176
+ if (bodyEl) {
177
+ bodyEl.style.overflow = '';
178
+ }
179
+ };
180
+
181
+ const checkAndAddBodyOverflow = (bodyEl: HTMLBodyElement) => {
182
+ const nodes = document.querySelectorAll('[id^="modal-root"]');
183
+ let isParentModalPresent = false;
184
+
185
+ if (nodes.length) {
186
+ isParentModalPresent = hasChildNode(nodes);
187
+ }
188
+
189
+ if (isParentModalPresent) {
190
+ fixBody(bodyEl);
191
+ } else {
192
+ unfixBody(bodyEl);
193
+ }
194
+ };
195
+
196
+ export const Modal = ({
197
+ classNames,
198
+ children,
199
+ isOpen = false,
200
+ onClose,
201
+ hideBackdrop = false,
202
+ tapOutsideToClose = true,
203
+ initialFocus = -1,
204
+ customAnimation,
205
+ allowBackgroundInteraction = false,
206
+ // Size is not set to default for backward compatibility. Don't change if you don't know this.
207
+ size,
208
+ }: BaseModalProps): React.ReactNode => {
209
+ const {refs, context} = useFloating({
210
+ open: isOpen,
211
+ });
212
+ const {isMounted, styles} = useTransitionStyles(
213
+ context,
214
+ customAnimation || DEFAULT_MODAL_ANIMATION,
215
+ );
216
+ const modalId = uuid();
217
+ const bodyRef = React.useRef(document.querySelector('body'));
218
+ const portalRootRef = React.useRef(
219
+ getModalRoot(modalId) || createPortalRoot(modalId),
220
+ );
221
+ const isTransitioning = useMountTransition(
222
+ isOpen,
223
+ parseInt(motionDurationNormal),
224
+ );
225
+ // Append portal root on mount
226
+ React.useEffect(() => {
227
+ bodyRef.current?.appendChild(portalRootRef.current);
228
+ const portal = portalRootRef.current;
229
+ const bodyEl = bodyRef.current;
230
+ return () => {
231
+ // Clean up the portal when modal component unmounts
232
+ portal.remove();
233
+
234
+ // Ensure scroll overflow is removed
235
+ if (bodyEl) {
236
+ unfixBody(bodyEl);
237
+ }
238
+ };
239
+ }, []);
240
+ // Prevent page scrolling when the modal is open
241
+ React.useEffect(() => {
242
+ const updatePageScroll = () => {
243
+ if (isOpen) {
244
+ if (bodyRef.current) {
245
+ fixBody(bodyRef.current);
246
+ }
247
+ } else {
248
+ if (bodyRef.current) {
249
+ unfixBody(bodyRef.current);
250
+ }
251
+ }
252
+ };
253
+
254
+ updatePageScroll();
255
+ }, [isOpen]);
256
+ // Allow Escape key to dismiss the modal
257
+ React.useEffect(() => {
258
+ const onKeyPress = (e) => {
259
+ if (e.key === 'Escape') {
260
+ tapOutsideToClose && onClose && onClose(e);
261
+ }
262
+ };
263
+
264
+ if (isOpen) {
265
+ window.addEventListener('keyup', onKeyPress);
266
+ }
267
+
268
+ return () => {
269
+ window.removeEventListener('keyup', onKeyPress);
270
+ };
271
+ }, [isOpen, onClose]);
272
+
273
+ if (!isTransitioning && !isOpen) {
274
+ // Check overflow after resetting the DOM for modal. This should always happen after DOM reset
275
+ // TODO(Nishant): Better way to do this?
276
+ setTimeout(() => {
277
+ if (bodyRef && !!bodyRef.current) {
278
+ checkAndAddBodyOverflow(bodyRef.current);
279
+ }
280
+ });
281
+ return null;
282
+ }
283
+
284
+ const onBackdropClick = (e: React.SyntheticEvent<HTMLElement>) => {
285
+ if (tapOutsideToClose && onClose) {
286
+ onClose(e);
287
+ }
288
+ };
289
+
290
+ return createPortal(
291
+ <FloatingFocusManager context={context} initialFocus={initialFocus}>
292
+ <div
293
+ ref={refs.setFloating}
294
+ aria-hidden={isOpen ? 'false' : 'true'}
295
+ className={classify(
296
+ css.modalContainer,
297
+ {
298
+ [css.in]: isTransitioning,
299
+ [css.open]: isOpen,
300
+ },
301
+ classNames?.container,
302
+ )}
303
+ >
304
+ {!allowBackgroundInteraction && (
305
+ <div
306
+ className={classify(
307
+ css.backdrop,
308
+ {
309
+ [css.darkBackdrop]: !hideBackdrop,
310
+ },
311
+ classNames?.backdrop,
312
+ )}
313
+ onClick={onBackdropClick}
314
+ />
315
+ )}
316
+
317
+ {isMounted && (
318
+ <div
319
+ className={classify(
320
+ css.modal,
321
+ {
322
+ [css.small]: size === 'small',
323
+ [css.medium]: size === 'medium',
324
+ [css.large]: size === 'large',
325
+ },
326
+ classNames?.content,
327
+ )}
328
+ role="dialog"
329
+ style={{
330
+ // Transition styles
331
+ ...styles,
332
+ }}
333
+ >
334
+ {children}
335
+ </div>
336
+ )}
337
+ </div>
338
+ </FloatingFocusManager>,
339
+ portalRootRef.current,
340
+ );
341
+ };
342
+ // FullScreen modal that just calls the Modal component with a wrapper className set which sets the width and height to sizeFluid
343
+ export const FullScreenModal = ({
344
+ classNames,
345
+ ...props
346
+ }: ModalProps): React.ReactNode => (
347
+ <Modal
348
+ classNames={{
349
+ ...classNames,
350
+ content: classify(css.fullscreenModalContainer, classNames?.content),
351
+ }}
352
+ {...props}
353
+ />
354
+ );
@@ -0,0 +1,14 @@
1
+ export type {
2
+ BaseModalProps,
3
+ ModalBodyProps,
4
+ ModalFooterProps,
5
+ ModalHeaderProps,
6
+ ModalProps,
7
+ } from './Modal';
8
+ export {
9
+ FullScreenModal,
10
+ Modal,
11
+ ModalBody,
12
+ ModalFooter,
13
+ ModalHeader,
14
+ } from './Modal';
@@ -0,0 +1,147 @@
1
+ import {$ReadOnly, $Values} from 'utility-types';
2
+ import * as React from 'react';
3
+ import {TEXT_COLORS} from '../../types/typography';
4
+ import {classify} from '../../utils/classify';
5
+ import type {IconType} from '../Icon';
6
+ import {Icon} from '../Icon';
7
+ import {SubTitleSmall} from '../Text';
8
+ import css from './Notification.module.css';
9
+ export const NOTIFICATION_SEMANTIC = Object.freeze({
10
+ success: 'success',
11
+ information: 'information',
12
+ danger: 'danger',
13
+ });
14
+ type ClassNames = $ReadOnly<{
15
+ wrapper?: string;
16
+ text?: string;
17
+ }>;
18
+ export type NotificationSemanticType = $Values<typeof NOTIFICATION_SEMANTIC>;
19
+ type BaseNotificationProps = {
20
+ classNames?: ClassNames;
21
+ semantic: NotificationSemanticType;
22
+ iconLeftName?: string;
23
+ iconLeftType?: IconType;
24
+ dismissable?: boolean;
25
+ children?: string;
26
+ onCloseClick?:
27
+ | ((arg0: React.SyntheticEvent<HTMLElement>) => unknown)
28
+ | null
29
+ | undefined;
30
+ };
31
+ type NotificationBaseProps = BaseNotificationProps &
32
+ (
33
+ | {
34
+ dismissable: true;
35
+ onCloseClick?:
36
+ | ((arg0: React.SyntheticEvent<HTMLElement>) => unknown)
37
+ | null
38
+ | undefined;
39
+ selfDismiss?: boolean;
40
+ }
41
+ | {
42
+ dismissable?: false;
43
+ }
44
+ ) & {
45
+ classNames?: ClassNames;
46
+ };
47
+ export type NotificationProps = NotificationBaseProps & {
48
+ semantic: NotificationSemanticType;
49
+ };
50
+ // Base Notification will not have any state.
51
+ const BaseNotification: React$AbstractComponent<
52
+ BaseNotificationProps,
53
+ HTMLDivElement
54
+ > = React.forwardRef<BaseNotificationProps, HTMLDivElement>(
55
+ (props: BaseNotificationProps, ref): React.ReactNode => {
56
+ const {
57
+ classNames,
58
+ semantic = NOTIFICATION_SEMANTIC.success,
59
+ iconLeftName,
60
+ iconLeftType = 'regular',
61
+ dismissable,
62
+ children,
63
+ onCloseClick,
64
+ } = props;
65
+ return (
66
+ <div
67
+ className={classify(
68
+ css.baseContainer,
69
+ {
70
+ [css.success]: semantic === NOTIFICATION_SEMANTIC.success,
71
+ [css.information]: semantic === NOTIFICATION_SEMANTIC.information,
72
+ [css.danger]: semantic === NOTIFICATION_SEMANTIC.danger,
73
+ [css.dismissable]: dismissable,
74
+ },
75
+ classNames?.wrapper,
76
+ )}
77
+ ref={ref}
78
+ >
79
+ {iconLeftName ? (
80
+ <Icon
81
+ name={iconLeftName}
82
+ color={TEXT_COLORS.inversePrimary}
83
+ size="small"
84
+ type={iconLeftType}
85
+ />
86
+ ) : null}
87
+ <SubTitleSmall
88
+ className={classify(classNames?.text)}
89
+ color={TEXT_COLORS.inversePrimary}
90
+ >
91
+ {children}
92
+ </SubTitleSmall>
93
+ {dismissable && (
94
+ <Icon
95
+ color={TEXT_COLORS.inversePrimary}
96
+ name="close"
97
+ size="small"
98
+ type="regular"
99
+ onClick={onCloseClick}
100
+ className={css.closeIcon}
101
+ />
102
+ )}
103
+ </div>
104
+ );
105
+ },
106
+ );
107
+ export const Notification: React$AbstractComponent<
108
+ NotificationProps,
109
+ HTMLDivElement
110
+ > = React.forwardRef<NotificationProps, HTMLDivElement>(
111
+ (props: NotificationProps, ref): React.ReactNode => {
112
+ const {
113
+ classNames,
114
+ semantic = NOTIFICATION_SEMANTIC.success,
115
+ iconLeftName,
116
+ iconLeftType,
117
+ dismissable,
118
+ children,
119
+ selfDismiss = false,
120
+ onCloseClick,
121
+ } = props;
122
+ const [dismissed, setDismissed] = React.useState(false);
123
+
124
+ const closeClickHandler = (e) => {
125
+ onCloseClick && onCloseClick(e);
126
+ selfDismiss && setDismissed(true);
127
+ };
128
+
129
+ return (
130
+ <>
131
+ {!dismissed && (
132
+ <BaseNotification
133
+ classNames={classNames}
134
+ iconLeftName={iconLeftName}
135
+ iconLeftType={iconLeftType}
136
+ semantic={semantic}
137
+ dismissable={dismissable}
138
+ onCloseClick={closeClickHandler}
139
+ ref={ref}
140
+ >
141
+ {children}
142
+ </BaseNotification>
143
+ )}
144
+ </>
145
+ );
146
+ },
147
+ );
@@ -0,0 +1 @@
1
+ export {Notification, NOTIFICATION_SEMANTIC} from './Notification';