@tldraw/editor 3.14.0-canary.49bb54bcf937 → 3.14.0-canary.4c533b76dc35

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 (122) hide show
  1. package/dist-cjs/index.d.ts +0 -17
  2. package/dist-cjs/index.js +8 -8
  3. package/dist-cjs/index.js.map +2 -2
  4. package/dist-cjs/lib/editor/Editor.js +19 -53
  5. package/dist-cjs/lib/editor/Editor.js.map +2 -2
  6. package/dist-cjs/lib/editor/derivations/bindingsIndex.js +22 -22
  7. package/dist-cjs/lib/editor/derivations/bindingsIndex.js.map +2 -2
  8. package/dist-cjs/lib/editor/derivations/parentsToChildren.js +16 -16
  9. package/dist-cjs/lib/editor/derivations/parentsToChildren.js.map +2 -2
  10. package/dist-cjs/lib/editor/managers/{ClickManager.js → ClickManager/ClickManager.js} +1 -1
  11. package/dist-cjs/lib/editor/managers/ClickManager/ClickManager.js.map +7 -0
  12. package/dist-cjs/lib/editor/managers/{EdgeScrollManager.js → EdgeScrollManager/EdgeScrollManager.js} +2 -2
  13. package/dist-cjs/lib/editor/managers/EdgeScrollManager/EdgeScrollManager.js.map +7 -0
  14. package/dist-cjs/lib/editor/managers/FocusManager/FocusManager.js.map +7 -0
  15. package/dist-cjs/lib/editor/managers/FontManager/FontManager.js.map +7 -0
  16. package/dist-cjs/lib/editor/managers/{HistoryManager.js → HistoryManager/HistoryManager.js} +64 -6
  17. package/dist-cjs/lib/editor/managers/HistoryManager/HistoryManager.js.map +7 -0
  18. package/dist-cjs/lib/editor/managers/{ScribbleManager.js → ScribbleManager/ScribbleManager.js} +1 -1
  19. package/dist-cjs/lib/editor/managers/ScribbleManager/ScribbleManager.js.map +7 -0
  20. package/dist-cjs/lib/editor/managers/TextManager/TextManager.js.map +7 -0
  21. package/dist-cjs/lib/editor/managers/{TickManager.js → TickManager/TickManager.js} +1 -1
  22. package/dist-cjs/lib/editor/managers/TickManager/TickManager.js.map +7 -0
  23. package/dist-cjs/lib/editor/managers/{UserPreferencesManager.js → UserPreferencesManager/UserPreferencesManager.js} +1 -1
  24. package/dist-cjs/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.js.map +7 -0
  25. package/dist-cjs/lib/editor/shapes/ShapeUtil.js.map +1 -1
  26. package/dist-cjs/lib/exports/getSvgJsx.js.map +1 -1
  27. package/dist-cjs/lib/utils/reorderShapes.js +11 -10
  28. package/dist-cjs/lib/utils/reorderShapes.js.map +2 -2
  29. package/dist-cjs/lib/utils/richText.js.map +1 -1
  30. package/dist-cjs/version.js +3 -3
  31. package/dist-cjs/version.js.map +1 -1
  32. package/dist-esm/index.d.mts +0 -17
  33. package/dist-esm/index.mjs +12 -8
  34. package/dist-esm/index.mjs.map +2 -2
  35. package/dist-esm/lib/editor/Editor.mjs +19 -53
  36. package/dist-esm/lib/editor/Editor.mjs.map +2 -2
  37. package/dist-esm/lib/editor/derivations/bindingsIndex.mjs +22 -22
  38. package/dist-esm/lib/editor/derivations/bindingsIndex.mjs.map +2 -2
  39. package/dist-esm/lib/editor/derivations/parentsToChildren.mjs +16 -16
  40. package/dist-esm/lib/editor/derivations/parentsToChildren.mjs.map +2 -2
  41. package/dist-esm/lib/editor/managers/{ClickManager.mjs → ClickManager/ClickManager.mjs} +1 -1
  42. package/dist-esm/lib/editor/managers/ClickManager/ClickManager.mjs.map +7 -0
  43. package/dist-esm/lib/editor/managers/{EdgeScrollManager.mjs → EdgeScrollManager/EdgeScrollManager.mjs} +2 -2
  44. package/dist-esm/lib/editor/managers/EdgeScrollManager/EdgeScrollManager.mjs.map +7 -0
  45. package/dist-esm/lib/editor/managers/FocusManager/FocusManager.mjs.map +7 -0
  46. package/dist-esm/lib/editor/managers/FontManager/FontManager.mjs.map +7 -0
  47. package/dist-esm/lib/editor/managers/{HistoryManager.mjs → HistoryManager/HistoryManager.mjs} +60 -2
  48. package/dist-esm/lib/editor/managers/HistoryManager/HistoryManager.mjs.map +7 -0
  49. package/dist-esm/lib/editor/managers/{ScribbleManager.mjs → ScribbleManager/ScribbleManager.mjs} +1 -1
  50. package/dist-esm/lib/editor/managers/ScribbleManager/ScribbleManager.mjs.map +7 -0
  51. package/dist-esm/lib/editor/managers/TextManager/TextManager.mjs.map +7 -0
  52. package/dist-esm/lib/editor/managers/{TickManager.mjs → TickManager/TickManager.mjs} +1 -1
  53. package/dist-esm/lib/editor/managers/TickManager/TickManager.mjs.map +7 -0
  54. package/dist-esm/lib/editor/managers/{UserPreferencesManager.mjs → UserPreferencesManager/UserPreferencesManager.mjs} +1 -1
  55. package/dist-esm/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.mjs.map +7 -0
  56. package/dist-esm/lib/editor/shapes/ShapeUtil.mjs.map +1 -1
  57. package/dist-esm/lib/exports/getSvgJsx.mjs.map +1 -1
  58. package/dist-esm/lib/utils/reorderShapes.mjs +11 -10
  59. package/dist-esm/lib/utils/reorderShapes.mjs.map +2 -2
  60. package/dist-esm/lib/utils/richText.mjs.map +1 -1
  61. package/dist-esm/version.mjs +3 -3
  62. package/dist-esm/version.mjs.map +1 -1
  63. package/package.json +7 -7
  64. package/src/index.ts +13 -7
  65. package/src/lib/editor/Editor.test.ts +252 -3
  66. package/src/lib/editor/Editor.ts +22 -55
  67. package/src/lib/editor/derivations/bindingsIndex.ts +27 -26
  68. package/src/lib/editor/derivations/parentsToChildren.ts +28 -25
  69. package/src/lib/editor/managers/ClickManager/ClickManager.test.ts +442 -0
  70. package/src/lib/editor/managers/{ClickManager.ts → ClickManager/ClickManager.ts} +3 -3
  71. package/src/lib/editor/managers/EdgeScrollManager/EdgeScrollManager.test.ts +374 -0
  72. package/src/lib/editor/managers/{EdgeScrollManager.ts → EdgeScrollManager/EdgeScrollManager.ts} +3 -3
  73. package/src/lib/editor/managers/FocusManager/FocusManager.test.ts +455 -0
  74. package/src/lib/editor/managers/{FocusManager.ts → FocusManager/FocusManager.ts} +1 -1
  75. package/src/lib/editor/managers/FontManager/FontManager.test.ts +263 -0
  76. package/src/lib/editor/managers/{FontManager.ts → FontManager/FontManager.ts} +1 -1
  77. package/src/lib/editor/managers/{HistoryManager.test.ts → HistoryManager/HistoryManager.test.ts} +388 -1
  78. package/src/lib/editor/managers/{HistoryManager.ts → HistoryManager/HistoryManager.ts} +73 -2
  79. package/src/lib/editor/managers/ScribbleManager/ScribbleManager.test.ts +624 -0
  80. package/src/lib/editor/managers/{ScribbleManager.ts → ScribbleManager/ScribbleManager.ts} +2 -2
  81. package/src/lib/editor/managers/SnapManager/SnapManager.test.ts +485 -0
  82. package/src/lib/editor/managers/TextManager/TextManager.test.ts +411 -0
  83. package/src/lib/editor/managers/{TextManager.ts → TextManager/TextManager.ts} +1 -1
  84. package/src/lib/editor/managers/TickManager/TickManager.test.ts +314 -0
  85. package/src/lib/editor/managers/{TickManager.ts → TickManager/TickManager.ts} +2 -2
  86. package/src/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.test.ts +591 -0
  87. package/src/lib/editor/managers/{UserPreferencesManager.ts → UserPreferencesManager/UserPreferencesManager.ts} +2 -2
  88. package/src/lib/editor/shapes/ShapeUtil.ts +1 -1
  89. package/src/lib/exports/getSvgJsx.tsx +1 -1
  90. package/src/lib/utils/reorderShapes.ts +10 -13
  91. package/src/lib/utils/richText.ts +1 -1
  92. package/src/version.ts +3 -3
  93. package/dist-cjs/lib/editor/managers/ClickManager.js.map +0 -7
  94. package/dist-cjs/lib/editor/managers/EdgeScrollManager.js.map +0 -7
  95. package/dist-cjs/lib/editor/managers/FocusManager.js.map +0 -7
  96. package/dist-cjs/lib/editor/managers/FontManager.js.map +0 -7
  97. package/dist-cjs/lib/editor/managers/HistoryManager.js.map +0 -7
  98. package/dist-cjs/lib/editor/managers/ScribbleManager.js.map +0 -7
  99. package/dist-cjs/lib/editor/managers/Stack.js +0 -82
  100. package/dist-cjs/lib/editor/managers/Stack.js.map +0 -7
  101. package/dist-cjs/lib/editor/managers/TextManager.js.map +0 -7
  102. package/dist-cjs/lib/editor/managers/TickManager.js.map +0 -7
  103. package/dist-cjs/lib/editor/managers/UserPreferencesManager.js.map +0 -7
  104. package/dist-esm/lib/editor/managers/ClickManager.mjs.map +0 -7
  105. package/dist-esm/lib/editor/managers/EdgeScrollManager.mjs.map +0 -7
  106. package/dist-esm/lib/editor/managers/FocusManager.mjs.map +0 -7
  107. package/dist-esm/lib/editor/managers/FontManager.mjs.map +0 -7
  108. package/dist-esm/lib/editor/managers/HistoryManager.mjs.map +0 -7
  109. package/dist-esm/lib/editor/managers/ScribbleManager.mjs.map +0 -7
  110. package/dist-esm/lib/editor/managers/Stack.mjs +0 -62
  111. package/dist-esm/lib/editor/managers/Stack.mjs.map +0 -7
  112. package/dist-esm/lib/editor/managers/TextManager.mjs.map +0 -7
  113. package/dist-esm/lib/editor/managers/TickManager.mjs.map +0 -7
  114. package/dist-esm/lib/editor/managers/UserPreferencesManager.mjs.map +0 -7
  115. package/src/lib/editor/managers/ScribbleManager.test.ts +0 -32
  116. package/src/lib/editor/managers/Stack.ts +0 -71
  117. /package/dist-cjs/lib/editor/managers/{FocusManager.js → FocusManager/FocusManager.js} +0 -0
  118. /package/dist-cjs/lib/editor/managers/{FontManager.js → FontManager/FontManager.js} +0 -0
  119. /package/dist-cjs/lib/editor/managers/{TextManager.js → TextManager/TextManager.js} +0 -0
  120. /package/dist-esm/lib/editor/managers/{FocusManager.mjs → FocusManager/FocusManager.mjs} +0 -0
  121. /package/dist-esm/lib/editor/managers/{FontManager.mjs → FontManager/FontManager.mjs} +0 -0
  122. /package/dist-esm/lib/editor/managers/{TextManager.mjs → TextManager/TextManager.mjs} +0 -0
