@cube-dev/ui-kit 0.140.1 → 0.142.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (456) hide show
  1. package/dist/CHANGELOG.md +96 -0
  2. package/dist/_internal/hooks/use-chained-callback.js +1 -1
  3. package/dist/_internal/hooks/use-debounced-value.js +1 -1
  4. package/dist/_internal/hooks/use-deprecation-warning.js +1 -1
  5. package/dist/_internal/hooks/use-event.js +1 -1
  6. package/dist/_internal/hooks/use-is-first-render.js +1 -1
  7. package/dist/_internal/hooks/use-sync-ref.js +1 -1
  8. package/dist/_internal/hooks/use-timer/timer.js +1 -1
  9. package/dist/_internal/hooks/use-timer/use-timer.js +1 -1
  10. package/dist/_internal/hooks/use-warn.js +1 -1
  11. package/dist/components/Block.js +1 -1
  12. package/dist/components/CollectionItem.js +1 -1
  13. package/dist/components/GlobalStyles.js +1 -1
  14. package/dist/components/GridProvider.js +1 -1
  15. package/dist/components/HiddenInput.js +1 -1
  16. package/dist/components/Root.js +1 -1
  17. package/dist/components/actions/Action/Action.js +1 -1
  18. package/dist/components/actions/Banner/Banner.js +1 -1
  19. package/dist/components/actions/Button/Button.js +17 -3
  20. package/dist/components/actions/Button/Button.js.map +1 -1
  21. package/dist/components/actions/ButtonGroup/ButtonGroup.js +1 -1
  22. package/dist/components/actions/ButtonSplit/ButtonSplit.js +1 -1
  23. package/dist/components/actions/ButtonSplit/context.js +1 -1
  24. package/dist/components/actions/CommandMenu/CommandMenu.js +3 -4
  25. package/dist/components/actions/CommandMenu/CommandMenu.js.map +1 -1
  26. package/dist/components/actions/CommandMenu/styled.js +1 -1
  27. package/dist/components/actions/ItemAction/ItemAction.js +1 -1
  28. package/dist/components/actions/ItemActionContext.js +1 -1
  29. package/dist/components/actions/ItemButton/ItemButton.js +18 -2
  30. package/dist/components/actions/ItemButton/ItemButton.js.map +1 -1
  31. package/dist/components/actions/Link/Link.js +1 -1
  32. package/dist/components/actions/Menu/Menu.js +3 -4
  33. package/dist/components/actions/Menu/Menu.js.map +1 -1
  34. package/dist/components/actions/Menu/MenuItem.js +1 -1
  35. package/dist/components/actions/Menu/MenuSection.js +1 -1
  36. package/dist/components/actions/Menu/MenuTrigger.js +8 -5
  37. package/dist/components/actions/Menu/MenuTrigger.js.map +1 -1
  38. package/dist/components/actions/Menu/SubMenuTrigger.js +10 -7
  39. package/dist/components/actions/Menu/SubMenuTrigger.js.map +1 -1
  40. package/dist/components/actions/Menu/SubmenuTriggerContext.js +1 -1
  41. package/dist/components/actions/Menu/context.js +1 -1
  42. package/dist/components/actions/Menu/styled.js +1 -1
  43. package/dist/components/actions/index.js +1 -1
  44. package/dist/components/actions/use-action.js +1 -1
  45. package/dist/components/actions/use-anchored-menu.js +4 -3
  46. package/dist/components/actions/use-anchored-menu.js.map +1 -1
  47. package/dist/components/actions/use-context-menu.js +4 -3
  48. package/dist/components/actions/use-context-menu.js.map +1 -1
  49. package/dist/components/content/ActiveZone/ActiveZone.js +1 -1
  50. package/dist/components/content/Alert/Alert.js +1 -1
  51. package/dist/components/content/Alert/use-alert.js +1 -1
  52. package/dist/components/content/Avatar/Avatar.js +1 -1
  53. package/dist/components/content/Badge/Badge.js +1 -1
  54. package/dist/components/content/Card/Card.js +1 -1
  55. package/dist/components/content/Content.js +1 -1
  56. package/dist/components/content/CopyPasteBlock/CopyPasteBlock.js +1 -1
  57. package/dist/components/content/CopySnippet/CopySnippet.js +1 -1
  58. package/dist/components/content/Disclosure/Disclosure.js +1 -1
  59. package/dist/components/content/Divider.js +1 -1
  60. package/dist/components/content/Footer.js +1 -1
  61. package/dist/components/content/Header.js +1 -1
  62. package/dist/components/content/HotKeys/HotKeys.js +1 -1
  63. package/dist/components/content/InlineInput/InlineInput.js +1 -1
  64. package/dist/components/content/Item/Item.js +1 -1
  65. package/dist/components/content/ItemBadge/ItemBadge.js +1 -1
  66. package/dist/components/content/ItemCard/ItemCard.js +1 -1
  67. package/dist/components/content/Layout/GridLayout.js +1 -1
  68. package/dist/components/content/Layout/Layout.js +1 -1
  69. package/dist/components/content/Layout/LayoutBlock.js +1 -1
  70. package/dist/components/content/Layout/LayoutCenter.js +1 -1
  71. package/dist/components/content/Layout/LayoutContainer.js +1 -1
  72. package/dist/components/content/Layout/LayoutContent.js +1 -1
  73. package/dist/components/content/Layout/LayoutContext.js +1 -1
  74. package/dist/components/content/Layout/LayoutFlex.js +1 -1
  75. package/dist/components/content/Layout/LayoutFooter.js +1 -1
  76. package/dist/components/content/Layout/LayoutGrid.js +1 -1
  77. package/dist/components/content/Layout/LayoutHeader.js +1 -1
  78. package/dist/components/content/Layout/LayoutPane.js +1 -1
  79. package/dist/components/content/Layout/LayoutPanel.js +1 -1
  80. package/dist/components/content/Layout/LayoutPanelHeader.js +1 -1
  81. package/dist/components/content/Layout/LayoutToolbar.js +1 -1
  82. package/dist/components/content/Layout/hooks/useTinyScrollbar.js +1 -1
  83. package/dist/components/content/Layout/index.js +1 -1
  84. package/dist/components/content/Layout/utils.js +1 -1
  85. package/dist/components/content/Paragraph.js +1 -1
  86. package/dist/components/content/Placeholder/Placeholder.js +1 -1
  87. package/dist/components/content/PrismCode/PrismCode.js +1 -1
  88. package/dist/components/content/PrismCode/prismSetup.js +1 -1
  89. package/dist/components/content/PrismDiffCode/PrismDiffCode.js +1 -1
  90. package/dist/components/content/Result/Result.js +1 -1
  91. package/dist/components/content/Skeleton/Skeleton.js +1 -1
  92. package/dist/components/content/Tag/Tag.js +1 -1
  93. package/dist/components/content/Text.js +1 -1
  94. package/dist/components/content/TextItem/TextItem.js +1 -1
  95. package/dist/components/content/Title.js +1 -1
  96. package/dist/components/content/Tree/Tree.js +1 -1
  97. package/dist/components/content/Tree/TreeNode.js +1 -1
  98. package/dist/components/content/Tree/styled.js +1 -1
  99. package/dist/components/content/Tree/tree-index.js +1 -1
  100. package/dist/components/content/Tree/use-checkbox-tree.js +1 -1
  101. package/dist/components/content/Tree/use-load-data.js +1 -1
  102. package/dist/components/content/highlightText.js +1 -1
  103. package/dist/components/content/use-auto-tooltip.js +1 -1
  104. package/dist/components/fields/Checkbox/Checkbox.js +1 -1
  105. package/dist/components/fields/Checkbox/CheckboxGroup.js +3 -3
  106. package/dist/components/fields/Checkbox/CheckboxGroup.js.map +1 -1
  107. package/dist/components/fields/Checkbox/context.js +1 -1
  108. package/dist/components/fields/ComboBox/ComboBox.js +11 -7
  109. package/dist/components/fields/ComboBox/ComboBox.js.map +1 -1
  110. package/dist/components/fields/DatePicker/DateInput.js +1 -1
  111. package/dist/components/fields/DatePicker/DateInputBase.js +1 -1
  112. package/dist/components/fields/DatePicker/DatePicker.js +1 -1
  113. package/dist/components/fields/DatePicker/DatePickerButton.js +1 -1
  114. package/dist/components/fields/DatePicker/DatePickerElement.js +1 -1
  115. package/dist/components/fields/DatePicker/DatePickerInput.js +1 -1
  116. package/dist/components/fields/DatePicker/DatePickerSegment.js +1 -1
  117. package/dist/components/fields/DatePicker/DateRangePicker.js +1 -1
  118. package/dist/components/fields/DatePicker/DateRangeSeparatedPicker.js +1 -1
  119. package/dist/components/fields/DatePicker/TimeInput.js +1 -1
  120. package/dist/components/fields/DatePicker/intl.js +1 -1
  121. package/dist/components/fields/DatePicker/parseDate.js +1 -1
  122. package/dist/components/fields/DatePicker/props.js +1 -1
  123. package/dist/components/fields/DatePicker/utils.js +1 -1
  124. package/dist/components/fields/FileInput/FileInput.js +1 -1
  125. package/dist/components/fields/FilterListBox/FilterListBox.js +1 -1
  126. package/dist/components/fields/FilterPicker/FilterPicker.js +1 -9
  127. package/dist/components/fields/FilterPicker/FilterPicker.js.map +1 -1
  128. package/dist/components/fields/Input/Input.js +1 -1
  129. package/dist/components/fields/ListBox/DraggableListBox.js +1 -1
  130. package/dist/components/fields/ListBox/ListBox.js +2 -1
  131. package/dist/components/fields/ListBox/ListBox.js.map +1 -1
  132. package/dist/components/fields/NumberInput/NumberInput.js +1 -1
  133. package/dist/components/fields/NumberInput/StepButton.js +1 -1
  134. package/dist/components/fields/PasswordInput/PasswordInput.js +1 -1
  135. package/dist/components/fields/Picker/Picker.js +1 -9
  136. package/dist/components/fields/Picker/Picker.js.map +1 -1
  137. package/dist/components/fields/RadioGroup/Radio.js +1 -1
  138. package/dist/components/fields/RadioGroup/RadioGroup.js +3 -3
  139. package/dist/components/fields/RadioGroup/RadioGroup.js.map +1 -1
  140. package/dist/components/fields/RadioGroup/context.js +1 -1
  141. package/dist/components/fields/SearchInput/SearchInput.js +1 -1
  142. package/dist/components/fields/Select/Select.js +10 -7
  143. package/dist/components/fields/Select/Select.js.map +1 -1
  144. package/dist/components/fields/Slider/Gradation.js +1 -1
  145. package/dist/components/fields/Slider/HueSlider.js +1 -1
  146. package/dist/components/fields/Slider/RangeSlider.js +1 -1
  147. package/dist/components/fields/Slider/Slider.js +1 -1
  148. package/dist/components/fields/Slider/SliderBase.js +1 -1
  149. package/dist/components/fields/Slider/SliderThumb.js +1 -1
  150. package/dist/components/fields/Slider/SliderTrack.js +1 -1
  151. package/dist/components/fields/Slider/elements.js +1 -1
  152. package/dist/components/fields/Slider/index.js +1 -1
  153. package/dist/components/fields/Switch/Switch.js +1 -1
  154. package/dist/components/fields/TextArea/TextArea.js +1 -1
  155. package/dist/components/fields/TextInput/TextInput.js +1 -1
  156. package/dist/components/fields/TextInput/TextInputBase.js +1 -1
  157. package/dist/components/fields/TextInputMapper/TextInputMapper.js +1 -1
  158. package/dist/components/form/FieldWrapper/FieldWrapper.js +1 -1
  159. package/dist/components/form/FieldWrapper/extract-field-wrapper-props.js +1 -1
  160. package/dist/components/form/Form/Field.js +1 -1
  161. package/dist/components/form/Form/Form.d.ts +1 -2
  162. package/dist/components/form/Form/Form.js +3 -3
  163. package/dist/components/form/Form/Form.js.map +1 -1
  164. package/dist/components/form/Form/ResetButton/ResetButton.js +1 -1
  165. package/dist/components/form/Form/SubmitButton/SubmitButton.js +1 -1
  166. package/dist/components/form/Form/SubmitError.js +1 -1
  167. package/dist/components/form/Form/index.d.ts +1 -1
  168. package/dist/components/form/Form/index.js +1 -1
  169. package/dist/components/form/Form/use-field/use-field-props.js +1 -1
  170. package/dist/components/form/Form/use-field/use-field.js +1 -1
  171. package/dist/components/form/Form/use-form.js +1 -1
  172. package/dist/components/form/Form/validation.js +1 -1
  173. package/dist/components/form/Label.js +3 -3
  174. package/dist/components/form/Label.js.map +1 -1
  175. package/dist/components/form/wrapper.js +1 -1
  176. package/dist/components/helpers/DisplayTransition/DisplayTransition.js +1 -1
  177. package/dist/components/helpers/IconSwitch/IconSwitch.js +1 -1
  178. package/dist/components/layout/Flex.js +1 -1
  179. package/dist/components/layout/Flow.js +1 -1
  180. package/dist/components/layout/Grid.js +1 -1
  181. package/dist/components/layout/Panel.js +1 -1
  182. package/dist/components/layout/Prefix.js +1 -1
  183. package/dist/components/layout/ResizablePanel.js +1 -1
  184. package/dist/components/layout/Space.js +1 -1
  185. package/dist/components/layout/Suffix.js +1 -1
  186. package/dist/components/navigation/Tabs/DraggableTabList.js +1 -1
  187. package/dist/components/navigation/Tabs/TabButton.js +2 -2
  188. package/dist/components/navigation/Tabs/TabButton.js.map +1 -1
  189. package/dist/components/navigation/Tabs/TabDropIndicator.js +1 -1
  190. package/dist/components/navigation/Tabs/TabPanel.js +1 -1
  191. package/dist/components/navigation/Tabs/TabPicker.js +1 -1
  192. package/dist/components/navigation/Tabs/Tabs.js +1 -1
  193. package/dist/components/navigation/Tabs/TabsAction.js +1 -1
  194. package/dist/components/navigation/Tabs/TabsContext.js +1 -1
  195. package/dist/components/navigation/Tabs/popover-placement.js +1 -1
  196. package/dist/components/navigation/Tabs/styled.js +1 -1
  197. package/dist/components/navigation/Tabs/types.d.ts +6 -2
  198. package/dist/components/navigation/Tabs/types.js +1 -1
  199. package/dist/components/navigation/Tabs/types.js.map +1 -1
  200. package/dist/components/navigation/Tabs/use-tab-editing.js +1 -1
  201. package/dist/components/navigation/Tabs/use-tab-indicator.js +1 -1
  202. package/dist/components/organisms/FileTabs/FileTabs.js +1 -1
  203. package/dist/components/organisms/StatsCard/StatsCard.js +1 -1
  204. package/dist/components/other/Calendar/Calendar.js +1 -1
  205. package/dist/components/other/Calendar/CalendarCell.js +1 -1
  206. package/dist/components/other/Calendar/CalendarGrid.js +1 -1
  207. package/dist/components/other/Calendar/RangeCalendar.js +1 -1
  208. package/dist/components/other/CloudLogo/CloudLogo.js +1 -1
  209. package/dist/components/overlays/AlertDialog/AlertDialog.js +1 -1
  210. package/dist/components/overlays/AlertDialog/AlertDialogApiProvider.js +1 -1
  211. package/dist/components/overlays/AlertDialog/AlertDialogZone.js +1 -1
  212. package/dist/components/overlays/Dialog/Dialog.d.ts +1 -2
  213. package/dist/components/overlays/Dialog/Dialog.js +3 -3
  214. package/dist/components/overlays/Dialog/Dialog.js.map +1 -1
  215. package/dist/components/overlays/Dialog/DialogContainer.js +1 -1
  216. package/dist/components/overlays/Dialog/DialogForm.js +1 -1
  217. package/dist/components/overlays/Dialog/DialogTrigger.js +34 -3
  218. package/dist/components/overlays/Dialog/DialogTrigger.js.map +1 -1
  219. package/dist/components/overlays/Dialog/context.js +1 -1
  220. package/dist/components/overlays/Dialog/use-dialog-container.js +1 -1
  221. package/dist/components/overlays/Modal/Modal.js +3 -3
  222. package/dist/components/overlays/Modal/Modal.js.map +1 -1
  223. package/dist/components/overlays/Modal/OpenTransitionContext.js +1 -1
  224. package/dist/components/overlays/Modal/Overlay.js +1 -1
  225. package/dist/components/overlays/Modal/Popover.js +1 -1
  226. package/dist/components/overlays/Modal/Tray.js +3 -3
  227. package/dist/components/overlays/Modal/Tray.js.map +1 -1
  228. package/dist/components/overlays/Modal/Underlay.js +1 -1
  229. package/dist/components/overlays/Notifications/Notification.js +1 -1
  230. package/dist/components/overlays/Notifications/NotificationAction.js +1 -1
  231. package/dist/components/overlays/Notifications/NotificationCard.js +1 -1
  232. package/dist/components/overlays/Notifications/NotificationContext.js +1 -1
  233. package/dist/components/overlays/Notifications/NotificationItem.js +1 -1
  234. package/dist/components/overlays/Notifications/OverlayContainer.js +1 -1
  235. package/dist/components/overlays/Notifications/OverlayProvider.js +1 -1
  236. package/dist/components/overlays/Notifications/PersistentNotificationsList.js +1 -1
  237. package/dist/components/overlays/Notifications/dismissed-storage.js +1 -1
  238. package/dist/components/overlays/Notifications/format-relative-time.js +1 -1
  239. package/dist/components/overlays/Notifications/index.js +1 -1
  240. package/dist/components/overlays/Notifications/use-notification-state.js +1 -1
  241. package/dist/components/overlays/Notifications/use-notifications.js +1 -1
  242. package/dist/components/overlays/Notifications/use-overlay-timers.js +1 -1
  243. package/dist/components/overlays/Notifications/use-persistent-notifications.js +1 -1
  244. package/dist/components/overlays/Notifications/use-persistent-state.js +1 -1
  245. package/dist/components/overlays/Notifications/use-toast-state.js +1 -1
  246. package/dist/components/overlays/Toast/ToastItem.js +1 -1
  247. package/dist/components/overlays/Toast/index.js +1 -1
  248. package/dist/components/overlays/Toast/useProgressToast.js +1 -1
  249. package/dist/components/overlays/Toast/useToast.js +1 -1
  250. package/dist/components/overlays/Tooltip/Tooltip.js +1 -1
  251. package/dist/components/overlays/Tooltip/TooltipProvider.js +1 -1
  252. package/dist/components/overlays/Tooltip/TooltipTrigger.js +1 -1
  253. package/dist/components/overlays/Tooltip/context.js +1 -1
  254. package/dist/components/portal/Portal.js +1 -1
  255. package/dist/components/portal/PortalProvider.js +1 -1
  256. package/dist/components/portal/usePortal.js +1 -1
  257. package/dist/components/shared/DraggableCollection.js +1 -1
  258. package/dist/components/shared/InvalidIcon.js +1 -1
  259. package/dist/components/shared/ValidIcon.js +1 -1
  260. package/dist/components/status/LoadingAnimation/LoadingAnimation.js +1 -1
  261. package/dist/components/status/Spin/Cube.js +1 -1
  262. package/dist/components/status/Spin/InternalSpinner.js +1 -1
  263. package/dist/components/status/Spin/Spin.js +1 -1
  264. package/dist/components/status/Spin/SpinsContainer.js +1 -1
  265. package/dist/data/item-themes.js +1 -1
  266. package/dist/data/themes.js +1 -1
  267. package/dist/icons/AdjustmentsHorizontalIcon.js +1 -1
  268. package/dist/icons/AdjustmentsIcon.js +1 -1
  269. package/dist/icons/AiIcon.js +1 -1
  270. package/dist/icons/AreaChartIcon.js +1 -1
  271. package/dist/icons/BackwardIcon.js +1 -1
  272. package/dist/icons/BarChartIcon.js +1 -1
  273. package/dist/icons/BellFilledIcon.js +1 -1
  274. package/dist/icons/BellIcon.js +1 -1
  275. package/dist/icons/BooleanIcon.js +1 -1
  276. package/dist/icons/CalendarEditIcon.js +1 -1
  277. package/dist/icons/CalendarIcon.js +1 -1
  278. package/dist/icons/CaretDownIcon.js +1 -1
  279. package/dist/icons/CaretUpIcon.js +1 -1
  280. package/dist/icons/ChartAreaStackedIcon.js +1 -1
  281. package/dist/icons/ChartAreaStackedPercentageIcon.js +1 -1
  282. package/dist/icons/ChartBarGroupedHorizontalIcon.js +1 -1
  283. package/dist/icons/ChartBarGroupedIcon.js +1 -1
  284. package/dist/icons/ChartBarHorizontalIcon.js +1 -1
  285. package/dist/icons/ChartBarLineIcon.js +1 -1
  286. package/dist/icons/ChartBarStackedHorizontalIcon.js +1 -1
  287. package/dist/icons/ChartBarStackedIcon.js +1 -1
  288. package/dist/icons/ChartBarStackedPercentageHorizontalIcon.js +1 -1
  289. package/dist/icons/ChartBarStackedPercentageIcon.js +1 -1
  290. package/dist/icons/ChartBoxPlot2Icon.js +1 -1
  291. package/dist/icons/ChartBoxPlotIcon.js +1 -1
  292. package/dist/icons/ChartBubbleIcon.js +1 -1
  293. package/dist/icons/ChartDonut2Icon.js +1 -1
  294. package/dist/icons/ChartFunnelIcon.js +1 -1
  295. package/dist/icons/ChartHeatmapIcon.js +1 -1
  296. package/dist/icons/ChartKPIIcon.js +1 -1
  297. package/dist/icons/ChartPie2Icon.js +1 -1
  298. package/dist/icons/ChartScatterIcon.js +1 -1
  299. package/dist/icons/CheckCircleFilledIcon.js +1 -1
  300. package/dist/icons/CheckCircleIcon.js +1 -1
  301. package/dist/icons/CheckIcon.js +1 -1
  302. package/dist/icons/CircleFilledIcon.js +1 -1
  303. package/dist/icons/ClearIcon.js +1 -1
  304. package/dist/icons/CloseCircleFilledIcon.js +1 -1
  305. package/dist/icons/CloseCircleIcon.js +1 -1
  306. package/dist/icons/CloseIcon.js +1 -1
  307. package/dist/icons/CodeIcon.js +1 -1
  308. package/dist/icons/ColumnTotalIcon.js +1 -1
  309. package/dist/icons/CopyIcon.js +1 -1
  310. package/dist/icons/CountIcon.js +1 -1
  311. package/dist/icons/CubeIcon.js +1 -1
  312. package/dist/icons/CubePauseIcon.js +1 -1
  313. package/dist/icons/CubePlayIcon.js +1 -1
  314. package/dist/icons/CurrencyDollarIcon.js +1 -1
  315. package/dist/icons/DangerIcon.js +1 -1
  316. package/dist/icons/DashboardIcon.js +1 -1
  317. package/dist/icons/DatabaseIcon.js +1 -1
  318. package/dist/icons/DecimalDecreaseIcon.js +1 -1
  319. package/dist/icons/DecimalIncreaseIcon.js +1 -1
  320. package/dist/icons/DirectionIcon.js +1 -1
  321. package/dist/icons/DonutIcon.js +1 -1
  322. package/dist/icons/DownIcon.js +1 -1
  323. package/dist/icons/EditIcon.js +1 -1
  324. package/dist/icons/ExclamationCircleFilledIcon.js +1 -1
  325. package/dist/icons/ExclamationCircleIcon.js +1 -1
  326. package/dist/icons/ExclamationIcon.js +1 -1
  327. package/dist/icons/EyeIcon.js +1 -1
  328. package/dist/icons/EyeInvisibleIcon.js +1 -1
  329. package/dist/icons/FilterIcon.js +1 -1
  330. package/dist/icons/FolderFilledIcon.js +1 -1
  331. package/dist/icons/FolderIcon.js +1 -1
  332. package/dist/icons/FolderOpenFilledIcon.js +1 -1
  333. package/dist/icons/FolderOpenIcon.js +1 -1
  334. package/dist/icons/ForwardIcon.js +1 -1
  335. package/dist/icons/GripVerticalIcon.js +1 -1
  336. package/dist/icons/HierarchyIcon.js +1 -1
  337. package/dist/icons/HierarchyOpenIcon.js +1 -1
  338. package/dist/icons/Icon.js +1 -1
  339. package/dist/icons/InfoCircleIcon.js +1 -1
  340. package/dist/icons/InfoIcon.js +1 -1
  341. package/dist/icons/KeyIcon.js +1 -1
  342. package/dist/icons/LeftIcon.js +1 -1
  343. package/dist/icons/LineChartIcon.js +1 -1
  344. package/dist/icons/LoadingIcon.js +1 -1
  345. package/dist/icons/LockFilledIcon.js +1 -1
  346. package/dist/icons/LockIcon.js +1 -1
  347. package/dist/icons/MoreIcon.js +1 -1
  348. package/dist/icons/NotAllowedIcon.js +1 -1
  349. package/dist/icons/Number123Icon.js +1 -1
  350. package/dist/icons/NumberIcon.js +1 -1
  351. package/dist/icons/PauseCircleFilledIcon.js +1 -1
  352. package/dist/icons/PauseCircleIcon.js +1 -1
  353. package/dist/icons/PauseIcon.js +1 -1
  354. package/dist/icons/PercentageIcon.js +1 -1
  355. package/dist/icons/PieChartIcon.js +1 -1
  356. package/dist/icons/PlayCircleIcon.js +1 -1
  357. package/dist/icons/PlayIcon.js +1 -1
  358. package/dist/icons/PlusIcon.js +1 -1
  359. package/dist/icons/ProgressBarIcon.js +1 -1
  360. package/dist/icons/ReloadIcon.js +1 -1
  361. package/dist/icons/ReportIcon.js +1 -1
  362. package/dist/icons/ReturnIcon.js +1 -1
  363. package/dist/icons/RightIcon.js +1 -1
  364. package/dist/icons/RowTotalsIcon.js +1 -1
  365. package/dist/icons/SchemeIcon.js +1 -1
  366. package/dist/icons/SearchIcon.js +1 -1
  367. package/dist/icons/SemanticQueryIcon.js +1 -1
  368. package/dist/icons/SettingsIcon.js +1 -1
  369. package/dist/icons/ShieldFilledIcon.js +1 -1
  370. package/dist/icons/ShieldIcon.js +1 -1
  371. package/dist/icons/SlashIcon.js +1 -1
  372. package/dist/icons/SparklesIcon.js +1 -1
  373. package/dist/icons/SqlIcon.js +1 -1
  374. package/dist/icons/StatsIcon.js +1 -1
  375. package/dist/icons/StopIcon.js +1 -1
  376. package/dist/icons/StringIcon.js +1 -1
  377. package/dist/icons/SubtotalsIcon.js +1 -1
  378. package/dist/icons/SwitchIcon.js +1 -1
  379. package/dist/icons/TableIcon.js +1 -1
  380. package/dist/icons/ThumbsDownIcon.js +1 -1
  381. package/dist/icons/ThumbsUpIcon.js +1 -1
  382. package/dist/icons/ThunderboltCrossedIcon.js +1 -1
  383. package/dist/icons/ThunderboltFilledIcon.js +1 -1
  384. package/dist/icons/ThunderboltIcon.js +1 -1
  385. package/dist/icons/TimeIcon.js +1 -1
  386. package/dist/icons/TrashIcon.js +1 -1
  387. package/dist/icons/UnlockIcon.js +1 -1
  388. package/dist/icons/UpIcon.js +1 -1
  389. package/dist/icons/UserGroupIcon.js +1 -1
  390. package/dist/icons/UserIcon.js +1 -1
  391. package/dist/icons/UserLockIcon.js +1 -1
  392. package/dist/icons/ViewIcon.js +1 -1
  393. package/dist/icons/WarningFilledIcon.js +1 -1
  394. package/dist/icons/WarningIcon.js +1 -1
  395. package/dist/icons/wrap-icon.js +1 -1
  396. package/dist/index.d.ts +2 -1
  397. package/dist/index.js +3 -2
  398. package/dist/index.js.map +1 -1
  399. package/dist/provider.js +1 -1
  400. package/dist/providers/TrackingProvider.js +1 -1
  401. package/dist/providers/navigationAdapter.default.js +1 -1
  402. package/dist/tokens/base.js +1 -1
  403. package/dist/tokens/colors.js +1 -1
  404. package/dist/tokens/index.js +1 -1
  405. package/dist/tokens/layout.js +1 -1
  406. package/dist/tokens/palette.js +1 -1
  407. package/dist/tokens/shadows.js +1 -1
  408. package/dist/tokens/sizes.js +1 -1
  409. package/dist/tokens/spacing.js +1 -1
  410. package/dist/tokens/typography.js +1 -1
  411. package/dist/utils/ResizeSensor.js +1 -1
  412. package/dist/utils/index.d.ts +1 -0
  413. package/dist/utils/is-dev-env.js +1 -1
  414. package/dist/utils/modules.js +1 -1
  415. package/dist/utils/promise.js +1 -1
  416. package/dist/utils/raf.js +1 -1
  417. package/dist/utils/random.js +1 -1
  418. package/dist/utils/range.js +1 -1
  419. package/dist/utils/react/RenderCache.js +1 -1
  420. package/dist/utils/react/Slots.js +1 -1
  421. package/dist/utils/react/chain.js +1 -1
  422. package/dist/utils/react/forwardRefWithGenerics.js +1 -1
  423. package/dist/utils/react/index.d.ts +1 -0
  424. package/dist/utils/react/index.js +2 -1
  425. package/dist/utils/react/interactions.js +1 -1
  426. package/dist/utils/react/isTextOnly.js +1 -1
  427. package/dist/utils/react/mapProps.js +1 -1
  428. package/dist/utils/react/mergeProps.js +1 -1
  429. package/dist/utils/react/nullableValue.js +1 -1
  430. package/dist/utils/react/resolveIcon.js +1 -1
  431. package/dist/utils/react/sharedStore.js +1 -1
  432. package/dist/utils/react/useCombinedRefs.js +1 -1
  433. package/dist/utils/react/useControlledFocusVisible.js +1 -1
  434. package/dist/utils/react/useEventBus.js +26 -17
  435. package/dist/utils/react/useEventBus.js.map +1 -1
  436. package/dist/utils/react/useId.js +1 -1
  437. package/dist/utils/react/useIsDarwin.js +1 -1
  438. package/dist/utils/react/useKeySymbols.js +1 -1
  439. package/dist/utils/react/useLayoutEffect.js +1 -1
  440. package/dist/utils/react/useLocalStorage.js +1 -1
  441. package/dist/utils/react/useMergeStyles.js +1 -1
  442. package/dist/utils/react/usePopoverSync.d.ts +116 -0
  443. package/dist/utils/react/usePopoverSync.js +131 -11
  444. package/dist/utils/react/usePopoverSync.js.map +1 -1
  445. package/dist/utils/react/useQaProps.js +1 -1
  446. package/dist/utils/react/useViewportSize.js +1 -1
  447. package/dist/utils/react/wrapNodeIfPlain.js +1 -1
  448. package/dist/utils/selection.js +1 -1
  449. package/dist/utils/styles.js +1 -1
  450. package/dist/utils/tree.js +1 -1
  451. package/dist/utils/warnings.js +1 -1
  452. package/dist/version.js +2 -2
  453. package/docs/components/actions/Button.md +102 -24
  454. package/docs/components/actions/ItemButton.md +71 -34
  455. package/docs/components/navigation/Tabs.md +22 -1
  456. package/package.json +1 -1
