tldraw 3.16.0-canary.dfdf6b7de8c2 → 3.16.0-canary.e9c30b532b82

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 (275) hide show
  1. package/dist-cjs/index.d.ts +98 -5
  2. package/dist-cjs/index.js +13 -1
  3. package/dist-cjs/index.js.map +2 -2
  4. package/dist-cjs/lib/canvas/TldrawScribble.js +1 -1
  5. package/dist-cjs/lib/canvas/TldrawScribble.js.map +2 -2
  6. package/dist-cjs/lib/shapes/arrow/ArrowShapeUtil.js +3 -3
  7. package/dist-cjs/lib/shapes/arrow/ArrowShapeUtil.js.map +2 -2
  8. package/dist-cjs/lib/shapes/arrow/arrowTargetState.js +1 -1
  9. package/dist-cjs/lib/shapes/arrow/arrowTargetState.js.map +2 -2
  10. package/dist-cjs/lib/shapes/arrow/elbow/ElbowArrowDebug.js +3 -3
  11. package/dist-cjs/lib/shapes/arrow/elbow/ElbowArrowDebug.js.map +1 -1
  12. package/dist-cjs/lib/shapes/draw/DrawShapeUtil.js +3 -3
  13. package/dist-cjs/lib/shapes/draw/DrawShapeUtil.js.map +2 -2
  14. package/dist-cjs/lib/shapes/embed/EmbedShapeUtil.js +1 -1
  15. package/dist-cjs/lib/shapes/embed/EmbedShapeUtil.js.map +1 -1
  16. package/dist-cjs/lib/shapes/frame/FrameShapeUtil.js +12 -12
  17. package/dist-cjs/lib/shapes/frame/FrameShapeUtil.js.map +2 -2
  18. package/dist-cjs/lib/shapes/frame/components/FrameHeading.js +1 -1
  19. package/dist-cjs/lib/shapes/frame/components/FrameHeading.js.map +2 -2
  20. package/dist-cjs/lib/shapes/geo/GeoShapeUtil.js +2 -2
  21. package/dist-cjs/lib/shapes/geo/GeoShapeUtil.js.map +2 -2
  22. package/dist-cjs/lib/shapes/geo/components/GeoShapeBody.js +2 -1
  23. package/dist-cjs/lib/shapes/geo/components/GeoShapeBody.js.map +2 -2
  24. package/dist-cjs/lib/shapes/highlight/HighlightShapeUtil.js +5 -1
  25. package/dist-cjs/lib/shapes/highlight/HighlightShapeUtil.js.map +2 -2
  26. package/dist-cjs/lib/shapes/image/ImageShapeUtil.js +3 -3
  27. package/dist-cjs/lib/shapes/image/ImageShapeUtil.js.map +1 -1
  28. package/dist-cjs/lib/shapes/line/LineShapeUtil.js +5 -1
  29. package/dist-cjs/lib/shapes/line/LineShapeUtil.js.map +2 -2
  30. package/dist-cjs/lib/shapes/note/NoteShapeUtil.js +4 -4
  31. package/dist-cjs/lib/shapes/note/NoteShapeUtil.js.map +2 -2
  32. package/dist-cjs/lib/shapes/shared/ShapeFill.js +4 -4
  33. package/dist-cjs/lib/shapes/shared/ShapeFill.js.map +2 -2
  34. package/dist-cjs/lib/shapes/shared/usePrefersReducedMotion.js +10 -1
  35. package/dist-cjs/lib/shapes/shared/usePrefersReducedMotion.js.map +2 -2
  36. package/dist-cjs/lib/shapes/text/TextShapeUtil.js +2 -2
  37. package/dist-cjs/lib/shapes/text/TextShapeUtil.js.map +2 -2
  38. package/dist-cjs/lib/shapes/video/VideoShapeUtil.js +3 -3
  39. package/dist-cjs/lib/shapes/video/VideoShapeUtil.js.map +1 -1
  40. package/dist-cjs/lib/ui/TldrawUi.js +14 -0
  41. package/dist-cjs/lib/ui/TldrawUi.js.map +3 -3
  42. package/dist-cjs/lib/ui/components/AccessibilityMenu.js +35 -0
  43. package/dist-cjs/lib/ui/components/AccessibilityMenu.js.map +7 -0
  44. package/dist-cjs/lib/ui/components/ActionsMenu/DefaultActionsMenu.js +12 -3
  45. package/dist-cjs/lib/ui/components/ActionsMenu/DefaultActionsMenu.js.map +2 -2
  46. package/dist-cjs/lib/ui/components/DefaultMenuPanel.js +3 -2
  47. package/dist-cjs/lib/ui/components/DefaultMenuPanel.js.map +2 -2
  48. package/dist-cjs/lib/ui/components/MainMenu/DefaultMainMenuContent.js +3 -3
  49. package/dist-cjs/lib/ui/components/MainMenu/DefaultMainMenuContent.js.map +2 -2
  50. package/dist-cjs/lib/ui/components/Minimap/MinimapManager.js +4 -4
  51. package/dist-cjs/lib/ui/components/Minimap/MinimapManager.js.map +2 -2
  52. package/dist-cjs/lib/ui/components/MobileStylePanel.js +5 -3
  53. package/dist-cjs/lib/ui/components/MobileStylePanel.js.map +2 -2
  54. package/dist-cjs/lib/ui/components/NavigationPanel/DefaultNavigationPanel.js +1 -1
  55. package/dist-cjs/lib/ui/components/NavigationPanel/DefaultNavigationPanel.js.map +2 -2
  56. package/dist-cjs/lib/ui/components/PageMenu/DefaultPageMenu.js +2 -1
  57. package/dist-cjs/lib/ui/components/PageMenu/DefaultPageMenu.js.map +2 -2
  58. package/dist-cjs/lib/ui/components/SharePanel/PeopleMenuItem.js +3 -2
  59. package/dist-cjs/lib/ui/components/SharePanel/PeopleMenuItem.js.map +2 -2
  60. package/dist-cjs/lib/ui/components/SharePanel/UserPresenceColorPicker.js +2 -2
  61. package/dist-cjs/lib/ui/components/SharePanel/UserPresenceColorPicker.js.map +2 -2
  62. package/dist-cjs/lib/ui/components/StylePanel/DefaultStylePanel.js +2 -0
  63. package/dist-cjs/lib/ui/components/StylePanel/DefaultStylePanel.js.map +2 -2
  64. package/dist-cjs/lib/ui/components/StylePanel/DefaultStylePanelContent.js +171 -140
  65. package/dist-cjs/lib/ui/components/StylePanel/DefaultStylePanelContent.js.map +2 -2
  66. package/dist-cjs/lib/ui/components/StylePanel/DoubleDropdownPicker.js +3 -3
  67. package/dist-cjs/lib/ui/components/StylePanel/DoubleDropdownPicker.js.map +2 -2
  68. package/dist-cjs/lib/ui/components/StylePanel/DropdownPicker.js +26 -25
  69. package/dist-cjs/lib/ui/components/StylePanel/DropdownPicker.js.map +3 -3
  70. package/dist-cjs/lib/ui/components/Toolbar/DefaultImageToolbarContent.js +1 -1
  71. package/dist-cjs/lib/ui/components/Toolbar/DefaultImageToolbarContent.js.map +2 -2
  72. package/dist-cjs/lib/ui/components/Toolbar/DefaultToolbar.js +66 -21
  73. package/dist-cjs/lib/ui/components/Toolbar/DefaultToolbar.js.map +3 -3
  74. package/dist-cjs/lib/ui/components/Toolbar/OverflowingToolbar.js +189 -80
  75. package/dist-cjs/lib/ui/components/Toolbar/OverflowingToolbar.js.map +3 -3
  76. package/dist-cjs/lib/ui/components/Toolbar/ToggleToolLockedButton.js +5 -4
  77. package/dist-cjs/lib/ui/components/Toolbar/ToggleToolLockedButton.js.map +2 -2
  78. package/dist-cjs/lib/ui/components/menu-items.js +6 -0
  79. package/dist-cjs/lib/ui/components/menu-items.js.map +2 -2
  80. package/dist-cjs/lib/ui/components/primitives/TldrawUiButtonPicker.js +5 -16
  81. package/dist-cjs/lib/ui/components/primitives/TldrawUiButtonPicker.js.map +3 -3
  82. package/dist-cjs/lib/ui/components/primitives/TldrawUiContextualToolbar.js +1 -1
  83. package/dist-cjs/lib/ui/components/primitives/TldrawUiContextualToolbar.js.map +2 -2
  84. package/dist-cjs/lib/ui/components/primitives/TldrawUiPopover.js +3 -2
  85. package/dist-cjs/lib/ui/components/primitives/TldrawUiPopover.js.map +3 -3
  86. package/dist-cjs/lib/ui/components/primitives/TldrawUiToolbar.js +30 -7
  87. package/dist-cjs/lib/ui/components/primitives/TldrawUiToolbar.js.map +2 -2
  88. package/dist-cjs/lib/ui/components/primitives/TldrawUiTooltip.js +253 -0
  89. package/dist-cjs/lib/ui/components/primitives/TldrawUiTooltip.js.map +7 -0
  90. package/dist-cjs/lib/ui/components/primitives/layout.js +76 -0
  91. package/dist-cjs/lib/ui/components/primitives/layout.js.map +7 -0
  92. package/dist-cjs/lib/ui/components/primitives/menus/TldrawUiMenuContext.js.map +2 -2
  93. package/dist-cjs/lib/ui/components/primitives/menus/TldrawUiMenuGroup.js +25 -12
  94. package/dist-cjs/lib/ui/components/primitives/menus/TldrawUiMenuGroup.js.map +2 -2
  95. package/dist-cjs/lib/ui/components/primitives/menus/TldrawUiMenuItem.js +11 -26
  96. package/dist-cjs/lib/ui/components/primitives/menus/TldrawUiMenuItem.js.map +2 -2
  97. package/dist-cjs/lib/ui/context/TldrawUiContextProvider.js +3 -2
  98. package/dist-cjs/lib/ui/context/TldrawUiContextProvider.js.map +2 -2
  99. package/dist-cjs/lib/ui/context/actions.js +15 -0
  100. package/dist-cjs/lib/ui/context/actions.js.map +2 -2
  101. package/dist-cjs/lib/ui/context/events.js.map +2 -2
  102. package/dist-cjs/lib/ui/hooks/useTools.js +21 -3
  103. package/dist-cjs/lib/ui/hooks/useTools.js.map +2 -2
  104. package/dist-cjs/lib/ui/hooks/useTranslation/TLUiTranslationKey.js.map +1 -1
  105. package/dist-cjs/lib/ui/hooks/useTranslation/defaultTranslation.js +3 -0
  106. package/dist-cjs/lib/ui/hooks/useTranslation/defaultTranslation.js.map +2 -2
  107. package/dist-cjs/lib/ui/version.js +3 -3
  108. package/dist-cjs/lib/ui/version.js.map +1 -1
  109. package/dist-esm/index.d.mts +98 -5
  110. package/dist-esm/index.mjs +23 -1
  111. package/dist-esm/index.mjs.map +2 -2
  112. package/dist-esm/lib/canvas/TldrawScribble.mjs +1 -1
  113. package/dist-esm/lib/canvas/TldrawScribble.mjs.map +2 -2
  114. package/dist-esm/lib/shapes/arrow/ArrowShapeUtil.mjs +4 -3
  115. package/dist-esm/lib/shapes/arrow/ArrowShapeUtil.mjs.map +2 -2
  116. package/dist-esm/lib/shapes/arrow/arrowTargetState.mjs +1 -1
  117. package/dist-esm/lib/shapes/arrow/arrowTargetState.mjs.map +2 -2
  118. package/dist-esm/lib/shapes/arrow/elbow/ElbowArrowDebug.mjs +3 -3
  119. package/dist-esm/lib/shapes/arrow/elbow/ElbowArrowDebug.mjs.map +1 -1
  120. package/dist-esm/lib/shapes/draw/DrawShapeUtil.mjs +4 -3
  121. package/dist-esm/lib/shapes/draw/DrawShapeUtil.mjs.map +2 -2
  122. package/dist-esm/lib/shapes/embed/EmbedShapeUtil.mjs +1 -1
  123. package/dist-esm/lib/shapes/embed/EmbedShapeUtil.mjs.map +1 -1
  124. package/dist-esm/lib/shapes/frame/FrameShapeUtil.mjs +13 -12
  125. package/dist-esm/lib/shapes/frame/FrameShapeUtil.mjs.map +2 -2
  126. package/dist-esm/lib/shapes/frame/components/FrameHeading.mjs +1 -1
  127. package/dist-esm/lib/shapes/frame/components/FrameHeading.mjs.map +2 -2
  128. package/dist-esm/lib/shapes/geo/GeoShapeUtil.mjs +3 -2
  129. package/dist-esm/lib/shapes/geo/GeoShapeUtil.mjs.map +2 -2
  130. package/dist-esm/lib/shapes/geo/components/GeoShapeBody.mjs +2 -1
  131. package/dist-esm/lib/shapes/geo/components/GeoShapeBody.mjs.map +2 -2
  132. package/dist-esm/lib/shapes/highlight/HighlightShapeUtil.mjs +6 -1
  133. package/dist-esm/lib/shapes/highlight/HighlightShapeUtil.mjs.map +2 -2
  134. package/dist-esm/lib/shapes/image/ImageShapeUtil.mjs +3 -3
  135. package/dist-esm/lib/shapes/image/ImageShapeUtil.mjs.map +1 -1
  136. package/dist-esm/lib/shapes/line/LineShapeUtil.mjs +6 -1
  137. package/dist-esm/lib/shapes/line/LineShapeUtil.mjs.map +2 -2
  138. package/dist-esm/lib/shapes/note/NoteShapeUtil.mjs +5 -4
  139. package/dist-esm/lib/shapes/note/NoteShapeUtil.mjs.map +2 -2
  140. package/dist-esm/lib/shapes/shared/ShapeFill.mjs +5 -4
  141. package/dist-esm/lib/shapes/shared/ShapeFill.mjs.map +2 -2
  142. package/dist-esm/lib/shapes/shared/usePrefersReducedMotion.mjs +10 -1
  143. package/dist-esm/lib/shapes/shared/usePrefersReducedMotion.mjs.map +2 -2
  144. package/dist-esm/lib/shapes/text/TextShapeUtil.mjs +3 -2
  145. package/dist-esm/lib/shapes/text/TextShapeUtil.mjs.map +2 -2
  146. package/dist-esm/lib/shapes/video/VideoShapeUtil.mjs +3 -3
  147. package/dist-esm/lib/shapes/video/VideoShapeUtil.mjs.map +1 -1
  148. package/dist-esm/lib/ui/TldrawUi.mjs +16 -2
  149. package/dist-esm/lib/ui/TldrawUi.mjs.map +3 -3
  150. package/dist-esm/lib/ui/components/AccessibilityMenu.mjs +19 -0
  151. package/dist-esm/lib/ui/components/AccessibilityMenu.mjs.map +7 -0
  152. package/dist-esm/lib/ui/components/ActionsMenu/DefaultActionsMenu.mjs +12 -3
  153. package/dist-esm/lib/ui/components/ActionsMenu/DefaultActionsMenu.mjs.map +2 -2
  154. package/dist-esm/lib/ui/components/DefaultMenuPanel.mjs +3 -2
  155. package/dist-esm/lib/ui/components/DefaultMenuPanel.mjs.map +2 -2
  156. package/dist-esm/lib/ui/components/MainMenu/DefaultMainMenuContent.mjs +3 -5
  157. package/dist-esm/lib/ui/components/MainMenu/DefaultMainMenuContent.mjs.map +2 -2
  158. package/dist-esm/lib/ui/components/Minimap/MinimapManager.mjs +4 -4
  159. package/dist-esm/lib/ui/components/Minimap/MinimapManager.mjs.map +2 -2
  160. package/dist-esm/lib/ui/components/MobileStylePanel.mjs +6 -3
  161. package/dist-esm/lib/ui/components/MobileStylePanel.mjs.map +2 -2
  162. package/dist-esm/lib/ui/components/NavigationPanel/DefaultNavigationPanel.mjs +1 -1
  163. package/dist-esm/lib/ui/components/NavigationPanel/DefaultNavigationPanel.mjs.map +2 -2
  164. package/dist-esm/lib/ui/components/PageMenu/DefaultPageMenu.mjs +2 -1
  165. package/dist-esm/lib/ui/components/PageMenu/DefaultPageMenu.mjs.map +2 -2
  166. package/dist-esm/lib/ui/components/SharePanel/PeopleMenuItem.mjs +3 -2
  167. package/dist-esm/lib/ui/components/SharePanel/PeopleMenuItem.mjs.map +2 -2
  168. package/dist-esm/lib/ui/components/SharePanel/UserPresenceColorPicker.mjs +2 -2
  169. package/dist-esm/lib/ui/components/SharePanel/UserPresenceColorPicker.mjs.map +2 -2
  170. package/dist-esm/lib/ui/components/StylePanel/DefaultStylePanel.mjs +3 -1
  171. package/dist-esm/lib/ui/components/StylePanel/DefaultStylePanel.mjs.map +2 -2
  172. package/dist-esm/lib/ui/components/StylePanel/DefaultStylePanelContent.mjs +171 -140
  173. package/dist-esm/lib/ui/components/StylePanel/DefaultStylePanelContent.mjs.map +2 -2
  174. package/dist-esm/lib/ui/components/StylePanel/DoubleDropdownPicker.mjs +3 -3
  175. package/dist-esm/lib/ui/components/StylePanel/DoubleDropdownPicker.mjs.map +2 -2
  176. package/dist-esm/lib/ui/components/StylePanel/DropdownPicker.mjs +26 -25
  177. package/dist-esm/lib/ui/components/StylePanel/DropdownPicker.mjs.map +2 -2
  178. package/dist-esm/lib/ui/components/Toolbar/DefaultImageToolbarContent.mjs +1 -1
  179. package/dist-esm/lib/ui/components/Toolbar/DefaultImageToolbarContent.mjs.map +2 -2
  180. package/dist-esm/lib/ui/components/Toolbar/DefaultToolbar.mjs +56 -21
  181. package/dist-esm/lib/ui/components/Toolbar/DefaultToolbar.mjs.map +2 -2
  182. package/dist-esm/lib/ui/components/Toolbar/OverflowingToolbar.mjs +192 -81
  183. package/dist-esm/lib/ui/components/Toolbar/OverflowingToolbar.mjs.map +3 -3
  184. package/dist-esm/lib/ui/components/Toolbar/ToggleToolLockedButton.mjs +5 -4
  185. package/dist-esm/lib/ui/components/Toolbar/ToggleToolLockedButton.mjs.map +2 -2
  186. package/dist-esm/lib/ui/components/menu-items.mjs +6 -0
  187. package/dist-esm/lib/ui/components/menu-items.mjs.map +2 -2
  188. package/dist-esm/lib/ui/components/primitives/TldrawUiButtonPicker.mjs +6 -6
  189. package/dist-esm/lib/ui/components/primitives/TldrawUiButtonPicker.mjs.map +2 -2
  190. package/dist-esm/lib/ui/components/primitives/TldrawUiContextualToolbar.mjs +1 -1
  191. package/dist-esm/lib/ui/components/primitives/TldrawUiContextualToolbar.mjs.map +2 -2
  192. package/dist-esm/lib/ui/components/primitives/TldrawUiPopover.mjs +3 -2
  193. package/dist-esm/lib/ui/components/primitives/TldrawUiPopover.mjs.map +2 -2
  194. package/dist-esm/lib/ui/components/primitives/TldrawUiToolbar.mjs +30 -7
  195. package/dist-esm/lib/ui/components/primitives/TldrawUiToolbar.mjs.map +2 -2
  196. package/dist-esm/lib/ui/components/primitives/TldrawUiTooltip.mjs +230 -0
  197. package/dist-esm/lib/ui/components/primitives/TldrawUiTooltip.mjs.map +7 -0
  198. package/dist-esm/lib/ui/components/primitives/layout.mjs +46 -0
  199. package/dist-esm/lib/ui/components/primitives/layout.mjs.map +7 -0
  200. package/dist-esm/lib/ui/components/primitives/menus/TldrawUiMenuContext.mjs.map +2 -2
  201. package/dist-esm/lib/ui/components/primitives/menus/TldrawUiMenuGroup.mjs +25 -12
  202. package/dist-esm/lib/ui/components/primitives/menus/TldrawUiMenuGroup.mjs.map +2 -2
  203. package/dist-esm/lib/ui/components/primitives/menus/TldrawUiMenuItem.mjs +11 -26
  204. package/dist-esm/lib/ui/components/primitives/menus/TldrawUiMenuItem.mjs.map +2 -2
  205. package/dist-esm/lib/ui/context/TldrawUiContextProvider.mjs +3 -2
  206. package/dist-esm/lib/ui/context/TldrawUiContextProvider.mjs.map +2 -2
  207. package/dist-esm/lib/ui/context/actions.mjs +15 -0
  208. package/dist-esm/lib/ui/context/actions.mjs.map +2 -2
  209. package/dist-esm/lib/ui/context/events.mjs.map +2 -2
  210. package/dist-esm/lib/ui/hooks/useTools.mjs +22 -3
  211. package/dist-esm/lib/ui/hooks/useTools.mjs.map +2 -2
  212. package/dist-esm/lib/ui/hooks/useTranslation/defaultTranslation.mjs +3 -0
  213. package/dist-esm/lib/ui/hooks/useTranslation/defaultTranslation.mjs.map +2 -2
  214. package/dist-esm/lib/ui/version.mjs +3 -3
  215. package/dist-esm/lib/ui/version.mjs.map +1 -1
  216. package/package.json +3 -3
  217. package/src/index.ts +18 -0
  218. package/src/lib/canvas/TldrawScribble.tsx +1 -1
  219. package/src/lib/shapes/arrow/ArrowShapeUtil.tsx +4 -3
  220. package/src/lib/shapes/arrow/arrowTargetState.ts +2 -1
  221. package/src/lib/shapes/arrow/elbow/ElbowArrowDebug.tsx +3 -3
  222. package/src/lib/shapes/draw/DrawShapeUtil.tsx +4 -3
  223. package/src/lib/shapes/embed/EmbedShapeUtil.tsx +1 -1
  224. package/src/lib/shapes/frame/FrameShapeUtil.tsx +21 -14
  225. package/src/lib/shapes/frame/components/FrameHeading.tsx +1 -1
  226. package/src/lib/shapes/geo/GeoShapeUtil.tsx +3 -2
  227. package/src/lib/shapes/geo/components/GeoShapeBody.tsx +2 -2
  228. package/src/lib/shapes/highlight/HighlightShapeUtil.tsx +7 -1
  229. package/src/lib/shapes/image/ImageShapeUtil.tsx +3 -3
  230. package/src/lib/shapes/line/LineShapeUtil.tsx +6 -1
  231. package/src/lib/shapes/note/NoteShapeUtil.tsx +9 -4
  232. package/src/lib/shapes/shared/ShapeFill.tsx +5 -4
  233. package/src/lib/shapes/shared/usePrefersReducedMotion.tsx +11 -1
  234. package/src/lib/shapes/text/TextShapeUtil.tsx +3 -2
  235. package/src/lib/shapes/video/VideoShapeUtil.tsx +3 -3
  236. package/src/lib/ui/TldrawUi.tsx +17 -2
  237. package/src/lib/ui/components/AccessibilityMenu.tsx +20 -0
  238. package/src/lib/ui/components/ActionsMenu/DefaultActionsMenu.tsx +15 -3
  239. package/src/lib/ui/components/DefaultMenuPanel.tsx +4 -3
  240. package/src/lib/ui/components/MainMenu/DefaultMainMenuContent.tsx +4 -4
  241. package/src/lib/ui/components/Minimap/MinimapManager.ts +4 -4
  242. package/src/lib/ui/components/MobileStylePanel.tsx +9 -6
  243. package/src/lib/ui/components/NavigationPanel/DefaultNavigationPanel.tsx +1 -1
  244. package/src/lib/ui/components/PageMenu/DefaultPageMenu.tsx +3 -2
  245. package/src/lib/ui/components/SharePanel/PeopleMenuItem.tsx +4 -3
  246. package/src/lib/ui/components/SharePanel/UserPresenceColorPicker.tsx +3 -3
  247. package/src/lib/ui/components/StylePanel/DefaultStylePanel.tsx +3 -1
  248. package/src/lib/ui/components/StylePanel/DefaultStylePanelContent.tsx +146 -107
  249. package/src/lib/ui/components/StylePanel/DoubleDropdownPicker.tsx +3 -3
  250. package/src/lib/ui/components/StylePanel/DropdownPicker.tsx +7 -6
  251. package/src/lib/ui/components/Toolbar/DefaultImageToolbarContent.tsx +1 -1
  252. package/src/lib/ui/components/Toolbar/DefaultToolbar.tsx +55 -23
  253. package/src/lib/ui/components/Toolbar/OverflowingToolbar.tsx +212 -61
  254. package/src/lib/ui/components/Toolbar/ToggleToolLockedButton.tsx +14 -11
  255. package/src/lib/ui/components/menu-items.tsx +8 -0
  256. package/src/lib/ui/components/primitives/TldrawUiButtonPicker.tsx +40 -37
  257. package/src/lib/ui/components/primitives/TldrawUiContextualToolbar.tsx +1 -1
  258. package/src/lib/ui/components/primitives/TldrawUiPopover.tsx +4 -2
  259. package/src/lib/ui/components/primitives/TldrawUiToolbar.tsx +51 -12
  260. package/src/lib/ui/components/primitives/TldrawUiTooltip.tsx +301 -0
  261. package/src/lib/ui/components/primitives/layout.tsx +107 -0
  262. package/src/lib/ui/components/primitives/menus/TldrawUiMenuContext.tsx +0 -1
  263. package/src/lib/ui/components/primitives/menus/TldrawUiMenuGroup.tsx +29 -16
  264. package/src/lib/ui/components/primitives/menus/TldrawUiMenuItem.tsx +15 -25
  265. package/src/lib/ui/context/TldrawUiContextProvider.tsx +23 -20
  266. package/src/lib/ui/context/actions.tsx +15 -0
  267. package/src/lib/ui/context/events.tsx +1 -0
  268. package/src/lib/ui/hooks/useTools.tsx +25 -3
  269. package/src/lib/ui/hooks/useTranslation/TLUiTranslationKey.ts +3 -0
  270. package/src/lib/ui/hooks/useTranslation/defaultTranslation.ts +3 -0
  271. package/src/lib/ui/version.ts +3 -3
  272. package/src/lib/ui.css +406 -292
  273. package/src/test/arrows-megabus.test.tsx +12 -6
  274. package/src/test/inner-outer-margin.test.ts +315 -0
  275. package/tldraw.css +699 -580