@@ -17,6 +17,7 @@ type ICustomShape = TLBaseShape<
17
17
  w: number
18
18
  h: number
19
19
  text: string | undefined
20
+ isFilled: boolean
20
21
  }
21
22
  >
22
23
 
@@ -26,19 +27,21 @@ class CustomShape extends ShapeUtil<ICustomShape> {
26
27
  w: T.number,
27
28
  h: T.number,
28
29
  text: T.string.optional(),
30
+ isFilled: T.boolean,
29
31
  }
30
32
  getDefaultProps(): ICustomShape['props'] {
31
33
  return {
32
34
  w: 200,
33
35
  h: 200,
34
36
  text: '',
37
+ isFilled: false,
35
38
  }
36
39
  }
37
40
  getGeometry(shape: ICustomShape): Geometry2d {
38
41
  return new Rectangle2d({
39
42
  width: shape.props.w,
40
43
  height: shape.props.h,
41
- isFilled: true,
44
+ isFilled: shape.props.isFilled,
42
45
  })
43
46
  }
44
47
  indicator() {}
@@ -81,11 +84,11 @@ describe('updateShape', () => {
81
84
  props: { w: 100, h: 100, text: 'Hello' },
82
85
  })
83
86
  const shape = editor.getShape(id) as ICustomShape