@@ -1 +1 @@
1
- {"version":3,"file":"useEventBus.js","names":[],"sources":["../../../src/utils/react/useEventBus.ts"],"sourcesContent":["import React, {\n createContext,\n ReactNode,\n useCallback,\n useContext,\n useEffect,\n useRef,\n} from 'react';\n\nexport type EventBusListener<T = any> = (data: T) => void;\n\nexport interface EventBusContextValue {\n emit: <T = any>(event: string, data?: T) => void;\n emitSync: <T = any>(event: string, data?: T) => void;\n on: <T = any>(event: string, listener: EventBusListener<T>) => () => void;\n off: <T = any>(event: string, listener: EventBusListener<T>) => void;\n}\n\nconst EventBusContext = createContext<EventBusContextValue | null>(null);\n\nexport interface EventBusProviderProps {\n children: ReactNode;\n}\n\n/**\n * EventBusProvider provides a global event system for the application.\n *\n * @example\n * ```tsx\n * function App() {\n * return (\n * <EventBusProvider>\n * <YourComponents />\n * </EventBusProvider>\n * );\n * }\n * ```\n */\nexport function EventBusProvider({ children }: EventBusProviderProps) {\n const listeners = useRef<Record<string, EventBusListener[]>>({});\n\n const off = useCallback(\n <T = any>(event: string, listener: EventBusListener<T>) => {\n const eventListeners = listeners.current[event];\n if (eventListeners) {\n listeners.current[event] = eventListeners.filter((l) => l !== listener);\n\n // Clean up empty event arrays\n if (listeners.current[event].length === 0) {\n delete listeners.current[event];\n }\n }\n },\n [],\n );\n\n const emitSync = useCallback(<T = any>(event: string, data?: T) => {\n const eventListeners = listeners.current[event];\n if (eventListeners) {\n eventListeners.forEach((listener) => listener(data));\n }\n }, []);\n\n const emit = useCallback(<T = any>(event: string, data?: T) => {\n // Use setTimeout to ensure async emission after current render cycle\n setTimeout(() => {\n emitSync(event, data);\n }, 0);\n }, []);\n\n const on = useCallback(\n <T = any>(event: string, listener: EventBusListener<T>) => {\n if (!listeners.current[event]) {\n listeners.current[event] = [];\n }\n listeners.current[event].push(listener);\n\n // Return cleanup function\n return () => {\n off(event, listener);\n };\n },\n [off],\n );\n\n const contextValue: EventBusContextValue = {\n emit,\n emitSync,\n on,\n off,\n };\n\n return React.createElement(\n EventBusContext.Provider,\n { value: contextValue },\n children,\n );\n}\n\n/**\n * Hook to access the event bus functionality.\n * Must be used within an EventBusProvider.\n *\n * @example\n * ```tsx\n * function Component() {\n * const { emit, emitSync, on } = useEventBus();\n *\n * const handleClick = () => {\n * emit('user-action', { type: 'click', target: 'button' });\n * };\n *\n * const handleSyncAction = () => {\n * emitSync('sync-action', { immediate: true });\n * };\n *\n * useEffect(() => {\n * const unsubscribe = on('data-updated', (data) => {\n * console.log('Data updated:', data);\n * });\n *\n * return unsubscribe;\n * }, [on]);\n *\n * return (\n * <div>\n * <button onClick={handleClick}>Async Event</button>\n * <button onClick={handleSyncAction}>Sync Event</button>\n * </div>\n * );\n * }\n * ```\n */\nexport function useEventBus(): EventBusContextValue {\n const context = useContext(EventBusContext);\n\n if (!context) {\n throw new Error('useEventBus must be used within an EventBusProvider');\n }\n\n return context;\n}\n\n/**\n * Convenience hook for subscribing to events with automatic cleanup.\n * The listener will be automatically unsubscribed when the component unmounts\n * or when the dependencies change.\n *\n * @param event - The event name to listen for\n * @param listener - The callback function to execute when the event is emitted\n * @param deps - Dependency array for the effect (similar to useEffect)\n *\n * @example\n * ```tsx\n * function NotificationComponent() {\n * const [message, setMessage] = useState('');\n *\n * useEventListener('notification', (data) => {\n * setMessage(data.message);\n * }, []);\n *\n * return <div>{message}</div>;\n * }\n * ```\n */\nexport function useEventListener<T = any>(\n event: string,\n listener: EventBusListener<T>,\n deps: React.DependencyList = [],\n) {\n const { on } = useEventBus();\n\n useEffect(() => {\n const unsubscribe = on(event, listener);\n return unsubscribe;\n }, [event, on, ...deps]);\n}\n"],"mappings":";;;;AAkBA,MAAM,kBAAkB,cAA2C,KAAK;;;;;;;;;;;;;;;AAoBxE,SAAgB,iBAAiB,EAAE,YAAmC;CACpE,MAAM,YAAY,OAA2C,EAAE,CAAC;CAEhE,MAAM,MAAM,aACA,OAAe,aAAkC;EACzD,MAAM,iBAAiB,UAAU,QAAQ;AACzC,MAAI,gBAAgB;AAClB,aAAU,QAAQ,SAAS,eAAe,QAAQ,MAAM,MAAM,SAAS;AAGvE,OAAI,UAAU,QAAQ,OAAO,WAAW,EACtC,QAAO,UAAU,QAAQ;;IAI/B,EAAE,CACH;CAED,MAAM,WAAW,aAAsB,OAAe,SAAa;EACjE,MAAM,iBAAiB,UAAU,QAAQ;AACzC,MAAI,eACF,gBAAe,SAAS,aAAa,SAAS,KAAK,CAAC;IAErD,EAAE,CAAC;CAwBN,MAAM,eAAqC;EACzC,MAvBW,aAAsB,OAAe,SAAa;AAE7D,oBAAiB;AACf,aAAS,OAAO,KAAK;MACpB,EAAE;KACJ,EAAE,CAAC;EAmBJ;EACA,IAlBS,aACC,OAAe,aAAkC;AACzD,OAAI,CAAC,UAAU,QAAQ,OACrB,WAAU,QAAQ,SAAS,EAAE;AAE/B,aAAU,QAAQ,OAAO,KAAK,SAAS;AAGvC,gBAAa;AACX,QAAI,OAAO,SAAS;;KAGxB,CAAC,IAAI,CACN;EAMC;EACD;AAED,QAAO,MAAM,cACX,gBAAgB,UAChB,EAAE,OAAO,cAAc,EACvB,SACD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAqCH,SAAgB,cAAoC;CAClD,MAAM,UAAU,WAAW,gBAAgB;AAE3C,KAAI,CAAC,QACH,OAAM,IAAI,MAAM,sDAAsD;AAGxE,QAAO;;;;;;;;;;;;;;;;;;;;;;;;AAyBT,SAAgB,iBACd,OACA,UACA,OAA6B,EAAE,EAC/B;CACA,MAAM,EAAE,OAAO,aAAa;AAE5B,iBAAgB;AAEd,SADoB,GAAG,OAAO,SAAS;IAEtC;EAAC;EAAO;EAAI,GAAG;EAAK,CAAC"}
1
+ {"version":3,"file":"useEventBus.js","names":[],"sources":["../../../src/utils/react/useEventBus.ts"],"sourcesContent":["import React, {\n createContext,\n ReactNode,\n useCallback,\n useContext,\n useEffect,\n useMemo,\n useRef,\n} from 'react';\n\nexport type EventBusListener<T = any> = (data: T) => void;\n\nexport interface EventBusContextValue {\n emit: <T = any>(event: string, data?: T) => void;\n emitSync: <T = any>(event: string, data?: T) => void;\n on: <T = any>(event: string, listener: EventBusListener<T>) => () => void;\n off: <T = any>(event: string, listener: EventBusListener<T>) => void;\n}\n\nexport const EventBusContext = createContext<EventBusContextValue | null>(null);\n\nexport interface EventBusProviderProps {\n children: ReactNode;\n}\n\n/**\n * EventBusProvider provides a global event system for the application.\n *\n * @example\n * ```tsx\n * function App() {\n * return (\n * <EventBusProvider>\n * <YourComponents />\n * </EventBusProvider>\n * );\n * }\n * ```\n */\nexport function EventBusProvider({ children }: EventBusProviderProps) {\n // If we're already inside a parent EventBusProvider (e.g. the global Root\n // one), DO NOT create a fresh bus — that would isolate listeners and\n // emitters across the boundary. This matters because overlays (Popover,\n // Modal, Tray) re-wrap their content with our `Provider` from\n // `provider.tsx`, which transparently nests an EventBusProvider. Cross-\n // overlay events such as `popover:dismiss-ancestor` (a Button inside a\n // popover footer dismissing the popover host) only work when both sides\n // share the same bus.\n const parentBus = useContext(EventBusContext);\n\n const listeners = useRef<Record<string, EventBusListener[]>>({});\n\n const off = useCallback(\n <T = any>(event: string, listener: EventBusListener<T>) => {\n const eventListeners = listeners.current[event];\n if (eventListeners) {\n listeners.current[event] = eventListeners.filter((l) => l !== listener);\n\n // Clean up empty event arrays\n if (listeners.current[event].length === 0) {\n delete listeners.current[event];\n }\n }\n },\n [],\n );\n\n const emitSync = useCallback(<T = any>(event: string, data?: T) => {\n const eventListeners = listeners.current[event];\n if (eventListeners) {\n eventListeners.forEach((listener) => listener(data));\n }\n }, []);\n\n const emit = useCallback(\n <T = any>(event: string, data?: T) => {\n setTimeout(() => {\n emitSync(event, data);\n }, 0);\n },\n [emitSync],\n );\n\n const on = useCallback(\n <T = any>(event: string, listener: EventBusListener<T>) => {\n if (!listeners.current[event]) {\n listeners.current[event] = [];\n }\n listeners.current[event].push(listener);\n\n // Return cleanup function\n return () => {\n off(event, listener);\n };\n },\n [off],\n );\n\n // Always compute the local contextValue so hook order stays stable, then\n // pick parent OR local. `useMemo` keeps the local value referentially\n // stable across renders — every consumer of `EventBusContext` (notably\n // `useDismissParentPopover` inside every `Button` / `ItemButton`) would\n // otherwise re-render on every render of this provider.\n const localContextValue = useMemo<EventBusContextValue>(\n () => ({ emit, emitSync, on, off }),\n [emit, emitSync, on, off],\n );\n\n const contextValue = parentBus ?? localContextValue;\n\n return React.createElement(\n EventBusContext.Provider,\n { value: contextValue },\n children,\n );\n}\n\n/**\n * Hook to access the event bus functionality.\n * Must be used within an EventBusProvider.\n *\n * @example\n * ```tsx\n * function Component() {\n * const { emit, emitSync, on } = useEventBus();\n *\n * const handleClick = () => {\n * emit('user-action', { type: 'click', target: 'button' });\n * };\n *\n * const handleSyncAction = () => {\n * emitSync('sync-action', { immediate: true });\n * };\n *\n * useEffect(() => {\n * const unsubscribe = on('data-updated', (data) => {\n * console.log('Data updated:', data);\n * });\n *\n * return unsubscribe;\n * }, [on]);\n *\n * return (\n * <div>\n * <button onClick={handleClick}>Async Event</button>\n * <button onClick={handleSyncAction}>Sync Event</button>\n * </div>\n * );\n * }\n * ```\n */\nexport function useEventBus(): EventBusContextValue {\n const context = useContext(EventBusContext);\n\n if (!context) {\n throw new Error('useEventBus must be used within an EventBusProvider');\n }\n\n return context;\n}\n\n/**\n * Convenience hook for subscribing to events with automatic cleanup.\n * The listener will be automatically unsubscribed when the component unmounts\n * or when the dependencies change.\n *\n * @param event - The event name to listen for\n * @param listener - The callback function to execute when the event is emitted\n * @param deps - Dependency array for the effect (similar to useEffect)\n *\n * @example\n * ```tsx\n * function NotificationComponent() {\n * const [message, setMessage] = useState('');\n *\n * useEventListener('notification', (data) => {\n * setMessage(data.message);\n * }, []);\n *\n * return <div>{message}</div>;\n * }\n * ```\n */\nexport function useEventListener<T = any>(\n event: string,\n listener: EventBusListener<T>,\n deps: React.DependencyList = [],\n) {\n const { on } = useEventBus();\n\n useEffect(() => {\n const unsubscribe = on(event, listener);\n return unsubscribe;\n }, [event, on, ...deps]);\n}\n"],"mappings":";;;;AAmBA,MAAa,kBAAkB,cAA2C,KAAK;;;;;;;;;;;;;;;AAoB/E,SAAgB,iBAAiB,EAAE,YAAmC;CASpE,MAAM,YAAY,WAAW,gBAAgB;CAE7C,MAAM,YAAY,OAA2C,EAAE,CAAC;CAEhE,MAAM,MAAM,aACA,OAAe,aAAkC;EACzD,MAAM,iBAAiB,UAAU,QAAQ;AACzC,MAAI,gBAAgB;AAClB,aAAU,QAAQ,SAAS,eAAe,QAAQ,MAAM,MAAM,SAAS;AAGvE,OAAI,UAAU,QAAQ,OAAO,WAAW,EACtC,QAAO,UAAU,QAAQ;;IAI/B,EAAE,CACH;CAED,MAAM,WAAW,aAAsB,OAAe,SAAa;EACjE,MAAM,iBAAiB,UAAU,QAAQ;AACzC,MAAI,eACF,gBAAe,SAAS,aAAa,SAAS,KAAK,CAAC;IAErD,EAAE,CAAC;CAEN,MAAM,OAAO,aACD,OAAe,SAAa;AACpC,mBAAiB;AACf,YAAS,OAAO,KAAK;KACpB,EAAE;IAEP,CAAC,SAAS,CACX;CAED,MAAM,KAAK,aACC,OAAe,aAAkC;AACzD,MAAI,CAAC,UAAU,QAAQ,OACrB,WAAU,QAAQ,SAAS,EAAE;AAE/B,YAAU,QAAQ,OAAO,KAAK,SAAS;AAGvC,eAAa;AACX,OAAI,OAAO,SAAS;;IAGxB,CAAC,IAAI,CACN;CAOD,MAAM,oBAAoB,eACjB;EAAE;EAAM;EAAU;EAAI;EAAK,GAClC;EAAC;EAAM;EAAU;EAAI;EAAI,CAC1B;CAED,MAAM,eAAe,aAAa;AAElC,QAAO,MAAM,cACX,gBAAgB,UAChB,EAAE,OAAO,cAAc,EACvB,SACD;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAqCH,SAAgB,cAAoC;CAClD,MAAM,UAAU,WAAW,gBAAgB;AAE3C,KAAI,CAAC,QACH,OAAM,IAAI,MAAM,sDAAsD;AAGxE,QAAO;;;;;;;;;;;;;;;;;;;;;;;;AAyBT,SAAgB,iBACd,OACA,UACA,OAA6B,EAAE,EAC/B;CACA,MAAM,EAAE,OAAO,aAAa;AAE5B,iBAAgB;AAEd,SADoB,GAAG,OAAO,SAAS;IAEtC;EAAC;EAAO;EAAI,GAAG;EAAK,CAAC"}
@@ -1,4 +1,4 @@
1
- /** @license MIT | @cube-dev/ui-kit v0.140.1 | Cube Dev Team */
1
+ /** @license MIT | @cube-dev/ui-kit v0.142.0 | Cube Dev Team */
2
2
  import { useLayoutEffect as useLayoutEffect$1 } from "./useLayoutEffect.js";
