@tldraw/editor 3.16.0-canary.39bd199a7aa6 → 3.16.0-canary.3be390323e1c

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 (171) hide show
  1. package/dist-cjs/index.d.ts +70 -101
  2. package/dist-cjs/index.js +3 -5
  3. package/dist-cjs/index.js.map +2 -2
  4. package/dist-cjs/lib/TldrawEditor.js +6 -6
  5. package/dist-cjs/lib/TldrawEditor.js.map +2 -2
  6. package/dist-cjs/lib/components/MenuClickCapture.js +0 -5
  7. package/dist-cjs/lib/components/MenuClickCapture.js.map +2 -2
  8. package/dist-cjs/lib/components/Shape.js +7 -10
  9. package/dist-cjs/lib/components/Shape.js.map +2 -2
  10. package/dist-cjs/lib/components/default-components/DefaultCanvas.js +4 -23
  11. package/dist-cjs/lib/components/default-components/DefaultCanvas.js.map +2 -2
  12. package/dist-cjs/lib/components/default-components/DefaultCollaboratorHint.js +1 -1
  13. package/dist-cjs/lib/components/default-components/DefaultCollaboratorHint.js.map +1 -1
  14. package/dist-cjs/lib/components/default-components/DefaultErrorFallback.js +1 -1
  15. package/dist-cjs/lib/components/default-components/DefaultErrorFallback.js.map +2 -2
  16. package/dist-cjs/lib/components/default-components/DefaultScribble.js +1 -1
  17. package/dist-cjs/lib/components/default-components/DefaultScribble.js.map +2 -2
  18. package/dist-cjs/lib/components/default-components/DefaultShapeIndicator.js +9 -1
  19. package/dist-cjs/lib/components/default-components/DefaultShapeIndicator.js.map +2 -2
  20. package/dist-cjs/lib/config/TLUserPreferences.js +9 -3
  21. package/dist-cjs/lib/config/TLUserPreferences.js.map +2 -2
  22. package/dist-cjs/lib/editor/Editor.js +58 -124
  23. package/dist-cjs/lib/editor/Editor.js.map +2 -2
  24. package/dist-cjs/lib/editor/derivations/notVisibleShapes.js +4 -0
  25. package/dist-cjs/lib/editor/derivations/notVisibleShapes.js.map +2 -2
  26. package/dist-cjs/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.js +9 -4
  27. package/dist-cjs/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.js.map +2 -2
  28. package/dist-cjs/lib/editor/shapes/ShapeUtil.js +23 -0
  29. package/dist-cjs/lib/editor/shapes/ShapeUtil.js.map +2 -2
  30. package/dist-cjs/lib/editor/types/misc-types.js.map +1 -1
  31. package/dist-cjs/lib/exports/getSvgJsx.js +35 -16
  32. package/dist-cjs/lib/exports/getSvgJsx.js.map +2 -2
  33. package/dist-cjs/lib/hooks/useCanvasEvents.js +31 -25
  34. package/dist-cjs/lib/hooks/useCanvasEvents.js.map +2 -2
  35. package/dist-cjs/lib/hooks/usePassThroughMouseOverEvents.js +4 -1
  36. package/dist-cjs/lib/hooks/usePassThroughMouseOverEvents.js.map +2 -2
  37. package/dist-cjs/lib/hooks/usePassThroughWheelEvents.js +4 -1
  38. package/dist-cjs/lib/hooks/usePassThroughWheelEvents.js.map +2 -2
  39. package/dist-cjs/lib/license/LicenseManager.js +120 -50
  40. package/dist-cjs/lib/license/LicenseManager.js.map +2 -2
  41. package/dist-cjs/lib/license/LicenseProvider.js +22 -0
  42. package/dist-cjs/lib/license/LicenseProvider.js.map +2 -2
  43. package/dist-cjs/lib/license/Watermark.js +72 -10
  44. package/dist-cjs/lib/license/Watermark.js.map +3 -3
  45. package/dist-cjs/lib/license/useLicenseManagerState.js.map +2 -2
  46. package/dist-cjs/lib/options.js +7 -0
  47. package/dist-cjs/lib/options.js.map +2 -2
  48. package/dist-cjs/lib/primitives/Box.js +3 -0
  49. package/dist-cjs/lib/primitives/Box.js.map +2 -2
  50. package/dist-cjs/lib/primitives/Vec.js +0 -4
  51. package/dist-cjs/lib/primitives/Vec.js.map +2 -2
  52. package/dist-cjs/lib/primitives/geometry/Geometry2d.js +26 -18
  53. package/dist-cjs/lib/primitives/geometry/Geometry2d.js.map +2 -2
  54. package/dist-cjs/lib/primitives/geometry/Group2d.js +3 -0
  55. package/dist-cjs/lib/primitives/geometry/Group2d.js.map +2 -2
  56. package/dist-cjs/lib/utils/reparenting.js +2 -35
  57. package/dist-cjs/lib/utils/reparenting.js.map +3 -3
  58. package/dist-cjs/version.js +3 -3
  59. package/dist-cjs/version.js.map +1 -1
  60. package/dist-esm/index.d.mts +70 -101
  61. package/dist-esm/index.mjs +3 -5
  62. package/dist-esm/index.mjs.map +2 -2
  63. package/dist-esm/lib/TldrawEditor.mjs +6 -6
  64. package/dist-esm/lib/TldrawEditor.mjs.map +2 -2
  65. package/dist-esm/lib/components/MenuClickCapture.mjs +0 -5
  66. package/dist-esm/lib/components/MenuClickCapture.mjs.map +2 -2
  67. package/dist-esm/lib/components/Shape.mjs +7 -10
  68. package/dist-esm/lib/components/Shape.mjs.map +2 -2
  69. package/dist-esm/lib/components/default-components/DefaultCanvas.mjs +4 -23
  70. package/dist-esm/lib/components/default-components/DefaultCanvas.mjs.map +2 -2
  71. package/dist-esm/lib/components/default-components/DefaultCollaboratorHint.mjs +1 -1
  72. package/dist-esm/lib/components/default-components/DefaultCollaboratorHint.mjs.map +1 -1
  73. package/dist-esm/lib/components/default-components/DefaultErrorFallback.mjs +1 -1
  74. package/dist-esm/lib/components/default-components/DefaultErrorFallback.mjs.map +2 -2
  75. package/dist-esm/lib/components/default-components/DefaultScribble.mjs +1 -1
  76. package/dist-esm/lib/components/default-components/DefaultScribble.mjs.map +2 -2
  77. package/dist-esm/lib/components/default-components/DefaultShapeIndicator.mjs +9 -1
  78. package/dist-esm/lib/components/default-components/DefaultShapeIndicator.mjs.map +2 -2
  79. package/dist-esm/lib/config/TLUserPreferences.mjs +9 -3
  80. package/dist-esm/lib/config/TLUserPreferences.mjs.map +2 -2
  81. package/dist-esm/lib/editor/Editor.mjs +58 -124
  82. package/dist-esm/lib/editor/Editor.mjs.map +2 -2
  83. package/dist-esm/lib/editor/derivations/notVisibleShapes.mjs +4 -0
  84. package/dist-esm/lib/editor/derivations/notVisibleShapes.mjs.map +2 -2
  85. package/dist-esm/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.mjs +9 -4
  86. package/dist-esm/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.mjs.map +2 -2
  87. package/dist-esm/lib/editor/shapes/ShapeUtil.mjs +23 -0
  88. package/dist-esm/lib/editor/shapes/ShapeUtil.mjs.map +2 -2
  89. package/dist-esm/lib/exports/getSvgJsx.mjs +36 -16
  90. package/dist-esm/lib/exports/getSvgJsx.mjs.map +2 -2
  91. package/dist-esm/lib/hooks/useCanvasEvents.mjs +32 -26
  92. package/dist-esm/lib/hooks/useCanvasEvents.mjs.map +2 -2
  93. package/dist-esm/lib/hooks/usePassThroughMouseOverEvents.mjs +4 -1
  94. package/dist-esm/lib/hooks/usePassThroughMouseOverEvents.mjs.map +2 -2
  95. package/dist-esm/lib/hooks/usePassThroughWheelEvents.mjs +4 -1
  96. package/dist-esm/lib/hooks/usePassThroughWheelEvents.mjs.map +2 -2
  97. package/dist-esm/lib/license/LicenseManager.mjs +121 -51
  98. package/dist-esm/lib/license/LicenseManager.mjs.map +2 -2
  99. package/dist-esm/lib/license/LicenseProvider.mjs +23 -1
  100. package/dist-esm/lib/license/LicenseProvider.mjs.map +2 -2
  101. package/dist-esm/lib/license/Watermark.mjs +72 -10
  102. package/dist-esm/lib/license/Watermark.mjs.map +3 -3
  103. package/dist-esm/lib/license/useLicenseManagerState.mjs.map +2 -2
  104. package/dist-esm/lib/options.mjs +7 -0
  105. package/dist-esm/lib/options.mjs.map +2 -2
  106. package/dist-esm/lib/primitives/Box.mjs +4 -1
  107. package/dist-esm/lib/primitives/Box.mjs.map +2 -2
  108. package/dist-esm/lib/primitives/Vec.mjs +0 -4
  109. package/dist-esm/lib/primitives/Vec.mjs.map +2 -2
  110. package/dist-esm/lib/primitives/geometry/Geometry2d.mjs +29 -19
  111. package/dist-esm/lib/primitives/geometry/Geometry2d.mjs.map +2 -2
  112. package/dist-esm/lib/primitives/geometry/Group2d.mjs +3 -0
  113. package/dist-esm/lib/primitives/geometry/Group2d.mjs.map +2 -2
  114. package/dist-esm/lib/utils/reparenting.mjs +3 -40
  115. package/dist-esm/lib/utils/reparenting.mjs.map +2 -2
  116. package/dist-esm/version.mjs +3 -3
  117. package/dist-esm/version.mjs.map +1 -1
  118. package/editor.css +308 -290
  119. package/package.json +14 -37
  120. package/src/index.ts +2 -9
  121. package/src/lib/TldrawEditor.tsx +11 -17
  122. package/src/lib/components/MenuClickCapture.tsx +0 -8
  123. package/src/lib/components/Shape.tsx +6 -12
  124. package/src/lib/components/default-components/DefaultCanvas.tsx +5 -22
  125. package/src/lib/components/default-components/DefaultCollaboratorHint.tsx +1 -1
  126. package/src/lib/components/default-components/DefaultErrorFallback.tsx +1 -1
  127. package/src/lib/components/default-components/DefaultScribble.tsx +1 -1
  128. package/src/lib/components/default-components/DefaultShapeIndicator.tsx +5 -1
  129. package/src/lib/config/TLUserPreferences.ts +8 -1
  130. package/src/lib/editor/Editor.test.ts +12 -11
  131. package/src/lib/editor/Editor.ts +75 -166
  132. package/src/lib/editor/derivations/notVisibleShapes.ts +6 -0
  133. package/src/lib/editor/managers/ClickManager/ClickManager.test.ts +15 -14
  134. package/src/lib/editor/managers/EdgeScrollManager/EdgeScrollManager.test.ts +16 -15
  135. package/src/lib/editor/managers/FocusManager/FocusManager.test.ts +49 -48
  136. package/src/lib/editor/managers/FontManager/FontManager.test.ts +24 -23
  137. package/src/lib/editor/managers/HistoryManager/HistoryManager.test.ts +7 -6
  138. package/src/lib/editor/managers/ScribbleManager/ScribbleManager.test.ts +12 -11
  139. package/src/lib/editor/managers/SnapManager/SnapManager.test.ts +57 -50
  140. package/src/lib/editor/managers/TextManager/TextManager.test.ts +51 -26
  141. package/src/lib/editor/managers/TickManager/TickManager.test.ts +14 -13
  142. package/src/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.test.ts +34 -26
  143. package/src/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.ts +6 -1
  144. package/src/lib/editor/shapes/ShapeUtil.ts +46 -0
  145. package/src/lib/editor/types/misc-types.ts +0 -6
  146. package/src/lib/exports/getSvgJsx.test.ts +868 -0
  147. package/src/lib/exports/getSvgJsx.tsx +78 -21
  148. package/src/lib/hooks/useCanvasEvents.ts +45 -38
  149. package/src/lib/hooks/usePassThroughMouseOverEvents.ts +4 -1
  150. package/src/lib/hooks/usePassThroughWheelEvents.ts +6 -1
  151. package/src/lib/license/LicenseManager.test.ts +648 -383
  152. package/src/lib/license/LicenseManager.ts +173 -53
  153. package/src/lib/license/LicenseProvider.tsx +34 -1
  154. package/src/lib/license/Watermark.test.tsx +2 -1
  155. package/src/lib/license/Watermark.tsx +77 -10
  156. package/src/lib/license/useLicenseManagerState.ts +2 -2
  157. package/src/lib/options.ts +8 -0
  158. package/src/lib/primitives/Box.test.ts +126 -0
  159. package/src/lib/primitives/Box.ts +10 -1
  160. package/src/lib/primitives/Vec.ts +0 -5
  161. package/src/lib/primitives/geometry/Geometry2d.ts +49 -19
  162. package/src/lib/primitives/geometry/Group2d.ts +4 -0
  163. package/src/lib/utils/reparenting.ts +3 -69
  164. package/src/lib/utils/sync/LocalIndexedDb.test.ts +2 -1
  165. package/src/lib/utils/sync/TLLocalSyncClient.test.ts +15 -15
  166. package/src/version.ts +3 -3
  167. package/dist-cjs/lib/utils/nearestMultiple.js +0 -34
  168. package/dist-cjs/lib/utils/nearestMultiple.js.map +0 -7
  169. package/dist-esm/lib/utils/nearestMultiple.mjs +0 -14
  170. package/dist-esm/lib/utils/nearestMultiple.mjs.map +0 -7
  171. package/src/lib/utils/nearestMultiple.ts +0 -13
