tldraw 3.16.0-canary.1647ca5bba28 → 3.16.0-canary.1e59e27864bb

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 (204) hide show
  1. package/dist-cjs/index.d.ts +128 -54
  2. package/dist-cjs/index.js +25 -9
  3. package/dist-cjs/index.js.map +2 -2
  4. package/dist-cjs/lib/defaultExternalContentHandlers.js +10 -0
  5. package/dist-cjs/lib/defaultExternalContentHandlers.js.map +2 -2
  6. package/dist-cjs/lib/shapes/arrow/arrowLabel.js +6 -0
  7. package/dist-cjs/lib/shapes/arrow/arrowLabel.js.map +3 -3
  8. package/dist-cjs/lib/shapes/bookmark/BookmarkShapeUtil.js +4 -4
  9. package/dist-cjs/lib/shapes/bookmark/BookmarkShapeUtil.js.map +2 -2
  10. package/dist-cjs/lib/shapes/frame/FrameShapeUtil.js +2 -1
  11. package/dist-cjs/lib/shapes/frame/FrameShapeUtil.js.map +2 -2
  12. package/dist-cjs/lib/shapes/frame/components/FrameLabelInput.js +2 -2
  13. package/dist-cjs/lib/shapes/frame/components/FrameLabelInput.js.map +2 -2
  14. package/dist-cjs/lib/shapes/geo/GeoShapeUtil.js +1 -0
  15. package/dist-cjs/lib/shapes/geo/GeoShapeUtil.js.map +2 -2
  16. package/dist-cjs/lib/shapes/note/NoteShapeUtil.js +2 -1
  17. package/dist-cjs/lib/shapes/note/NoteShapeUtil.js.map +2 -2
  18. package/dist-cjs/lib/shapes/shared/HyperlinkButton.js +4 -4
  19. package/dist-cjs/lib/shapes/shared/HyperlinkButton.js.map +2 -2
  20. package/dist-cjs/lib/shapes/shared/useEditablePlainText.js +3 -2
  21. package/dist-cjs/lib/shapes/shared/useEditablePlainText.js.map +2 -2
  22. package/dist-cjs/lib/shapes/text/PlainTextArea.js +2 -2
  23. package/dist-cjs/lib/shapes/text/PlainTextArea.js.map +2 -2
  24. package/dist-cjs/lib/shapes/text/RichTextArea.js +3 -3
  25. package/dist-cjs/lib/shapes/text/RichTextArea.js.map +2 -2
  26. package/dist-cjs/lib/ui/components/A11y.js +1 -1
  27. package/dist-cjs/lib/ui/components/A11y.js.map +2 -2
  28. package/dist-cjs/lib/ui/components/LanguageMenu.js +1 -0
  29. package/dist-cjs/lib/ui/components/LanguageMenu.js.map +2 -2
  30. package/dist-cjs/lib/ui/components/Minimap/DefaultMinimap.js +1 -0
  31. package/dist-cjs/lib/ui/components/Minimap/DefaultMinimap.js.map +2 -2
  32. package/dist-cjs/lib/ui/components/PageMenu/DefaultPageMenu.js +1 -1
  33. package/dist-cjs/lib/ui/components/PageMenu/DefaultPageMenu.js.map +2 -2
  34. package/dist-cjs/lib/ui/components/StylePanel/DefaultStylePanel.js +9 -4
  35. package/dist-cjs/lib/ui/components/StylePanel/DefaultStylePanel.js.map +2 -2
  36. package/dist-cjs/lib/ui/components/StylePanel/DefaultStylePanelContent.js +255 -316
  37. package/dist-cjs/lib/ui/components/StylePanel/DefaultStylePanelContent.js.map +2 -2
  38. package/dist-cjs/lib/ui/components/StylePanel/StylePanelButtonPicker.js +147 -0
  39. package/dist-cjs/lib/ui/components/StylePanel/StylePanelButtonPicker.js.map +7 -0
  40. package/dist-cjs/lib/ui/components/StylePanel/StylePanelContext.js +68 -0
  41. package/dist-cjs/lib/ui/components/StylePanel/StylePanelContext.js.map +7 -0
  42. package/dist-cjs/lib/ui/components/StylePanel/{DoubleDropdownPicker.js → StylePanelDoubleDropdownPicker.js} +23 -22
  43. package/dist-cjs/lib/ui/components/StylePanel/StylePanelDoubleDropdownPicker.js.map +7 -0
  44. package/dist-cjs/lib/ui/components/StylePanel/{DropdownPicker.js → StylePanelDropdownPicker.js} +23 -20
  45. package/dist-cjs/lib/ui/components/StylePanel/StylePanelDropdownPicker.js.map +7 -0
  46. package/dist-cjs/lib/ui/components/StylePanel/StylePanelSubheading.js +28 -0
  47. package/dist-cjs/lib/ui/components/StylePanel/StylePanelSubheading.js.map +7 -0
  48. package/dist-cjs/lib/ui/components/Toolbar/AltTextEditor.js +1 -0
  49. package/dist-cjs/lib/ui/components/Toolbar/AltTextEditor.js.map +2 -2
  50. package/dist-cjs/lib/ui/components/Toolbar/LinkEditor.js +2 -1
  51. package/dist-cjs/lib/ui/components/Toolbar/LinkEditor.js.map +2 -2
  52. package/dist-cjs/lib/ui/components/primitives/TldrawUiContextualToolbar.js +1 -1
  53. package/dist-cjs/lib/ui/components/primitives/TldrawUiContextualToolbar.js.map +1 -1
  54. package/dist-cjs/lib/ui/components/primitives/TldrawUiInput.js +2 -2
  55. package/dist-cjs/lib/ui/components/primitives/TldrawUiInput.js.map +2 -2
  56. package/dist-cjs/lib/ui/components/primitives/TldrawUiSlider.js +1 -1
  57. package/dist-cjs/lib/ui/components/primitives/TldrawUiSlider.js.map +2 -2
  58. package/dist-cjs/lib/ui/components/primitives/TldrawUiToolbar.js +3 -0
  59. package/dist-cjs/lib/ui/components/primitives/TldrawUiToolbar.js.map +2 -2
  60. package/dist-cjs/lib/ui/components/primitives/TldrawUiTooltip.js +44 -5
  61. package/dist-cjs/lib/ui/components/primitives/TldrawUiTooltip.js.map +2 -2
  62. package/dist-cjs/lib/ui/components/primitives/menus/TldrawUiMenuCheckboxItem.js +3 -0
  63. package/dist-cjs/lib/ui/components/primitives/menus/TldrawUiMenuCheckboxItem.js.map +2 -2
  64. package/dist-cjs/lib/ui/components/primitives/menus/TldrawUiMenuItem.js +5 -5
  65. package/dist-cjs/lib/ui/components/primitives/menus/TldrawUiMenuItem.js.map +1 -1
  66. package/dist-cjs/lib/ui/context/actions.js +6 -0
  67. package/dist-cjs/lib/ui/context/actions.js.map +2 -2
  68. package/dist-cjs/lib/ui/hooks/useClipboardEvents.js +1 -1
  69. package/dist-cjs/lib/ui/hooks/useClipboardEvents.js.map +2 -2
  70. package/dist-cjs/lib/ui/hooks/useTools.js +1 -1
  71. package/dist-cjs/lib/ui/hooks/useTools.js.map +2 -2
  72. package/dist-cjs/lib/ui/hooks/useTranslation/TLUiTranslationKey.js.map +1 -1
  73. package/dist-cjs/lib/ui/hooks/useTranslation/defaultTranslation.js +4 -2
  74. package/dist-cjs/lib/ui/hooks/useTranslation/defaultTranslation.js.map +2 -2
  75. package/dist-cjs/lib/ui/version.js +3 -3
  76. package/dist-cjs/lib/ui/version.js.map +1 -1
  77. package/dist-esm/index.d.mts +128 -54
  78. package/dist-esm/index.mjs +49 -17
  79. package/dist-esm/index.mjs.map +2 -2
  80. package/dist-esm/lib/defaultExternalContentHandlers.mjs +10 -0
  81. package/dist-esm/lib/defaultExternalContentHandlers.mjs.map +2 -2
  82. package/dist-esm/lib/shapes/arrow/arrowLabel.mjs +6 -0
  83. package/dist-esm/lib/shapes/arrow/arrowLabel.mjs.map +3 -3
  84. package/dist-esm/lib/shapes/bookmark/BookmarkShapeUtil.mjs +5 -5
  85. package/dist-esm/lib/shapes/bookmark/BookmarkShapeUtil.mjs.map +2 -2
  86. package/dist-esm/lib/shapes/frame/FrameShapeUtil.mjs +2 -1
  87. package/dist-esm/lib/shapes/frame/FrameShapeUtil.mjs.map +2 -2
  88. package/dist-esm/lib/shapes/frame/components/FrameLabelInput.mjs +3 -3
  89. package/dist-esm/lib/shapes/frame/components/FrameLabelInput.mjs.map +2 -2
  90. package/dist-esm/lib/shapes/geo/GeoShapeUtil.mjs +1 -0
  91. package/dist-esm/lib/shapes/geo/GeoShapeUtil.mjs.map +2 -2
  92. package/dist-esm/lib/shapes/note/NoteShapeUtil.mjs +2 -1
  93. package/dist-esm/lib/shapes/note/NoteShapeUtil.mjs.map +2 -2
  94. package/dist-esm/lib/shapes/shared/HyperlinkButton.mjs +5 -5
  95. package/dist-esm/lib/shapes/shared/HyperlinkButton.mjs.map +2 -2
  96. package/dist-esm/lib/shapes/shared/useEditablePlainText.mjs +4 -3
  97. package/dist-esm/lib/shapes/shared/useEditablePlainText.mjs.map +2 -2
  98. package/dist-esm/lib/shapes/text/PlainTextArea.mjs +3 -3
  99. package/dist-esm/lib/shapes/text/PlainTextArea.mjs.map +2 -2
  100. package/dist-esm/lib/shapes/text/RichTextArea.mjs +3 -4
  101. package/dist-esm/lib/shapes/text/RichTextArea.mjs.map +2 -2
  102. package/dist-esm/lib/ui/components/A11y.mjs +2 -2
  103. package/dist-esm/lib/ui/components/A11y.mjs.map +2 -2
  104. package/dist-esm/lib/ui/components/LanguageMenu.mjs +1 -0
  105. package/dist-esm/lib/ui/components/LanguageMenu.mjs.map +2 -2
  106. package/dist-esm/lib/ui/components/Minimap/DefaultMinimap.mjs +1 -0
  107. package/dist-esm/lib/ui/components/Minimap/DefaultMinimap.mjs.map +2 -2
  108. package/dist-esm/lib/ui/components/PageMenu/DefaultPageMenu.mjs +2 -2
  109. package/dist-esm/lib/ui/components/PageMenu/DefaultPageMenu.mjs.map +2 -2
  110. package/dist-esm/lib/ui/components/StylePanel/DefaultStylePanel.mjs +14 -5
  111. package/dist-esm/lib/ui/components/StylePanel/DefaultStylePanel.mjs.map +2 -2
  112. package/dist-esm/lib/ui/components/StylePanel/DefaultStylePanelContent.mjs +257 -320
  113. package/dist-esm/lib/ui/components/StylePanel/DefaultStylePanelContent.mjs.map +2 -2
  114. package/dist-esm/lib/ui/components/StylePanel/StylePanelButtonPicker.mjs +135 -0
  115. package/dist-esm/lib/ui/components/StylePanel/StylePanelButtonPicker.mjs.map +7 -0
  116. package/dist-esm/lib/ui/components/StylePanel/StylePanelContext.mjs +48 -0
  117. package/dist-esm/lib/ui/components/StylePanel/StylePanelContext.mjs.map +7 -0
  118. package/dist-esm/lib/ui/components/StylePanel/{DoubleDropdownPicker.mjs → StylePanelDoubleDropdownPicker.mjs} +20 -19
  119. package/dist-esm/lib/ui/components/StylePanel/StylePanelDoubleDropdownPicker.mjs.map +7 -0
  120. package/dist-esm/lib/ui/components/StylePanel/{DropdownPicker.mjs → StylePanelDropdownPicker.mjs} +20 -17
  121. package/dist-esm/lib/ui/components/StylePanel/StylePanelDropdownPicker.mjs.map +7 -0
  122. package/dist-esm/lib/ui/components/StylePanel/StylePanelSubheading.mjs +8 -0
  123. package/dist-esm/lib/ui/components/StylePanel/StylePanelSubheading.mjs.map +7 -0
  124. package/dist-esm/lib/ui/components/Toolbar/AltTextEditor.mjs +1 -0
  125. package/dist-esm/lib/ui/components/Toolbar/AltTextEditor.mjs.map +2 -2
  126. package/dist-esm/lib/ui/components/Toolbar/LinkEditor.mjs +2 -1
  127. package/dist-esm/lib/ui/components/Toolbar/LinkEditor.mjs.map +2 -2
  128. package/dist-esm/lib/ui/components/primitives/TldrawUiContextualToolbar.mjs +2 -2
  129. package/dist-esm/lib/ui/components/primitives/TldrawUiContextualToolbar.mjs.map +1 -1
  130. package/dist-esm/lib/ui/components/primitives/TldrawUiInput.mjs +3 -3
  131. package/dist-esm/lib/ui/components/primitives/TldrawUiInput.mjs.map +2 -2
  132. package/dist-esm/lib/ui/components/primitives/TldrawUiSlider.mjs +1 -1
  133. package/dist-esm/lib/ui/components/primitives/TldrawUiSlider.mjs.map +2 -2
  134. package/dist-esm/lib/ui/components/primitives/TldrawUiToolbar.mjs +3 -0
  135. package/dist-esm/lib/ui/components/primitives/TldrawUiToolbar.mjs.map +2 -2
  136. package/dist-esm/lib/ui/components/primitives/TldrawUiTooltip.mjs +45 -5
  137. package/dist-esm/lib/ui/components/primitives/TldrawUiTooltip.mjs.map +2 -2
  138. package/dist-esm/lib/ui/components/primitives/menus/TldrawUiMenuCheckboxItem.mjs +3 -0
  139. package/dist-esm/lib/ui/components/primitives/menus/TldrawUiMenuCheckboxItem.mjs.map +2 -2
  140. package/dist-esm/lib/ui/components/primitives/menus/TldrawUiMenuItem.mjs +5 -5
  141. package/dist-esm/lib/ui/components/primitives/menus/TldrawUiMenuItem.mjs.map +1 -1
  142. package/dist-esm/lib/ui/context/actions.mjs +6 -0
  143. package/dist-esm/lib/ui/context/actions.mjs.map +2 -2
  144. package/dist-esm/lib/ui/hooks/useClipboardEvents.mjs +2 -2
  145. package/dist-esm/lib/ui/hooks/useClipboardEvents.mjs.map +2 -2
  146. package/dist-esm/lib/ui/hooks/useTools.mjs +1 -1
  147. package/dist-esm/lib/ui/hooks/useTools.mjs.map +2 -2
  148. package/dist-esm/lib/ui/hooks/useTranslation/defaultTranslation.mjs +4 -2
  149. package/dist-esm/lib/ui/hooks/useTranslation/defaultTranslation.mjs.map +2 -2
  150. package/dist-esm/lib/ui/version.mjs +3 -3
  151. package/dist-esm/lib/ui/version.mjs.map +1 -1
  152. package/package.json +3 -3
  153. package/src/index.ts +36 -13
  154. package/src/lib/defaultExternalContentHandlers.ts +14 -0
  155. package/src/lib/shapes/arrow/ArrowShapeTool.test.ts +2 -2
  156. package/src/lib/shapes/arrow/ArrowShapeUtil.test.ts +41 -0
  157. package/src/lib/shapes/arrow/arrowLabel.ts +8 -0
  158. package/src/lib/shapes/bookmark/BookmarkShapeUtil.tsx +5 -5
  159. package/src/lib/shapes/frame/FrameShapeUtil.tsx +1 -0
  160. package/src/lib/shapes/frame/components/FrameLabelInput.tsx +3 -3
  161. package/src/lib/shapes/geo/GeoShapeUtil.tsx +1 -0
  162. package/src/lib/shapes/note/NoteShapeUtil.tsx +1 -0
  163. package/src/lib/shapes/shared/HyperlinkButton.tsx +5 -5
  164. package/src/lib/shapes/shared/useEditablePlainText.ts +5 -3
  165. package/src/lib/shapes/text/PlainTextArea.tsx +3 -3
  166. package/src/lib/shapes/text/RichTextArea.tsx +3 -4
  167. package/src/lib/ui/components/A11y.tsx +2 -2
  168. package/src/lib/ui/components/LanguageMenu.tsx +1 -0
  169. package/src/lib/ui/components/Minimap/DefaultMinimap.tsx +1 -0
  170. package/src/lib/ui/components/PageMenu/DefaultPageMenu.tsx +2 -2
  171. package/src/lib/ui/components/StylePanel/DefaultStylePanel.tsx +27 -13
  172. package/src/lib/ui/components/StylePanel/DefaultStylePanelContent.tsx +260 -381
  173. package/src/lib/ui/components/{primitives/TldrawUiButtonPicker.tsx → StylePanel/StylePanelButtonPicker.tsx} +70 -50
  174. package/src/lib/ui/components/StylePanel/StylePanelContext.tsx +63 -0
  175. package/src/lib/ui/components/StylePanel/{DoubleDropdownPicker.tsx → StylePanelDoubleDropdownPicker.tsx} +28 -19
  176. package/src/lib/ui/components/StylePanel/StylePanelDropdownPicker.tsx +119 -0
  177. package/src/lib/ui/components/StylePanel/StylePanelSubheading.tsx +9 -0
  178. package/src/lib/ui/components/Toolbar/AltTextEditor.tsx +1 -0
  179. package/src/lib/ui/components/Toolbar/LinkEditor.tsx +1 -0
  180. package/src/lib/ui/components/primitives/TldrawUiContextualToolbar.tsx +2 -2
  181. package/src/lib/ui/components/primitives/TldrawUiInput.tsx +3 -3
  182. package/src/lib/ui/components/primitives/TldrawUiSlider.tsx +2 -2
  183. package/src/lib/ui/components/primitives/TldrawUiToolbar.tsx +5 -1
  184. package/src/lib/ui/components/primitives/TldrawUiTooltip.tsx +63 -16
  185. package/src/lib/ui/components/primitives/menus/TldrawUiMenuCheckboxItem.tsx +4 -0
  186. package/src/lib/ui/components/primitives/menus/TldrawUiMenuItem.tsx +6 -6
  187. package/src/lib/ui/context/actions.tsx +13 -0
  188. package/src/lib/ui/hooks/useClipboardEvents.ts +2 -2
  189. package/src/lib/ui/hooks/useTools.tsx +1 -1
  190. package/src/lib/ui/hooks/useTranslation/TLUiTranslationKey.ts +2 -0
  191. package/src/lib/ui/hooks/useTranslation/defaultTranslation.ts +4 -2
  192. package/src/lib/ui/version.ts +3 -3
  193. package/src/lib/ui.css +26 -2
  194. package/src/test/getCulledShapes.test.tsx +71 -2
  195. package/tldraw.css +34 -5
  196. package/dist-cjs/lib/ui/components/StylePanel/DoubleDropdownPicker.js.map +0 -7
  197. package/dist-cjs/lib/ui/components/StylePanel/DropdownPicker.js.map +0 -7
  198. package/dist-cjs/lib/ui/components/primitives/TldrawUiButtonPicker.js +0 -131
  199. package/dist-cjs/lib/ui/components/primitives/TldrawUiButtonPicker.js.map +0 -7
  200. package/dist-esm/lib/ui/components/StylePanel/DoubleDropdownPicker.mjs.map +0 -7
  201. package/dist-esm/lib/ui/components/StylePanel/DropdownPicker.mjs.map +0 -7
  202. package/dist-esm/lib/ui/components/primitives/TldrawUiButtonPicker.mjs +0 -115
  203. package/dist-esm/lib/ui/components/primitives/TldrawUiButtonPicker.mjs.map +0 -7
  204. package/src/lib/ui/components/StylePanel/DropdownPicker.tsx +0 -110