3
3
  import { useEffect, useRef, useState } from "react";
4
4
  import { useSSRSafeId } from "@react-aria/ssr";
@@ -1,4 +1,4 @@
1
- /** @license MIT | @cube-dev/ui-kit v0.140.1 | Cube Dev Team */
1
+ /** @license MIT | @cube-dev/ui-kit v0.142.0 | Cube Dev Team */
2
2
  import { useEffect, useState } from "react";
3
3
 
4
4
  //#region src/utils/react/useIsDarwin.ts
@@ -1,4 +1,4 @@
1
- /** @license MIT | @cube-dev/ui-kit v0.140.1 | Cube Dev Team */
1
+ /** @license MIT | @cube-dev/ui-kit v0.142.0 | Cube Dev Team */
2
2
  import { useIsDarwin } from "./useIsDarwin.js";
3
3
  import { useMemo } from "react";
4
4
 
@@ -1,4 +1,4 @@
1
- /** @license MIT | @cube-dev/ui-kit v0.140.1 | Cube Dev Team */
1
+ /** @license MIT | @cube-dev/ui-kit v0.142.0 | Cube Dev Team */
2
2
  import * as React$1 from "react";
3
3
 
4
4
  //#region src/utils/react/useLayoutEffect.tsx
@@ -1,4 +1,4 @@
1
- /** @license MIT | @cube-dev/ui-kit v0.140.1 | Cube Dev Team */
1
+ /** @license MIT | @cube-dev/ui-kit v0.142.0 | Cube Dev Team */
2
2
  import { useEvent } from "../../_internal/hooks/use-event.js";