84
- expect(shape.props).toEqual({ w: 100, h: 100, text: 'Hello' })
87
+ expect(shape.props).toEqual({ w: 100, h: 100, text: 'Hello', isFilled: false })
85
88
 
86
89
  editor.updateShape({ ...shape, props: { ...shape.props, text: undefined } })
87
90
  const updatedShape = editor.getShape(id) as ICustomShape
88
- expect(updatedShape.props).toEqual({ w: 100, h: 100, text: undefined })
91
+ expect(updatedShape.props).toEqual({ w: 100, h: 100, text: undefined, isFilled: false })
89
92
  })
90
93
  })
91
94
 
@@ -176,3 +179,249 @@ describe('zoomToBounds', () => {
176
179
  expect(editor.setCamera).toHaveBeenCalled()
177
180
  })
178
181
  })
182
+
183
+ describe('getShapesAtPoint', () => {
184
+ const ids = {
185
+ shape1: createShapeId('shape1'),
186
+ shape2: createShapeId('shape2'),
187
+ shape3: createShapeId('shape3'),
188
+ shape4: createShapeId('shape4'),
189
+ shape5: createShapeId('shape5'),
190
+ overlap1: createShapeId('overlap1'),
191
+ overlap2: createShapeId('overlap2'),
192
+ filledShape: createShapeId('filledShape'),
193
+ hollowShape: createShapeId('hollowShape'),
194
+ hiddenShape: createShapeId('hiddenShape'),
195
+ }
196
+
197
+ beforeEach(() => {
198
+ // Create test shapes with different z-index positions
199
+ // Shape 1: Bottom layer, large square
200
+ editor.createShape({
201
+ id: ids.shape1,
202
+ type: 'my-custom-shape',
203
+ x: 0,
204
+ y: 0,
205
+ props: { w: 200, h: 200, text: 'Bottom' },
206
+ })
207
+
208
+ // Shape 2: Middle layer, overlapping square
209
+ editor.createShape({
210
+ id: ids.shape2,
211
+ type: 'my-custom-shape',
212
+ x: 100,
213
+ y: 0,
214
+ props: { w: 200, h: 200, text: 'Middle' },
215
+ })
216
+
217
+ // Shape 3: Top layer, small square
218
+ editor.createShape({
219
+ id: ids.shape3,
220
+ type: 'my-custom-shape',
221
+ x: 50,
222
+ y: 50,
223
+ props: { w: 100, h: 100, text: 'Top' },
224
+ })
225
+
226
+ // Shape 4: Separate area, no overlap
227
+ editor.createShape({
228
+ id: ids.shape4,
229
+ type: 'my-custom-shape',
230
+ x: 50,
231
+ y: 100,
232
+ props: { w: 100, h: 100, text: 'Separate' },
233
+ })
234
+ })
235
+
236
+ it('returns shapes at a point in order of their index', () => {
237
+ // Point at (50, 50) should hit shape3's edge (since it's at 50,50 with size 100x100)
238
+ // This point is exactly at the top-left corner of shape3
239
+ const shapes = editor.getShapesAtPoint({ x: 50, y: 50 })
240
+ const shapeIds = shapes.map((s) => s.id)
241
+
242
+ expect(shapeIds).toEqual([ids.shape3])
243
+ expect(shapes).toHaveLength(1)
244
+ })
245
+
246
+ it('returns empty array when no shapes at point', () => {
247
+ const shapes = editor.getShapesAtPoint({ x: 1000, y: 1000 })
248
+ expect(shapes).toEqual([])
249
+ })
250
+
251
+ it('returns single shape when point hits only one shape', () => {
252
+ // Point at right edge of shape2 where it doesn't overlap with other shapes
253
+ // Shape2 is at (100,0) with size 200x200, so right edge is at x=300
254
+ const shapes = editor.getShapesAtPoint({ x: 300, y: 100 })
255
+ expect(shapes).toHaveLength(1)
256
+ expect(shapes[0].id).toBe(ids.shape2)
257
+ })
258
+
259
+ it('returns shapes on edge when point is exactly on boundary', () => {
260
+ // Point at exact edge of shape1
261
+ const shapes = editor.getShapesAtPoint({ x: 0, y: 0 })
262
+ expect(shapes).toHaveLength(1)
263
+ expect(shapes[0].id).toBe(ids.shape1)
264
+ })
265
+
266
+ it('respects hitInside option when false (default)', () => {
267
+ // Point inside shape1 (at 0,0 with size 200x200) but with hitInside false should not hit
268
+ const shapes = editor.getShapesAtPoint({ x: 25, y: 25 }, { hitInside: false })
269
+ expect(shapes).toEqual([])
270
+ })
271
+
272
+ it('respects hitInside option when true', () => {
273
+ // Point inside shape1 (at 0,0 with size 200x200) with hitInside true should hit
274
+ const shapes = editor.getShapesAtPoint({ x: 25, y: 25 }, { hitInside: true })
275
+ expect(shapes).toHaveLength(1)
276
+ expect(shapes[0].id).toBe(ids.shape1)
277
+ })
278
+
279
+ it('respects margin option', () => {
280
+ // Point slightly outside shape1 at bottom edge but within margin should hit only shape1
281
+ // Shape1 is at (0,0) with size 200x200, shape2 goes to (300,200) so avoid overlap at (200,200)
282
+ const shapes = editor.getShapesAtPoint({ x: 205, y: 100 }, { margin: 10 })
283
+ expect(shapes).toHaveLength(1)
284
+ expect(shapes[0].id).toBe(ids.shape1)
285
+ })
286
+
287
+ it('filters out hidden shapes', () => {
288
+ // Create a spy to mock isShapeHidden
289
+ const isShapeHiddenSpy = jest.spyOn(editor, 'isShapeHidden')
290
+ isShapeHiddenSpy.mockImplementation((shape) => {
291
+ return typeof shape === 'string' ? shape === ids.shape3 : shape.id === ids.shape3
292
+ })
293
+
294
+ const shapes = editor.getShapesAtPoint({ x: 50, y: 50 })
295
+ const shapeIds = shapes.map((s) => s.id)
296
+
297
+ // Should not include shape3 since it's hidden, and no other shapes are at this point
298
+ expect(shapeIds).toEqual([])
299
+ expect(shapes).toHaveLength(0)
300
+
301
+ isShapeHiddenSpy.mockRestore()
302
+ })
303
+
304
+ it('handles point exactly at shape corner', () => {
305
+ // Point at bottom-left corner of shape1 where it doesn't overlap with other shapes
306
+ const shapes = editor.getShapesAtPoint({ x: 0, y: 200 })
307
+ expect(shapes).toHaveLength(1)
308
+ expect(shapes[0].id).toBe(ids.shape1)
309
+ })
310
+
311
+ it('handles overlapping shapes with different hit areas', () => {
312
+ // Point that hits both shape1 and shape2 edges (they overlap at x=100,y=0)
313
+ const shapes = editor.getShapesAtPoint({ x: 100, y: 0 })
314
+ const shapeIds = shapes.map((s) => s.id)
315
+
316
+ // Both shapes should be detected at this overlapping point
317
+ expect(shapeIds).toEqual([ids.shape1, ids.shape2])
318
+ expect(shapes).toHaveLength(2)
319
+ })
320
+
321
+ it('maintains shape order from getCurrentPageShapesSorted', () => {
322
+ // Create filled shape that overlaps with shape2
323
+ editor.createShape({
324
+ id: ids.shape5,
325
+ type: 'my-custom-shape',
326
+ x: 110,
327
+ y: 110,
328
+ props: { w: 200, h: 200, isFilled: true, text: 'Shape5' },
329
+ })
330
+
331
+ // Test with hitInside to detect multiple shapes
332
+ // Point (120,120) will hit shape1, shape2, shape3, shape4, and shape5 with hitInside: true
333
+ const shapes = editor.getShapesAtPoint({ x: 120, y: 120 }, { hitInside: true })
334
+ const shapeIds = shapes.map((s) => s.id)
335
+
336
+ // All shapes that contain this point should be returned in z-index order
337
+ expect(shapeIds).toEqual([ids.shape1, ids.shape2, ids.shape3, ids.shape4, ids.shape5])
338
+
339
+ // After bringing shape2 to front, order should change
340
+ editor.bringToFront([ids.shape2])
341
+ const shapes2 = editor.getShapesAtPoint({ x: 120, y: 120 }, { hitInside: true })
342
+ const shapeIds2 = shapes2.map((s) => s.id)
343
+ expect(shapeIds2).toEqual([ids.shape1, ids.shape3, ids.shape4, ids.shape5, ids.shape2])
344
+ })
345
+
346
+ it('combines hitInside and margin options', () => {
347
+ // Point inside shape1 (at 0,0 with size 200x200) with hitInside and margin
348
+ const shapes = editor.getShapesAtPoint({ x: 25, y: 25 }, { hitInside: true, margin: 5 })
349
+ expect(shapes).toHaveLength(1)
350
+ expect(shapes[0].id).toBe(ids.shape1)
351
+ })
352
+
353
+ it('returns empty array when all shapes are hidden', () => {
354
+ // Mock all shapes as hidden
355
+ const isShapeHiddenSpy = jest.spyOn(editor, 'isShapeHidden')
356
+ isShapeHiddenSpy.mockReturnValue(true)
357
+
358
+ const shapes = editor.getShapesAtPoint({ x: 50, y: 50 })
359
+ expect(shapes).toEqual([])
360
+
361
+ isShapeHiddenSpy.mockRestore()
362
+ })
363
+
364
+ it('returns multiple shapes at same point in z-index order', () => {
365
+ // Create two shapes at exactly the same position (away from existing shapes)
366
+ editor.createShape({
367
+ id: ids.overlap1,
368
+ type: 'my-custom-shape',
369
+ x: 600,
370
+ y: 600,
371
+ props: { w: 100, h: 100, text: 'First' },
372
+ })
373
+
374
+ editor.createShape({
375
+ id: ids.overlap2,
376
+ type: 'my-custom-shape',
377
+ x: 600,
378
+ y: 600,
379
+ props: { w: 100, h: 100, text: 'Second' },
380
+ })
381
+
382
+ // Test at corner where both shapes' edges meet
383
+ const shapes = editor.getShapesAtPoint({ x: 600, y: 600 })
384
+ const shapeIds = shapes.map((s) => s.id)
385
+
386
+ // Should return both shapes in z-index order
387
+ expect(shapeIds).toEqual([ids.overlap1, ids.overlap2])
388
+ expect(shapes).toHaveLength(2)
389
+ })
390
+
391
+ it('respects isFilled property for hit detection', () => {
392
+ // Create a filled shape
393
+ editor.createShape({
394
+ id: ids.filledShape,
395
+ type: 'my-custom-shape',
396
+ x: 300,
397
+ y: 300,
398
+ props: { w: 100, h: 100, isFilled: true, text: 'Filled' },
399
+ })
400
+
401
+ // Create a hollow shape at the same position
402
+ editor.createShape({
403
+ id: ids.hollowShape,
404
+ type: 'my-custom-shape',
405
+ x: 400,
406
+ y: 300,
407
+ props: { w: 100, h: 100, isFilled: false, text: 'Hollow' },
408
+ })
409
+
410
+ // Test point inside filled shape - should hit without hitInside option
411
+ const filledShapes = editor.getShapesAtPoint({ x: 350, y: 350 })
412
+ expect(filledShapes).toHaveLength(1)
413
+ expect(filledShapes[0].id).toBe(ids.filledShape)
414
+
415
+ // Test point inside hollow shape - should not hit without hitInside option
416
+ const hollowShapes = editor.getShapesAtPoint({ x: 450, y: 350 })
417
+ expect(hollowShapes).toHaveLength(0)
418
+
419
+ // Test point inside hollow shape with hitInside - should hit
420
+ const hollowShapesWithHitInside = editor.getShapesAtPoint(
421
+ { x: 450, y: 350 },
422
+ { hitInside: true }
423
+ )
424
+ expect(hollowShapesWithHitInside).toHaveLength(1)
425
+ expect(hollowShapesWithHitInside[0].id).toBe(ids.hollowShape)
426
+ })
427
+ })
@@ -148,16 +148,16 @@ import { bindingsIndex } from './derivations/bindingsIndex'
148
148
  import { notVisibleShapes } from './derivations/notVisibleShapes'