package/src/index.ts CHANGED
@@ -360,10 +360,6 @@ export {
360
360
  TldrawUiMenuSubmenu,
361
361
  type TLUiMenuSubmenuProps,
362
362
  } from './lib/ui/components/primitives/menus/TldrawUiMenuSubmenu'
363
- export {
364
- TldrawUiButtonPicker,
365
- type TLUiButtonPickerProps,
366
- } from './lib/ui/components/primitives/TldrawUiButtonPicker'
367
363
  export {
368
364
  TldrawUiContextualToolbar,
369
365
  type TLUiContextualToolbarProps,
@@ -443,17 +439,44 @@ export {
443
439
  type TLUiStylePanelProps,
444
440
  } from './lib/ui/components/StylePanel/DefaultStylePanel'
445
441
  export {
446
- ArrowheadStylePickerSet,
447
- CommonStylePickerSet,
448
442
  DefaultStylePanelContent,
449
- GeoStylePickerSet,
450
- OpacitySlider,
451
- SplineStylePickerSet,
452
- TextStylePickerSet,
453
- type StylePickerSetProps,
454
- type ThemeStylePickerSetProps,
455
- type TLUiStylePanelContentProps,
443
+ StylePanelArrowheadPicker,
444
+ StylePanelArrowKindPicker,
445
+ StylePanelColorPicker,
446
+ StylePanelDashPicker,
447
+ StylePanelFillPicker,
448
+ StylePanelFontPicker,
449
+ StylePanelGeoShapePicker,
450
+ StylePanelLabelAlignPicker,
451
+ StylePanelOpacityPicker,
452
+ StylePanelSection,
453
+ StylePanelSizePicker,
454
+ StylePanelSplinePicker,
455
+ StylePanelTextAlignPicker,
456
+ type StylePanelSectionProps,
456
457
  } from './lib/ui/components/StylePanel/DefaultStylePanelContent'
458
+ export {
459
+ StylePanelButtonPicker,
460
+ type StylePanelButtonPickerProps,
461
+ } from './lib/ui/components/StylePanel/StylePanelButtonPicker'
462
+ export {
463
+ StylePanelContextProvider,
464
+ useStylePanelContext,
465
+ type StylePanelContext,
466
+ type StylePanelContextProviderProps,
467
+ } from './lib/ui/components/StylePanel/StylePanelContext'
468
+ export {
469
+ StylePanelDoubleDropdownPicker,
470
+ type StylePanelDoubleDropdownPickerProps,
471
+ } from './lib/ui/components/StylePanel/StylePanelDoubleDropdownPicker'
472
+ export {
473
+ StylePanelDropdownPicker,
474
+ type StylePanelDropdownPickerProps,
475
+ } from './lib/ui/components/StylePanel/StylePanelDropdownPicker'
476
+ export {
477
+ StylePanelSubheading,
478
+ type StylePanelSubheadingProps,
479
+ } from './lib/ui/components/StylePanel/StylePanelSubheading'
457
480
  export {
458
481
  DefaultImageToolbar,
459
482
  type TLUiImageToolbarProps,
@@ -901,8 +901,22 @@ export function notifyIfFileNotAllowed(file: File, options: TLDefaultExternalCon
901
901
  }
902
902
 
903
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
+
904
917
  toasts.addToast({
905
918
  title: msg('assets.files.size-too-big'),
919
+ description: msg('assets.files.maximum-size').replace('{size}', formatBytes(maxAssetSize)),
906
920
  severity: 'error',
907
921
  })
908
922
  return false
@@ -569,12 +569,12 @@ describe('reparenting issue', () => {
569
569
  const arrow1BoundIndex = editor.getShape(arrow1Id)!.index
570
570
  const arrow2BoundIndex = editor.getShape(arrow2Id)!.index
571
571
  expect(arrow1BoundIndex).toBe('a1V')
572
- expect(arrow2BoundIndex).toBe('a1F')
572
+ expect(arrow2BoundIndex).toBe('a1G')
573
573
 
574
574
  // nudge everything around and make sure we all stay in the right order
575
575
  editor.selectAll().nudgeShapes(editor.getSelectedShapeIds(), { x: -1, y: 0 })
576
576
  expect(editor.getShape(arrow1Id)!.index).toBe('a1V')
577
- expect(editor.getShape(arrow2Id)!.index).toBe('a1F')
577
+ expect(editor.getShape(arrow2Id)!.index).toBe('a1G')
578
578
  })
579
579
  })