3
3
  import { useCallback, useSyncExternalStore } from "react";
4
4
 
@@ -1,4 +1,4 @@
1
- /** @license MIT | @cube-dev/ui-kit v0.140.1 | Cube Dev Team */
1
+ /** @license MIT | @cube-dev/ui-kit v0.142.0 | Cube Dev Team */
2
2
  import { mergeStyles } from "@tenphi/tasty";
3
3
  import { useMemo } from "react";
4
4
 
@@ -0,0 +1,116 @@
1
+
2
+ import { RefObject } from "react";
3
+
4
+ //#region src/utils/react/usePopoverSync.d.ts
5
+ interface UsePopoverSyncOptions {
6
+ /** Stable identifier for this popover instance (typically a generateRandomId() memo). */
7
+ menuId: string;
8
+ /** Current open state of this popover. */
9
+ isOpen: boolean;
10
+ /** Called when another popover opens while this one is open. */
11
+ onClose: () => void;
12
+ /**
13
+ * When `false`, this popover does not participate in the sync (no listening,
14
+ * no emitting). Useful for "dummy" triggers that proxy a real one (see
15
+ * `MenuTrigger`'s `isDummy`). Defaults to `true`.
16
+ */
17
+ enabled?: boolean;
18
+ /**
19
+ * Ref to the popover's trigger element. When provided, the element is
20
+ * included in the `popover:open` emit payload so peers can detect whether
21
+ * the new opener is nested inside their own overlay (and skip closing in
22
+ * that case). Optional — omitting it preserves the legacy "always close on
23
+ * peer open" behaviour.
24
+ */
25
+ triggerRef?: RefObject<HTMLElement | null>;
26
+ /**
27
+ * Ref to the overlay/container element that hosts this popover's content.
28
+ * When provided, the listener performs a DOM `contains()` check on incoming
29
+ * peer triggers: peers whose trigger lives inside this container are
30
+ * considered nested children and do NOT close us.
31
+ */
32
+ containerRef?: RefObject<HTMLElement | null>;
33
+ /**
34
+ * Whether this overlay closes when a Button/ItemButton inside its container
35
+ * is pressed. Defaults to `true` (popover semantics — popovers are transient
36
+ * surfaces and any action inside them should dismiss them). Set to `false`
37
+ * for modals/trays/fullscreen dialogs — buttons inside a Dialog should not
38
+ * auto-close it. Requires `containerRef` to be set; without it the listener
39
+ * has no way to determine whether the dispatching button is "inside" this
40
+ * overlay and is effectively a no-op.
41
+ */
42
+ dismissOnInnerButtonPress?: boolean;
43
+ /**
44
+ * Whether this overlay closes when a peer popover opens. Defaults to `true`
45
+ * (popover semantics — only one popover open at a time). Set to `false` for
46
+ * modals/trays/fullscreen dialogs so a peer popover opening cannot bypass
47
+ * the dialog's own `isDismissable` / `onClose` handling and yank it shut.
48
+ *
49
+ * The host still EMITS `popover:open` regardless of this flag, so opening a
50
+ * modal/tray correctly dismisses any peer popover that was open before.
51
+ */
52
+ closeOnPeerOpen?: boolean;
53
+ }
54
+ /**
55
+ * Coordinates the "only one popover open at a time" invariant via the EventBus.
56
+ *
57
+ * - When `isOpen` flips `false -> true`, emits `popover:open` once.
58
+ * - While open, listens for peers' `popover:open` events and calls `onClose`.
59
+ *
60
+ * Implementation notes (ALL of these matter — losing any one re-introduces a
61
+ * race that surfaces only under rapid trigger switching, which is hard to
62
+ * reproduce in tests):
63
+ *
64
+ * 1. `isOpen` and `onClose` are read through refs inside the listener, so the
65
+ * subscription effect's dep array does NOT include `isOpen`/`onClose`. This
66
+ * keeps the listener identity stable across open/close transitions and
67
+ * avoids the unsubscribe-emit-resubscribe window where an emit can be
68
+ * delivered to a stale listener (or no listener).
69
+ * 2. The emit fires only on the `false -> true` transition, gated by
70
+ * `wasOpenRef`. A re-render where `isOpen` is still `true` must NOT
71
+ * re-emit, otherwise it could re-trigger listeners on peers that just
72
+ * opened in the same render flush.
73
+ * 3. The `enabled` flag short-circuits both effects symmetrically. When it
74
+ * flips off, `wasOpenRef` is reset so re-enabling later still emits if
75
+ * `isOpen` is true at that moment.
76
+ */
77
+ declare function usePopoverSync({
78
+ menuId,
79
+ isOpen,
80
+ onClose,
81
+ enabled,
82
+ triggerRef,
83
+ containerRef,
84
+ dismissOnInnerButtonPress,
85
+ closeOnPeerOpen
86
+ }: UsePopoverSyncOptions): void;
87
+ /**
88
+ * Hook that returns a dispatcher to close the popover that contains a given
89
+ * DOM element. Used by `Button` / `ItemButton` to implement the default
90
+ * "press inside a popover closes the popover" behaviour. Custom (non-Cube)
91
+ * interactive controls can call this directly:
92
+ *
93
+ * ```tsx
94
+ * const dismiss = useDismissParentPopover();
95
+ * <MyCustomPressable onPress={(e) => { doThing(); dismiss(e.currentTarget); }} />
96
+ * ```
97
+ *
98
+ * The actual dismiss is dispatched through the EventBus, which defers via
99
+ * `setTimeout(0)` — so the user's synchronous handler (and any React state
100
+ * updates it triggers) flushes BEFORE the popover closes. This is critical
101
+ * for the "open a hoisted modal from a popover footer" case: the modal
102
+ * mounts first, then the popover closes.
103
+ *
104
+ * Only popover-type containers (those that pass `dismissOnInnerButtonPress`
105
+ * as `true`, the default) react to the event. Modal/tray/fullscreen Dialog
106
+ * containers explicitly opt out so a Button inside their content does not
107
+ * auto-close them.
108
+ *
109
+ * When called outside an `EventBusProvider` (e.g. in unit tests that render
110
+ * a Button without wrapping in `<Root>`), the returned function is a no-op —
111
+ * the dismiss flow gracefully degrades rather than throwing.
112
+ */
113
+ declare function useDismissParentPopover(): (from: Element | null) => void;
114
+ //#endregion
115
+ export { UsePopoverSyncOptions, useDismissParentPopover, usePopoverSync };
116
+ //# sourceMappingURL=usePopoverSync.d.ts.map
@@ -1,9 +1,56 @@
1
- /** @license MIT | @cube-dev/ui-kit v0.140.1 | Cube Dev Team */
2
- import { useEventBus } from "./useEventBus.js";
3
- import { useEffect, useRef } from "react";
1
+ /** @license MIT | @cube-dev/ui-kit v0.142.0 | Cube Dev Team */
2
+ import { EventBusContext, useEventBus } from "./useEventBus.js";
3
+ import { useCallback, useContext, useEffect, useRef } from "react";
4
4
 