@@ -4,6 +4,7 @@ import {
4
4
  TLGroupShape,
5
5
  TLShape,
6
6
  TLShapeId,
7
+ getColorValue,
7
8
  getDefaultColorTheme,
8
9
  } from '@tldraw/tlschema'
9
10
  import { hasOwnProperty, promiseWithResolve, uniqueId } from '@tldraw/utils'
@@ -56,33 +57,21 @@ export function getSvgJsx(editor: Editor, ids: TLShapeId[], opts: TLImageExportO
56
57
  .filter(({ id }) => shapeIdsToInclude.has(id))
57
58
 
58
59
  // --- Common bounding box of all shapes
60
+ const singleFrameShapeId =
61
+ ids.length === 1 && editor.isShapeOfType<TLFrameShape>(editor.getShape(ids[0])!, 'frame')
62
+ ? ids[0]
63
+ : null
64
+
59
65
  let bbox: null | Box = null
60
66
  if (opts.bounds) {
61
- bbox = opts.bounds
67
+ bbox = opts.bounds.clone().expandBy(padding)
62
68
  } else {
63
- for (const { id } of renderingShapes) {
64
- const maskedPageBounds = editor.getShapeMaskedPageBounds(id)
65
- if (!maskedPageBounds) continue
66
- if (bbox) {
67
- bbox.union(maskedPageBounds)
68
- } else {
69
- bbox = maskedPageBounds.clone()
70
- }
71
- }
69
+ bbox = getExportDefaultBounds(editor, renderingShapes, padding, singleFrameShapeId)
72
70
  }
73
71
 
74
72
  // no unmasked shapes to export
75
73
  if (!bbox) return
76
74
 
77
- const singleFrameShapeId =
78
- ids.length === 1 && editor.isShapeOfType<TLFrameShape>(editor.getShape(ids[0])!, 'frame')
79
- ? ids[0]
80
- : null
81
- if (!singleFrameShapeId) {
82
- // Expand by an extra 32 pixels
83
- bbox.expandBy(padding)
84
- }
85
-
86
75
  // We want the svg image to be BIGGER THAN USUAL to account for image quality
87
76
  const w = bbox.width * scale
88
77
  const h = bbox.height * scale
@@ -119,6 +108,75 @@ export function getSvgJsx(editor: Editor, ids: TLShapeId[], opts: TLImageExportO
119
108
  return { jsx: svg, width: w, height: h, exportDelay }
120
109
  }
121
110
 
111
+ /**
112
+ * Calculates the default bounds for an SVG export. This function handles:
113
+ * 1. Computing masked page bounds for each shape
114
+ * 2. Container logic: if a shape is marked as an export bounds container and it
115
+ * contains all other shapes, use its bounds and skip padding
116
+ * 3. Otherwise, create a union of all shape bounds and apply padding
117
+ *
118
+ * The container logic is useful for cases like annotating on an image - if the image
119
+ * contains all annotations, we want to export exactly the image bounds without extra padding.
120
+ *
121
+ * @param editor - The editor instance
122
+ * @param renderingShapes - The shapes to include in the export
123
+ * @param padding - Padding to add around the bounds (only applied if no container bounds)
124
+ * @param singleFrameShapeId - If exporting a single frame, this is its ID (skips padding)
125
+ * @returns The calculated bounds box, or null if no shapes to export
126
+ */
127
+ export function getExportDefaultBounds(
128
+ editor: Editor,
129
+ renderingShapes: TLRenderingShape[],
130
+ padding: number,
131
+ singleFrameShapeId: TLShapeId | null
132
+ ) {
133
+ let isBoundedByContainer = false
134
+ let bbox: null | Box = null
135
+
136
+ for (const { id } of renderingShapes) {
137
+ const maskedPageBounds = editor.getShapeMaskedPageBounds(id)
138
+ if (!maskedPageBounds) continue
139
+
140
+ // Check if this shape is an export bounds container (e.g., an image being annotated)
141
+ const shape = editor.getShape(id)!
142
+ const isContainer = editor.getShapeUtil(shape).isExportBoundsContainer(shape)
143
+
144
+ if (bbox) {
145
+ // Container logic: if this is a container and it contains all shapes processed so far,
146
+ // use the container's bounds instead of the union. This prevents extra padding around
147
+ // things like annotated images.
148
+ if (isContainer && Box.ContainsApproximately(maskedPageBounds, bbox)) {
149
+ isBoundedByContainer = true
150
+ bbox = maskedPageBounds.clone()
151
+ } else {
152
+ // If we were previously bounded by a container but this shape extends outside it,
153
+ // we're no longer bounded by a container
154
+ if (isBoundedByContainer && !Box.ContainsApproximately(bbox, maskedPageBounds)) {
155
+ isBoundedByContainer = false
156
+ }
157
+ // Expand the bounding box to include this shape
158
+ bbox.union(maskedPageBounds)
159
+ }
160
+ } else {
161
+ // First shape sets the initial bounds
162
+ isBoundedByContainer = isContainer
163
+ bbox = maskedPageBounds.clone()
164
+ }
165
+ }
166
+
167
+ // No unmasked shapes to export
168
+ if (!bbox) return null
169
+
170
+ // Only apply padding if:
171
+ // - Not exporting a single frame (frames have their own padding rules)
172
+ // - Not bounded by a container (containers define their own bounds precisely)
173
+ if (!singleFrameShapeId && !isBoundedByContainer) {
174
+ bbox.expandBy(padding)
175
+ }
176
+
177
+ return bbox
178
+ }
179
+
122
180
  function SvgExport({
123
181
  editor,
124
182
  preserveAspectRatio,
@@ -373,8 +431,7 @@ function SvgExport({
373
431
  | { options: { showColors: boolean } }
374
432
  if (frameShapeUtil?.options.showColors) {
375
433
  const shape = editor.getShape(singleFrameShapeId)! as TLFrameShape
376
- const color = theme[shape.props.color]
377
- backgroundColor = color.frame.fill
434
+ backgroundColor = getColorValue(theme, shape.props.color, 'frameFill')
378
435
  } else {
379
436
  backgroundColor = theme.solid
380
437
  }
@@ -1,5 +1,5 @@
1
1
  import { useValue } from '@tldraw/state-react'
2
- import React, { useMemo } from 'react'
2
+ import React, { useEffect, useMemo } from 'react'
3
3
  import { RIGHT_MOUSE_BUTTON } from '../constants'
4
4
  import {
5
5
  preventDefault,
@@ -16,9 +16,6 @@ export function useCanvasEvents() {
16
16
 
17
17
  const events = useMemo(
18
18
  function canvasEvents() {
19
- // Track the last screen point
20
- let lastX: number, lastY: number
21
-
22
19
  function onPointerDown(e: React.PointerEvent) {
23
20
  if ((e as any).isKilled) return
24
21
 
@@ -44,35 +41,9 @@ export function useCanvasEvents() {
44
41
  })
45
42
  }
46
43
 
47
- function onPointerMove(e: React.PointerEvent) {
48
- if ((e as any).isKilled) return
49
-
50
- if (e.clientX === lastX && e.clientY === lastY) return
51
- lastX = e.clientX
52
- lastY = e.clientY
53
-
54
- // For tools that benefit from a higher fidelity of events,
55
- // we dispatch the coalesced events.
56
- // N.B. Sometimes getCoalescedEvents isn't present on iOS, ugh.
57
- const events =
58
- currentTool.useCoalescedEvents && e.nativeEvent.getCoalescedEvents
59
- ? e.nativeEvent.getCoalescedEvents()
60
- : [e]
61
- for (const singleEvent of events) {
62
- editor.dispatch({
63
- type: 'pointer',
64
- target: 'canvas',
65
- name: 'pointer_move',
66
- ...getPointerInfo(singleEvent),
67
- })
68
- }
69
- }
70
-
71
44
  function onPointerUp(e: React.PointerEvent) {
72
45
  if ((e as any).isKilled) return
73
46
  if (e.button !== 0 && e.button !== 1 && e.button !== 2 && e.button !== 5) return
74
- lastX = e.clientX
75
- lastY = e.clientY
76
47
 
77
48
  releasePointerCapture(e.currentTarget, e)
78
49
 
@@ -108,15 +79,15 @@ export function useCanvasEvents() {
108
79
  // check that e.target is an HTMLElement
109
80
  if (!(e.target instanceof HTMLElement)) return
110
81
 
82
+ const editingShapeId = editor.getEditingShape()?.id
111
83
  if (
84
+ // if the target is not inside the editing shape
85
+ !(editingShapeId && e.target.closest(`[data-shape-id="${editingShapeId}"]`)) &&
86
+ // and the target is not an clickable element
112
87
  e.target.tagName !== 'A' &&
88
+ // or a TextArea.tsx ?
113
89
  e.target.tagName !== 'TEXTAREA' &&
114
- !e.target.isContentEditable &&
115
- // When in EditingShape state, we are actually clicking on a 'DIV'
116
- // not A/TEXTAREA/contenteditable element yet. So, to preserve cursor position
117
- // for edit mode on mobile we need to not preventDefault.
118
- // TODO: Find out if we still need this preventDefault in general though.
119
- !(editor.getEditingShape() && e.target.className.includes('tl-text-content'))
90
+ !e.target.isContentEditable
120
91
  ) {
121
92
  preventDefault(e)
122
93
  }
@@ -158,7 +129,6 @@ export function useCanvasEvents() {
158
129
 
159
130
  return {
160
131
  onPointerDown,
161
- onPointerMove,
162
132
  onPointerUp,
163
133
  onPointerEnter,
164
134
  onPointerLeave,
@@ -169,8 +139,45 @@ export function useCanvasEvents() {
169
139
  onClick,
170
140
  }
171
141
  },
172
- [editor, currentTool]
142
+ [editor]
173
143
  )
174
144
 
145
+ // onPointerMove is special: where we're only interested in the other events when they're
146
+ // happening _on_ the canvas (as opposed to outside of it, or on UI floating over it), we want
147
+ // the pointer position to be up to date regardless of whether it's over the tldraw canvas or
148
+ // not. So instead of returning a listener to be attached to the canvas, we directly attach a
149
+ // listener to the whole document instead.
150
+ useEffect(() => {
151
+ let lastX: number, lastY: number
152
+
153
+ function onPointerMove(e: PointerEvent) {
154
+ if ((e as any).isKilled) return
155
+ ;(e as any).isKilled = true
156
+
157
+ if (e.clientX === lastX && e.clientY === lastY) return
158
+ lastX = e.clientX
159
+ lastY = e.clientY
160
+
161
+ // For tools that benefit from a higher fidelity of events,
162
+ // we dispatch the coalesced events.
163
+ // N.B. Sometimes getCoalescedEvents isn't present on iOS, ugh.
164
+ const events =
165
+ currentTool.useCoalescedEvents && e.getCoalescedEvents ? e.getCoalescedEvents() : [e]
166
+ for (const singleEvent of events) {
167
+ editor.dispatch({
168
+ type: 'pointer',
169
+ target: 'canvas',
170
+ name: 'pointer_move',
171
+ ...getPointerInfo(singleEvent),
172
+ })
173
+ }
174
+ }
175
+
176
+ document.body.addEventListener('pointermove', onPointerMove)
177
+ return () => {
178
+ document.body.removeEventListener('pointermove', onPointerMove)
179
+ }
180
+ }, [editor, currentTool])
181
+
175
182
  return events
176
183
  }
@@ -1,14 +1,17 @@
1
1
  import { RefObject, useEffect } from 'react'
2
2
  import { preventDefault } from '../utils/dom'
3
3
  import { useContainer } from './useContainer'
4
+ import { useMaybeEditor } from './useEditor'
4
5
 
5
6
  /** @public */
6
7
  export function usePassThroughMouseOverEvents(ref: RefObject<HTMLElement>) {
7
8
  if (!ref) throw Error('usePassThroughWheelEvents must be passed a ref')
8
9
  const container = useContainer()
10
+ const editor = useMaybeEditor()
9
11
 
10
12
  useEffect(() => {
11
13
  function onMouseOver(e: MouseEvent) {
14
+ if (!editor?.getInstanceState().isFocused) return
12
15
  if ((e as any).isSpecialRedispatchedEvent) return
13
16
  preventDefault(e)
14
17
  const cvs = container.querySelector('.tl-canvas')
@@ -25,5 +28,5 @@ export function usePassThroughMouseOverEvents(ref: RefObject<HTMLElement>) {
25
28
  return () => {
26
29
  elm.removeEventListener('mouseover', onMouseOver)
27
30
  }
28
- }, [container, ref])
31
+ }, [container, editor, ref])
29
32
  }
@@ -1,14 +1,19 @@
1
1
  import { RefObject, useEffect } from 'react'
2
2
  import { preventDefault } from '../utils/dom'
3
3
  import { useContainer } from './useContainer'
4
+ import { useMaybeEditor } from './useEditor'
4
5
 
5
6
  /** @public */
6
7
  export function usePassThroughWheelEvents(ref: RefObject<HTMLElement>) {
7
8
  if (!ref) throw Error('usePassThroughWheelEvents must be passed a ref')
8
9
  const container = useContainer()
10
+ const editor = useMaybeEditor()
9
11
 
10
12
  useEffect(() => {
11
13
  function onWheel(e: WheelEvent) {
14
+ // Only pass through wheel events if the editor is focused
15
+ if (!editor?.getInstanceState().isFocused) return
16
+
12
17
  if ((e as any).isSpecialRedispatchedEvent) return
13
18
 
14
19
  // if the element is scrollable, don't redispatch the event
@@ -32,5 +37,5 @@ export function usePassThroughWheelEvents(ref: RefObject<HTMLElement>) {
32
37
  return () => {
33
38
  elm.removeEventListener('wheel', onWheel)
34
39
  }
35
- }, [container, ref])
40
+ }, [container, editor, ref])
36
41
  }