@@ -1,6 +1,8 @@
1
1
  import classnames from 'classnames'
2
2
  import { Toolbar as _Toolbar } from 'radix-ui'
3
3
  import React from 'react'
4
+ import { TldrawUiColumn, TldrawUiGrid, TldrawUiRow } from './layout'
5
+ import { TldrawUiTooltip } from './TldrawUiTooltip'
4
6
 
5
7
  /** @public */
6
8
  export interface TLUiToolbarProps extends React.HTMLAttributes<HTMLDivElement> {
@@ -8,20 +10,42 @@ export interface TLUiToolbarProps extends React.HTMLAttributes<HTMLDivElement> {
8
10
  className?: string
9
11
  dir?: 'ltr' | 'rtl'
10
12
  label: string
13
+ orientation?: 'horizontal' | 'vertical' | 'grid'
14
+ tooltipSide?: 'top' | 'right' | 'bottom' | 'left'
15
+ }
16
+
17
+ const LayoutByOrientation = {
18
+ horizontal: TldrawUiRow,
19
+ vertical: TldrawUiColumn,
20
+ grid: TldrawUiGrid,
11
21
  }
12
22
 
13
23
  /** @public @react */
14
24
  export const TldrawUiToolbar = React.forwardRef<HTMLDivElement, TLUiToolbarProps>(
15
- ({ children, className, label, ...props }: TLUiToolbarProps, ref) => {
25
+ (
26
+ {
27
+ children,
28
+ className,
29
+ label,
30
+ orientation = 'horizontal',
31
+ tooltipSide,
32
+ ...props
33
+ }: TLUiToolbarProps,
34
+ ref
35
+ ) => {
36
+ const Layout = LayoutByOrientation[orientation]
16
37
  return (
17
- <_Toolbar.Root
18
- ref={ref}
19
- {...props}
20
- className={classnames('tlui-toolbar-container', className)}
21
- aria-label={label}
22
- >
23
- {children}
24
- </_Toolbar.Root>
38
+ <Layout asChild tooltipSide={tooltipSide}>
39
+ <_Toolbar.Root
40
+ ref={ref}
41
+ {...props}
42
+ className={classnames('tlui-toolbar', className)}
43
+ aria-label={label}
44
+ orientation={orientation === 'grid' ? 'horizontal' : orientation}
45
+ >
46
+ {children}
47
+ </_Toolbar.Root>
48
+ </Layout>
25
49
  )
26
50
  }
27
51
  )
@@ -34,23 +58,30 @@ export interface TLUiToolbarButtonProps extends React.HTMLAttributes<HTMLButtonE
34
58
  disabled?: boolean
35
59
  isActive?: boolean
36
60
  type: 'icon' | 'tool' | 'menu'
61
+ tooltip?: string
37
62
  }
38
63
 
39
64
  /** @public @react */
40
65
  export const TldrawUiToolbarButton = React.forwardRef<HTMLButtonElement, TLUiToolbarButtonProps>(
41
- ({ asChild, children, type, isActive, ...props }: TLUiToolbarButtonProps, ref) => {
42
- return (
66
+ ({ asChild, children, type, isActive, tooltip, ...props }: TLUiToolbarButtonProps, ref) => {
67
+ const button = (
43
68
  <_Toolbar.Button
44
69
  ref={ref}
45
70
  asChild={asChild}
46
71
  draggable={false}
47
72
  data-isactive={isActive}
48
73
  {...props}
74
+ // The tooltip takes care of this.
75
+ title={undefined}
49
76
  className={classnames('tlui-button', `tlui-button__${type}`, props.className)}
50
77
  >
51
78
  {children}
52
79
  </_Toolbar.Button>
53
80
  )
81
+
82
+ const tooltipContent = tooltip || props.title
83
+
84
+ return <TldrawUiTooltip content={tooltipContent}>{button}</TldrawUiTooltip>
54
85
  }
55
86
  )
56
87
 
@@ -93,6 +124,7 @@ export interface TLUiToolbarToggleItemProps extends React.HTMLAttributes<HTMLBut
93
124
  className?: string
94
125
  type: 'icon' | 'tool'
95
126
  value: string
127
+ tooltip?: string
96
128
  }
97
129
 
98
130
  /** @public @react */
@@ -101,11 +133,14 @@ export const TldrawUiToolbarToggleItem = ({
101
133
  className,
102
134
  type,
103
135
  value,
136
+ tooltip,
104
137
  ...props
105
138
  }: TLUiToolbarToggleItemProps) => {
106
- return (
139
+ const toggleItem = (
107
140
  <_Toolbar.ToggleItem
108
141
  {...props}
142
+ // The tooltip takes care of this.
143
+ title={undefined}
109
144
  className={classnames(
110
145
  'tlui-button',
111
146
  `tlui-button__${type}`,
@@ -117,4 +152,8 @@ export const TldrawUiToolbarToggleItem = ({
117
152
  {children}
118
153
  </_Toolbar.ToggleItem>
119
154
  )
155
+
156
+ const tooltipContent = tooltip || props.title
157
+
158
+ return <TldrawUiTooltip content={tooltipContent}>{toggleItem}</TldrawUiTooltip>
120
159
  }
@@ -0,0 +1,301 @@
1
+ import { assert, Atom, atom, Editor, uniqueId, useMaybeEditor, useValue } from '@tldraw/editor'
2
+ import { Tooltip as _Tooltip } from 'radix-ui'
3
+ import React, {
4
+ createContext,
5
+ forwardRef,
6
+ ReactNode,
7
+ useContext,
8
+ useEffect,
9
+ useRef,
10
+ useState,
11
+ } from 'react'
12
+ import { useTldrawUiOrientation } from './layout'
13
+
14
+ const DEFAULT_TOOLTIP_DELAY_MS = 700
15
+
16
+ /** @public */
17
+ export interface TldrawUiTooltipProps {
18
+ children: React.ReactNode
19
+ content?: string | React.ReactNode
20
+ side?: 'top' | 'right' | 'bottom' | 'left'
21
+ sideOffset?: number
22
+ disabled?: boolean
23
+ }
24
+
25
+ // Singleton tooltip manager
26
+ class TooltipManager {
27
+ private static instance: TooltipManager | null = null
28
+ private currentTooltip = atom<{
29
+ id: string
30
+ content: ReactNode
31
+ side: 'top' | 'right' | 'bottom' | 'left'
32
+ sideOffset: number
33
+ targetElement: HTMLElement
34
+ } | null>('current tooltip', null)
35
+ private destroyTimeoutId: number | null = null
36
+ private subscribers: Set<() => void> = new Set()
37
+
38
+ static getInstance(): TooltipManager {
39
+ if (!TooltipManager.instance) {
40
+ TooltipManager.instance = new TooltipManager()
41
+ }
42
+ return TooltipManager.instance
43
+ }
44
+
45
+ showTooltip(
46
+ tooltipId: string,
47
+ content: string | React.ReactNode,
48
+ targetElement: HTMLElement,
49
+ side: 'top' | 'right' | 'bottom' | 'left' = 'bottom',
50
+ sideOffset: number = 5
51
+ ) {
52
+ // Clear any existing destroy timeout
53
+ if (this.destroyTimeoutId) {
54
+ clearTimeout(this.destroyTimeoutId)
55
+ this.destroyTimeoutId = null
56
+ }
57
+
58
+ // Update current tooltip
59
+ this.currentTooltip.set({
60
+ id: tooltipId,
61
+ content,
62
+ side,
63
+ sideOffset,
64
+ targetElement,
65
+ })
66
+ }
67
+
68
+ hideTooltip(editor: Editor | null, tooltipId: string, instant: boolean = false) {
69
+ const hide = () => {
70
+ // Only hide if this is the current tooltip
71
+ if (this.currentTooltip.get()?.id === tooltipId) {
72
+ this.currentTooltip.set(null)
73
+ this.destroyTimeoutId = null
74
+ }
75
+ }
76
+
77
+ if (editor && !instant) {
78
+ // Start destroy timeout (1 second)
79
+ this.destroyTimeoutId = editor.timers.setTimeout(hide, 300)
80
+ } else {
81
+ hide()
82
+ }
83
+ }
84
+
85
+ hideAllTooltips() {
86
+ this.currentTooltip.set(null)
87
+ this.destroyTimeoutId = null
88
+ }
89
+
90
+ getCurrentTooltipData() {
91
+ if (!this.supportsHover()) return null
92
+ return this.currentTooltip.get()
93
+ }
94
+
95
+ private supportsHoverAtom: Atom<boolean> | null = null
96
+ supportsHover() {
97
+ if (!this.supportsHoverAtom) {
98
+ const mediaQuery = window.matchMedia('(hover: hover)')
99
+ const supportsHover = atom('has hover', mediaQuery.matches)
100
+ this.supportsHoverAtom = supportsHover
101
+ mediaQuery.addEventListener('change', (e) => {
102
+ supportsHover.set(e.matches)
103
+ })
104
+ }
105
+ return this.supportsHoverAtom.get()
106
+ }
107
+ }
108
+
109
+ export const tooltipManager = TooltipManager.getInstance()
110
+
111
+ // Context for the tooltip singleton
112
+ const TooltipSingletonContext = createContext<boolean>(false)
113
+
114
+ /** @public */
115
+ export interface TldrawUiTooltipProviderProps {
116
+ children: React.ReactNode
117
+ }
118
+
119
+ /** @public @react */
120
+ export function TldrawUiTooltipProvider({ children }: TldrawUiTooltipProviderProps) {
121
+ return (
122
+ <_Tooltip.Provider skipDelayDuration={700}>
123
+ <TooltipSingletonContext.Provider value={true}>
124
+ {children}
125
+ <TooltipSingleton />
126
+ </TooltipSingletonContext.Provider>
127
+ </_Tooltip.Provider>
128
+ )
129
+ }
130
+
131
+ // The singleton tooltip component that renders once
132
+ function TooltipSingleton() {
133
+ const editor = useMaybeEditor()
134
+ const [isOpen, setIsOpen] = useState(false)
135
+ const triggerRef = useRef<HTMLDivElement>(null)
136
+ const isFirstShowRef = useRef(true)
137
+ const showTimeoutRef = useRef<number | null>(null)
138
+
139
+ const currentTooltip = useValue(
140
+ 'current tooltip',
141
+ () => tooltipManager.getCurrentTooltipData(),
142
+ []
143
+ )
144
+
145
+ // Update open state and trigger position
146
+ useEffect(() => {
147
+ // Clear any existing show timeout
148
+ if (showTimeoutRef.current) {
149
+ clearTimeout(showTimeoutRef.current)
150
+ showTimeoutRef.current = null
151
+ }
152
+
153
+ if (currentTooltip && triggerRef.current) {
154
+ // Position the invisible trigger element over the active element
155
+ const activeRect = currentTooltip.targetElement.getBoundingClientRect()
156
+ const trigger = triggerRef.current
157
+
158
+ trigger.style.position = 'fixed'
159
+ trigger.style.left = `${activeRect.left}px`
160
+ trigger.style.top = `${activeRect.top}px`
161
+ trigger.style.width = `${activeRect.width}px`
162
+ trigger.style.height = `${activeRect.height}px`
163
+ trigger.style.pointerEvents = 'none'
164
+ trigger.style.zIndex = '9999'
165
+
166
+ // Handle delay for first show
167
+ if (isFirstShowRef.current && editor) {
168
+ showTimeoutRef.current = editor.timers.setTimeout(() => {
169
+ setIsOpen(true)
170
+ isFirstShowRef.current = false
171
+ }, editor.options.tooltipDelayMs)
172
+ } else {
173
+ // Subsequent tooltips show immediately
174
+ setIsOpen(true)
175
+ }
176
+ } else {
177
+ // Hide tooltip immediately
178
+ setIsOpen(false)
179
+ // Reset first show state after tooltip is hidden
180
+ isFirstShowRef.current = true
181
+ }
182
+ }, [editor, currentTooltip])
183
+
184
+ if (!currentTooltip) {
185
+ return null
186
+ }
187
+
188
+ return (
189
+ <_Tooltip.Root open={isOpen} delayDuration={0}>
190
+ <_Tooltip.Trigger asChild>
191
+ <div ref={triggerRef} />
192
+ </_Tooltip.Trigger>
193
+ <_Tooltip.Content
194
+ className="tlui-tooltip"
195
+ side={currentTooltip.side}
196
+ sideOffset={currentTooltip.sideOffset}
197
+ avoidCollisions
198
+ collisionPadding={8}
199
+ dir="ltr"
200
+ >
201
+ {currentTooltip.content}
202
+ <_Tooltip.Arrow className="tlui-tooltip__arrow" />
203
+ </_Tooltip.Content>
204
+ </_Tooltip.Root>
205
+ )
206
+ }
207
+
208
+ /** @public @react */
209
+ export const TldrawUiTooltip = forwardRef<HTMLButtonElement, TldrawUiTooltipProps>(
210
+ ({ children, content, side, sideOffset = 5, disabled = false }, ref) => {
211
+ const editor = useMaybeEditor()
212
+ const tooltipId = useRef<string>(uniqueId())
213
+ const hasProvider = useContext(TooltipSingletonContext)
214
+
215
+ const orientationCtx = useTldrawUiOrientation()
216
+ const sideToUse = side ?? orientationCtx.tooltipSide
217
+
218
+ useEffect(() => {
219
+ const currentTooltipId = tooltipId.current
220
+ return () => {
221
+ if (hasProvider) {
222
+ tooltipManager.hideTooltip(editor, currentTooltipId, true)
223
+ }
224
+ }
225
+ }, [editor, hasProvider])
226
+
227
+ // Don't show tooltip if disabled, no content, or UI labels are disabled
228
+ if (disabled || !content) {
229
+ return <>{children}</>
230
+ }
231
+
232
+ // Fallback to old behavior if no provider
233
+ if (!hasProvider) {
234
+ return (
235
+ <_Tooltip.Root
236
+ delayDuration={editor?.options.tooltipDelayMs || DEFAULT_TOOLTIP_DELAY_MS}
237
+ disableHoverableContent
238
+ >
239
+ <_Tooltip.Trigger asChild ref={ref}>
240
+ {children}
241
+ </_Tooltip.Trigger>
242
+ <_Tooltip.Content
243
+ className="tlui-tooltip"
244
+ side={sideToUse}
245
+ sideOffset={sideOffset}
246
+ avoidCollisions
247
+ collisionPadding={8}
248
+ dir="ltr"
249
+ >
250
+ {content}
251
+ <_Tooltip.Arrow className="tlui-tooltip__arrow" />
252
+ </_Tooltip.Content>
253
+ </_Tooltip.Root>
254
+ )
255
+ }
256
+
257
+ const child = React.Children.only(children)
258
+ assert(React.isValidElement(child), 'TldrawUiTooltip children must be a single element')
259
+
260
+ const handleMouseEnter = (event: React.MouseEvent<HTMLElement>) => {
261
+ child.props.onMouseEnter?.(event)
262
+ tooltipManager.showTooltip(
263
+ tooltipId.current,
264
+ content,
265
+ event.currentTarget as HTMLElement,
266
+ sideToUse,
267
+ sideOffset
268
+ )
269
+ }
270
+
271
+ const handleMouseLeave = (event: React.MouseEvent<HTMLElement>) => {
272
+ child.props.onMouseLeave?.(event)
273
+ tooltipManager.hideTooltip(editor, tooltipId.current)
274
+ }
275
+
276
+ const handleFocus = (event: React.FocusEvent<HTMLElement>) => {
277
+ child.props.onFocus?.(event)
278
+ tooltipManager.showTooltip(
279
+ tooltipId.current,
280
+ content,
281
+ event.currentTarget as HTMLElement,
282
+ sideToUse,
283
+ sideOffset
284
+ )
285
+ }
286
+
287
+ const handleBlur = (event: React.FocusEvent<HTMLElement>) => {
288
+ child.props.onBlur?.(event)
289
+ tooltipManager.hideTooltip(editor, tooltipId.current)
290
+ }
291
+
292
+ const childrenWithHandlers = React.cloneElement(children as React.ReactElement, {
293
+ onMouseEnter: handleMouseEnter,
294
+ onMouseLeave: handleMouseLeave,
295
+ onFocus: handleFocus,
296
+ onBlur: handleBlur,
297
+ })
298
+
299
+ return childrenWithHandlers
300
+ }
301
+ )
@@ -0,0 +1,107 @@
1
+ import classNames from 'classnames'
2
+ import { Slot } from 'radix-ui'
3
+ import { HTMLAttributes, ReactNode, createContext, forwardRef, useContext } from 'react'
4
+
5
+ /** @public */
6
+ export interface TldrawUiOrientationContext {
7
+ orientation: 'horizontal' | 'vertical'
8
+ tooltipSide: 'top' | 'right' | 'bottom' | 'left'
9
+ }
10
+
11
+ const TldrawUiOrientationContext = createContext<TldrawUiOrientationContext>({
12
+ orientation: 'horizontal',
13
+ tooltipSide: 'bottom',
14
+ })
15
+
16
+ /** @public */
17
+ export interface TldrawUiOrientationProviderProps {
18
+ children: ReactNode
19
+ orientation: 'horizontal' | 'vertical'
20
+ tooltipSide?: 'top' | 'right' | 'bottom' | 'left'
21
+ }
22
+ /** @public @react */
23
+ export function TldrawUiOrientationProvider({
24
+ children,
25
+ orientation,
26
+ tooltipSide,
27
+ }: TldrawUiOrientationProviderProps) {
28
+ const prevContext = useTldrawUiOrientation()
29
+ // generally, we want tooltip side to cascade down through the layout - apart from when the
30
+ // orientation changes. If the tooltip side is "bottom", and then I include some vertical layout
31
+ // elements, keeping the tooltip side as bottom will cause the tooltip to overlap elements
32
+ // stacked on top of each other. In the absence of a tooltip side, we pick a default side based
33
+ // on the orientation whenever the orientation changes.
34
+ const tooltipSideToUse =
35
+ tooltipSide ??
36
+ (orientation === prevContext.orientation
37
+ ? prevContext.tooltipSide
38
+ : orientation === 'horizontal'
39
+ ? 'bottom'
40
+ : 'right')
41
+
42
+ return (
43
+ <TldrawUiOrientationContext.Provider value={{ orientation, tooltipSide: tooltipSideToUse }}>
44
+ {children}
45
+ </TldrawUiOrientationContext.Provider>
46
+ )
47
+ }
48
+
49
+ /** @public */
50
+ export function useTldrawUiOrientation() {
51
+ return useContext(TldrawUiOrientationContext)
52
+ }
53
+
54
+ /** @public */
55
+ export interface TLUiLayoutProps extends HTMLAttributes<HTMLDivElement> {
56
+ children: ReactNode
57
+ tooltipSide?: 'top' | 'right' | 'bottom' | 'left'
58
+ asChild?: boolean
59
+ }
60
+
61
+ /**
62
+ * A row, usually of UI controls like buttons, select dropdown, checkboxes, etc.
63
+ *
64
+ * @public @react
65
+ */
66
+ export const TldrawUiRow = forwardRef<HTMLDivElement, TLUiLayoutProps>(
67
+ ({ asChild, className, tooltipSide, ...props }, ref) => {
68
+ const Component = asChild ? Slot.Root : 'div'
69
+ return (
70
+ <TldrawUiOrientationProvider orientation="horizontal" tooltipSide={tooltipSide}>
71
+ <Component ref={ref} className={classNames('tlui-row', className)} {...props} />
72
+ </TldrawUiOrientationProvider>
73
+ )
74
+ }
75
+ )
76
+
77
+ /**
78
+ * A column, usually of UI controls like buttons, select dropdown, checkboxes, etc.
79
+ *
80
+ * @public @react
81
+ */
82
+ export const TldrawUiColumn = forwardRef<HTMLDivElement, TLUiLayoutProps>(
83
+ ({ asChild, className, tooltipSide, ...props }, ref) => {
84
+ const Component = asChild ? Slot.Root : 'div'
85
+ return (
86
+ <TldrawUiOrientationProvider orientation="vertical" tooltipSide={tooltipSide}>
87
+ <Component ref={ref} className={classNames('tlui-column', className)} {...props} />
88
+ </TldrawUiOrientationProvider>
89
+ )
90
+ }
91
+ )
92
+
93
+ /**
94
+ * A tight grid 4 elements wide, usually of UI controls like buttons, select dropdown, checkboxes,
95
+ * etc.
96
+ *
97
+ * @public @react */
98
+ export const TldrawUiGrid = forwardRef<HTMLDivElement, TLUiLayoutProps>(
99
+ ({ asChild, className, tooltipSide, ...props }, ref) => {
100
+ const Component = asChild ? Slot.Root : 'div'
101
+ return (
102
+ <TldrawUiOrientationProvider orientation="horizontal" tooltipSide={tooltipSide}>
103
+ <Component ref={ref} className={classNames('tlui-grid', className)} {...props} />
104
+ </TldrawUiOrientationProvider>
105
+ )
106
+ }
107
+ )
@@ -3,7 +3,6 @@ import { TLUiEventSource } from '../../../context/events'
3
3
 
4
4
  /** @public */
5
5
  export type TLUiMenuContextType =
6
- | 'panel'
7
6
  | 'menu'
8
7
  | 'small-icons'
9
8
  | 'context-menu'
@@ -3,6 +3,7 @@ import { ReactNode } from 'react'
3
3
  import { unwrapLabel } from '../../../context/actions'
4
4
  import { TLUiTranslationKey } from '../../../hooks/useTranslation/TLUiTranslationKey'
5
5
  import { useTranslation } from '../../../hooks/useTranslation/useTranslation'
6
+ import { TldrawUiColumn, TldrawUiGrid, TldrawUiRow, useTldrawUiOrientation } from '../layout'
6
7
  import { TldrawUiDropdownMenuGroup } from '../TldrawUiDropdownMenu'
7
8
  import { useTldrawUiMenuContext } from './TldrawUiMenuContext'
8
9
 
@@ -19,25 +20,19 @@ export interface TLUiMenuGroupProps<TranslationKey extends string = string> {
19
20
 
20
21
  /** @public @react */
21
22
  export function TldrawUiMenuGroup({ id, label, className, children }: TLUiMenuGroupProps) {
22
- const { type: menuType, sourceId } = useTldrawUiMenuContext()
23
+ const menu = useTldrawUiMenuContext()
24
+ const { orientation } = useTldrawUiOrientation()
23
25
  const msg = useTranslation()
24
- const labelToUse = unwrapLabel(label, menuType)
26
+ const labelToUse = unwrapLabel(label, menu.type)
25
27
  const labelStr = labelToUse ? msg(labelToUse as TLUiTranslationKey) : undefined
26
28
 
27
- switch (menuType) {
28
- case 'panel': {
29
- return (
30
- <div
31
- className={classNames('tlui-menu__group', className)}
32
- data-testid={`${sourceId}-group.${id}`}
33
- >
34
- {children}
35
- </div>
36
- )
37
- }
29
+ switch (menu.type) {
38
30
  case 'menu': {
39
31
  return (
40
- <TldrawUiDropdownMenuGroup className={className} data-testid={`${sourceId}-group.${id}`}>
32
+ <TldrawUiDropdownMenuGroup
33
+ className={className}
34
+ data-testid={`${menu.sourceId}-group.${id}`}
35
+ >
41
36
  {children}
42
37
  </TldrawUiDropdownMenuGroup>
43
38
  )
@@ -47,7 +42,7 @@ export function TldrawUiMenuGroup({ id, label, className, children }: TLUiMenuGr
47
42
  <div
48
43
  dir="ltr"
49
44
  className={classNames('tlui-menu__group', className)}
50
- data-testid={`${sourceId}-group.${id}`}
45
+ data-testid={`${menu.sourceId}-group.${id}`}
51
46
  >
52
47
  {children}
53
48
  </div>
@@ -56,12 +51,30 @@ export function TldrawUiMenuGroup({ id, label, className, children }: TLUiMenuGr
56
51
  case 'keyboard-shortcuts': {
57
52
  // todo: if groups need a label, let's give em a label
58
53
  return (
59
- <div className="tlui-shortcuts-dialog__group" data-testid={`${sourceId}-group.${id}`}>
54
+ <div className="tlui-shortcuts-dialog__group" data-testid={`${menu.sourceId}-group.${id}`}>
60
55
  <h2 className="tlui-shortcuts-dialog__group__title">{labelStr}</h2>
61
56
  <div className="tlui-shortcuts-dialog__group__content">{children}</div>
62
57
  </div>
63
58
  )
64
59
  }
60
+ case 'toolbar': {
61
+ const Layout = orientation === 'horizontal' ? TldrawUiRow : TldrawUiColumn
62
+ return (
63
+ <Layout className="tlui-main-toolbar__group" data-testid={`${menu.sourceId}-group.${id}`}>
64
+ {children}
65
+ </Layout>
66
+ )
67
+ }
68
+ case 'toolbar-overflow': {
69
+ return (
70
+ <TldrawUiGrid
71
+ className="tlui-main-toolbar__group"
72
+ data-testid={`${menu.sourceId}-group.${id}`}
73
+ >
74
+ {children}
75
+ </TldrawUiGrid>
76
+ )
77
+ }
65
78
  default: {
66
79
  return children
67
80
  }