5
5
  //#region src/utils/react/usePopoverSync.ts
6
6
  /**
7
+ * Module-level registry of currently open popovers. Lets us resolve the
8
+ * LOGICAL parent of an arbitrary element across portal boundaries.
9
+ *
10
+ * Popover content is portaled to a shared root (`document.body` by default —
11
+ * see `Overlay.tsx`), so a grandchild popover's trigger lives inside a
12
+ * sibling portal rather than physically inside its grandparent's container.
13
+ * A naive `container.contains(triggerEl)` check therefore misses the
14
+ * relationship and closes the grandparent — the bug that surfaces with 3+
15
+ * levels of `SubMenuTrigger` nesting.
16
+ *
17
+ * The registry stores closures over the host's refs so reads always see the
18
+ * latest `.current` value without forcing a re-register per render.
19
+ */
20
+ const openPopovers = /* @__PURE__ */ new Map();
21
+ /**
22
+ * Find the open popover whose container directly contains `target`. Returns
23
+ * `null` for elements that live outside every registered popover (e.g. a
24
+ * top-level trigger button rendered next to its overlay root).
25
+ */
26
+ function findOwningPopover(target) {
27
+ if (!target) return null;
28
+ for (const entry of openPopovers) if (entry[1].getContainerEl()?.contains(target)) return entry;
29
+ return null;
30
+ }
31
+ /**
32
+ * Returns `true` when `target` is nested inside the overlay identified by
33
+ * `ancestorMenuId` — either as a direct DOM descendant of `ancestorContainer`
34
+ * or as a descendant of a chain of popovers whose triggers eventually land
35
+ * inside it. Used by `popover:open` peers to avoid closing themselves when a
36
+ * grand-child overlay opens through a portal.
37
+ */
38
+ function isLogicalDescendantOf(target, ancestorMenuId, ancestorContainer) {
39
+ if (ancestorContainer && target && ancestorContainer.contains(target)) return true;
40
+ let cur = target;
41
+ const visited = /* @__PURE__ */ new Set();
42
+ while (cur) {
43
+ const owner = findOwningPopover(cur);
44
+ if (!owner) return false;
45
+ const [id, entry] = owner;
46
+ if (visited.has(id)) return false;
47
+ visited.add(id);
48
+ if (id === ancestorMenuId) return true;
49
+ cur = entry.getTriggerEl();
50
+ }
51
+ return false;
52
+ }
53
+ /**
7
54
  * Coordinates the "only one popover open at a time" invariant via the EventBus.
8
55
  *
9
56
  * - When `isOpen` flips `false -> true`, emits `popover:open` once.
@@ -26,7 +73,7 @@ import { useEffect, useRef } from "react";
26
73
  * flips off, `wasOpenRef` is reset so re-enabling later still emits if
27
74
  * `isOpen` is true at that moment.
28
75
  */