580
580
 
@@ -579,3 +579,44 @@ describe("an arrow's parents", () => {
579
579
  })
580
580
  })
581
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
+ })
@@ -227,6 +227,14 @@ interface ArrowheadInfo {
227
227
  hasEndArrowhead: boolean
228
228
  }
229
229
  export function getArrowLabelPosition(editor: Editor, shape: TLArrowShape) {
230
+ const isEditing = editor.getEditingShapeId() === shape.id
231
+ if (!isEditing && isEmptyRichText(shape.props.richText)) {
232
+ // Short-circuit for empty labels.
233
+ const bodyGeom = getArrowBodyGeometry(editor, shape)
234
+ const labelCenter = bodyGeom.interpolateAlongEdge(0.5)
235
+ return { box: Box.FromCenter(labelCenter, new Vec(0, 0)), debugGeom: [] }
236
+ }
237
+
230
238
  const debugGeom: Geometry2d[] = []
231
239
  const info = getArrowInfo(editor, shape)!
232
240
 
@@ -13,7 +13,7 @@ import {
13
13
  debounce,
14
14
  getHashForString,
15
15
  lerp,
16
- stopEventPropagation,
16
+ markEventAsHandled,
17
17
  tlenv,
18
18
  toDomPrecision,
19
19
  useEditor,
@@ -132,9 +132,9 @@ function BookmarkShapeComponent({ shape }: { shape: TLBookmarkShape }) {
132
132
  const [isFaviconValid, setIsFaviconValid] = useState(true)
133
133
  const onFaviconError = () => setIsFaviconValid(false)
134
134
 
135
- const useStopPropagationOnShiftKey = useCallback<PointerEventHandler>(
135
+ const markAsHandledOnShiftKey = useCallback<PointerEventHandler>(
136
136
  (e) => {
137
- if (!editor.inputs.shiftKey) stopEventPropagation(e)
137
+ if (!editor.inputs.shiftKey) markEventAsHandled(e)
138
138
  },
139
139
  [editor]
140
140
  )
@@ -182,8 +182,8 @@ function BookmarkShapeComponent({ shape }: { shape: TLBookmarkShape }) {
182
182
  target="_blank"
183
183
  rel="noopener noreferrer"
184
184
  draggable={false}
185
- onPointerDown={useStopPropagationOnShiftKey}
186
- onPointerUp={useStopPropagationOnShiftKey}
185
+ onPointerDown={markAsHandledOnShiftKey}
186
+ onPointerUp={markAsHandledOnShiftKey}
187
187
  >
188
188
  {isFaviconValid && asset?.props.favicon ? (
189
189
  <img
@@ -196,6 +196,7 @@ export class FrameShapeUtil extends BaseBoxShapeUtil<TLFrameShape> {
196
196
  height,
197
197
  isFilled: true,
198
198
  isLabel: true,
199
+ excludeFromShapeBounds: true,
199
200
  }),
200
201
  ],
201
202
  })
@@ -1,4 +1,4 @@
1
- import { TLFrameShape, TLShapeId, stopEventPropagation, useEditor } from '@tldraw/editor'
1
+ import { TLFrameShape, TLShapeId, markEventAsHandled, useEditor } from '@tldraw/editor'
2
2
  import { forwardRef, useCallback } from 'react'
3
3
  import { defaultEmptyAs } from '../FrameShapeUtil'
4
4
 
@@ -13,7 +13,7 @@ export const FrameLabelInput = forwardRef<
13
13
  if (e.key === 'Enter' && !e.nativeEvent.isComposing) {
14
14
  // need to prevent the enter keydown making it's way up to the Idle state
15
15
  // and sending us back into edit mode
16
- stopEventPropagation(e)
16
+ markEventAsHandled(e)
17
17
  e.currentTarget.blur()
18
18
  editor.setEditingShape(null)
19
19
  }
@@ -74,7 +74,7 @@ export const FrameLabelInput = forwardRef<
74
74
  onKeyDown={handleKeyDown}
75
75
  onBlur={handleBlur}
76
76
  onChange={handleChange}
77
- onPointerDown={isEditing ? stopEventPropagation : undefined}
77
+ onPointerDown={isEditing ? markEventAsHandled : undefined}
78
78
  draggable={false}
79
79
  />
80
80
  {defaultEmptyAs(name, 'Frame') + String.fromCharCode(8203)}
@@ -126,6 +126,7 @@ export class GeoShapeUtil extends BaseBoxShapeUtil<TLGeoShape> {
126
126
  height: unscaledLabelHeight * shape.props.scale,
127
127
  isFilled: true,
128
128
  isLabel: true,
129
+ excludeFromShapeBounds: true,
129
130
  isEmptyLabel: isEmptyRichText(shape.props.richText),
130
131
  }),
131
132
  ],
@@ -147,6 +147,7 @@ export class NoteShapeUtil extends ShapeUtil<TLNoteShape> {
147
147
  height: lh,
148
148
  isFilled: true,
149
149
  isLabel: true,
150
+ excludeFromShapeBounds: true,
150
151
  }),