149
149
  import { parentsToChildren } from './derivations/parentsToChildren'
150
150
  import { deriveShapeIdsInCurrentPage } from './derivations/shapeIdsInCurrentPage'
151
- import { ClickManager } from './managers/ClickManager'
152
- import { EdgeScrollManager } from './managers/EdgeScrollManager'
153
- import { FocusManager } from './managers/FocusManager'
154
- import { FontManager } from './managers/FontManager'
155
- import { HistoryManager } from './managers/HistoryManager'
156
- import { ScribbleManager } from './managers/ScribbleManager'
151
+ import { ClickManager } from './managers/ClickManager/ClickManager'
152
+ import { EdgeScrollManager } from './managers/EdgeScrollManager/EdgeScrollManager'
153
+ import { FocusManager } from './managers/FocusManager/FocusManager'
154
+ import { FontManager } from './managers/FontManager/FontManager'
155
+ import { HistoryManager } from './managers/HistoryManager/HistoryManager'
156
+ import { ScribbleManager } from './managers/ScribbleManager/ScribbleManager'
157
157
  import { SnapManager } from './managers/SnapManager/SnapManager'
158
- import { TextManager } from './managers/TextManager'
159
- import { TickManager } from './managers/TickManager'
160
- import { UserPreferencesManager } from './managers/UserPreferencesManager'
158
+ import { TextManager } from './managers/TextManager/TextManager'
159
+ import { TickManager } from './managers/TickManager/TickManager'
160
+ import { UserPreferencesManager } from './managers/UserPreferencesManager/UserPreferencesManager'
161
161
  import { ShapeUtil, TLGeometryOpts, TLResizeMode } from './shapes/ShapeUtil'