29
- function usePopoverSync({ menuId, isOpen, onClose, enabled = true }) {
76
+ function usePopoverSync({ menuId, isOpen, onClose, enabled = true, triggerRef, containerRef, dismissOnInnerButtonPress = true, closeOnPeerOpen = true }) {
30
77
  const { emit, on } = useEventBus();
31
78
  const isOpenRef = useRef(isOpen);
32
79
  useEffect(() => {
@@ -36,34 +83,107 @@ function usePopoverSync({ menuId, isOpen, onClose, enabled = true }) {
36
83
  useEffect(() => {
37
84
  onCloseRef.current = onClose;
38
85
  }, [onClose]);
86
+ const containerRefRef = useRef(containerRef);
87
+ useEffect(() => {
88
+ containerRefRef.current = containerRef;
89
+ }, [containerRef]);
90
+ const triggerRefRef = useRef(triggerRef);
91
+ useEffect(() => {
92
+ triggerRefRef.current = triggerRef;
93
+ }, [triggerRef]);
39
94
  useEffect(() => {
40
95
  if (!enabled) return;
41
- return on("popover:open", (data) => {
42
- if (data.menuId !== menuId && isOpenRef.current) onCloseRef.current();
43
- });
96
+ const offOpen = closeOnPeerOpen ? on("popover:open", (data) => {
97
+ if (data.menuId === menuId || !isOpenRef.current) return;
98
+ const container = containerRefRef.current?.current ?? null;
99
+ if (isLogicalDescendantOf(data.triggerEl, menuId, container)) return;
100
+ onCloseRef.current();
101
+ }) : null;
102
+ const offDismiss = dismissOnInnerButtonPress ? on("popover:dismiss-ancestor", (data) => {
103
+ if (!isOpenRef.current) return;
104
+ const container = containerRefRef.current?.current;
105
+ const from = data?.from;
106
+ if (!container || !from) return;
107
+ if (container.contains(from)) onCloseRef.current();
108
+ }) : null;
109
+ return () => {
110
+ offOpen?.();
111
+ offDismiss?.();
112
+ };
44
113
  }, [
45
114
  on,
46
115
  menuId,
47
- enabled
116
+ enabled,
117
+ dismissOnInnerButtonPress,
118
+ closeOnPeerOpen
48
119
  ]);
49
120
  const wasOpenRef = useRef(false);
50
121
  useEffect(() => {
51
122
  if (!enabled) {
123
+ if (wasOpenRef.current) openPopovers.delete(menuId);
52
124
  wasOpenRef.current = false;
53
125
  return;
54
126
  }
55
127
  if (isOpen && !wasOpenRef.current) {
56
128
  wasOpenRef.current = true;
57
- emit("popover:open", { menuId });
58
- } else if (!isOpen) wasOpenRef.current = false;
129
+ openPopovers.set(menuId, {
130
+ getContainerEl: () => containerRefRef.current?.current ?? null,
131
+ getTriggerEl: () => triggerRefRef.current?.current ?? null
132
+ });
133
+ emit("popover:open", {
134
+ menuId,
135
+ triggerEl: triggerRefRef.current?.current ?? null
136
+ });
137
+ } else if (!isOpen && wasOpenRef.current) {
138
+ wasOpenRef.current = false;
139
+ openPopovers.delete(menuId);
140
+ }
59
141
  }, [
60
142
  isOpen,
61
143
  emit,
62
144
  menuId,
63
145
  enabled
64
146
  ]);
147
+ useEffect(() => {
148
+ return () => {
149
+ openPopovers.delete(menuId);
150
+ };
151
+ }, [menuId]);
152
+ }
153
+ /**
154
+ * Hook that returns a dispatcher to close the popover that contains a given
155
+ * DOM element. Used by `Button` / `ItemButton` to implement the default
156
+ * "press inside a popover closes the popover" behaviour. Custom (non-Cube)
157
+ * interactive controls can call this directly:
158
+ *
159
+ * ```tsx
160
+ * const dismiss = useDismissParentPopover();
161
+ * <MyCustomPressable onPress={(e) => { doThing(); dismiss(e.currentTarget); }} />
162
+ * ```
163
+ *
164
+ * The actual dismiss is dispatched through the EventBus, which defers via
165
+ * `setTimeout(0)` — so the user's synchronous handler (and any React state
166
+ * updates it triggers) flushes BEFORE the popover closes. This is critical
167
+ * for the "open a hoisted modal from a popover footer" case: the modal
168
+ * mounts first, then the popover closes.
169
+ *
170
+ * Only popover-type containers (those that pass `dismissOnInnerButtonPress`
171
+ * as `true`, the default) react to the event. Modal/tray/fullscreen Dialog
172
+ * containers explicitly opt out so a Button inside their content does not
173
+ * auto-close them.
174
+ *
175
+ * When called outside an `EventBusProvider` (e.g. in unit tests that render
176
+ * a Button without wrapping in `<Root>`), the returned function is a no-op —
177
+ * the dismiss flow gracefully degrades rather than throwing.
178
+ */
179
+ function useDismissParentPopover() {
180
+ const emit = useContext(EventBusContext)?.emit;
181
+ return useCallback((from) => {
182
+ if (!from || !emit) return;
183
+ emit("popover:dismiss-ancestor", { from });
184
+ }, [emit]);
65
185
  }
66
186
 
67
187
  //#endregion
68
- export { usePopoverSync };
188
+ export { useDismissParentPopover, usePopoverSync };
69
189
  //# sourceMappingURL=usePopoverSync.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"usePopoverSync.js","names":[],"sources":["../../../src/utils/react/usePopoverSync.ts"],"sourcesContent":["import { useEffect, useRef } from 'react';\n\nimport { useEventBus } from './useEventBus';\n\nexport interface UsePopoverSyncOptions {\n /** Stable identifier for this popover instance (typically a generateRandomId() memo). */\n menuId: string;\n /** Current open state of this popover. */\n isOpen: boolean;\n /** Called when another popover opens while this one is open. */\n onClose: () => void;\n /**\n * When `false`, this popover does not participate in the sync (no listening,\n * no emitting). Useful for \"dummy\" triggers that proxy a real one (see\n * `MenuTrigger`'s `isDummy`). Defaults to `true`.\n */\n enabled?: boolean;\n}\n\n/**\n * Coordinates the \"only one popover open at a time\" invariant via the EventBus.\n *\n * - When `isOpen` flips `false -> true`, emits `popover:open` once.\n * - While open, listens for peers' `popover:open` events and calls `onClose`.\n *\n * Implementation notes (ALL of these matter — losing any one re-introduces a\n * race that surfaces only under rapid trigger switching, which is hard to\n * reproduce in tests):\n *\n * 1. `isOpen` and `onClose` are read through refs inside the listener, so the\n * subscription effect's dep array does NOT include `isOpen`/`onClose`. This\n * keeps the listener identity stable across open/close transitions and\n * avoids the unsubscribe-emit-resubscribe window where an emit can be\n * delivered to a stale listener (or no listener).\n * 2. The emit fires only on the `false -> true` transition, gated by\n * `wasOpenRef`. A re-render where `isOpen` is still `true` must NOT\n * re-emit, otherwise it could re-trigger listeners on peers that just\n * opened in the same render flush.\n * 3. The `enabled` flag short-circuits both effects symmetrically. When it\n * flips off, `wasOpenRef` is reset so re-enabling later still emits if\n * `isOpen` is true at that moment.\n */\nexport function usePopoverSync({\n menuId,\n isOpen,\n onClose,\n enabled = true,\n}: UsePopoverSyncOptions): void {\n const { emit, on } = useEventBus();\n\n const isOpenRef = useRef(isOpen);\n useEffect(() => {\n isOpenRef.current = isOpen;\n }, [isOpen]);\n\n const onCloseRef = useRef(onClose);\n useEffect(() => {\n onCloseRef.current = onClose;\n }, [onClose]);\n\n useEffect(() => {\n if (!enabled) return;\n return on('popover:open', (data: { menuId: string }) => {\n if (data.menuId !== menuId && isOpenRef.current) {\n onCloseRef.current();\n }\n });\n }, [on, menuId, enabled]);\n\n const wasOpenRef = useRef(false);\n useEffect(() => {\n if (!enabled) {\n wasOpenRef.current = false;\n return;\n }\n if (isOpen && !wasOpenRef.current) {\n wasOpenRef.current = true;\n emit('popover:open', { menuId });\n } else if (!isOpen) {\n wasOpenRef.current = false;\n }\n }, [isOpen, emit, menuId, enabled]);\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;AA0CA,SAAgB,eAAe,EAC7B,QACA,QACA,SACA,UAAU,QACoB;CAC9B,MAAM,EAAE,MAAM,OAAO,aAAa;CAElC,MAAM,YAAY,OAAO,OAAO;AAChC,iBAAgB;AACd,YAAU,UAAU;IACnB,CAAC,OAAO,CAAC;CAEZ,MAAM,aAAa,OAAO,QAAQ;AAClC,iBAAgB;AACd,aAAW,UAAU;IACpB,CAAC,QAAQ,CAAC;AAEb,iBAAgB;AACd,MAAI,CAAC,QAAS;AACd,SAAO,GAAG,iBAAiB,SAA6B;AACtD,OAAI,KAAK,WAAW,UAAU,UAAU,QACtC,YAAW,SAAS;IAEtB;IACD;EAAC;EAAI;EAAQ;EAAQ,CAAC;CAEzB,MAAM,aAAa,OAAO,MAAM;AAChC,iBAAgB;AACd,MAAI,CAAC,SAAS;AACZ,cAAW,UAAU;AACrB;;AAEF,MAAI,UAAU,CAAC,WAAW,SAAS;AACjC,cAAW,UAAU;AACrB,QAAK,gBAAgB,EAAE,QAAQ,CAAC;aACvB,CAAC,OACV,YAAW,UAAU;IAEtB;EAAC;EAAQ;EAAM;EAAQ;EAAQ,CAAC"}
1
+ {"version":3,"file":"usePopoverSync.js","names":[],"sources":["../../../src/utils/react/usePopoverSync.ts"],"sourcesContent":["import { RefObject, useCallback, useContext, useEffect, useRef } from 'react';\n\nimport { EventBusContext, useEventBus } from './useEventBus';\n\nexport interface UsePopoverSyncOptions {\n /** Stable identifier for this popover instance (typically a generateRandomId() memo). */\n menuId: string;\n /** Current open state of this popover. */\n isOpen: boolean;\n /** Called when another popover opens while this one is open. */\n onClose: () => void;\n /**\n * When `false`, this popover does not participate in the sync (no listening,\n * no emitting). Useful for \"dummy\" triggers that proxy a real one (see\n * `MenuTrigger`'s `isDummy`). Defaults to `true`.\n */\n enabled?: boolean;\n /**\n * Ref to the popover's trigger element. When provided, the element is\n * included in the `popover:open` emit payload so peers can detect whether\n * the new opener is nested inside their own overlay (and skip closing in\n * that case). Optional — omitting it preserves the legacy \"always close on\n * peer open\" behaviour.\n */\n triggerRef?: RefObject<HTMLElement | null>;\n /**\n * Ref to the overlay/container element that hosts this popover's content.\n * When provided, the listener performs a DOM `contains()` check on incoming\n * peer triggers: peers whose trigger lives inside this container are\n * considered nested children and do NOT close us.\n */\n containerRef?: RefObject<HTMLElement | null>;\n /**\n * Whether this overlay closes when a Button/ItemButton inside its container\n * is pressed. Defaults to `true` (popover semantics — popovers are transient\n * surfaces and any action inside them should dismiss them). Set to `false`\n * for modals/trays/fullscreen dialogs — buttons inside a Dialog should not\n * auto-close it. Requires `containerRef` to be set; without it the listener\n * has no way to determine whether the dispatching button is \"inside\" this\n * overlay and is effectively a no-op.\n */\n dismissOnInnerButtonPress?: boolean;\n /**\n * Whether this overlay closes when a peer popover opens. Defaults to `true`\n * (popover semantics — only one popover open at a time). Set to `false` for\n * modals/trays/fullscreen dialogs so a peer popover opening cannot bypass\n * the dialog's own `isDismissable` / `onClose` handling and yank it shut.\n *\n * The host still EMITS `popover:open` regardless of this flag, so opening a\n * modal/tray correctly dismisses any peer popover that was open before.\n */\n closeOnPeerOpen?: boolean;\n}\n\ninterface PopoverOpenPayload {\n menuId: string;\n triggerEl: Element | null;\n}\n\ninterface PopoverDismissAncestorPayload {\n /** The DOM node of the button that requested the dismiss. */\n from: Element | null;\n}\n\ninterface PopoverRegistryEntry {\n getContainerEl: () => Element | null;\n getTriggerEl: () => Element | null;\n}\n\n/**\n * Module-level registry of currently open popovers. Lets us resolve the\n * LOGICAL parent of an arbitrary element across portal boundaries.\n *\n * Popover content is portaled to a shared root (`document.body` by default —\n * see `Overlay.tsx`), so a grandchild popover's trigger lives inside a\n * sibling portal rather than physically inside its grandparent's container.\n * A naive `container.contains(triggerEl)` check therefore misses the\n * relationship and closes the grandparent — the bug that surfaces with 3+\n * levels of `SubMenuTrigger` nesting.\n *\n * The registry stores closures over the host's refs so reads always see the\n * latest `.current` value without forcing a re-register per render.\n */\nconst openPopovers = new Map<string, PopoverRegistryEntry>();\n\n/**\n * Find the open popover whose container directly contains `target`. Returns\n * `null` for elements that live outside every registered popover (e.g. a\n * top-level trigger button rendered next to its overlay root).\n */\nfunction findOwningPopover(\n target: Element | null,\n): [string, PopoverRegistryEntry] | null {\n if (!target) return null;\n for (const entry of openPopovers) {\n if (entry[1].getContainerEl()?.contains(target)) {\n return entry;\n }\n }\n return null;\n}\n\n/**\n * Returns `true` when `target` is nested inside the overlay identified by\n * `ancestorMenuId` — either as a direct DOM descendant of `ancestorContainer`\n * or as a descendant of a chain of popovers whose triggers eventually land\n * inside it. Used by `popover:open` peers to avoid closing themselves when a\n * grand-child overlay opens through a portal.\n */\nfunction isLogicalDescendantOf(\n target: Element | null,\n ancestorMenuId: string,\n ancestorContainer: Element | null,\n): boolean {\n if (ancestorContainer && target && ancestorContainer.contains(target)) {\n return true;\n }\n let cur: Element | null = target;\n const visited = new Set<string>();\n while (cur) {\n const owner = findOwningPopover(cur);\n if (!owner) return false;\n const [id, entry] = owner;\n if (visited.has(id)) return false;\n visited.add(id);\n if (id === ancestorMenuId) return true;\n cur = entry.getTriggerEl();\n }\n return false;\n}\n\n/**\n * Coordinates the \"only one popover open at a time\" invariant via the EventBus.\n *\n * - When `isOpen` flips `false -> true`, emits `popover:open` once.\n * - While open, listens for peers' `popover:open` events and calls `onClose`.\n *\n * Implementation notes (ALL of these matter — losing any one re-introduces a\n * race that surfaces only under rapid trigger switching, which is hard to\n * reproduce in tests):\n *\n * 1. `isOpen` and `onClose` are read through refs inside the listener, so the\n * subscription effect's dep array does NOT include `isOpen`/`onClose`. This\n * keeps the listener identity stable across open/close transitions and\n * avoids the unsubscribe-emit-resubscribe window where an emit can be\n * delivered to a stale listener (or no listener).\n * 2. The emit fires only on the `false -> true` transition, gated by\n * `wasOpenRef`. A re-render where `isOpen` is still `true` must NOT\n * re-emit, otherwise it could re-trigger listeners on peers that just\n * opened in the same render flush.\n * 3. The `enabled` flag short-circuits both effects symmetrically. When it\n * flips off, `wasOpenRef` is reset so re-enabling later still emits if\n * `isOpen` is true at that moment.\n */\nexport function usePopoverSync({\n menuId,\n isOpen,\n onClose,\n enabled = true,\n triggerRef,\n containerRef,\n dismissOnInnerButtonPress = true,\n closeOnPeerOpen = true,\n}: UsePopoverSyncOptions): void {\n const { emit, on } = useEventBus();\n\n const isOpenRef = useRef(isOpen);\n useEffect(() => {\n isOpenRef.current = isOpen;\n }, [isOpen]);\n\n const onCloseRef = useRef(onClose);\n useEffect(() => {\n onCloseRef.current = onClose;\n }, [onClose]);\n\n // Track the latest containerRef via a stable ref-of-refs so the listener\n // never resubscribes when callers pass freshly-created ref wrappers across\n // renders. Same pattern as `onCloseRef` above — without this the listener's\n // effect would churn for any caller that doesn't memoize its refs.\n const containerRefRef = useRef(containerRef);\n useEffect(() => {\n containerRefRef.current = containerRef;\n }, [containerRef]);\n\n const triggerRefRef = useRef(triggerRef);\n useEffect(() => {\n triggerRefRef.current = triggerRef;\n }, [triggerRef]);\n\n useEffect(() => {\n if (!enabled) return;\n\n // `popover:open` listener — gated on `closeOnPeerOpen`. Modals/trays opt\n // out so a peer popover opening cannot bypass the dialog's\n // `isDismissable` / `onClose` handling and call `state.close()` directly.\n // Note: the EMIT side (lower in this hook) still fires regardless, so\n // opening a modal still correctly dismisses peer popovers.\n const offOpen = closeOnPeerOpen\n ? on('popover:open', (data: PopoverOpenPayload) => {\n if (data.menuId === menuId || !isOpenRef.current) return;\n const container = containerRefRef.current?.current ?? null;\n // Nested-popover guard: stay open when the opening peer's trigger is\n // a LOGICAL descendant of our overlay. Direct DOM containment only\n // covers the first level — for grand-child popovers the trigger lives\n // in a sibling portal, so we walk the registered popover chain back\n // up via each parent's trigger element. Without this, opening a\n // third-level `SubMenuTrigger` would close every ancestor menu.\n if (isLogicalDescendantOf(data.triggerEl, menuId, container)) return;\n onCloseRef.current();\n })\n : null;\n\n // `popover:dismiss-ancestor` is emitted by `Button` / `ItemButton` (and any\n // consumer using `useDismissParentPopover`) after their `onPress` runs.\n // Only popover-type overlays subscribe; modals/trays opt out via\n // `dismissOnInnerButtonPress: false` so a Button inside a Dialog content\n // does NOT auto-close the Dialog.\n const offDismiss = dismissOnInnerButtonPress\n ? on(\n 'popover:dismiss-ancestor',\n (data: PopoverDismissAncestorPayload) => {\n if (!isOpenRef.current) return;\n const container = containerRefRef.current?.current;\n const from = data?.from;\n // Require both a container and an originating element so we can do\n // the contains-check. Hosts without `containerRef` (e.g.\n // `use-anchored-menu`, `use-context-menu`) are silently no-op.\n if (!container || !from) return;\n if (container.contains(from)) onCloseRef.current();\n },\n )\n : null;\n\n return () => {\n offOpen?.();\n offDismiss?.();\n };\n }, [on, menuId, enabled, dismissOnInnerButtonPress, closeOnPeerOpen]);\n\n const wasOpenRef = useRef(false);\n useEffect(() => {\n if (!enabled) {\n if (wasOpenRef.current) {\n openPopovers.delete(menuId);\n }\n wasOpenRef.current = false;\n return;\n }\n if (isOpen && !wasOpenRef.current) {\n wasOpenRef.current = true;\n // Register BEFORE emitting so peers' listeners (deferred via the\n // bus's `setTimeout(0)`) can already see us in the registry when\n // they walk the logical chain.\n openPopovers.set(menuId, {\n getContainerEl: () => containerRefRef.current?.current ?? null,\n getTriggerEl: () => triggerRefRef.current?.current ?? null,\n });\n emit<PopoverOpenPayload>('popover:open', {\n menuId,\n triggerEl: triggerRefRef.current?.current ?? null,\n });\n } else if (!isOpen && wasOpenRef.current) {\n wasOpenRef.current = false;\n openPopovers.delete(menuId);\n }\n }, [isOpen, emit, menuId, enabled]);\n\n // Final cleanup on unmount: covers the case where a host unmounts while\n // still flagged open (e.g. a parent unmount tears down a child popover\n // before its close transition runs).\n useEffect(() => {\n return () => {\n openPopovers.delete(menuId);\n };\n }, [menuId]);\n}\n\n/**\n * Hook that returns a dispatcher to close the popover that contains a given\n * DOM element. Used by `Button` / `ItemButton` to implement the default\n * \"press inside a popover closes the popover\" behaviour. Custom (non-Cube)\n * interactive controls can call this directly:\n *\n * ```tsx\n * const dismiss = useDismissParentPopover();\n * <MyCustomPressable onPress={(e) => { doThing(); dismiss(e.currentTarget); }} />\n * ```\n *\n * The actual dismiss is dispatched through the EventBus, which defers via\n * `setTimeout(0)` — so the user's synchronous handler (and any React state\n * updates it triggers) flushes BEFORE the popover closes. This is critical\n * for the \"open a hoisted modal from a popover footer\" case: the modal\n * mounts first, then the popover closes.\n *\n * Only popover-type containers (those that pass `dismissOnInnerButtonPress`\n * as `true`, the default) react to the event. Modal/tray/fullscreen Dialog\n * containers explicitly opt out so a Button inside their content does not\n * auto-close them.\n *\n * When called outside an `EventBusProvider` (e.g. in unit tests that render\n * a Button without wrapping in `<Root>`), the returned function is a no-op —\n * the dismiss flow gracefully degrades rather than throwing.\n */\nexport function useDismissParentPopover() {\n // Read context defensively: `Button` / `ItemButton` use this hook\n // unconditionally, but tests (and standalone usages outside `<Root>`) may\n // mount them without an `EventBusProvider`. A throw would crash every\n // Button render in those cases.\n const bus = useContext(EventBusContext);\n const emit = bus?.emit;\n return useCallback(\n (from: Element | null) => {\n if (!from || !emit) return;\n emit<PopoverDismissAncestorPayload>('popover:dismiss-ancestor', { from });\n },\n [emit],\n );\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;AAmFA,MAAM,+BAAe,IAAI,KAAmC;;;;;;AAO5D,SAAS,kBACP,QACuC;AACvC,KAAI,CAAC,OAAQ,QAAO;AACpB,MAAK,MAAM,SAAS,aAClB,KAAI,MAAM,GAAG,gBAAgB,EAAE,SAAS,OAAO,CAC7C,QAAO;AAGX,QAAO;;;;;;;;;AAUT,SAAS,sBACP,QACA,gBACA,mBACS;AACT,KAAI,qBAAqB,UAAU,kBAAkB,SAAS,OAAO,CACnE,QAAO;CAET,IAAI,MAAsB;CAC1B,MAAM,0BAAU,IAAI,KAAa;AACjC,QAAO,KAAK;EACV,MAAM,QAAQ,kBAAkB,IAAI;AACpC,MAAI,CAAC,MAAO,QAAO;EACnB,MAAM,CAAC,IAAI,SAAS;AACpB,MAAI,QAAQ,IAAI,GAAG,CAAE,QAAO;AAC5B,UAAQ,IAAI,GAAG;AACf,MAAI,OAAO,eAAgB,QAAO;AAClC,QAAM,MAAM,cAAc;;AAE5B,QAAO;;;;;;;;;;;;;;;;;;;;;;;;;AA0BT,SAAgB,eAAe,EAC7B,QACA,QACA,SACA,UAAU,MACV,YACA,cACA,4BAA4B,MAC5B,kBAAkB,QACY;CAC9B,MAAM,EAAE,MAAM,OAAO,aAAa;CAElC,MAAM,YAAY,OAAO,OAAO;AAChC,iBAAgB;AACd,YAAU,UAAU;IACnB,CAAC,OAAO,CAAC;CAEZ,MAAM,aAAa,OAAO,QAAQ;AAClC,iBAAgB;AACd,aAAW,UAAU;IACpB,CAAC,QAAQ,CAAC;CAMb,MAAM,kBAAkB,OAAO,aAAa;AAC5C,iBAAgB;AACd,kBAAgB,UAAU;IACzB,CAAC,aAAa,CAAC;CAElB,MAAM,gBAAgB,OAAO,WAAW;AACxC,iBAAgB;AACd,gBAAc,UAAU;IACvB,CAAC,WAAW,CAAC;AAEhB,iBAAgB;AACd,MAAI,CAAC,QAAS;EAOd,MAAM,UAAU,kBACZ,GAAG,iBAAiB,SAA6B;AAC/C,OAAI,KAAK,WAAW,UAAU,CAAC,UAAU,QAAS;GAClD,MAAM,YAAY,gBAAgB,SAAS,WAAW;AAOtD,OAAI,sBAAsB,KAAK,WAAW,QAAQ,UAAU,CAAE;AAC9D,cAAW,SAAS;IACpB,GACF;EAOJ,MAAM,aAAa,4BACf,GACE,6BACC,SAAwC;AACvC,OAAI,CAAC,UAAU,QAAS;GACxB,MAAM,YAAY,gBAAgB,SAAS;GAC3C,MAAM,OAAO,MAAM;AAInB,OAAI,CAAC,aAAa,CAAC,KAAM;AACzB,OAAI,UAAU,SAAS,KAAK,CAAE,YAAW,SAAS;IAErD,GACD;AAEJ,eAAa;AACX,cAAW;AACX,iBAAc;;IAEf;EAAC;EAAI;EAAQ;EAAS;EAA2B;EAAgB,CAAC;CAErE,MAAM,aAAa,OAAO,MAAM;AAChC,iBAAgB;AACd,MAAI,CAAC,SAAS;AACZ,OAAI,WAAW,QACb,cAAa,OAAO,OAAO;AAE7B,cAAW,UAAU;AACrB;;AAEF,MAAI,UAAU,CAAC,WAAW,SAAS;AACjC,cAAW,UAAU;AAIrB,gBAAa,IAAI,QAAQ;IACvB,sBAAsB,gBAAgB,SAAS,WAAW;IAC1D,oBAAoB,cAAc,SAAS,WAAW;IACvD,CAAC;AACF,QAAyB,gBAAgB;IACvC;IACA,WAAW,cAAc,SAAS,WAAW;IAC9C,CAAC;aACO,CAAC,UAAU,WAAW,SAAS;AACxC,cAAW,UAAU;AACrB,gBAAa,OAAO,OAAO;;IAE5B;EAAC;EAAQ;EAAM;EAAQ;EAAQ,CAAC;AAKnC,iBAAgB;AACd,eAAa;AACX,gBAAa,OAAO,OAAO;;IAE5B,CAAC,OAAO,CAAC;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA6Bd,SAAgB,0BAA0B;CAMxC,MAAM,OADM,WAAW,gBAAgB,EACrB;AAClB,QAAO,aACJ,SAAyB;AACxB,MAAI,CAAC,QAAQ,CAAC,KAAM;AACpB,OAAoC,4BAA4B,EAAE,MAAM,CAAC;IAE3E,CAAC,KAAK,CACP"}
@@ -1,4 +1,4 @@
1
- /** @license MIT | @cube-dev/ui-kit v0.140.1 | Cube Dev Team */
1
+ /** @license MIT | @cube-dev/ui-kit v0.142.0 | Cube Dev Team */
2
2
  //#region src/utils/react/useQaProps.ts
3
3
  /**
4
4
  * Processes QA props.
@@ -1,4 +1,4 @@
1
- /** @license MIT | @cube-dev/ui-kit v0.140.1 | Cube Dev Team */
1
+ /** @license MIT | @cube-dev/ui-kit v0.142.0 | Cube Dev Team */
2
2
  import { useEffect, useState } from "react";
3
3
 
4
4
  //#region src/utils/react/useViewportSize.ts
@@ -1,4 +1,4 @@
1
- /** @license MIT | @cube-dev/ui-kit v0.140.1 | Cube Dev Team */
1
+ /** @license MIT | @cube-dev/ui-kit v0.142.0 | Cube Dev Team */
2
2
  import { isValidElement } from "react";
3
3
  import { isFragment } from "react-is";
4
4
 
@@ -1,4 +1,4 @@
1
- /** @license MIT | @cube-dev/ui-kit v0.140.1 | Cube Dev Team */
1
+ /** @license MIT | @cube-dev/ui-kit v0.142.0 | Cube Dev Team */
2
2
  //#region src/utils/selection.ts
3
3
  /**
4
4
  * Deduplicate/toggle keys in a selection: if a key appears twice it is
@@ -1,4 +1,4 @@
1
- /** @license MIT | @cube-dev/ui-kit v0.140.1 | Cube Dev Team */
1
+ /** @license MIT | @cube-dev/ui-kit v0.142.0 | Cube Dev Team */
2
2
  //#region src/utils/styles.ts
3
3
  /**
4
4
  * Split properties into style and non-style properties.
@@ -1,4 +1,4 @@
1
- /** @license MIT | @cube-dev/ui-kit v0.140.1 | Cube Dev Team */
1
+ /** @license MIT | @cube-dev/ui-kit v0.142.0 | Cube Dev Team */
2
2
  //#region src/utils/tree.ts
3
3
  function toFlatTree(tree, includingFolders) {
4
4
  return tree.reduce((result, node) => {
@@ -1,4 +1,4 @@
1
- /** @license MIT | @cube-dev/ui-kit v0.140.1 | Cube Dev Team */
1
+ /** @license MIT | @cube-dev/ui-kit v0.142.0 | Cube Dev Team */
2
2
  //#region src/utils/warnings.ts
3
3
  const SUGGESTED_PROP_MAP = {
4
4
  disabled: "isDisabled",
package/dist/version.js CHANGED
@@ -1,6 +1,6 @@
1
- /** @license MIT | @cube-dev/ui-kit v0.140.1 | Cube Dev Team */
1
+ /** @license MIT | @cube-dev/ui-kit v0.142.0 | Cube Dev Team */
2
2
  //#region src/version.ts
3
- const VERSION = "0.140.1";
3
+ const VERSION = "0.142.0";
4
4
  const TASTY_VERSION = "2.6.2";
5
5
  if (typeof window !== "undefined") {
6
6
  const version = VERSION;
@@ -41,6 +41,7 @@ Supports [Base properties](../../BaseProperties.md)
41
41
  Customises the root element of the component.
42
42
 
43
43
  **Sub-elements:**
44
+
44
45
  - `Icon` – wrapper around the leading icon
45
46
  - `RightIcon` – wrapper around the trailing icon
46
47
  - `Label` – the button text content
@@ -114,27 +115,35 @@ The `mods` prop accepts the following modifiers you can override:
114
115
  ```jsx
115
116
  import { IconPlus } from '@tabler/icons-react';
116
117
 
117
- {/* Standard icon */}
118
- <Button icon={<IconPlus />} aria-label="Add item" />
119
-
120
- {/* Empty slot (reserves space but shows nothing) */}
121
- <Button icon={true} aria-label="Empty slot" />
122
-
123
- {/* Dynamic icon based on mods */}
124
- <Button
125
- icon={({ loading }) => loading ? <SpinnerIcon /> : <IconPlus />}
118
+ {
119
+ /* Standard icon */
120
+ }
121
+ <Button icon={<IconPlus />} aria-label="Add item" />;
122
+
123
+ {
124
+ /* Empty slot (reserves space but shows nothing) */
125
+ }
126
+ <Button icon={true} aria-label="Empty slot" />;
127
+
128
+ {
129
+ /* Dynamic icon based on mods */
130
+ }
131
+ <Button
132
+ icon={({ loading }) => (loading ? <SpinnerIcon /> : <IconPlus />)}
126
133
  aria-label="Dynamic icon"
127
134
  >
128
135
  Save
129
- </Button>
136
+ </Button>;
130
137
 
131
- {/* Return true from function for empty slot */}
132
- <Button
133
- icon={({ selected }) => selected ? <CheckIcon /> : true}
138
+ {
139
+ /* Return true from function for empty slot */
140
+ }
141
+ <Button
142
+ icon={({ selected }) => (selected ? <CheckIcon /> : true)}
134
143
  aria-label="Conditional icon"
135
144
  >
136
145
  Option
137
- </Button>
146
+ </Button>;
138
147
  ```
139
148
 
140
149
  ### Link Button
@@ -159,21 +168,31 @@ import { IconPlus } from '@tabler/icons-react';
159
168
  ### With Tooltip
160
169
 
161
170
  ```jsx
162
- {/* Simple tooltip text */}
163
- <Button icon={<IconPlus />} tooltip="Add new item" aria-label="Add" />
164
-
165
- {/* Auto tooltip on overflow */}
166
- <Button tooltip={true} width="100px">This text will be truncated</Button>
171
+ {
172
+ /* Simple tooltip text */
173
+ }
174
+ <Button icon={<IconPlus />} tooltip="Add new item" aria-label="Add" />;
175
+
176
+ {
177
+ /* Auto tooltip on overflow */
178
+ }
179
+ <Button tooltip={true} width="100px">
180
+ This text will be truncated
181
+ </Button>;
167
182
  ```
168
183
 
169
184
  ### Custom Size
170
185
 
171
186
  ```jsx
172
- {/* Using a number (pixels) */}
173
- <Button size={64}>Large Custom</Button>
174
-
175
- {/* Using a string multiplier */}
176
- <Button size="8x">8x Size</Button>
187
+ {
188
+ /* Using a number (pixels) */
189
+ }
190
+ <Button size={64}>Large Custom</Button>;
191
+
192
+ {
193
+ /* Using a string multiplier */
194
+ }
195
+ <Button size="8x">8x Size</Button>;
177
196
  ```
178
197
 
179
198
  ## Accessibility
@@ -195,6 +214,65 @@ import { IconPlus } from '@tabler/icons-react';
195
214
  - `aria-pressed` – Managed automatically when `isSelected` is used.
196
215
  - Additional ARIA attributes supported by React Aria's `useButton`.
197
216
 
217
+ ## Behaviour Inside Popovers
218
+
219
+ By default, pressing a `Button` inside an open popover (any overlay that
220
+ uses `usePopoverSync` — including the popovers rendered by `FilterPicker`,
221
+ `Picker`, `ComboBox`, `Select`, `MenuTrigger`, and
222
+ `DialogTrigger type="popover"`) **dismisses that popover** after the user's
223
+ `onPress` runs. This matches the common UX where a popover is a transient
224
+ surface: any action inside it should close it.
225
+
226
+ Modal / tray / fullscreen Dialogs are **not** affected — a Button inside a
227
+ modal Dialog does not auto-close it.
228
+
229
+ ### Opt-outs
230
+
231
+ There are three ways to keep the popover open when a Button is pressed:
232
+
233
+ 1. **Automatic for popover triggers.** Buttons that act as popover triggers
234
+ already carry `data-popover-trigger` (applied by `MenuTrigger`,
235
+ `DialogTrigger type="popover"`, `FilterPicker`, `Picker`, `ComboBox`,
236
+ `Select`). Modal-type `DialogTrigger` children are **not** marked, so they
237
+ correctly dismiss the parent popover and let the modal take over.
238
+ 2. **Manual via `data-popover-keep`.** Add the attribute to the button
239
+ (toggle case) or to any ancestor element (subtree case — useful when a
240
+ custom inline editor inside a popover footer should not collapse on
241
+ every interaction):
242
+
243
+ ```jsx
244
+ <Button data-popover-keep onPress={() => setMode((m) => !m)}>
245
+ Toggle
246
+ </Button>
247
+
248
+ <div data-popover-keep>
249
+ {/* All Buttons inside this wrapper opt out. */}
250
+ </div>
251
+ ```
252
+
253
+ 3. **Custom non-Cube triggers.** Call `useDismissParentPopover()` yourself if
254
+ your control should dismiss the popover but does not extend `Button` /
255
+ `ItemButton`:
256
+ ```jsx
257
+ const dismiss = useDismissParentPopover();
258
+ <MyCustomPressable
259
+ onPress={(e) => {
260
+ doThing();
261
+ dismiss(e.currentTarget);
262
+ }}
263
+ />;
264
+ ```
265
+
266
+ ### Async `onPress`
267
+
268
+ The dismiss is dispatched via the EventBus, which defers with `setTimeout(0)`.
269
+ This means synchronous `onPress` handlers (and any React state updates they
270
+ trigger) flush **before** the popover closes — including the common case of
271
+ opening a hoisted modal via `useDialogContainer` from a popover footer (the
272
+ modal mounts first, then the popover closes behind it). If your `onPress` is
273
+ async and `await`s before opening another overlay, the popover will close
274
+ first and the new overlay will appear after — usually fine, but worth noting.
275
+
198
276
  ## Best Practices
199
277
 
200
278
  1. **Do**: Provide a label for icon-only buttons