151
152
  ],
152
153
  })
@@ -1,4 +1,4 @@
1
- import { stopEventPropagation, useEditor, useValue } from '@tldraw/editor'
1
+ import { markEventAsHandled, useEditor, useValue } from '@tldraw/editor'
2
2
  import classNames from 'classnames'
3
3
  import { PointerEventHandler, useCallback } from 'react'
4
4
 
@@ -8,9 +8,9 @@ const LINK_ICON =
8
8
  export function HyperlinkButton({ url }: { url: string }) {
9
9
  const editor = useEditor()
10
10
  const hideButton = useValue('zoomLevel', () => editor.getZoomLevel() < 0.32, [editor])
11
- const useStopPropagationOnShiftKey = useCallback<PointerEventHandler>(
11
+ const markAsHandledOnShiftKey = useCallback<PointerEventHandler>(
12
12
  (e) => {
13
- if (!editor.inputs.shiftKey) stopEventPropagation(e)
13
+ if (!editor.inputs.shiftKey) markEventAsHandled(e)
14
14
  },
15
15
  [editor]
16
16
  )
@@ -22,8 +22,8 @@ export function HyperlinkButton({ url }: { url: string }) {
22
22
  href={url}
23
23
  target="_blank"
24
24
  rel="noopener noreferrer"
25
- onPointerDown={useStopPropagationOnShiftKey}
26
- onPointerUp={useStopPropagationOnShiftKey}
25
+ onPointerDown={markAsHandledOnShiftKey}
26
+ onPointerUp={markAsHandledOnShiftKey}
27
27
  title={url}
28
28
  draggable={false}
29
29
  >
@@ -3,9 +3,9 @@ import {
3
3
  TLShapeId,
4
4
  TLUnknownShape,
5
5
  getPointerInfo,
6
+ markEventAsHandled,
6
7
  noop,
7
8
  preventDefault,
8
- stopEventPropagation,
9
9
  tlenv,
10
10
  useEditor,
11
11
  useValue,
@@ -136,7 +136,7 @@ export function useEditableTextCommon(shapeId: TLShapeId) {
136
136
  shape: editor.getShape(shapeId)!,
137
137
  })
138
138
 
139
- stopEventPropagation(e) // we need to prevent blurring the input
139
+ e.stopPropagation() // we need to prevent blurring the input
140
140
  },
141
141
  [editor, shapeId]
142
142
  )
@@ -157,11 +157,13 @@ export function useEditableTextCommon(shapeId: TLShapeId) {
157
157
  [editor, shapeId]
158
158
  )
159
159
 
160
+ const handleDoubleClick: (e: React.MouseEvent) => void = markEventAsHandled
161
+
160
162
  return {
161
163
  handleFocus: noop,
162
164
  handleBlur: noop,
163
165
  handleInputPointerDown,
164
- handleDoubleClick: stopEventPropagation,
166
+ handleDoubleClick,
165
167
  handlePaste,
166
168
  isEditing,
167
169
  isReadyForEditing,
@@ -1,4 +1,4 @@
1
- import { preventDefault, stopEventPropagation } from '@tldraw/editor'
1
+ import { markEventAsHandled, preventDefault } from '@tldraw/editor'
2
2
  import React from 'react'
3
3
  import { TextAreaProps } from './RichTextArea'
4
4
 
@@ -46,8 +46,8 @@ export const PlainTextArea = React.forwardRef<HTMLTextAreaElement, TextAreaProps
46
46
  onChange={onChange}
47
47
  onKeyDown={(e) => handleKeyDown(e.nativeEvent)}
48
48
  onBlur={handleBlur}
49
- onTouchEnd={stopEventPropagation}
50
- onContextMenu={isEditing ? stopEventPropagation : undefined}
49
+ onTouchEnd={markEventAsHandled}
50
+ onContextMenu={isEditing ? (e) => e.stopPropagation() : undefined}
51
51
  onPointerDown={handleInputPointerDown}
52
52
  onPaste={handlePaste}
53
53
  onDoubleClick={handleDoubleClick}
@@ -10,7 +10,6 @@ import {
10
10
  TLRichText,
11
11
  TLShapeId,
12
12
  preventDefault,
13
- stopEventPropagation,
14
13
  useEditor,
15
14
  useEvent,
16
15
  useUniqueSafeId,
@@ -233,13 +232,13 @@ export const RichTextArea = React.forwardRef<HTMLDivElement, TextAreaProps>(func
233
232
  tabIndex={-1}
234
233
  data-testid="rich-text-area"
235
234
  className="tl-rich-text tl-text tl-text-input"
236
- onContextMenu={isEditing ? stopEventPropagation : undefined}
235
+ onContextMenu={isEditing ? (e) => e.stopPropagation() : undefined}
237
236
  // N.B. When PointerStateExtension was introduced, this was moved there.
238
237
  // However, that caused selecting over list items to break.
239
238
  // The handleDOMEvents in TipTap don't seem to support the pointerDownCapture event.
240
- onPointerDownCapture={stopEventPropagation}
239
+ onPointerDownCapture={(e) => e.stopPropagation()}
241
240
  // This onTouchEnd is important for Android to be able to change selection on text.
242
- onTouchEnd={stopEventPropagation}
241
+ onTouchEnd={(e) => e.stopPropagation()}
243
242
  // On FF, there's a behavior where dragging a selection will grab that selection into
244
243
  // the drag event. However, once the drag is over, and you select away from the textarea,
245
244
  // starting a drag over the textarea will restart a selection drag instead of a shape drag.
@@ -1,7 +1,7 @@
1
1
  import {
2
2
  debugFlags,
3
3
  Editor,
4
- stopEventPropagation,
4
+ markEventAsHandled,
5
5
  TLGeoShape,
6
6
  TLShapeId,
7
7
  unsafe__withoutCapture,
@@ -23,7 +23,7 @@ export function SkipToMainContent() {
23
23
 
24
24
  const handleNavigateToFirstShape = useCallback(
25
25
  (e: MouseEvent | KeyboardEvent) => {
26
- stopEventPropagation(e)
26
+ markEventAsHandled(e)
27
27
  button.current?.blur()
28
28
  const shapes = editor.getCurrentPageShapesInReadingOrder()
29
29
  if (!shapes.length) return
@@ -18,6 +18,7 @@ export function LanguageMenu() {
18
18
  {LANGUAGES.map(({ locale, label }) => (
19
19
  <TldrawUiMenuCheckboxItem
20
20
  id={`language-${locale}`}
21
+ lang={locale}
21
22
  key={locale}
22
23
  title={locale}
23
24
  label={label}
@@ -204,6 +204,7 @@ export function DefaultMinimap() {
204
204
  <canvas
205
205
  role="img"
206
206
  aria-label={msg('navigation-zone.minimap')}
207
+ data-testid="minimap.canvas"
207
208
  ref={rCanvas}
208
209
  className="tlui-minimap__canvas"
209
210
  onDoubleClick={onDoubleClick}
@@ -1,9 +1,9 @@
1
1
  import {
2
2
  PageRecordType,
3
3
  TLPageId,
4
+ markEventAsHandled,
4
5
  releasePointerCapture,
5
6
  setPointerCapture,
6
- stopEventPropagation,
7
7
  tlenv,
8
8
  useEditor,
9
9
  useValue,
@@ -451,7 +451,7 @@ export const DefaultPageMenu = memo(function DefaultPageMenu() {
451
451
  if (e.key === 'Enter') {
452
452
  if (page.id === currentPage.id) {
453
453
  toggleEditing()
454
- stopEventPropagation(e)
454
+ markEventAsHandled(e)
455
455
  }
456
456
  }
457
457
  }}
@@ -1,18 +1,26 @@
1
- import { useEditor, usePassThroughWheelEvents, useValue } from '@tldraw/editor'
1
+ import {
2
+ ReadonlySharedStyleMap,
3
+ useEditor,
4
+ usePassThroughWheelEvents,
5
+ useValue,
6
+ } from '@tldraw/editor'
2
7
  import classNames from 'classnames'
3
8
  import { ReactNode, memo, useCallback, useEffect, useRef } from 'react'
4
9
  import { useRelevantStyles } from '../../hooks/useRelevantStyles'
5
10
  import { DefaultStylePanelContent } from './DefaultStylePanelContent'
11
+ import { StylePanelContextProvider } from './StylePanelContext'
6
12
 
7
13
  /** @public */
8
14
  export interface TLUiStylePanelProps {
9
15
  isMobile?: boolean
16
+ styles?: ReadonlySharedStyleMap | null
10
17
  children?: ReactNode
11
18
  }
12
19
 
13
20
  /** @public @react */
14
21
  export const DefaultStylePanel = memo(function DefaultStylePanel({
15
22
  isMobile,
23
+ styles,
16
24
  children,
17
25
  }: TLUiStylePanelProps) {
18
26
  const editor = useEditor()
@@ -21,15 +29,16 @@ export const DefaultStylePanel = memo(function DefaultStylePanel({
21
29
  const ref = useRef<HTMLDivElement>(null)
22
30
  usePassThroughWheelEvents(ref)
23
31
 
24
- const styles = useRelevantStyles()
25
-
26
32
  const handlePointerOut = useCallback(() => {
27
33
  if (!isMobile) {
28
34
  editor.updateInstanceState({ isChangingStyle: false })
29
35
  }
30
36
  }, [editor, isMobile])
31
37
 
32
- const content = children ?? <DefaultStylePanelContent styles={styles} />
38
+ const defaultStyles = useRelevantStyles()
39
+ if (styles === undefined) {
40
+ styles = defaultStyles
41
+ }
33
42
 
34
43
  useEffect(() => {
35
44
  function handleKeyDown(event: KeyboardEvent) {
@@ -47,14 +56,19 @@ export const DefaultStylePanel = memo(function DefaultStylePanel({
47
56
  }, [editor])
48
57
 
49
58
  return (
50
- <div
51
- ref={ref}
52
- className={classNames('tlui-style-panel', { 'tlui-style-panel__wrapper': !isMobile })}
53
- data-ismobile={isMobile}
54
- data-show-ui-labels={showUiLabels}
55
- onPointerLeave={handlePointerOut}
56
- >
57
- {content}
58
- </div>
59
+ styles && (
60
+ <div
61
+ ref={ref}
62
+ data-testid="style.panel"
63
+ className={classNames('tlui-style-panel', { 'tlui-style-panel__wrapper': !isMobile })}
64
+ data-ismobile={isMobile}
65
+ data-show-ui-labels={showUiLabels}
66
+ onPointerLeave={handlePointerOut}
67
+ >
68
+ <StylePanelContextProvider styles={styles}>
69
+ {children ?? <DefaultStylePanelContent />}
70
+ </StylePanelContextProvider>
71
+ </div>
72
+ )
59
73
  )
60
74
  })