tldraw 3.16.0-canary.6f3aedaa1c01 → 3.16.0-canary.7379d3553d7e

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 (463) hide show
  1. package/dist-cjs/index.d.ts +362 -113
  2. package/dist-cjs/index.js +40 -14
  3. package/dist-cjs/index.js.map +2 -2
  4. package/dist-cjs/lib/Tldraw.js +12 -2
  5. package/dist-cjs/lib/Tldraw.js.map +2 -2
  6. package/dist-cjs/lib/canvas/TldrawScribble.js +1 -1
  7. package/dist-cjs/lib/canvas/TldrawScribble.js.map +2 -2
  8. package/dist-cjs/lib/defaultExternalContentHandlers.js +15 -4
  9. package/dist-cjs/lib/defaultExternalContentHandlers.js.map +2 -2
  10. package/dist-cjs/lib/shapes/arrow/arrow-types.js.map +1 -1
  11. package/dist-cjs/lib/shapes/arrow/arrowLabel.js +6 -0
  12. package/dist-cjs/lib/shapes/arrow/arrowLabel.js.map +3 -3
  13. package/dist-cjs/lib/shapes/arrow/arrowTargetState.js +3 -2
  14. package/dist-cjs/lib/shapes/arrow/arrowTargetState.js.map +2 -2
  15. package/dist-cjs/lib/shapes/arrow/elbow/ElbowArrowDebug.js +3 -3
  16. package/dist-cjs/lib/shapes/arrow/elbow/ElbowArrowDebug.js.map +1 -1
  17. package/dist-cjs/lib/shapes/arrow/toolStates/Pointing.js +1 -1
  18. package/dist-cjs/lib/shapes/arrow/toolStates/Pointing.js.map +2 -2
  19. package/dist-cjs/lib/shapes/bookmark/BookmarkShapeUtil.js +4 -4
  20. package/dist-cjs/lib/shapes/bookmark/BookmarkShapeUtil.js.map +2 -2
  21. package/dist-cjs/lib/shapes/embed/EmbedShapeUtil.js +1 -1
  22. package/dist-cjs/lib/shapes/embed/EmbedShapeUtil.js.map +1 -1
  23. package/dist-cjs/lib/shapes/frame/FrameShapeUtil.js +12 -5
  24. package/dist-cjs/lib/shapes/frame/FrameShapeUtil.js.map +2 -2
  25. package/dist-cjs/lib/shapes/frame/components/FrameHeading.js +1 -1
  26. package/dist-cjs/lib/shapes/frame/components/FrameHeading.js.map +2 -2
  27. package/dist-cjs/lib/shapes/frame/components/FrameLabelInput.js +8 -2
  28. package/dist-cjs/lib/shapes/frame/components/FrameLabelInput.js.map +2 -2
  29. package/dist-cjs/lib/shapes/geo/GeoShapeUtil.js +1 -0
  30. package/dist-cjs/lib/shapes/geo/GeoShapeUtil.js.map +2 -2
  31. package/dist-cjs/lib/shapes/image/ImageShapeUtil.js +6 -3
  32. package/dist-cjs/lib/shapes/image/ImageShapeUtil.js.map +2 -2
  33. package/dist-cjs/lib/shapes/note/NoteShapeUtil.js +2 -1
  34. package/dist-cjs/lib/shapes/note/NoteShapeUtil.js.map +2 -2
  35. package/dist-cjs/lib/shapes/shared/HyperlinkButton.js +4 -4
  36. package/dist-cjs/lib/shapes/shared/HyperlinkButton.js.map +2 -2
  37. package/dist-cjs/lib/shapes/shared/PlainTextLabel.js +1 -3
  38. package/dist-cjs/lib/shapes/shared/PlainTextLabel.js.map +2 -2
  39. package/dist-cjs/lib/shapes/shared/ShapeFill.js +1 -1
  40. package/dist-cjs/lib/shapes/shared/ShapeFill.js.map +2 -2
  41. package/dist-cjs/lib/shapes/shared/freehand/svg.js.map +2 -2
  42. package/dist-cjs/lib/shapes/shared/useEditablePlainText.js +3 -5
  43. package/dist-cjs/lib/shapes/shared/useEditablePlainText.js.map +2 -2
  44. package/dist-cjs/lib/shapes/shared/useImageOrVideoAsset.js +0 -2
  45. package/dist-cjs/lib/shapes/shared/useImageOrVideoAsset.js.map +2 -2
  46. package/dist-cjs/lib/shapes/text/PlainTextArea.js +3 -2
  47. package/dist-cjs/lib/shapes/text/PlainTextArea.js.map +2 -2
  48. package/dist-cjs/lib/shapes/text/RichTextArea.js +3 -3
  49. package/dist-cjs/lib/shapes/text/RichTextArea.js.map +2 -2
  50. package/dist-cjs/lib/shapes/video/VideoShapeUtil.js +3 -3
  51. package/dist-cjs/lib/shapes/video/VideoShapeUtil.js.map +1 -1
  52. package/dist-cjs/lib/tools/EraserTool/childStates/Erasing.js +25 -1
  53. package/dist-cjs/lib/tools/EraserTool/childStates/Erasing.js.map +2 -2
  54. package/dist-cjs/lib/tools/EraserTool/childStates/Pointing.js +12 -0
  55. package/dist-cjs/lib/tools/EraserTool/childStates/Pointing.js.map +2 -2
  56. package/dist-cjs/lib/tools/SelectTool/childStates/DraggingHandle.js +3 -1
  57. package/dist-cjs/lib/tools/SelectTool/childStates/DraggingHandle.js.map +2 -2
  58. package/dist-cjs/lib/ui/TldrawUi.js +27 -12
  59. package/dist-cjs/lib/ui/TldrawUi.js.map +3 -3
  60. package/dist-cjs/lib/ui/assetUrls.js +13 -10
  61. package/dist-cjs/lib/ui/assetUrls.js.map +2 -2
  62. package/dist-cjs/lib/ui/components/A11y.js +1 -1
  63. package/dist-cjs/lib/ui/components/A11y.js.map +2 -2
  64. package/dist-cjs/lib/ui/components/ActionsMenu/DefaultActionsMenu.js +10 -2
  65. package/dist-cjs/lib/ui/components/ActionsMenu/DefaultActionsMenu.js.map +2 -2
  66. package/dist-cjs/lib/ui/components/{FollowingIndicator.js → DefaultFollowingIndicator.js} +6 -6
  67. package/dist-cjs/lib/ui/components/DefaultFollowingIndicator.js.map +7 -0
  68. package/dist-cjs/lib/ui/components/InputModeMenu.js +77 -0
  69. package/dist-cjs/lib/ui/components/InputModeMenu.js.map +7 -0
  70. package/dist-cjs/lib/ui/components/KeyboardShortcutsDialog/DefaultKeyboardShortcutsDialogContent.js +6 -6
  71. package/dist-cjs/lib/ui/components/KeyboardShortcutsDialog/DefaultKeyboardShortcutsDialogContent.js.map +1 -1
  72. package/dist-cjs/lib/ui/components/LanguageMenu.js +1 -0
  73. package/dist-cjs/lib/ui/components/LanguageMenu.js.map +2 -2
  74. package/dist-cjs/lib/ui/components/MainMenu/DefaultMainMenuContent.js +2 -0
  75. package/dist-cjs/lib/ui/components/MainMenu/DefaultMainMenuContent.js.map +2 -2
  76. package/dist-cjs/lib/ui/components/Minimap/DefaultMinimap.js +2 -1
  77. package/dist-cjs/lib/ui/components/Minimap/DefaultMinimap.js.map +2 -2
  78. package/dist-cjs/lib/ui/components/Minimap/MinimapManager.js +4 -4
  79. package/dist-cjs/lib/ui/components/Minimap/MinimapManager.js.map +2 -2
  80. package/dist-cjs/lib/ui/components/MobileStylePanel.js +4 -2
  81. package/dist-cjs/lib/ui/components/MobileStylePanel.js.map +2 -2
  82. package/dist-cjs/lib/ui/components/PageMenu/DefaultPageMenu.js +1 -1
  83. package/dist-cjs/lib/ui/components/PageMenu/DefaultPageMenu.js.map +2 -2
  84. package/dist-cjs/lib/ui/components/StylePanel/DefaultStylePanel.js +9 -4
  85. package/dist-cjs/lib/ui/components/StylePanel/DefaultStylePanel.js.map +2 -2
  86. package/dist-cjs/lib/ui/components/StylePanel/DefaultStylePanelContent.js +255 -316
  87. package/dist-cjs/lib/ui/components/StylePanel/DefaultStylePanelContent.js.map +2 -2
  88. package/dist-cjs/lib/ui/components/StylePanel/StylePanelButtonPicker.js +147 -0
  89. package/dist-cjs/lib/ui/components/StylePanel/StylePanelButtonPicker.js.map +7 -0
  90. package/dist-cjs/lib/ui/components/StylePanel/StylePanelContext.js +68 -0
  91. package/dist-cjs/lib/ui/components/StylePanel/StylePanelContext.js.map +7 -0
  92. package/dist-cjs/lib/ui/components/StylePanel/{DoubleDropdownPicker.js → StylePanelDoubleDropdownPicker.js} +23 -22
  93. package/dist-cjs/lib/ui/components/StylePanel/StylePanelDoubleDropdownPicker.js.map +7 -0
  94. package/dist-cjs/lib/ui/components/StylePanel/{DropdownPicker.js → StylePanelDropdownPicker.js} +24 -21
  95. package/dist-cjs/lib/ui/components/StylePanel/StylePanelDropdownPicker.js.map +7 -0
  96. package/dist-cjs/lib/ui/components/StylePanel/StylePanelSubheading.js +28 -0
  97. package/dist-cjs/lib/ui/components/StylePanel/StylePanelSubheading.js.map +7 -0
  98. package/dist-cjs/lib/ui/components/Toolbar/AltTextEditor.js +2 -0
  99. package/dist-cjs/lib/ui/components/Toolbar/AltTextEditor.js.map +2 -2
  100. package/dist-cjs/lib/ui/components/Toolbar/DefaultImageToolbarContent.js +39 -10
  101. package/dist-cjs/lib/ui/components/Toolbar/DefaultImageToolbarContent.js.map +2 -2
  102. package/dist-cjs/lib/ui/components/Toolbar/DefaultToolbar.js +66 -22
  103. package/dist-cjs/lib/ui/components/Toolbar/DefaultToolbar.js.map +3 -3
  104. package/dist-cjs/lib/ui/components/Toolbar/DefaultVideoToolbarContent.js +15 -3
  105. package/dist-cjs/lib/ui/components/Toolbar/DefaultVideoToolbarContent.js.map +2 -2
  106. package/dist-cjs/lib/ui/components/Toolbar/LinkEditor.js +2 -1
  107. package/dist-cjs/lib/ui/components/Toolbar/LinkEditor.js.map +2 -2
  108. package/dist-cjs/lib/ui/components/Toolbar/OverflowingToolbar.js +188 -78
  109. package/dist-cjs/lib/ui/components/Toolbar/OverflowingToolbar.js.map +3 -3
  110. package/dist-cjs/lib/ui/components/Toolbar/ToggleToolLockedButton.js +6 -2
  111. package/dist-cjs/lib/ui/components/Toolbar/ToggleToolLockedButton.js.map +2 -2
  112. package/dist-cjs/lib/ui/components/primitives/TldrawUiContextualToolbar.js +11 -2
  113. package/dist-cjs/lib/ui/components/primitives/TldrawUiContextualToolbar.js.map +2 -2
  114. package/dist-cjs/lib/ui/components/primitives/TldrawUiInput.js +5 -3
  115. package/dist-cjs/lib/ui/components/primitives/TldrawUiInput.js.map +2 -2
  116. package/dist-cjs/lib/ui/components/primitives/TldrawUiSlider.js +18 -5
  117. package/dist-cjs/lib/ui/components/primitives/TldrawUiSlider.js.map +2 -2
  118. package/dist-cjs/lib/ui/components/primitives/TldrawUiToolbar.js +18 -3
  119. package/dist-cjs/lib/ui/components/primitives/TldrawUiToolbar.js.map +2 -2
  120. package/dist-cjs/lib/ui/components/primitives/TldrawUiTooltip.js +187 -165
  121. package/dist-cjs/lib/ui/components/primitives/TldrawUiTooltip.js.map +2 -2
  122. package/dist-cjs/lib/ui/components/primitives/layout.js +30 -5
  123. package/dist-cjs/lib/ui/components/primitives/layout.js.map +2 -2
  124. package/dist-cjs/lib/ui/components/primitives/menus/TldrawUiMenuCheckboxItem.js +3 -0
  125. package/dist-cjs/lib/ui/components/primitives/menus/TldrawUiMenuCheckboxItem.js.map +2 -2
  126. package/dist-cjs/lib/ui/components/primitives/menus/TldrawUiMenuContext.js.map +2 -2
  127. package/dist-cjs/lib/ui/components/primitives/menus/TldrawUiMenuGroup.js +25 -12
  128. package/dist-cjs/lib/ui/components/primitives/menus/TldrawUiMenuGroup.js.map +2 -2
  129. package/dist-cjs/lib/ui/components/primitives/menus/TldrawUiMenuItem.js +11 -27
  130. package/dist-cjs/lib/ui/components/primitives/menus/TldrawUiMenuItem.js.map +2 -2
  131. package/dist-cjs/lib/ui/context/actions.js +29 -30
  132. package/dist-cjs/lib/ui/context/actions.js.map +2 -2
  133. package/dist-cjs/lib/ui/context/components.js +2 -0
  134. package/dist-cjs/lib/ui/context/components.js.map +2 -2
  135. package/dist-cjs/lib/ui/context/events.js.map +2 -2
  136. package/dist-cjs/lib/ui/hooks/useClipboardEvents.js +1 -1
  137. package/dist-cjs/lib/ui/hooks/useClipboardEvents.js.map +2 -2
  138. package/dist-cjs/lib/ui/hooks/useExportAs.js +3 -2
  139. package/dist-cjs/lib/ui/hooks/useExportAs.js.map +2 -2
  140. package/dist-cjs/lib/ui/hooks/useTools.js +22 -4
  141. package/dist-cjs/lib/ui/hooks/useTools.js.map +2 -2
  142. package/dist-cjs/lib/ui/hooks/useTranslation/TLUiTranslationKey.js.map +1 -1
  143. package/dist-cjs/lib/ui/hooks/useTranslation/defaultTranslation.js +12 -2
  144. package/dist-cjs/lib/ui/hooks/useTranslation/defaultTranslation.js.map +2 -2
  145. package/dist-cjs/lib/ui/kbd-utils.js +9 -3
  146. package/dist-cjs/lib/ui/kbd-utils.js.map +2 -2
  147. package/dist-cjs/lib/ui/version.js +3 -3
  148. package/dist-cjs/lib/ui/version.js.map +1 -1
  149. package/dist-cjs/lib/utils/export/copyAs.js +1 -2
  150. package/dist-cjs/lib/utils/export/copyAs.js.map +2 -2
  151. package/dist-cjs/lib/utils/export/export.js +0 -20
  152. package/dist-cjs/lib/utils/export/export.js.map +2 -2
  153. package/dist-cjs/lib/utils/export/exportAs.js +1 -2
  154. package/dist-cjs/lib/utils/export/exportAs.js.map +2 -2
  155. package/dist-esm/index.d.mts +362 -113
  156. package/dist-esm/index.mjs +76 -30
  157. package/dist-esm/index.mjs.map +2 -2
  158. package/dist-esm/lib/Tldraw.mjs +14 -4
  159. package/dist-esm/lib/Tldraw.mjs.map +2 -2
  160. package/dist-esm/lib/canvas/TldrawScribble.mjs +1 -1
  161. package/dist-esm/lib/canvas/TldrawScribble.mjs.map +2 -2
  162. package/dist-esm/lib/defaultExternalContentHandlers.mjs +15 -4
  163. package/dist-esm/lib/defaultExternalContentHandlers.mjs.map +2 -2
  164. package/dist-esm/lib/shapes/arrow/arrowLabel.mjs +6 -0
  165. package/dist-esm/lib/shapes/arrow/arrowLabel.mjs.map +3 -3
  166. package/dist-esm/lib/shapes/arrow/arrowTargetState.mjs +3 -2
  167. package/dist-esm/lib/shapes/arrow/arrowTargetState.mjs.map +2 -2
  168. package/dist-esm/lib/shapes/arrow/elbow/ElbowArrowDebug.mjs +3 -3
  169. package/dist-esm/lib/shapes/arrow/elbow/ElbowArrowDebug.mjs.map +1 -1
  170. package/dist-esm/lib/shapes/arrow/toolStates/Pointing.mjs +1 -1
  171. package/dist-esm/lib/shapes/arrow/toolStates/Pointing.mjs.map +2 -2
  172. package/dist-esm/lib/shapes/bookmark/BookmarkShapeUtil.mjs +4 -5
  173. package/dist-esm/lib/shapes/bookmark/BookmarkShapeUtil.mjs.map +2 -2
  174. package/dist-esm/lib/shapes/embed/EmbedShapeUtil.mjs +1 -1
  175. package/dist-esm/lib/shapes/embed/EmbedShapeUtil.mjs.map +1 -1
  176. package/dist-esm/lib/shapes/frame/FrameShapeUtil.mjs +12 -5
  177. package/dist-esm/lib/shapes/frame/FrameShapeUtil.mjs.map +2 -2
  178. package/dist-esm/lib/shapes/frame/components/FrameHeading.mjs +1 -1
  179. package/dist-esm/lib/shapes/frame/components/FrameHeading.mjs.map +2 -2
  180. package/dist-esm/lib/shapes/frame/components/FrameLabelInput.mjs +9 -3
  181. package/dist-esm/lib/shapes/frame/components/FrameLabelInput.mjs.map +2 -2
  182. package/dist-esm/lib/shapes/geo/GeoShapeUtil.mjs +1 -0
  183. package/dist-esm/lib/shapes/geo/GeoShapeUtil.mjs.map +2 -2
  184. package/dist-esm/lib/shapes/image/ImageShapeUtil.mjs +6 -3
  185. package/dist-esm/lib/shapes/image/ImageShapeUtil.mjs.map +2 -2
  186. package/dist-esm/lib/shapes/note/NoteShapeUtil.mjs +2 -1
  187. package/dist-esm/lib/shapes/note/NoteShapeUtil.mjs.map +2 -2
  188. package/dist-esm/lib/shapes/shared/HyperlinkButton.mjs +5 -5
  189. package/dist-esm/lib/shapes/shared/HyperlinkButton.mjs.map +2 -2
  190. package/dist-esm/lib/shapes/shared/PlainTextLabel.mjs +1 -3
  191. package/dist-esm/lib/shapes/shared/PlainTextLabel.mjs.map +2 -2
  192. package/dist-esm/lib/shapes/shared/ShapeFill.mjs +1 -1
  193. package/dist-esm/lib/shapes/shared/ShapeFill.mjs.map +2 -2
  194. package/dist-esm/lib/shapes/shared/freehand/svg.mjs.map +2 -2
  195. package/dist-esm/lib/shapes/shared/useEditablePlainText.mjs +3 -6
  196. package/dist-esm/lib/shapes/shared/useEditablePlainText.mjs.map +2 -2
  197. package/dist-esm/lib/shapes/shared/useImageOrVideoAsset.mjs +0 -2
  198. package/dist-esm/lib/shapes/shared/useImageOrVideoAsset.mjs.map +2 -2
  199. package/dist-esm/lib/shapes/text/PlainTextArea.mjs +4 -3
  200. package/dist-esm/lib/shapes/text/PlainTextArea.mjs.map +2 -2
  201. package/dist-esm/lib/shapes/text/RichTextArea.mjs +3 -4
  202. package/dist-esm/lib/shapes/text/RichTextArea.mjs.map +2 -2
  203. package/dist-esm/lib/shapes/video/VideoShapeUtil.mjs +3 -3
  204. package/dist-esm/lib/shapes/video/VideoShapeUtil.mjs.map +1 -1
  205. package/dist-esm/lib/tools/EraserTool/childStates/Erasing.mjs +26 -1
  206. package/dist-esm/lib/tools/EraserTool/childStates/Erasing.mjs.map +2 -2
  207. package/dist-esm/lib/tools/EraserTool/childStates/Pointing.mjs +13 -0
  208. package/dist-esm/lib/tools/EraserTool/childStates/Pointing.mjs.map +2 -2
  209. package/dist-esm/lib/tools/SelectTool/childStates/DraggingHandle.mjs +3 -1
  210. package/dist-esm/lib/tools/SelectTool/childStates/DraggingHandle.mjs.map +2 -2
  211. package/dist-esm/lib/ui/TldrawUi.mjs +29 -14
  212. package/dist-esm/lib/ui/TldrawUi.mjs.map +3 -3
  213. package/dist-esm/lib/ui/assetUrls.mjs +13 -10
  214. package/dist-esm/lib/ui/assetUrls.mjs.map +2 -2
  215. package/dist-esm/lib/ui/components/A11y.mjs +1 -2
  216. package/dist-esm/lib/ui/components/A11y.mjs.map +2 -2
  217. package/dist-esm/lib/ui/components/ActionsMenu/DefaultActionsMenu.mjs +10 -2
  218. package/dist-esm/lib/ui/components/ActionsMenu/DefaultActionsMenu.mjs.map +2 -2
  219. package/dist-esm/lib/ui/components/{FollowingIndicator.mjs → DefaultFollowingIndicator.mjs} +3 -3
  220. package/dist-esm/lib/ui/components/DefaultFollowingIndicator.mjs.map +7 -0
  221. package/dist-esm/lib/ui/components/InputModeMenu.mjs +57 -0
  222. package/dist-esm/lib/ui/components/InputModeMenu.mjs.map +7 -0
  223. package/dist-esm/lib/ui/components/KeyboardShortcutsDialog/DefaultKeyboardShortcutsDialogContent.mjs +6 -6
  224. package/dist-esm/lib/ui/components/KeyboardShortcutsDialog/DefaultKeyboardShortcutsDialogContent.mjs.map +1 -1
  225. package/dist-esm/lib/ui/components/LanguageMenu.mjs +1 -0
  226. package/dist-esm/lib/ui/components/LanguageMenu.mjs.map +2 -2
  227. package/dist-esm/lib/ui/components/MainMenu/DefaultMainMenuContent.mjs +2 -0
  228. package/dist-esm/lib/ui/components/MainMenu/DefaultMainMenuContent.mjs.map +2 -2
  229. package/dist-esm/lib/ui/components/Minimap/DefaultMinimap.mjs +2 -1
  230. package/dist-esm/lib/ui/components/Minimap/DefaultMinimap.mjs.map +2 -2
  231. package/dist-esm/lib/ui/components/Minimap/MinimapManager.mjs +4 -4
  232. package/dist-esm/lib/ui/components/Minimap/MinimapManager.mjs.map +2 -2
  233. package/dist-esm/lib/ui/components/MobileStylePanel.mjs +4 -2
  234. package/dist-esm/lib/ui/components/MobileStylePanel.mjs.map +2 -2
  235. package/dist-esm/lib/ui/components/PageMenu/DefaultPageMenu.mjs +1 -2
  236. package/dist-esm/lib/ui/components/PageMenu/DefaultPageMenu.mjs.map +2 -2
  237. package/dist-esm/lib/ui/components/StylePanel/DefaultStylePanel.mjs +14 -5
  238. package/dist-esm/lib/ui/components/StylePanel/DefaultStylePanel.mjs.map +2 -2
  239. package/dist-esm/lib/ui/components/StylePanel/DefaultStylePanelContent.mjs +257 -320
  240. package/dist-esm/lib/ui/components/StylePanel/DefaultStylePanelContent.mjs.map +2 -2
  241. package/dist-esm/lib/ui/components/StylePanel/StylePanelButtonPicker.mjs +135 -0
  242. package/dist-esm/lib/ui/components/StylePanel/StylePanelButtonPicker.mjs.map +7 -0
  243. package/dist-esm/lib/ui/components/StylePanel/StylePanelContext.mjs +48 -0
  244. package/dist-esm/lib/ui/components/StylePanel/StylePanelContext.mjs.map +7 -0
  245. package/dist-esm/lib/ui/components/StylePanel/{DoubleDropdownPicker.mjs → StylePanelDoubleDropdownPicker.mjs} +20 -19
  246. package/dist-esm/lib/ui/components/StylePanel/StylePanelDoubleDropdownPicker.mjs.map +7 -0
  247. package/dist-esm/lib/ui/components/StylePanel/{DropdownPicker.mjs → StylePanelDropdownPicker.mjs} +21 -18
  248. package/dist-esm/lib/ui/components/StylePanel/StylePanelDropdownPicker.mjs.map +7 -0
  249. package/dist-esm/lib/ui/components/StylePanel/StylePanelSubheading.mjs +8 -0
  250. package/dist-esm/lib/ui/components/StylePanel/StylePanelSubheading.mjs.map +7 -0
  251. package/dist-esm/lib/ui/components/Toolbar/AltTextEditor.mjs +2 -0
  252. package/dist-esm/lib/ui/components/Toolbar/AltTextEditor.mjs.map +2 -2
  253. package/dist-esm/lib/ui/components/Toolbar/DefaultImageToolbarContent.mjs +39 -10
  254. package/dist-esm/lib/ui/components/Toolbar/DefaultImageToolbarContent.mjs.map +2 -2
  255. package/dist-esm/lib/ui/components/Toolbar/DefaultToolbar.mjs +56 -22
  256. package/dist-esm/lib/ui/components/Toolbar/DefaultToolbar.mjs.map +2 -2
  257. package/dist-esm/lib/ui/components/Toolbar/DefaultVideoToolbarContent.mjs +15 -3
  258. package/dist-esm/lib/ui/components/Toolbar/DefaultVideoToolbarContent.mjs.map +2 -2
  259. package/dist-esm/lib/ui/components/Toolbar/LinkEditor.mjs +2 -1
  260. package/dist-esm/lib/ui/components/Toolbar/LinkEditor.mjs.map +2 -2
  261. package/dist-esm/lib/ui/components/Toolbar/OverflowingToolbar.mjs +192 -80
  262. package/dist-esm/lib/ui/components/Toolbar/OverflowingToolbar.mjs.map +3 -3
  263. package/dist-esm/lib/ui/components/Toolbar/ToggleToolLockedButton.mjs +6 -2
  264. package/dist-esm/lib/ui/components/Toolbar/ToggleToolLockedButton.mjs.map +2 -2
  265. package/dist-esm/lib/ui/components/primitives/TldrawUiContextualToolbar.mjs +11 -3
  266. package/dist-esm/lib/ui/components/primitives/TldrawUiContextualToolbar.mjs.map +2 -2
  267. package/dist-esm/lib/ui/components/primitives/TldrawUiInput.mjs +6 -4
  268. package/dist-esm/lib/ui/components/primitives/TldrawUiInput.mjs.map +2 -2
  269. package/dist-esm/lib/ui/components/primitives/TldrawUiSlider.mjs +18 -5
  270. package/dist-esm/lib/ui/components/primitives/TldrawUiSlider.mjs.map +2 -2
  271. package/dist-esm/lib/ui/components/primitives/TldrawUiToolbar.mjs +19 -4
  272. package/dist-esm/lib/ui/components/primitives/TldrawUiToolbar.mjs.map +2 -2
  273. package/dist-esm/lib/ui/components/primitives/TldrawUiTooltip.mjs +197 -167
  274. package/dist-esm/lib/ui/components/primitives/TldrawUiTooltip.mjs.map +2 -2
  275. package/dist-esm/lib/ui/components/primitives/layout.mjs +31 -6
  276. package/dist-esm/lib/ui/components/primitives/layout.mjs.map +2 -2
  277. package/dist-esm/lib/ui/components/primitives/menus/TldrawUiMenuCheckboxItem.mjs +3 -0
  278. package/dist-esm/lib/ui/components/primitives/menus/TldrawUiMenuCheckboxItem.mjs.map +2 -2
  279. package/dist-esm/lib/ui/components/primitives/menus/TldrawUiMenuContext.mjs.map +2 -2
  280. package/dist-esm/lib/ui/components/primitives/menus/TldrawUiMenuGroup.mjs +25 -12
  281. package/dist-esm/lib/ui/components/primitives/menus/TldrawUiMenuGroup.mjs.map +2 -2
  282. package/dist-esm/lib/ui/components/primitives/menus/TldrawUiMenuItem.mjs +11 -27
  283. package/dist-esm/lib/ui/components/primitives/menus/TldrawUiMenuItem.mjs.map +2 -2
  284. package/dist-esm/lib/ui/context/actions.mjs +29 -30
  285. package/dist-esm/lib/ui/context/actions.mjs.map +2 -2
  286. package/dist-esm/lib/ui/context/components.mjs +2 -0
  287. package/dist-esm/lib/ui/context/components.mjs.map +2 -2
  288. package/dist-esm/lib/ui/context/events.mjs.map +2 -2
  289. package/dist-esm/lib/ui/hooks/useClipboardEvents.mjs +1 -2
  290. package/dist-esm/lib/ui/hooks/useClipboardEvents.mjs.map +2 -2
  291. package/dist-esm/lib/ui/hooks/useExportAs.mjs +3 -2
  292. package/dist-esm/lib/ui/hooks/useExportAs.mjs.map +2 -2
  293. package/dist-esm/lib/ui/hooks/useTools.mjs +23 -4
  294. package/dist-esm/lib/ui/hooks/useTools.mjs.map +2 -2
  295. package/dist-esm/lib/ui/hooks/useTranslation/defaultTranslation.mjs +12 -2
  296. package/dist-esm/lib/ui/hooks/useTranslation/defaultTranslation.mjs.map +2 -2
  297. package/dist-esm/lib/ui/kbd-utils.mjs +9 -3
  298. package/dist-esm/lib/ui/kbd-utils.mjs.map +2 -2
  299. package/dist-esm/lib/ui/version.mjs +3 -3
  300. package/dist-esm/lib/ui/version.mjs.map +1 -1
  301. package/dist-esm/lib/utils/export/copyAs.mjs +1 -2
  302. package/dist-esm/lib/utils/export/copyAs.mjs.map +2 -2
  303. package/dist-esm/lib/utils/export/export.mjs +0 -20
  304. package/dist-esm/lib/utils/export/export.mjs.map +2 -2
  305. package/dist-esm/lib/utils/export/exportAs.mjs +1 -2
  306. package/dist-esm/lib/utils/export/exportAs.mjs.map +2 -2
  307. package/package.json +11 -34
  308. package/src/index.ts +56 -22
  309. package/src/lib/Tldraw.tsx +15 -2
  310. package/src/lib/canvas/TldrawScribble.tsx +1 -1
  311. package/src/lib/defaultExternalContentHandlers.ts +26 -4
  312. package/src/lib/shapes/arrow/ArrowShapeOptions.test.ts +85 -14
  313. package/src/lib/shapes/arrow/ArrowShapeTool.test.ts +103 -8
  314. package/src/lib/shapes/arrow/ArrowShapeUtil.test.ts +48 -6
  315. package/src/lib/shapes/arrow/arrow-types.ts +3 -5
  316. package/src/lib/shapes/arrow/arrowLabel.ts +8 -0
  317. package/src/lib/shapes/arrow/arrowTargetState.ts +34 -3
  318. package/src/lib/shapes/arrow/elbow/ElbowArrowDebug.tsx +3 -3
  319. package/src/lib/shapes/arrow/toolStates/Pointing.tsx +1 -1
  320. package/src/lib/shapes/bookmark/BookmarkShapeUtil.tsx +4 -5
  321. package/src/lib/shapes/draw/DrawShapeTool.test.ts +0 -5
  322. package/src/lib/shapes/embed/EmbedShapeUtil.tsx +1 -1
  323. package/src/lib/shapes/frame/FrameShapeUtil.tsx +21 -4
  324. package/src/lib/shapes/frame/components/FrameHeading.tsx +1 -1
  325. package/src/lib/shapes/frame/components/FrameLabelInput.tsx +10 -3
  326. package/src/lib/shapes/geo/GeoShapeUtil.tsx +1 -0
  327. package/src/lib/shapes/image/ImageShapeUtil.tsx +6 -3
  328. package/src/lib/shapes/line/LineShapeUtil.test.tsx +4 -3
  329. package/src/lib/shapes/line/__snapshots__/LineShapeUtil.test.tsx.snap +2 -2
  330. package/src/lib/shapes/note/NoteShapeUtil.tsx +1 -0
  331. package/src/lib/shapes/shared/HyperlinkButton.tsx +5 -5
  332. package/src/lib/shapes/shared/PlainTextLabel.tsx +0 -6
  333. package/src/lib/shapes/shared/ShapeFill.tsx +1 -1
  334. package/src/lib/shapes/shared/freehand/svg.ts +2 -0
  335. package/src/lib/shapes/shared/useEditablePlainText.ts +3 -10
  336. package/src/lib/shapes/shared/useImageOrVideoAsset.ts +0 -7
  337. package/src/lib/shapes/text/PlainTextArea.tsx +4 -3
  338. package/src/lib/shapes/text/RichTextArea.tsx +3 -4
  339. package/src/lib/shapes/text/TextShapeTool.test.ts +6 -5
  340. package/src/lib/shapes/video/VideoShapeUtil.tsx +3 -3
  341. package/src/lib/tools/EraserTool/childStates/Erasing.ts +34 -1
  342. package/src/lib/tools/EraserTool/childStates/Pointing.ts +20 -0
  343. package/src/lib/tools/SelectTool/childStates/DraggingHandle.tsx +6 -2
  344. package/src/lib/ui/TldrawUi.tsx +33 -12
  345. package/src/lib/ui/assetUrls.ts +13 -10
  346. package/src/lib/ui/components/A11y.tsx +1 -2
  347. package/src/lib/ui/components/ActionsMenu/DefaultActionsMenu.tsx +13 -2
  348. package/src/lib/ui/components/{FollowingIndicator.tsx → DefaultFollowingIndicator.tsx} +2 -1
  349. package/src/lib/ui/components/InputModeMenu.tsx +65 -0
  350. package/src/lib/ui/components/KeyboardShortcutsDialog/DefaultKeyboardShortcutsDialogContent.tsx +6 -6
  351. package/src/lib/ui/components/LanguageMenu.tsx +1 -0
  352. package/src/lib/ui/components/MainMenu/DefaultMainMenuContent.tsx +4 -0
  353. package/src/lib/ui/components/Minimap/DefaultMinimap.tsx +2 -1
  354. package/src/lib/ui/components/Minimap/MinimapManager.ts +4 -4
  355. package/src/lib/ui/components/MobileStylePanel.tsx +4 -3
  356. package/src/lib/ui/components/PageMenu/DefaultPageMenu.tsx +1 -2
  357. package/src/lib/ui/components/StylePanel/DefaultStylePanel.tsx +27 -13
  358. package/src/lib/ui/components/StylePanel/DefaultStylePanelContent.tsx +260 -381
  359. package/src/lib/ui/components/{primitives/TldrawUiButtonPicker.tsx → StylePanel/StylePanelButtonPicker.tsx} +70 -50
  360. package/src/lib/ui/components/StylePanel/StylePanelContext.tsx +63 -0
  361. package/src/lib/ui/components/StylePanel/{DoubleDropdownPicker.tsx → StylePanelDoubleDropdownPicker.tsx} +28 -19
  362. package/src/lib/ui/components/StylePanel/StylePanelDropdownPicker.tsx +119 -0
  363. package/src/lib/ui/components/StylePanel/StylePanelSubheading.tsx +9 -0
  364. package/src/lib/ui/components/Toolbar/AltTextEditor.tsx +2 -0
  365. package/src/lib/ui/components/Toolbar/DefaultImageToolbarContent.tsx +33 -16
  366. package/src/lib/ui/components/Toolbar/DefaultToolbar.tsx +55 -24
  367. package/src/lib/ui/components/Toolbar/DefaultVideoToolbarContent.tsx +12 -4
  368. package/src/lib/ui/components/Toolbar/LinkEditor.tsx +1 -0
  369. package/src/lib/ui/components/Toolbar/OverflowingToolbar.tsx +208 -56
  370. package/src/lib/ui/components/Toolbar/ToggleToolLockedButton.tsx +9 -2
  371. package/src/lib/ui/components/primitives/TldrawUiContextualToolbar.tsx +7 -3
  372. package/src/lib/ui/components/primitives/TldrawUiInput.tsx +6 -3
  373. package/src/lib/ui/components/primitives/TldrawUiSlider.tsx +52 -32
  374. package/src/lib/ui/components/primitives/TldrawUiToolbar.tsx +27 -6
  375. package/src/lib/ui/components/primitives/TldrawUiTooltip.tsx +232 -185
  376. package/src/lib/ui/components/primitives/layout.tsx +79 -5
  377. package/src/lib/ui/components/primitives/menus/TldrawUiMenuCheckboxItem.tsx +4 -0
  378. package/src/lib/ui/components/primitives/menus/TldrawUiMenuContext.tsx +0 -1
  379. package/src/lib/ui/components/primitives/menus/TldrawUiMenuGroup.tsx +29 -16
  380. package/src/lib/ui/components/primitives/menus/TldrawUiMenuItem.tsx +14 -27
  381. package/src/lib/ui/context/actions.tsx +36 -32
  382. package/src/lib/ui/context/components.tsx +3 -0
  383. package/src/lib/ui/context/events.tsx +2 -1
  384. package/src/lib/ui/hooks/useClipboardEvents.ts +1 -2
  385. package/src/lib/ui/hooks/useExportAs.ts +3 -2
  386. package/src/lib/ui/hooks/useTools.tsx +26 -4
  387. package/src/lib/ui/hooks/useTranslation/TLUiTranslationKey.ts +10 -0
  388. package/src/lib/ui/hooks/useTranslation/defaultTranslation.ts +12 -2
  389. package/src/lib/ui/kbd-utils.ts +10 -3
  390. package/src/lib/ui/version.ts +3 -3
  391. package/src/lib/ui.css +389 -246
  392. package/src/lib/utils/excalidraw/__snapshots__/putExcalidrawContent.test.tsx.snap +5 -5
  393. package/src/lib/utils/export/copyAs.ts +1 -24
  394. package/src/lib/utils/export/export.ts +0 -36
  395. package/src/lib/utils/export/exportAs.ts +1 -32
  396. package/src/lib/utils/tldr/__snapshots__/buildFromV1Document.test.ts.snap +4 -4
  397. package/src/test/A11y.test.tsx +3 -2
  398. package/src/test/ClickManager.test.ts +7 -6
  399. package/src/test/Editor.test.tsx +20 -19
  400. package/src/test/EraserTool.test.ts +184 -13
  401. package/src/test/HandTool.test.ts +10 -9
  402. package/src/test/HighlightShape.test.ts +2 -1
  403. package/src/test/SelectTool.test.ts +3 -2
  404. package/src/test/TLUserPreferences.test.ts +4 -3
  405. package/src/test/TestEditor.ts +21 -17
  406. package/src/test/TldrawEditor.test.tsx +11 -10
  407. package/src/test/ZoomTool.test.ts +7 -6
  408. package/src/test/__snapshots__/drawing.test.ts.snap +2 -2
  409. package/src/test/__snapshots__/groups.test.tsx.snap +6 -6
  410. package/src/test/__snapshots__/resizing.test.ts.snap +2 -2
  411. package/src/test/arrows-megabus.test.tsx +5 -4
  412. package/src/test/bindings.test.tsx +24 -37
  413. package/src/test/bookmark-shapes.test.ts +1 -8
  414. package/src/test/commands/__snapshots__/getSvgString.test.ts.snap +23 -7
  415. package/src/test/commands/__snapshots__/packShapes.test.ts.snap +8 -8
  416. package/src/test/commands/__snapshots__/zoomToFit.test.ts.snap +2 -2
  417. package/src/test/commands/alignShapes.test.tsx +25 -24
  418. package/src/test/commands/animationSpeed.test.ts +2 -1
  419. package/src/test/commands/centerOnPoint.test.ts +3 -2
  420. package/src/test/commands/clipboard.test.ts +3 -2
  421. package/src/test/commands/createShapes.test.ts +2 -1
  422. package/src/test/commands/deleteShapes.test.ts +2 -1
  423. package/src/test/commands/distributeShapes.test.tsx +11 -10
  424. package/src/test/commands/getSvgString.test.ts +2 -1
  425. package/src/test/commands/packShapes.test.ts +5 -4
  426. package/src/test/commands/resizeShape.test.ts +2 -1
  427. package/src/test/commands/rotateShapes.test.ts +7 -6
  428. package/src/test/commands/setCamera.test.ts +17 -3
  429. package/src/test/commands/setCurrentPage.test.ts +3 -2
  430. package/src/test/commands/stackShapes.test.ts +11 -10
  431. package/src/test/commands/stretch.test.tsx +13 -12
  432. package/src/test/createDeepLink.test.tsx +2 -1
  433. package/src/test/cropping.test.ts +3 -2
  434. package/src/test/custom-clipping.test.ts +436 -0
  435. package/src/test/drawing.test.ts +2 -1
  436. package/src/test/flipShapes.test.ts +4 -3
  437. package/src/test/frames.test.ts +40 -24
  438. package/src/test/getCulledShapes.test.tsx +74 -4
  439. package/src/test/groups.test.tsx +1 -1
  440. package/src/test/handleDeepLink.test.tsx +2 -1
  441. package/src/test/maxShapes.test.ts +3 -2
  442. package/src/test/modifiers.test.ts +5 -4
  443. package/src/test/navigation.test.ts +12 -11
  444. package/src/test/panning.test.ts +2 -1
  445. package/src/test/perf/perf.test.ts +2 -1
  446. package/src/test/registerDeepLinkListener.test.tsx +10 -9
  447. package/src/test/resizing.test.ts +39 -38
  448. package/src/test/select.test.tsx +4 -3
  449. package/src/test/selection-omnibus.test.ts +11 -10
  450. package/src/test/shapeutils.test.ts +4 -3
  451. package/src/test/translating.test.ts +9 -8
  452. package/tldraw.css +697 -538
  453. package/dist-cjs/lib/ui/components/FollowingIndicator.js.map +0 -7
  454. package/dist-cjs/lib/ui/components/StylePanel/DoubleDropdownPicker.js.map +0 -7
  455. package/dist-cjs/lib/ui/components/StylePanel/DropdownPicker.js.map +0 -7
  456. package/dist-cjs/lib/ui/components/primitives/TldrawUiButtonPicker.js +0 -131
  457. package/dist-cjs/lib/ui/components/primitives/TldrawUiButtonPicker.js.map +0 -7
  458. package/dist-esm/lib/ui/components/FollowingIndicator.mjs.map +0 -7
  459. package/dist-esm/lib/ui/components/StylePanel/DoubleDropdownPicker.mjs.map +0 -7
  460. package/dist-esm/lib/ui/components/StylePanel/DropdownPicker.mjs.map +0 -7
  461. package/dist-esm/lib/ui/components/primitives/TldrawUiButtonPicker.mjs +0 -115
  462. package/dist-esm/lib/ui/components/primitives/TldrawUiButtonPicker.mjs.map +0 -7
  463. package/src/lib/ui/components/StylePanel/DropdownPicker.tsx +0 -110