162
162
  import { RootState } from './tools/RootState'
163
163
  import { StateNode, TLStateNodeConstructor } from './tools/StateNode'
@@ -328,7 +328,7 @@ export class Editor extends EventEmitter<TLEventMap> {
328
328
  this.store = store
329
329
  this.history = new HistoryManager<TLRecord>({
330
330
  store,
331
- annotateError: (error) => {
331
+ annotateError: (error: any) => {
332
332
  this.annotateError(error, { origin: 'history.batch', willCrashApp: true })
333
333
  this.crash(error)
334
334
  },
@@ -4643,44 +4643,6 @@ export class Editor extends EventEmitter<TLEventMap> {
4643
4643
  )! as T
4644
4644
  }
4645
4645
 
4646
- private _shapePageGeometryCaches: Record<string, ComputedCache<Geometry2d, TLShape>> = {}
4647
-
4648
- /**
4649
- * Get the geometry of a shape in page-space.
4650
- *
4651
- * @example
4652
- * ```ts
4653
- * editor.getShapePageGeometry(myShape)
4654
- * editor.getShapePageGeometry(myShapeId)
4655
- * editor.getShapePageGeometry(myShapeId, { context: "arrow" })
4656
- * ```
4657
- *
4658
- * @param shape - The shape (or shape id) to get the geometry for.
4659
- * @param opts - Additional options about the request for geometry. Passed to {@link ShapeUtil.getGeometry}.
4660
- *
4661
- * @public
4662
- */
4663
- getShapePageGeometry<T extends Geometry2d>(shape: TLShape | TLShapeId, opts?: TLGeometryOpts): T {
4664
- const context = opts?.context ?? 'none'
4665
- if (!this._shapePageGeometryCaches[context]) {
4666
- this._shapePageGeometryCaches[context] = this.store.createComputedCache(
4667
- 'bounds',
4668
- (shape) => {
4669
- const geometry = this.getShapeGeometry(shape.id, opts)
4670
- const pageTransform = this.getShapePageTransform(shape.id)
4671
- return geometry.transform(pageTransform)
4672
- },
4673
- {
4674
- // we only depend directly on the shape id, and changing geometry/transform will update us anyway
4675
- areRecordsEqual: () => true,
4676
- }
4677
- )
4678
- }
4679
- return this._shapePageGeometryCaches[context].get(
4680
- typeof shape === 'string' ? shape : shape.id
4681
- )! as T
4682
- }
4683
-
4684
4646
  /** @internal */
4685
4647
  @computed private _getShapeHandlesCache(): ComputedCache<TLHandle[] | undefined, TLShape> {
4686
4648
  return this.store.createComputedCache(
@@ -4793,7 +4755,10 @@ export class Editor extends EventEmitter<TLEventMap> {
4793
4755
  /** @internal */
4794
4756
  @computed private _getShapePageBoundsCache(): ComputedCache<Box, TLShape> {
4795
4757
  return this.store.createComputedCache<Box, TLShape>('pageBoundsCache', (shape) => {
4796
- return this.getShapePageGeometry(shape).bounds
4758
+ const pageTransform = this.getShapePageTransform(shape)
4759
+ if (!pageTransform) return undefined
4760
+ const geometry = this.getShapeGeometry(shape)
4761
+ return Box.FromPoints(pageTransform.applyToPoints(geometry.vertices))
4797
4762
  })
4798
4763
  }
4799
4764
 
@@ -4867,11 +4832,12 @@ export class Editor extends EventEmitter<TLEventMap> {
4867
4832
  if (frameAncestors.length === 0) return undefined
4868
4833
 
4869
4834
  const pageMask = frameAncestors
4870
- .map<Vec[] | undefined>(
4871
- (s) =>
4872
- // Apply the frame transform to the frame outline to get the frame outline in the current page space
4873
- this.getShapePageGeometry(s.id).vertices
4874
- )
4835
+ .map<Vec[] | undefined>((s) => {
4836
+ // Apply the frame transform to the frame outline to get the frame outline in the current page space
4837
+ const geometry = this.getShapeGeometry(s.id)
4838
+ const pageTransform = this.getShapePageTransform(s.id)
4839
+ return pageTransform.applyToPoints(geometry.vertices)
4840
+ })
4875
4841
  .reduce((acc, b) => {
4876
4842
  if (!(b && acc)) return undefined
4877
4843
  const intersection = intersectPolygonPolygon(acc, b)
@@ -5351,7 +5317,7 @@ export class Editor extends EventEmitter<TLEventMap> {
5351
5317
  point: VecLike,
5352
5318
  opts = {} as { margin?: number; hitInside?: boolean }
5353
5319
  ): TLShape[] {
5354
- return this.getCurrentPageShapes().filter(
5320
+ return this.getCurrentPageShapesSorted().filter(
5355
5321
  (shape) => !this.isShapeHidden(shape) && this.isPointInShape(shape, point, opts)
5356
5322
  )
5357
5323
  }
@@ -9302,6 +9268,7 @@ export class Editor extends EventEmitter<TLEventMap> {
9302
9268
  if (rootShapes.length === 1) {
9303
9269
  const onlyRoot = rootShapes[0] as TLFrameShape
9304
9270
  // If the old bounds are in the viewport...
9271
+ // todo: replace frame references with shapes that can accept children
9305
9272
  if (this.isShapeOfType<TLFrameShape>(onlyRoot, 'frame')) {
9306
9273
  while (
9307
9274
  this.getShapesAtPoint(point).some(
@@ -1,41 +1,42 @@
1
1
  import { Computed, RESET_VALUE, computed, isUninitialized } from '@tldraw/state'
2
- import { TLBinding, TLShapeId } from '@tldraw/tlschema'
2
+ import { TLArrowBinding, TLBinding, TLShapeId, TLUnknownBinding } from '@tldraw/tlschema'
3
3
  import { objectMapValues } from '@tldraw/utils'
4
4
  import { Editor } from '../Editor'
5
5
 
6
6
  type TLBindingsIndex = Map<TLShapeId, TLBinding[]>
7
7
 
8
- export const bindingsIndex = (editor: Editor): Computed<TLBindingsIndex> => {
9
- const { store } = editor
10
- const bindingsHistory = store.query.filterHistory('binding')
11
- const bindingsQuery = store.query.records('binding')
12
- function fromScratch() {
13
- const allBindings = bindingsQuery.get() as TLBinding[]
8
+ function fromScratch(bindingsQuery: Computed<(TLArrowBinding | TLUnknownBinding)[], unknown>) {
9
+ const allBindings = bindingsQuery.get() as TLBinding[]
14
10
 
15
- const shape2Binding: TLBindingsIndex = new Map()
11
+ const shapesToBindings: TLBindingsIndex = new Map()
16
12
 
17
- for (const binding of allBindings) {
18
- const { fromId, toId } = binding
19
- const bindingsForFromShape = shape2Binding.get(fromId)
20
- if (!bindingsForFromShape) {
21
- shape2Binding.set(fromId, [binding])
22
- } else {
23
- bindingsForFromShape.push(binding)
24
- }
25
- const bindingsForToShape = shape2Binding.get(toId)
26
- if (!bindingsForToShape) {
27
- shape2Binding.set(toId, [binding])
28
- } else {
29
- bindingsForToShape.push(binding)
30
- }
13
+ for (const binding of allBindings) {
14
+ const { fromId, toId } = binding
15
+ const bindingsForFromShape = shapesToBindings.get(fromId)
16
+ if (!bindingsForFromShape) {
17
+ shapesToBindings.set(fromId, [binding])
18
+ } else {
19
+ bindingsForFromShape.push(binding)
20
+ }
21
+ const bindingsForToShape = shapesToBindings.get(toId)
22
+ if (!bindingsForToShape) {
23
+ shapesToBindings.set(toId, [binding])
24
+ } else {
25
+ bindingsForToShape.push(binding)
31
26
  }
32
-
33
- return shape2Binding
34
27
  }
35
28
 
29
+ return shapesToBindings
30
+ }
31
+
32
+ export const bindingsIndex = (editor: Editor): Computed<TLBindingsIndex> => {
33
+ const { store } = editor
34
+ const bindingsHistory = store.query.filterHistory('binding')
35
+ const bindingsQuery = store.query.records('binding')
36
+
36
37
  return computed<TLBindingsIndex>('arrowBindingsIndex', (_lastValue, lastComputedEpoch) => {
37
38
  if (isUninitialized(_lastValue)) {
38
- return fromScratch()
39
+ return fromScratch(bindingsQuery)
39
40
  }
40
41
 
41
42
  const lastValue = _lastValue
@@ -43,7 +44,7 @@ export const bindingsIndex = (editor: Editor): Computed<TLBindingsIndex> => {
43
44
  const diff = bindingsHistory.getDiffSince(lastComputedEpoch)
44
45
 
45
46
  if (diff === RESET_VALUE) {
46
- return fromScratch()
47
+ return fromScratch(bindingsQuery)
47
48
  }
48
49
 
49
50
  let nextValue: TLBindingsIndex | undefined = undefined
@@ -1,45 +1,48 @@
1
- import { computed, isUninitialized, RESET_VALUE } from '@tldraw/state'
2
- import { RecordsDiff } from '@tldraw/store'
1
+ import { Computed, computed, isUninitialized, RESET_VALUE } from '@tldraw/state'
2
+ import { CollectionDiff, RecordsDiff } from '@tldraw/store'
3
3
  import { isShape, TLParentId, TLRecord, TLShape, TLShapeId, TLStore } from '@tldraw/tlschema'
4
4
  import { compact, sortByIndex } from '@tldraw/utils'
5
5
 
6
- type Parents2Children = Record<TLParentId, TLShapeId[]>
6
+ type ParentShapeIdsToChildShapeIds = Record<TLParentId, TLShapeId[]>
7
7
 
8
- export const parentsToChildren = (store: TLStore) => {
9
- const shapeIdsQuery = store.query.ids<'shape'>('shape')
10
- const shapeHistory = store.query.filterHistory('shape')
8
+ function fromScratch(
9
+ shapeIdsQuery: Computed<Set<TLShapeId>, CollectionDiff<TLShapeId>>,
10
+ store: TLStore
11
+ ) {
12
+ const result: ParentShapeIdsToChildShapeIds = {}
13
+ const shapeIds = shapeIdsQuery.get()
14
+ const shapes = Array(shapeIds.size) as TLShape[]
15
+ shapeIds.forEach((id) => shapes.push(store.get(id)!))
11
16
 
12
- function fromScratch() {
13
- const result: Parents2Children = {}
14
- const shapeIds = shapeIdsQuery.get()
15
- const shapes = Array(shapeIds.size) as TLShape[]
16
- shapeIds.forEach((id) => shapes.push(store.get(id)!))
17
+ // Sort the shapes by index
18
+ shapes.sort(sortByIndex)
17
19
 
18
- // Sort the shapes by index
19
- shapes.sort(sortByIndex)
20
+ // Populate the result object with an array for each parent.
21
+ shapes.forEach((shape) => {
22
+ if (!result[shape.parentId]) {
23
+ result[shape.parentId] = []
24
+ }
25
+ result[shape.parentId].push(shape.id)
26
+ })
20
27
 
21
- // Populate the result object with an array for each parent.
22
- shapes.forEach((shape) => {
23
- if (!result[shape.parentId]) {
24
- result[shape.parentId] = []
25
- }
26
- result[shape.parentId].push(shape.id)
27
- })
28
+ return result
29
+ }
28
30
 
29
- return result
30
- }
31
+ export const parentsToChildren = (store: TLStore) => {
32
+ const shapeIdsQuery = store.query.ids<'shape'>('shape')
33
+ const shapeHistory = store.query.filterHistory('shape')
31
34
 
32
- return computed<Parents2Children>(
35
+ return computed<ParentShapeIdsToChildShapeIds>(
33
36
  'parentsToChildrenWithIndexes',
34
37
  (lastValue, lastComputedEpoch) => {
35
38
  if (isUninitialized(lastValue)) {
36
- return fromScratch()
39
+ return fromScratch(shapeIdsQuery, store)
37
40
  }
38
41
 
39
42
  const diff = shapeHistory.getDiffSince(lastComputedEpoch)
40
43
 
41
44
  if (diff === RESET_VALUE) {
42
- return fromScratch()
45
+ return fromScratch(shapeIdsQuery, store)
43
46
  }
44
47
 
45
48
  if (diff.length === 0) return lastValue