package/src/index.ts CHANGED
@@ -24,10 +24,16 @@ export { usePrefersReducedMotion } from './lib/shapes/shared/usePrefersReducedMo
24
24
  export { DefaultA11yAnnouncer, useSelectedShapesAnnouncer } from './lib/ui/components/A11y'
25
25
  export { AccessibilityMenu } from './lib/ui/components/AccessibilityMenu'
26
26
  export { ColorSchemeMenu } from './lib/ui/components/ColorSchemeMenu'
27
+ export { DefaultFollowingIndicator } from './lib/ui/components/DefaultFollowingIndicator'
27
28
  export { DefaultDialogs } from './lib/ui/components/Dialogs'
28
29
  export {
30
+ TldrawUiColumn,
29
31
  TldrawUiGrid,
32
+ TldrawUiOrientationProvider,
30
33
  TldrawUiRow,
34
+ useTldrawUiOrientation,
35
+ type TldrawUiOrientationContext,
36
+ type TldrawUiOrientationProviderProps,
31
37
  type TLUiLayoutProps,
32
38
  } from './lib/ui/components/primitives/layout'
33
39
  export {
@@ -80,6 +86,7 @@ export {
80
86
  defaultHandleExternalUrlContent,
81
87
  getAssetInfo,
82
88
  getMediaAssetInfoPartial,
89
+ notifyIfFileNotAllowed,
83
90
  registerDefaultExternalContentHandlers,
84
91
  type TLDefaultExternalContentHandlerOpts,
85
92
  type TLExternalContentProps,
@@ -99,6 +106,13 @@ export {
99
106
  } from './lib/shapes/arrow/arrow-types'
100
107
  export { ArrowShapeTool } from './lib/shapes/arrow/ArrowShapeTool'
101
108
  export { ArrowShapeUtil } from './lib/shapes/arrow/ArrowShapeUtil'
109
+ export {
110
+ clearArrowTargetState,
111
+ getArrowTargetState,
112
+ updateArrowTargetState,
113
+ type ArrowTargetState,
114
+ type UpdateArrowTargetStateOpts,
115
+ } from './lib/shapes/arrow/arrowTargetState'
102
116
  export {
103
117
  type ElbowArrowBox,
104
118
  type ElbowArrowBoxEdges,
@@ -161,11 +175,10 @@ export {
161
175
  type TLDefaultFont,
162
176
  type TLDefaultFonts,
163
177
  } from './lib/shapes/shared/defaultFonts'
164
- export {
165
- PlainTextLabel,
166
- TextLabel,
167
- type PlainTextLabelProps,
168
- } from './lib/shapes/shared/PlainTextLabel'
178
+ export { getStrokePoints } from './lib/shapes/shared/freehand/getStrokePoints'
179
+ export { getSvgPathFromStrokePoints } from './lib/shapes/shared/freehand/svg'
180
+ export { type StrokeOptions, type StrokePoint } from './lib/shapes/shared/freehand/types'
181
+ export { PlainTextLabel, type PlainTextLabelProps } from './lib/shapes/shared/PlainTextLabel'
169
182
  export {
170
183
  RichTextLabel,
171
184
  RichTextSVG,
@@ -173,10 +186,9 @@ export {
173
186
  type RichTextSVGProps,
174
187
  } from './lib/shapes/shared/RichTextLabel'
175
188
  export { useDefaultColorTheme } from './lib/shapes/shared/useDefaultColorTheme'
176
- export { useEditablePlainText, useEditableText } from './lib/shapes/shared/useEditablePlainText'
189
+ export { useEditablePlainText } from './lib/shapes/shared/useEditablePlainText'
177
190
  export { useEditableRichText } from './lib/shapes/shared/useEditableRichText'
178
191
  export {
179
- useAsset,
180
192
  useImageOrVideoAsset,
181
193
  type UseImageOrVideoAssetOptions,
182
194
  } from './lib/shapes/shared/useImageOrVideoAsset'
@@ -355,10 +367,6 @@ export {
355
367
  TldrawUiMenuSubmenu,
356
368
  type TLUiMenuSubmenuProps,
357
369
  } from './lib/ui/components/primitives/menus/TldrawUiMenuSubmenu'
358
- export {
359
- TldrawUiButtonPicker,
360
- type TLUiButtonPickerProps,
361
- } from './lib/ui/components/primitives/TldrawUiButtonPicker'
362
370
  export {
363
371
  TldrawUiContextualToolbar,
364
372
  type TLUiContextualToolbarProps,
@@ -438,17 +446,44 @@ export {
438
446
  type TLUiStylePanelProps,
439
447
  } from './lib/ui/components/StylePanel/DefaultStylePanel'
440
448
  export {
441
- ArrowheadStylePickerSet,
442
- CommonStylePickerSet,
443
449
  DefaultStylePanelContent,
444
- GeoStylePickerSet,
445
- OpacitySlider,
446
- SplineStylePickerSet,
447
- TextStylePickerSet,
448
- type StylePickerSetProps,
449
- type ThemeStylePickerSetProps,
450
- type TLUiStylePanelContentProps,
450
+ StylePanelArrowheadPicker,
451
+ StylePanelArrowKindPicker,
452
+ StylePanelColorPicker,
453
+ StylePanelDashPicker,
454
+ StylePanelFillPicker,
455
+ StylePanelFontPicker,
456
+ StylePanelGeoShapePicker,
457
+ StylePanelLabelAlignPicker,
458
+ StylePanelOpacityPicker,
459
+ StylePanelSection,
460
+ StylePanelSizePicker,
461
+ StylePanelSplinePicker,
462
+ StylePanelTextAlignPicker,
463
+ type StylePanelSectionProps,
451
464
  } from './lib/ui/components/StylePanel/DefaultStylePanelContent'
465
+ export {
466
+ StylePanelButtonPicker,
467
+ type StylePanelButtonPickerProps,
468
+ } from './lib/ui/components/StylePanel/StylePanelButtonPicker'
469
+ export {
470
+ StylePanelContextProvider,
471
+ useStylePanelContext,
472
+ type StylePanelContext,
473
+ type StylePanelContextProviderProps,
474
+ } from './lib/ui/components/StylePanel/StylePanelContext'
475
+ export {
476
+ StylePanelDoubleDropdownPicker,
477
+ type StylePanelDoubleDropdownPickerProps,
478
+ } from './lib/ui/components/StylePanel/StylePanelDoubleDropdownPicker'
479
+ export {
480
+ StylePanelDropdownPicker,
481
+ type StylePanelDropdownPickerProps,
482
+ } from './lib/ui/components/StylePanel/StylePanelDropdownPicker'
483
+ export {
484
+ StylePanelSubheading,
485
+ type StylePanelSubheadingProps,
486
+ } from './lib/ui/components/StylePanel/StylePanelSubheading'
452
487
  export {
453
488
  DefaultImageToolbar,
454
489
  type TLUiImageToolbarProps,
@@ -620,13 +655,12 @@ export {
620
655
  } from './lib/ui/hooks/useTranslation/useTranslation'
621
656
  export { type TLUiIconType } from './lib/ui/icon-types'
622
657
  export { useDefaultHelpers, type TLUiOverrideHelpers, type TLUiOverrides } from './lib/ui/overrides'
623
- export { TldrawUi, type TldrawUiProps } from './lib/ui/TldrawUi'
658
+ export { TldrawUi, TldrawUiInFrontOfTheCanvas, type TldrawUiProps } from './lib/ui/TldrawUi'
624
659
  export { containBoxSize, downsizeImage, type BoxWidthHeight } from './lib/utils/assets/assets'
625
660
  export { preloadFont, type TLTypeFace } from './lib/utils/assets/preload-font'
626
661
  export { getEmbedInfo, type TLEmbedResult } from './lib/utils/embeds/embeds'
627
662
  export { putExcalidrawContent } from './lib/utils/excalidraw/putExcalidrawContent'
628
663
  export { copyAs, type CopyAsOptions, type TLCopyType } from './lib/utils/export/copyAs'
629
- export { exportToBlob } from './lib/utils/export/export'
630
664
  export { downloadFile, exportAs, type ExportAsOptions } from './lib/utils/export/exportAs'
631
665
  export { fitFrameToContent, removeFrame } from './lib/utils/frames/frames'
632
666
  export {
@@ -33,7 +33,7 @@ import { registerDefaultSideEffects } from './defaultSideEffects'
33
33
  import { defaultTools } from './defaultTools'
34
34
  import { EmbedShapeUtil } from './shapes/embed/EmbedShapeUtil'
35
35
  import { allDefaultFontFaces } from './shapes/shared/defaultFonts'
36
- import { TldrawUi, TldrawUiProps } from './ui/TldrawUi'
36
+ import { TldrawUi, TldrawUiInFrontOfTheCanvas, TldrawUiProps } from './ui/TldrawUi'
37
37
  import { TLUiAssetUrlOverrides, useDefaultUiAssetUrlsWithOverrides } from './ui/assetUrls'
38
38
  import { LoadingScreen } from './ui/components/LoadingScreen'
39
39
  import { Spinner } from './ui/components/Spinner'
@@ -118,6 +118,18 @@ export function Tldraw(props: TldrawProps) {
118
118
 
119
119
  const _components = useShallowObjectIdentity(components)
120
120
 
121
+ const CustomInFrontOfTheCanvas = components?.InFrontOfTheCanvas
122
+ const InFrontOfTheCanvas = useMemo(() => {
123
+ if (rest.hideUi) return CustomInFrontOfTheCanvas ?? null
124
+ if (!CustomInFrontOfTheCanvas) return TldrawUiInFrontOfTheCanvas
125
+
126
+ return () => (
127
+ <>
128
+ <TldrawUiInFrontOfTheCanvas />
129
+ <CustomInFrontOfTheCanvas />
130
+ </>
131
+ )
132
+ }, [rest.hideUi, CustomInFrontOfTheCanvas])
121
133
  const componentsWithDefault = useMemo(
122
134
  () => ({
123
135
  Scribble: TldrawScribble,
@@ -129,8 +141,9 @@ export function Tldraw(props: TldrawProps) {
129
141
  Spinner,
130
142
  LoadingScreen,
131
143
  ..._components,
144
+ InFrontOfTheCanvas,
132
145
  }),
133
- [_components]
146
+ [_components, InFrontOfTheCanvas]
134
147
  )
135
148
 
136
149
  const _shapeUtils = useShallowArrayIdentity(shapeUtils)
@@ -31,7 +31,7 @@ export function TldrawScribble({ scribble, zoom, color, opacity, className }: TL
31
31
  <path
32
32
  className="tl-scribble"
33
33
  d={d}
34
- fill={color ?? `var(--color-${scribble.color})`}
34
+ fill={color ?? `var(--tl-color-${scribble.color})`}
35
35
  opacity={opacity ?? scribble.opacity}
36
36
  />
37
37
  </svg>
@@ -144,7 +144,7 @@ export async function defaultHandleExternalFileAsset(
144
144
  { file, assetId }: TLFileExternalAsset,
145
145
  options: TLDefaultExternalContentHandlerOpts
146
146
  ) {
147
- const isSuccess = runFileChecks(file, options)
147
+ const isSuccess = notifyIfFileNotAllowed(file, options)
148
148
  if (!isSuccess) assert(false, 'File checks failed')
149
149
 
150
150
  const assetInfo = await getAssetInfo(file, options, assetId)
@@ -161,7 +161,7 @@ export async function defaultHandleExternalFileReplaceContent(
161
161
  { file, shapeId, isImage }: TLFileReplaceExternalContent,
162
162
  options: TLDefaultExternalContentHandlerOpts
163
163
  ) {
164
- const isSuccess = runFileChecks(file, options)
164
+ const isSuccess = notifyIfFileNotAllowed(file, options)
165
165
  if (!isSuccess) assert(false, 'File checks failed')
166
166
 
167
167
  const shape = editor.getShape(shapeId)
@@ -399,7 +399,7 @@ export async function defaultHandleExternalFileContent(
399
399
  file: File
400
400
  }[] = []
401
401
  for (const file of files) {
402
- const isSuccess = runFileChecks(file, options)
402
+ const isSuccess = notifyIfFileNotAllowed(file, options)
403
403
  if (!isSuccess) continue
404
404
 
405
405
  const assetInfo = await getAssetInfo(file, options)
@@ -873,7 +873,15 @@ export function createEmptyBookmarkShape(
873
873
  return editor.getShape(partial.id) as TLBookmarkShape
874
874
  }
875
875
 
876
- function runFileChecks(file: File, options: TLDefaultExternalContentHandlerOpts) {
876
+ /**
877
+ * Checks if a file is allowed to be uploaded. If it is not, it will show a toast explaining why to the user.
878
+ *
879
+ * @param file - The file to check
880
+ * @param options - The options for the external content handler
881
+ * @returns True if the file is allowed, false otherwise
882
+ * @public
883
+ */
884
+ export function notifyIfFileNotAllowed(file: File, options: TLDefaultExternalContentHandlerOpts) {
877
885
  const {
878
886
  acceptedImageMimeTypes = DEFAULT_SUPPORTED_IMAGE_TYPES,
879
887
  acceptedVideoMimeTypes = DEFAULT_SUPPORT_VIDEO_TYPES,
@@ -893,8 +901,22 @@ function runFileChecks(file: File, options: TLDefaultExternalContentHandlerOpts)
893
901
  }
894
902
 
895
903
  if (file.size > maxAssetSize) {
904
+ const formatBytes = (bytes: number): string => {
905
+ if (bytes === 0) return '0 bytes'
906
+
907
+ const units = ['bytes', 'KB', 'MB', 'GB', 'TB', 'PB']
908
+ const base = 1024
909
+ const unitIndex = Math.floor(Math.log(bytes) / Math.log(base))
910
+
911
+ const value = bytes / Math.pow(base, unitIndex)
912
+ const formatted = value % 1 === 0 ? value.toString() : value.toFixed(1)
913
+
914
+ return `${formatted} ${units[unitIndex]}`
915
+ }
916
+
896
917
  toasts.addToast({
897
918
  title: msg('assets.files.size-too-big'),
919
+ description: msg('assets.files.maximum-size').replace('{size}', formatBytes(maxAssetSize)),
898
920
  severity: 'error',
899
921
  })
900
922
  return false
@@ -1,4 +1,5 @@
1
1
  import { TLArrowShape, createShapeId } from '@tldraw/editor'
2
+ import { vi } from 'vitest'
2
3
  import { TestEditor } from '../../../test/TestEditor'
3
4
  import { ArrowShapeUtil } from './ArrowShapeUtil'
4
5
  import { updateArrowTargetState } from './arrowTargetState'
@@ -12,7 +13,7 @@ const ids = {
12
13
  arrow1: createShapeId('arrow1'),
13
14
  }
14
15
 
15
- jest.useFakeTimers()
16
+ vi.useFakeTimers()
16
17
 
17
18
  window.requestAnimationFrame = function requestAnimationFrame(cb) {
18
19
  return setTimeout(cb, 1000 / 60)
@@ -46,13 +47,21 @@ describe('ArrowShapeOptions', () => {
46
47
  it('should have correct default shouldBeExact behavior (alt key)', () => {
47
48
  const util = editor.getShapeUtil<ArrowShapeUtil>('arrow')
48
49
 
49
- // Test without alt key
50
+ // Test without alt key, not precise
50
51
  editor.inputs.altKey = false
51
- expect(util.options.shouldBeExact(editor)).toBe(false)
52
+ expect(util.options.shouldBeExact(editor, false)).toBe(false)
52
53
 
53
- // Test with alt key
54
+ // Test without alt key, precise
55
+ editor.inputs.altKey = false
56
+ expect(util.options.shouldBeExact(editor, true)).toBe(false)
57
+
58
+ // Test with alt key, not precise
59
+ editor.inputs.altKey = true
60
+ expect(util.options.shouldBeExact(editor, false)).toBe(true)
61
+
62
+ // Test with alt key, precise
54
63
  editor.inputs.altKey = true
55
- expect(util.options.shouldBeExact(editor)).toBe(true)
64
+ expect(util.options.shouldBeExact(editor, true)).toBe(true)
56
65
  })
57
66
 
58
67
  it('should have correct default shouldIgnoreTargets behavior (ctrl key)', () => {
@@ -185,7 +194,7 @@ describe('ArrowShapeOptions', () => {
185
194
  class CustomArrowShapeUtil extends ArrowShapeUtil {
186
195
  override options = {
187
196
  ...baseUtil.options,
188
- shouldBeExact: (editor: any) => editor.inputs.shiftKey, // Use shift instead of alt
197
+ shouldBeExact: (editor: any, _isPrecise: boolean) => editor.inputs.shiftKey, // Use shift instead of alt
189
198
  }
190
199
  }
191
200
 
@@ -194,12 +203,14 @@ describe('ArrowShapeOptions', () => {
194
203
  // Test with shift key
195
204
  editor.inputs.shiftKey = true
196
205
  editor.inputs.altKey = false
197
- expect(customUtil.options.shouldBeExact(editor)).toBe(true)
206
+ expect(customUtil.options.shouldBeExact(editor, false)).toBe(true)
207
+ expect(customUtil.options.shouldBeExact(editor, true)).toBe(true)
198
208
 
199
209
  // Test without shift key
200
210
  editor.inputs.shiftKey = false
201
211
  editor.inputs.altKey = true // Alt key should not matter for custom implementation
202
- expect(customUtil.options.shouldBeExact(editor)).toBe(false)
212
+ expect(customUtil.options.shouldBeExact(editor, false)).toBe(false)
213
+ expect(customUtil.options.shouldBeExact(editor, true)).toBe(false)
203
214
  })
204
215
 
205
216
  it('should allow customizing shouldIgnoreTargets behavior', () => {
@@ -231,9 +242,9 @@ describe('ArrowShapeOptions', () => {
231
242
  class CustomArrowShapeUtil extends ArrowShapeUtil {
232
243
  override options = {
233
244
  ...baseUtil.options,
234
- shouldBeExact: (editor: any) => {
235
- // Custom logic: exact when both alt and shift are pressed
236
- return editor.inputs.altKey && editor.inputs.shiftKey
245
+ shouldBeExact: (editor: any, isPrecise: boolean) => {
246
+ // Custom logic: exact when both alt and shift are pressed, and only if precise
247
+ return editor.inputs.altKey && editor.inputs.shiftKey && isPrecise
237
248
  },
238
249
  shouldIgnoreTargets: (editor: any) => {
239
250
  // Custom logic: ignore targets when any modifier key is pressed
@@ -244,15 +255,20 @@ describe('ArrowShapeOptions', () => {
244
255
 
245
256
  const customUtil = new CustomArrowShapeUtil(editor)
246
257
 
247
- // Test shouldBeExact with both keys
258
+ // Test shouldBeExact with both keys and precise
248
259
  editor.inputs.altKey = true
249
260
  editor.inputs.shiftKey = true
250
- expect(customUtil.options.shouldBeExact(editor)).toBe(true)
261
+ expect(customUtil.options.shouldBeExact(editor, true)).toBe(true)
262
+
263
+ // Test shouldBeExact with both keys but not precise
264
+ editor.inputs.altKey = true
265
+ editor.inputs.shiftKey = true
266
+ expect(customUtil.options.shouldBeExact(editor, false)).toBe(false)
251
267
 
252
268
  // Test shouldBeExact with only one key
253
269
  editor.inputs.altKey = true
254
270
  editor.inputs.shiftKey = false
255
- expect(customUtil.options.shouldBeExact(editor)).toBe(false)
271
+ expect(customUtil.options.shouldBeExact(editor, true)).toBe(false)
256
272
 
257
273
  // Test shouldIgnoreTargets with any key
258
274
  editor.inputs.altKey = false
@@ -282,6 +298,61 @@ describe('ArrowShapeOptions', () => {
282
298
  expect(editor.getCurrentToolId()).toBe('arrow')
283
299
  })
284
300
 
301
+ it('should allow custom shouldBeExact logic based on isPrecise - example from arrow precise-exact', () => {
302
+ // This replicates the logic from the arrows-precise-exact example
303
+ const baseUtil = editor.getShapeUtil<ArrowShapeUtil>('arrow')
304
+ class ExampleArrowShapeUtil extends ArrowShapeUtil {
305
+ override options = {
306
+ ...baseUtil.options,
307
+ shouldBeExact: (_editor: any, isPrecise: boolean) => isPrecise,
308
+ }
309
+ }
310
+
311
+ // Replace the util temporarily for testing
312
+ const customUtil = new ExampleArrowShapeUtil(editor)
313
+ const originalShouldBeExact = baseUtil.options.shouldBeExact
314
+ baseUtil.options.shouldBeExact = customUtil.options.shouldBeExact
315
+
316
+ try {
317
+ editor.setCurrentTool('arrow')
318
+ editor.inputs.ctrlKey = false // Allow binding
319
+
320
+ // Set up fast pointer velocity to ensure precise remains false
321
+ editor.inputs.pointerVelocity = { x: 2, y: 2, len: () => 2.8 } as any
322
+
323
+ const targetState = updateArrowTargetState({
324
+ editor,
325
+ pointInPageSpace: { x: 150, y: 150 },
326
+ arrow: undefined,
327
+ isPrecise: true, // Input precise
328
+ currentBinding: undefined,
329
+ oppositeBinding: undefined,
330
+ })
331
+
332
+ // With the custom logic, precise arrows should be exact
333
+ expect(targetState?.isExact).toBe(true)
334
+ expect(targetState?.isPrecise).toBe(true)
335
+
336
+ // Test with non-precise movement (and fast velocity to avoid auto-precise)
337
+ const nonPreciseTargetState = updateArrowTargetState({
338
+ editor,
339
+ pointInPageSpace: { x: 150, y: 150 },
340
+ arrow: undefined,
341
+ isPrecise: false, // Not precise
342
+ currentBinding: undefined,
343
+ oppositeBinding: undefined,
344
+ })
345
+
346
+ // Non-precise arrows should not be exact with this custom logic,
347
+ // but they might still become precise due to internal logic
348
+ // The key test is that shouldBeExact gets the final computed precise value
349
+ expect(nonPreciseTargetState).toBeDefined()
350
+ } finally {
351
+ // Restore original function
352
+ baseUtil.options.shouldBeExact = originalShouldBeExact
353
+ }
354
+ })
355
+
285
356
  it('should respect shouldIgnoreTargets when starting arrow creation', () => {
286
357
  editor.setCurrentTool('arrow')
287
358
 
@@ -1,5 +1,8 @@
1
1
  import { IndexKey, TLArrowShape, TLShapeId, Vec, createShapeId } from '@tldraw/editor'
2
+ import { vi } from 'vitest'
2
3
  import { TestEditor } from '../../../test/TestEditor'
4
+ import { defaultShapeUtils } from '../../defaultShapeUtils'
5
+ import { ArrowShapeUtil } from './ArrowShapeUtil'
3
6
  import { getArrowTargetState } from './arrowTargetState'
4
7
  import { getArrowBindings } from './shared'
5
8
 
@@ -13,7 +16,7 @@ global.cancelAnimationFrame = function cancelAnimationFrame(id) {
13
16
  clearTimeout(id)
14
17
  }
15
18
 
16
- jest.useFakeTimers()
19
+ vi.useFakeTimers()
17
20
 
18
21
  const ids = {
19
22
  box1: createShapeId('box1'),
@@ -25,8 +28,8 @@ function bindings(id: TLShapeId) {
25
28
  return getArrowBindings(editor, editor.getShape(id) as TLArrowShape)
26
29
  }
27
30
 
28
- beforeEach(() => {
29
- editor = new TestEditor()
31
+ function init(opts?: ConstructorParameters<typeof TestEditor>[0]) {
32
+ editor = new TestEditor(opts)
30
33
  editor
31
34
  .selectAll()
32
35
  .deleteShapes(editor.getSelectedShapeIds())
@@ -35,7 +38,9 @@ beforeEach(() => {
35
38
  { id: ids.box2, type: 'geo', x: 300, y: 300, props: { w: 100, h: 100 } },
36
39
  { id: ids.box3, type: 'geo', x: 350, y: 350, props: { w: 50, h: 50 } }, // overlapping box2, but smaller!
37
40
  ])
38
- })
41
+ }
42
+
43
+ beforeEach(init)
39
44
 
40
45
  it('enters the arrow state', () => {
41
46
  editor.setCurrentTool('arrow')
@@ -242,7 +247,7 @@ describe('When pointing an end shape', () => {
242
247
  },
243
248
  })
244
249
 
245
- jest.advanceTimersByTime(1000)
250
+ vi.advanceTimersByTime(1000)
246
251
 
247
252
  arrow = editor.getCurrentPageShapes()[editor.getCurrentPageShapes().length - 1]
248
253
 
@@ -306,7 +311,7 @@ describe('When pointing an end shape', () => {
306
311
  })
307
312
 
308
313
  // Give time for the velocity to die down
309
- jest.advanceTimersByTime(1000)
314
+ vi.advanceTimersByTime(1000)
310
315
 
311
316
  arrow = editor.getCurrentPageShapes()[editor.getCurrentPageShapes().length - 1]
312
317
 
@@ -568,12 +573,102 @@ describe('reparenting issue', () => {
568
573
  const arrow1BoundIndex = editor.getShape(arrow1Id)!.index
569
574
  const arrow2BoundIndex = editor.getShape(arrow2Id)!.index
570
575
  expect(arrow1BoundIndex).toBe('a1V')
571
- expect(arrow2BoundIndex).toBe('a1F')
576
+ expect(arrow2BoundIndex).toBe('a1G')
572
577
 
573
578
  // nudge everything around and make sure we all stay in the right order
574
579
  editor.selectAll().nudgeShapes(editor.getSelectedShapeIds(), { x: -1, y: 0 })
575
580
  expect(editor.getShape(arrow1Id)!.index).toBe('a1V')
576
- expect(editor.getShape(arrow2Id)!.index).toBe('a1F')
581
+ expect(editor.getShape(arrow2Id)!.index).toBe('a1G')
582
+ })
583
+ })
584
+
585
+ describe('precision timeout configuration', () => {
586
+ it('uses a timeout when dragging arrow handles', () => {
587
+ // Create an arrow first
588
+
589
+ editor.setCurrentTool('arrow').pointerDown(0, 0)
590
+ // Use high velocity to avoid precise mode immediately
591
+ editor.inputs.pointerVelocity = new Vec(1, 1)
592
+ editor.pointerMove(100, 100)
593
+
594
+ const arrow = editor.getCurrentPageShapes()[
595
+ editor.getCurrentPageShapes().length - 1
596
+ ] as TLArrowShape
597
+
598
+ editor.expectToBeIn('select.dragging_handle')
599
+
600
+ expect(bindings(arrow.id)).toMatchObject({
601
+ end: {
602
+ toId: ids.box1,
603
+ props: {
604
+ isPrecise: false,
605
+ },
606
+ },
607
+ })
608
+
609
+ vi.advanceTimersByTime(1000)
610
+
611
+ expect(bindings(arrow.id)).toMatchObject({
612
+ end: {
613
+ toId: ids.box1,
614
+ props: {
615
+ isPrecise: true,
616
+ },
617
+ },
618
+ })
619
+ })
620
+
621
+ it('allows configuring the pointingPreciseTimeout', () => {
622
+ init({
623
+ shapeUtils: [
624
+ ...defaultShapeUtils.map((s) =>
625
+ s.type === 'arrow' ? ArrowShapeUtil.configure({ pointingPreciseTimeout: 2000 }) : s
626
+ ),
627
+ ],
628
+ })
629
+ // Create an arrow first
630
+
631
+ editor.setCurrentTool('arrow').pointerDown(0, 0)
632
+ // Use high velocity to avoid precise mode immediately
633
+ editor.inputs.pointerVelocity = new Vec(1, 1)
634
+ editor.pointerMove(100, 100)
635
+
636
+ const arrow = editor.getCurrentPageShapes()[
637
+ editor.getCurrentPageShapes().length - 1
638
+ ] as TLArrowShape
639
+
640
+ editor.expectToBeIn('select.dragging_handle')
641
+
642
+ expect(bindings(arrow.id)).toMatchObject({
643
+ end: {
644
+ toId: ids.box1,
645
+ props: {
646
+ isPrecise: false,
647
+ },
648
+ },
649
+ })
650
+
651
+ vi.advanceTimersByTime(1000)
652
+
653
+ expect(bindings(arrow.id)).toMatchObject({
654
+ end: {
655
+ toId: ids.box1,
656
+ props: {
657
+ isPrecise: false,
658
+ },
659
+ },
660
+ })
661
+
662
+ vi.advanceTimersByTime(1000)
663
+
664
+ expect(bindings(arrow.id)).toMatchObject({
665
+ end: {
666
+ toId: ids.box1,
667
+ props: {
668
+ isPrecise: true,
669
+ },
670
+ },
671
+ })
577
672
  })
578
673
  })
579
674
 
@@ -1,4 +1,5 @@
1
1
  import { HALF_PI, TLArrowShape, TLShapeId, createShapeId, toRichText } from '@tldraw/editor'
2
+ import { vi } from 'vitest'
2
3
  import { TestEditor } from '../../../test/TestEditor'
3
4
  import { createOrUpdateArrowBinding, getArrowBindings } from './shared'
4
5
 
@@ -12,7 +13,7 @@ const ids = {
12
13
  arrow1: createShapeId('arrow1'),
13
14
  }
14
15
 
15
- jest.useFakeTimers()
16
+ vi.useFakeTimers()
16
17
 
17
18
  window.requestAnimationFrame = function requestAnimationFrame(cb) {
18
19
  return setTimeout(cb, 1000 / 60)
@@ -217,7 +218,7 @@ describe('Other cases when arrow are moved', () => {
217
218
  // When box one is not selected, unbinds box1 and keeps binding to box2
218
219
  editor.select(ids.arrow1, ids.box2, ids.box3)
219
220
  editor.alignShapes(editor.getSelectedShapeIds(), 'right')
220
- jest.advanceTimersByTime(1000)
221
+ vi.advanceTimersByTime(1000)
221
222
 
222
223
  expect(bindings()).toMatchObject({
223
224
  start: { toId: ids.box1, props: { isPrecise: false } },
@@ -227,7 +228,7 @@ describe('Other cases when arrow are moved', () => {
227
228
  // maintains bindings if they would still be over the same shape (but makes them precise), but unbinds others
228
229
  editor.select(ids.arrow1, ids.box3)
229
230
  editor.alignShapes(editor.getSelectedShapeIds(), 'top')
230
- jest.advanceTimersByTime(1000)
231
+ vi.advanceTimersByTime(1000)
231
232
 
232
233
  expect(bindings()).toMatchObject({
233
234
  start: { toId: ids.box1, props: { isPrecise: true } },
@@ -244,7 +245,7 @@ describe('Other cases when arrow are moved', () => {
244
245
  // When box one is not selected, unbinds box1 and keeps binding to box2
245
246
  editor.select(ids.arrow1, ids.box2, ids.box3)
246
247
  editor.distributeShapes(editor.getSelectedShapeIds(), 'horizontal')
247
- jest.advanceTimersByTime(1000)
248
+ vi.advanceTimersByTime(1000)
248
249
 
249
250
  expect(bindings()).toMatchObject({
250
251
  start: { toId: ids.box1, props: { isPrecise: false } },
@@ -254,7 +255,7 @@ describe('Other cases when arrow are moved', () => {
254
255
  // unbinds when only the arrow is selected (not its bound shapes) if the arrow itself has moved
255
256
  editor.select(ids.arrow1, ids.box3, ids.box4)
256
257
  editor.distributeShapes(editor.getSelectedShapeIds(), 'vertical')
257
- jest.advanceTimersByTime(1000)
258
+ vi.advanceTimersByTime(1000)
258
259
 
259
260
  // The arrow didn't actually move
260
261
  expect(bindings()).toMatchObject({
@@ -265,7 +266,7 @@ describe('Other cases when arrow are moved', () => {
265
266
  // The arrow will not move because it is still bound to another shape
266
267
  editor.updateShapes([{ id: ids.box4, type: 'geo', y: -600 }])
267
268
  editor.distributeShapes(editor.getSelectedShapeIds(), 'vertical')
268
- jest.advanceTimersByTime(1000)
269
+ vi.advanceTimersByTime(1000)
269
270
 
270
271
  expect(bindings()).toMatchObject({
271
272
  start: undefined,
@@ -578,3 +579,44 @@ describe("an arrow's parents", () => {
578
579
  })
579
580
  })
580
581
  })
582
+
583
+ describe('Arrow export bounds', () => {
584
+ it('excludes labels from shape bounds for export', () => {
585
+ editor.selectAll().deleteShapes(editor.getSelectedShapeIds())
586
+
587
+ // Create shapes for the arrow to bind to
588
+ editor.createShapes([
589
+ { id: ids.box1, type: 'geo', x: 100, y: 100, props: { w: 100, h: 100 } },
590
+ { id: ids.box2, type: 'geo', x: 300, y: 100, props: { w: 100, h: 100 } },
591
+ ])
592
+
593
+ // Create an arrow with a label
594
+ editor.createShapes([
595
+ {
596
+ id: ids.arrow1,
597
+ type: 'arrow',
598
+ x: 0,
599
+ y: 0,
600
+ props: {
601
+ start: { x: 0, y: 0 },
602
+ end: { x: 0, y: 100 },
603
+ richText: toRichText('Test Label'),
604
+ },
605
+ },
606
+ ])
607
+
608
+ // Get the page bounds (should exclude labels due to excludeFromShapeBounds flag)
609
+ const pageBounds = editor.getShapePageBounds(ids.arrow1)
610
+ expect(pageBounds).toBeDefined()
611
+
612
+ // The bounds should be smaller than if labels were included
613
+ // Since the arrow has a label that's excluded, the bounds should be minimal
614
+ expect(pageBounds!.width).toBeLessThan(200) // Should not include label width
615
+ expect(pageBounds!.height).toBeLessThan(200) // Should not include label height
616
+
617
+ // Verify that the arrow has a label (which should be excluded from shape bounds)
618
+ const arrow = editor.getShape(ids.arrow1) as TLArrowShape
619
+ expect(arrow.props.richText).toBeDefined()
620
+ expect(arrow.props.richText).not.toBeNull()
621
+ })
622
+ })