@tldraw/editor 3.13.0-canary.37377fbef46d → 3.13.0-canary.409fd05a1a56

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 (156) hide show
  1. package/dist-cjs/index.d.ts +124 -113
  2. package/dist-cjs/index.js +7 -22
  3. package/dist-cjs/index.js.map +2 -2
  4. package/dist-cjs/lib/components/Shape.js +12 -8
  5. package/dist-cjs/lib/components/Shape.js.map +2 -2
  6. package/dist-cjs/lib/components/default-components/DefaultCanvas.js +37 -8
  7. package/dist-cjs/lib/components/default-components/DefaultCanvas.js.map +2 -2
  8. package/dist-cjs/lib/components/default-components/DefaultShapeIndicator.js +17 -11
  9. package/dist-cjs/lib/components/default-components/DefaultShapeIndicator.js.map +2 -2
  10. package/dist-cjs/lib/editor/Editor.js +85 -24
  11. package/dist-cjs/lib/editor/Editor.js.map +2 -2
  12. package/dist-cjs/lib/editor/managers/SnapManager/HandleSnaps.js.map +2 -2
  13. package/dist-cjs/lib/editor/managers/TextManager.js +10 -0
  14. package/dist-cjs/lib/editor/managers/TextManager.js.map +2 -2
  15. package/dist-cjs/lib/editor/shapes/ShapeUtil.js +1 -1
  16. package/dist-cjs/lib/editor/shapes/ShapeUtil.js.map +2 -2
  17. package/dist-cjs/lib/editor/shapes/group/GroupShapeUtil.js +0 -3
  18. package/dist-cjs/lib/editor/shapes/group/GroupShapeUtil.js.map +2 -2
  19. package/dist-cjs/lib/editor/shapes/shared/getPerfectDashProps.js.map +2 -2
  20. package/dist-cjs/lib/exports/getSvgJsx.js +12 -3
  21. package/dist-cjs/lib/exports/getSvgJsx.js.map +2 -2
  22. package/dist-cjs/lib/hooks/useDocumentEvents.js +3 -2
  23. package/dist-cjs/lib/hooks/useDocumentEvents.js.map +2 -2
  24. package/dist-cjs/lib/hooks/useEditorComponents.js +16 -16
  25. package/dist-cjs/lib/hooks/useEditorComponents.js.map +2 -2
  26. package/dist-cjs/lib/primitives/Box.js +16 -0
  27. package/dist-cjs/lib/primitives/Box.js.map +2 -2
  28. package/dist-cjs/lib/primitives/Mat.js +1 -1
  29. package/dist-cjs/lib/primitives/Mat.js.map +2 -2
  30. package/dist-cjs/lib/primitives/Vec.js +20 -0
  31. package/dist-cjs/lib/primitives/Vec.js.map +2 -2
  32. package/dist-cjs/lib/primitives/geometry/Arc2d.js +2 -2
  33. package/dist-cjs/lib/primitives/geometry/Arc2d.js.map +2 -2
  34. package/dist-cjs/lib/primitives/geometry/Circle2d.js +1 -1
  35. package/dist-cjs/lib/primitives/geometry/Circle2d.js.map +2 -2
  36. package/dist-cjs/lib/primitives/geometry/CubicBezier2d.js +1 -1
  37. package/dist-cjs/lib/primitives/geometry/CubicBezier2d.js.map +2 -2
  38. package/dist-cjs/lib/primitives/geometry/CubicSpline2d.js.map +2 -2
  39. package/dist-cjs/lib/primitives/geometry/Edge2d.js +1 -1
  40. package/dist-cjs/lib/primitives/geometry/Edge2d.js.map +2 -2
  41. package/dist-cjs/lib/primitives/geometry/Ellipse2d.js.map +2 -2
  42. package/dist-cjs/lib/primitives/geometry/Geometry2d.js +91 -20
  43. package/dist-cjs/lib/primitives/geometry/Geometry2d.js.map +2 -2
  44. package/dist-cjs/lib/primitives/geometry/Group2d.js +55 -2
  45. package/dist-cjs/lib/primitives/geometry/Group2d.js.map +2 -2
  46. package/dist-cjs/lib/primitives/geometry/Point2d.js.map +2 -2
  47. package/dist-cjs/lib/primitives/geometry/Polyline2d.js.map +2 -2
  48. package/dist-cjs/lib/primitives/geometry/Stadium2d.js.map +2 -2
  49. package/dist-cjs/lib/utils/areShapesContentEqual.js +25 -0
  50. package/dist-cjs/lib/utils/areShapesContentEqual.js.map +7 -0
  51. package/dist-cjs/lib/utils/debug-flags.js +5 -2
  52. package/dist-cjs/lib/utils/debug-flags.js.map +2 -2
  53. package/dist-cjs/lib/utils/nearestMultiple.js +34 -0
  54. package/dist-cjs/lib/utils/nearestMultiple.js.map +7 -0
  55. package/dist-cjs/lib/utils/rotation.js +5 -5
  56. package/dist-cjs/lib/utils/rotation.js.map +2 -2
  57. package/dist-cjs/version.js +3 -3
  58. package/dist-cjs/version.js.map +1 -1
  59. package/dist-esm/index.d.mts +124 -113
  60. package/dist-esm/index.mjs +9 -41
  61. package/dist-esm/index.mjs.map +2 -2
  62. package/dist-esm/lib/components/Shape.mjs +12 -8
  63. package/dist-esm/lib/components/Shape.mjs.map +2 -2
  64. package/dist-esm/lib/components/default-components/DefaultCanvas.mjs +37 -8
  65. package/dist-esm/lib/components/default-components/DefaultCanvas.mjs.map +2 -2
  66. package/dist-esm/lib/components/default-components/DefaultShapeIndicator.mjs +17 -11
  67. package/dist-esm/lib/components/default-components/DefaultShapeIndicator.mjs.map +2 -2
  68. package/dist-esm/lib/editor/Editor.mjs +85 -24
  69. package/dist-esm/lib/editor/Editor.mjs.map +2 -2
  70. package/dist-esm/lib/editor/managers/SnapManager/HandleSnaps.mjs.map +2 -2
  71. package/dist-esm/lib/editor/managers/TextManager.mjs +10 -0
  72. package/dist-esm/lib/editor/managers/TextManager.mjs.map +2 -2
  73. package/dist-esm/lib/editor/shapes/ShapeUtil.mjs +1 -1
  74. package/dist-esm/lib/editor/shapes/ShapeUtil.mjs.map +2 -2
  75. package/dist-esm/lib/editor/shapes/group/GroupShapeUtil.mjs +0 -3
  76. package/dist-esm/lib/editor/shapes/group/GroupShapeUtil.mjs.map +2 -2
  77. package/dist-esm/lib/editor/shapes/shared/getPerfectDashProps.mjs.map +2 -2
  78. package/dist-esm/lib/exports/getSvgJsx.mjs +12 -3
  79. package/dist-esm/lib/exports/getSvgJsx.mjs.map +2 -2
  80. package/dist-esm/lib/hooks/useDocumentEvents.mjs +3 -2
  81. package/dist-esm/lib/hooks/useDocumentEvents.mjs.map +2 -2
  82. package/dist-esm/lib/hooks/useEditorComponents.mjs +16 -18
  83. package/dist-esm/lib/hooks/useEditorComponents.mjs.map +2 -2
  84. package/dist-esm/lib/primitives/Box.mjs +16 -0
  85. package/dist-esm/lib/primitives/Box.mjs.map +2 -2
  86. package/dist-esm/lib/primitives/Mat.mjs +1 -1
  87. package/dist-esm/lib/primitives/Mat.mjs.map +2 -2
  88. package/dist-esm/lib/primitives/Vec.mjs +20 -0
  89. package/dist-esm/lib/primitives/Vec.mjs.map +2 -2
  90. package/dist-esm/lib/primitives/geometry/Arc2d.mjs +2 -2
  91. package/dist-esm/lib/primitives/geometry/Arc2d.mjs.map +2 -2
  92. package/dist-esm/lib/primitives/geometry/Circle2d.mjs +1 -1
  93. package/dist-esm/lib/primitives/geometry/Circle2d.mjs.map +2 -2
  94. package/dist-esm/lib/primitives/geometry/CubicBezier2d.mjs +1 -1
  95. package/dist-esm/lib/primitives/geometry/CubicBezier2d.mjs.map +2 -2
  96. package/dist-esm/lib/primitives/geometry/CubicSpline2d.mjs.map +2 -2
  97. package/dist-esm/lib/primitives/geometry/Edge2d.mjs +1 -1
  98. package/dist-esm/lib/primitives/geometry/Edge2d.mjs.map +2 -2
  99. package/dist-esm/lib/primitives/geometry/Ellipse2d.mjs.map +2 -2
  100. package/dist-esm/lib/primitives/geometry/Geometry2d.mjs +92 -21
  101. package/dist-esm/lib/primitives/geometry/Geometry2d.mjs.map +2 -2
  102. package/dist-esm/lib/primitives/geometry/Group2d.mjs +55 -2
  103. package/dist-esm/lib/primitives/geometry/Group2d.mjs.map +2 -2
  104. package/dist-esm/lib/primitives/geometry/Point2d.mjs.map +2 -2
  105. package/dist-esm/lib/primitives/geometry/Polyline2d.mjs.map +2 -2
  106. package/dist-esm/lib/primitives/geometry/Stadium2d.mjs.map +2 -2
  107. package/dist-esm/lib/utils/areShapesContentEqual.mjs +5 -0
  108. package/dist-esm/lib/utils/areShapesContentEqual.mjs.map +7 -0
  109. package/dist-esm/lib/utils/debug-flags.mjs +5 -2
  110. package/dist-esm/lib/utils/debug-flags.mjs.map +2 -2
  111. package/dist-esm/lib/utils/nearestMultiple.mjs +14 -0
  112. package/dist-esm/lib/utils/nearestMultiple.mjs.map +7 -0
  113. package/dist-esm/lib/utils/rotation.mjs +5 -5
  114. package/dist-esm/lib/utils/rotation.mjs.map +2 -2
  115. package/dist-esm/version.mjs +3 -3
  116. package/dist-esm/version.mjs.map +1 -1
  117. package/editor.css +41 -4
  118. package/package.json +7 -7
  119. package/src/index.ts +16 -31
  120. package/src/lib/components/Shape.tsx +14 -10
  121. package/src/lib/components/default-components/DefaultCanvas.tsx +43 -8
  122. package/src/lib/components/default-components/DefaultShapeIndicator.tsx +17 -8
  123. package/src/lib/editor/Editor.test.ts +1 -1
  124. package/src/lib/editor/Editor.ts +96 -24
  125. package/src/lib/editor/managers/SnapManager/HandleSnaps.ts +0 -1
  126. package/src/lib/editor/managers/TextManager.ts +12 -0
  127. package/src/lib/editor/shapes/ShapeUtil.ts +23 -3
  128. package/src/lib/editor/shapes/group/GroupShapeUtil.tsx +0 -4
  129. package/src/lib/editor/shapes/shared/getPerfectDashProps.ts +9 -9
  130. package/src/lib/exports/getSvgJsx.tsx +16 -7
  131. package/src/lib/hooks/useDocumentEvents.ts +7 -2
  132. package/src/lib/hooks/useEditorComponents.tsx +33 -32
  133. package/src/lib/primitives/Box.ts +20 -0
  134. package/src/lib/primitives/Mat.ts +5 -4
  135. package/src/lib/primitives/Vec.ts +23 -0
  136. package/src/lib/primitives/geometry/Arc2d.ts +5 -5
  137. package/src/lib/primitives/geometry/Circle2d.ts +4 -4
  138. package/src/lib/primitives/geometry/CubicBezier2d.ts +4 -4
  139. package/src/lib/primitives/geometry/CubicSpline2d.ts +3 -3
  140. package/src/lib/primitives/geometry/Edge2d.ts +3 -3
  141. package/src/lib/primitives/geometry/Ellipse2d.ts +3 -3
  142. package/src/lib/primitives/geometry/Geometry2d.test.ts +42 -0
  143. package/src/lib/primitives/geometry/Geometry2d.ts +123 -35
  144. package/src/lib/primitives/geometry/Group2d.ts +70 -7
  145. package/src/lib/primitives/geometry/Point2d.ts +2 -2
  146. package/src/lib/primitives/geometry/Polyline2d.ts +3 -3
  147. package/src/lib/primitives/geometry/Stadium2d.ts +3 -3
  148. package/src/lib/test/currentToolIdMask.test.ts +1 -1
  149. package/src/lib/test/user.test.ts +1 -1
  150. package/src/lib/utils/areShapesContentEqual.ts +4 -0
  151. package/src/lib/utils/debug-flags.ts +7 -2
  152. package/src/lib/utils/nearestMultiple.ts +13 -0
  153. package/src/lib/utils/rotation.ts +8 -6
  154. package/src/lib/utils/sync/LocalIndexedDb.test.ts +1 -1
  155. package/src/lib/utils/sync/TLLocalSyncClient.test.ts +1 -1
  156. package/src/version.ts +3 -3
package/editor.css CHANGED
@@ -44,6 +44,7 @@
44
44
  /* User handles need to be above selection edges / corners, matters for sticky note clone handles */
45
45
  --layer-overlays-user-handles: 105;
46
46
  --layer-overlays-user-indicator-hint: 110;
47
+ --layer-overlays-custom: 115;
47
48
  --layer-overlays-collaborator-cursor-hint: 120;
48
49
  --layer-overlays-collaborator-cursor: 130;
49
50
 
@@ -156,6 +157,7 @@
156
157
  --color-panel-contrast: hsl(0, 0%, 100%);
157
158
  --color-panel-overlay: hsl(0, 0%, 100%, 82%);
158
159
  --color-panel: hsl(0, 0%, 99%);
160
+ --color-panel-transparent: hsla(0, 0%, 99%, 0%);
159
161
  --color-focus: hsl(219, 65%, 50%);
160
162
  --color-selected: hsl(214, 84%, 56%);
161
163
  --color-selected-contrast: hsl(0, 0%, 100%);
@@ -207,6 +209,7 @@
207
209
  --color-panel-contrast: hsl(245, 12%, 23%);
208
210
  --color-panel: hsl(235, 6.8%, 13.5%);
209
211
  --color-panel-overlay: hsl(210, 10%, 24%, 82%);
212
+ --color-panel-transparent: hsla(235, 6.8%, 13.5%, 0%);
210
213
  --color-focus: hsl(217, 76%, 80%);
211
214
  --color-selected: hsl(217, 89%, 61%);
212
215
  --color-selected-contrast: hsl(0, 0%, 100%);
@@ -472,6 +475,10 @@ input,
472
475
  stroke-width: calc(2.5px * var(--tl-scale));
473
476
  }
474
477
 
478
+ .tl-custom-overlays {
479
+ z-index: var(--layer-overlays-custom);
480
+ }
481
+
475
482
  /* behind collaborator cursor */
476
483
  .tl-collaborator__cursor-hint {
477
484
  z-index: var(--layer-overlays-collaborator-cursor-hint);
@@ -595,6 +602,36 @@ input,
595
602
  }
596
603
  }
597
604
 
605
+ .tl-rotate-corner:not(:hover),
606
+ .tl-resize-handle:not(:hover) {
607
+ cursor: none;
608
+ }
609
+
610
+ /* --------------------- Arrow Hints -------------------- */
611
+
612
+ .tl-arrow-hint-handle {
613
+ fill: var(--color-selected-contrast);
614
+ stroke: var(--color-selection-stroke);
615
+ stroke-width: calc(1.5px * var(--tl-scale));
616
+ r: calc(4px * var(--tl-scale));
617
+ }
618
+
619
+ .tl-arrow-hint-snap {
620
+ stroke: transparent;
621
+ fill: var(--color-selection-fill);
622
+ r: calc(12px * var(--tl-scale));
623
+ }
624
+
625
+ .tl-arrow-hint-snap__none,
626
+ .tl-arrow-hint-snap__center,
627
+ .tl-arrow-hint-snap__axis {
628
+ display: none;
629
+ }
630
+
631
+ .tl-arrow-hint-snap__edge {
632
+ r: calc(8px * var(--tl-scale));
633
+ }
634
+
598
635
  /* ------------------ Bounds Detail ----------------- */
599
636
 
600
637
  .tl-image,
@@ -964,6 +1001,8 @@ input,
964
1001
 
965
1002
  .tl-rich-text p {
966
1003
  margin: 0;
1004
+ /* Depending on the extensions, <p> tags can be empty, without a <br />. */
1005
+ min-height: 1lh;
967
1006
  }
968
1007
 
969
1008
  .tl-rich-text ul,
@@ -971,6 +1010,8 @@ input,
971
1010
  text-align: left;
972
1011
  margin: 0;
973
1012
  padding-left: 3.25ch;
1013
+ /* Some resets, like Tailwind, nix the list styling. */
1014
+ list-style: revert;
974
1015
  }
975
1016
 
976
1017
  .tl-rich-text ol:has(> li:nth-child(10)) {
@@ -1348,10 +1389,6 @@ input,
1348
1389
  opacity: 0;
1349
1390
  }
1350
1391
 
1351
- .tl-arrow-label[data-isediting='true'] > .tl-arrow-label__inner {
1352
- background-color: var(--color-background);
1353
- }
1354
-
1355
1392
  .tl-arrow-label__inner {
1356
1393
  border-radius: var(--radius-1);
1357
1394
  box-sizing: content-box;
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@tldraw/editor",
3
3
  "description": "A tiny little drawing app (editor).",
4
- "version": "3.13.0-canary.37377fbef46d",
4
+ "version": "3.13.0-canary.409fd05a1a56",
5
5
  "author": {
6
6
  "name": "tldraw Inc.",
7
7
  "email": "hello@tldraw.com"
@@ -48,12 +48,12 @@
48
48
  "@tiptap/core": "^2.9.1",
49
49
  "@tiptap/pm": "^2.9.1",
50
50
  "@tiptap/react": "^2.9.1",
51
- "@tldraw/state": "3.13.0-canary.37377fbef46d",
52
- "@tldraw/state-react": "3.13.0-canary.37377fbef46d",
53
- "@tldraw/store": "3.13.0-canary.37377fbef46d",
54
- "@tldraw/tlschema": "3.13.0-canary.37377fbef46d",
55
- "@tldraw/utils": "3.13.0-canary.37377fbef46d",
56
- "@tldraw/validate": "3.13.0-canary.37377fbef46d",
51
+ "@tldraw/state": "3.13.0-canary.409fd05a1a56",
52
+ "@tldraw/state-react": "3.13.0-canary.409fd05a1a56",
53
+ "@tldraw/store": "3.13.0-canary.409fd05a1a56",
54
+ "@tldraw/tlschema": "3.13.0-canary.409fd05a1a56",
55
+ "@tldraw/utils": "3.13.0-canary.409fd05a1a56",
56
+ "@tldraw/validate": "3.13.0-canary.409fd05a1a56",
57
57
  "@types/core-js": "^2.5.8",
58
58
  "@use-gesture/react": "^10.3.1",
59
59
  "classnames": "^2.5.1",
package/src/index.ts CHANGED
@@ -4,37 +4,11 @@ import 'core-js/stable/array/flat-map.js'
4
4
  import 'core-js/stable/array/flat.js'
5
5
  import 'core-js/stable/string/at.js'
6
6
  import 'core-js/stable/string/replace-all.js'
7
- export {
8
- EMPTY_ARRAY,
9
- EffectScheduler,
10
- atom,
11
- computed,
12
- react,
13
- transact,
14
- transaction,
15
- whyAmIRunning,
16
- type Atom,
17
- type Signal,
18
- } from '@tldraw/state'
19
- export {
20
- track,
21
- useAtom,
22
- useComputed,
23
- useQuickReactor,
24
- useReactor,
25
- useStateTracking,
26
- useValue,
27
- } from '@tldraw/state-react'
28
- export { resizeScaled } from './lib/editor/shapes/shared/resizeScaled'
29
- export {
30
- getFontsFromRichText,
31
- type RichTextFontVisitor,
32
- type RichTextFontVisitorState,
33
- type TLTextOptions,
34
- type TiptapEditor,
35
- type TiptapNode,
36
- } from './lib/utils/richText'
37
- export { LocalIndexedDb, Table, type StoreName } from './lib/utils/sync/LocalIndexedDb'
7
+
8
+ // eslint-disable-next-line local/no-export-star
9
+ export * from '@tldraw/state'
10
+ // eslint-disable-next-line local/no-export-star
11
+ export * from '@tldraw/state-react'
38
12
  // eslint-disable-next-line local/no-export-star
39
13
  export * from '@tldraw/store'
40
14
  // eslint-disable-next-line local/no-export-star
@@ -43,6 +17,7 @@ export * from '@tldraw/tlschema'
43
17
  export * from '@tldraw/utils'
44
18
  // eslint-disable-next-line local/no-export-star
45
19
  export * from '@tldraw/validate'
20
+
46
21
  export {
47
22
  ErrorScreen,
48
23
  LoadingScreen,
@@ -212,6 +187,7 @@ export {
212
187
  export { GroupShapeUtil } from './lib/editor/shapes/group/GroupShapeUtil'
213
188
  export { getPerfectDashProps } from './lib/editor/shapes/shared/getPerfectDashProps'
214
189
  export { resizeBox, type ResizeBoxOptions } from './lib/editor/shapes/shared/resizeBox'
190
+ export { resizeScaled } from './lib/editor/shapes/shared/resizeScaled'
215
191
  export { BaseBoxShapeTool } from './lib/editor/tools/BaseBoxShapeTool/BaseBoxShapeTool'
216
192
  export { maybeSnapToGrid } from './lib/editor/tools/BaseBoxShapeTool/children/Pointing'
217
193
  export { StateNode, type TLStateNodeConstructor } from './lib/editor/tools/StateNode'
@@ -459,12 +435,21 @@ export { hardResetEditor } from './lib/utils/hardResetEditor'
459
435
  export { isAccelKey } from './lib/utils/keyboard'
460
436
  export { normalizeWheel } from './lib/utils/normalizeWheel'
461
437
  export { refreshPage } from './lib/utils/refreshPage'
438
+ export {
439
+ getFontsFromRichText,
440
+ type RichTextFontVisitor,
441
+ type RichTextFontVisitorState,
442
+ type TLTextOptions,
443
+ type TiptapEditor,
444
+ type TiptapNode,
445
+ } from './lib/utils/richText'
462
446
  export {
463
447
  applyRotationToSnapshotShapes,
464
448
  getRotationSnapshot,
465
449
  type TLRotationSnapshot,
466
450
  } from './lib/utils/rotation'
467
451
  export { runtime, setRuntimeOverrides } from './lib/utils/runtime'
452
+ export { LocalIndexedDb, Table, type StoreName } from './lib/utils/sync/LocalIndexedDb'
468
453
  export { type TLStoreWithStatus } from './lib/utils/sync/StoreWithStatus'
469
454
  export { hardReset } from './lib/utils/sync/hardReset'
470
455
  export { uniq } from './lib/utils/uniq'
@@ -6,6 +6,7 @@ import { ShapeUtil } from '../editor/shapes/ShapeUtil'
6
6
  import { useEditor } from '../hooks/useEditor'
7
7
  import { useEditorComponents } from '../hooks/useEditorComponents'
8
8
  import { Mat } from '../primitives/Mat'
9
+ import { areShapesContentEqual } from '../utils/areShapesContentEqual'
9
10
  import { setStyleProperty } from '../utils/dom'
10
11
  import { OptionalErrorBoundary } from './ErrorBoundary'
11
12
 
@@ -27,6 +28,7 @@ export const Shape = memo(function Shape({
27
28
  index,
28
29
  backgroundIndex,
29
30
  opacity,
31
+ dprMultiple,
30
32
  }: {
31
33
  id: TLShapeId
32
34
  shape: TLShape
@@ -34,6 +36,7 @@ export const Shape = memo(function Shape({
34
36
  index: number
35
37
  backgroundIndex: number
36
38
  opacity: number
39
+ dprMultiple: number
37
40
  }) {
38
41
  const editor = useEditor()
39
42
 
@@ -88,14 +91,18 @@ export const Shape = memo(function Shape({
88
91
  }
89
92
 
90
93
  // Width / Height
91
- const width = Math.max(bounds.width, 1)
92
- const height = Math.max(bounds.height, 1)
94
+ // We round the shape width and height up to the nearest multiple of dprMultiple
95
+ // to avoid the browser making miscalculations when applying the transform.
96
+ const widthRemainder = bounds.w % dprMultiple
97
+ const heightRemainder = bounds.h % dprMultiple
98
+ const width = widthRemainder === 0 ? bounds.w : bounds.w + (dprMultiple - widthRemainder)
99
+ const height = heightRemainder === 0 ? bounds.h : bounds.h + (dprMultiple - heightRemainder)
93
100
 
94
101
  if (width !== prev.width || height !== prev.height) {
95
- setStyleProperty(containerRef.current, 'width', width + 'px')
96
- setStyleProperty(containerRef.current, 'height', height + 'px')
97
- setStyleProperty(bgContainerRef.current, 'width', width + 'px')
98
- setStyleProperty(bgContainerRef.current, 'height', height + 'px')
102
+ setStyleProperty(containerRef.current, 'width', Math.max(width, dprMultiple) + 'px')
103
+ setStyleProperty(containerRef.current, 'height', Math.max(height, dprMultiple) + 'px')
104
+ setStyleProperty(bgContainerRef.current, 'width', Math.max(width, dprMultiple) + 'px')
105
+ setStyleProperty(bgContainerRef.current, 'height', Math.max(height, dprMultiple) + 'px')
99
106
  prev.width = width
100
107
  prev.height = height
101
108
  }
@@ -184,10 +191,7 @@ export const InnerShape = memo(
184
191
  [util, shape.id]
185
192
  )
186
193
  },
187
- (prev, next) =>
188
- prev.shape.props === next.shape.props &&
189
- prev.shape.meta === next.shape.meta &&
190
- prev.util === next.util
194
+ (prev, next) => areShapesContentEqual(prev.shape, next.shape) && prev.util === next.util
191
195
  )
192
196
 
193
197
  export const InnerShapeBackground = memo(
@@ -22,6 +22,7 @@ import { Vec } from '../../primitives/Vec'
22
22
  import { toDomPrecision } from '../../primitives/utils'
23
23
  import { debugFlags } from '../../utils/debug-flags'
24
24
  import { setStyleProperty } from '../../utils/dom'
25
+ import { nearestMultiple } from '../../utils/nearestMultiple'
25
26
  import { GeometryDebuggingView } from '../GeometryDebuggingView'
26
27
  import { LiveCollaborators } from '../LiveCollaborators'
27
28
  import { MenuClickCapture } from '../MenuClickCapture'
@@ -36,7 +37,7 @@ export interface TLCanvasComponentProps {
36
37
  export function DefaultCanvas({ className }: TLCanvasComponentProps) {
37
38
  const editor = useEditor()
38
39
 
39
- const { Background, SvgDefs, ShapeIndicators } = useEditorComponents()
40
+ const { SelectionBackground, Background, SvgDefs, ShapeIndicators } = useEditorComponents()
40
41
 
41
42
  const rCanvas = useRef<HTMLDivElement>(null)
42
43
  const rHtmlLayer = useRef<HTMLDivElement>(null)
@@ -154,7 +155,7 @@ export function DefaultCanvas({ className }: TLCanvasComponentProps) {
154
155
  <GridWrapper />
155
156
  <div ref={rHtmlLayer} className="tl-html-layer tl-shapes" draggable={false}>
156
157
  <OnTheCanvasWrapper />
157
- <SelectionBackgroundWrapper />
158
+ {SelectionBackground && <SelectionBackgroundWrapper />}
158
159
  {hideShapes ? null : debugSvg ? <ShapesWithSVGs /> : <ShapesToDisplay />}
159
160
  </div>
160
161
  <div className="tl-overlays">
@@ -168,6 +169,7 @@ export function DefaultCanvas({ className }: TLCanvasComponentProps) {
168
169
  <SnapIndicatorWrapper />
169
170
  <SelectionForegroundWrapper />
170
171
  <HandlesWrapper />
172
+ <OverlaysWrapper />
171
173
  <LiveCollaborators />
172
174
  </div>
173
175
  </div>
@@ -363,7 +365,8 @@ function HandleWrapper({
363
365
  return (
364
366
  <g
365
367
  role="button"
366
- aria-label="handle"
368
+ // TODO(mime): handle.label needs to be required in the future.
369
+ aria-label={handle.label || 'handle'}
367
370
  transform={`translate(${handle.x}, ${handle.y})`}
368
371
  {...events}
369
372
  >
@@ -372,14 +375,33 @@ function HandleWrapper({
372
375
  )
373
376
  }
374
377
 
378
+ function OverlaysWrapper() {
379
+ const { Overlays } = useEditorComponents()
380
+ if (!Overlays) return null
381
+ return (
382
+ <div className="tl-custom-overlays tl-overlays__item">
383
+ <Overlays />
384
+ </div>
385
+ )
386
+ }
387
+
375
388
  function ShapesWithSVGs() {
376
389
  const editor = useEditor()
377
390
 
378
391
  const renderingShapes = useValue('rendering shapes', () => editor.getRenderingShapes(), [editor])
379
392
 
393
+ const dprMultiple = useValue(
394
+ 'dpr multiple',
395
+ () =>
396
+ // dprMultiple is the smallest number we can multiply dpr by to get an integer
397
+ // it's usually 1, 2, or 4 (for e.g. dpr of 2, 2.5 and 2.25 respectively)
398
+ nearestMultiple(Math.floor(editor.getInstanceState().devicePixelRatio * 100) / 100),
399
+ [editor]
400
+ )
401
+
380
402
  return renderingShapes.map((result) => (
381
403
  <Fragment key={result.id + '_fragment'}>
382
- <Shape {...result} />
404
+ <Shape {...result} dprMultiple={dprMultiple} />
383
405
  <DebugSvgCopy id={result.id} mode="iframe" />
384
406
  </Fragment>
385
407
  ))
@@ -414,10 +436,19 @@ function ShapesToDisplay() {
414
436
 
415
437
  const renderingShapes = useValue('rendering shapes', () => editor.getRenderingShapes(), [editor])
416
438
 
439
+ const dprMultiple = useValue(
440
+ 'dpr multiple',
441
+ () =>
442
+ // dprMultiple is the smallest number we can multiply dpr by to get an integer
443
+ // it's usually 1, 2, or 4 (for e.g. dpr of 2, 2.5 and 2.25 respectively)
444
+ nearestMultiple(Math.floor(editor.getInstanceState().devicePixelRatio * 100) / 100),
445
+ [editor]
446
+ )
447
+
417
448
  return (
418
449
  <>
419
450
  {renderingShapes.map((result) => (
420
- <Shape key={result.id + '_shape'} {...result} />
451
+ <Shape key={result.id + '_shape'} {...result} dprMultiple={dprMultiple} />
421
452
  ))}
422
453
  {tlenv.isSafari && <ReflowIfNeeded />}
423
454
  </>
@@ -545,9 +576,13 @@ function DebugSvgCopy({ id, mode }: { id: TLShapeId; mode: 'img' | 'iframe' }) {
545
576
 
546
577
  function SelectionForegroundWrapper() {
547
578
  const editor = useEditor()
548
- const selectionRotation = useValue('selection rotation', () => editor.getSelectionRotation(), [
549
- editor,
550
- ])
579
+ const selectionRotation = useValue(
580
+ 'selection rotation',
581
+ function getSelectionRotation() {
582
+ return editor.getSelectionRotation()
583
+ },
584
+ [editor]
585
+ )
551
586
  const selectionBounds = useValue(
552
587
  'selection bounds',
553
588
  () => editor.getSelectionRotatedPageBounds(),
@@ -9,13 +9,21 @@ import { useEditorComponents } from '../../hooks/useEditorComponents'
9
9
  import { OptionalErrorBoundary } from '../ErrorBoundary'
10
10
 
11
11
  // need an extra layer of indirection here to allow hooks to be used inside the indicator render
12
- const EvenInnererIndicator = memo(({ shape, util }: { shape: TLShape; util: ShapeUtil<any> }) => {
13
- return useStateTracking('Indicator: ' + shape.type, () =>
14
- // always fetch the latest shape from the store even if the props/meta have not changed, to avoid
15
- // calling the render method with stale data.
16
- util.indicator(util.editor.store.unsafeGetWithoutCapture(shape.id) as TLShape)
17
- )
18
- })
12
+ const EvenInnererIndicator = memo(
13
+ ({ shape, util }: { shape: TLShape; util: ShapeUtil<any> }) => {
14
+ return useStateTracking('Indicator: ' + shape.type, () =>
15
+ // always fetch the latest shape from the store even if the props/meta have not changed, to avoid
16
+ // calling the render method with stale data.
17
+ util.indicator(util.editor.store.unsafeGetWithoutCapture(shape.id) as TLShape)
18
+ )
19
+ },
20
+ (prevProps, nextProps) => {
21
+ return (
22
+ prevProps.shape.props === nextProps.shape.props &&
23
+ prevProps.shape.meta === nextProps.shape.meta
24
+ )
25
+ }
26
+ )
19
27
 
20
28
  const InnerIndicator = memo(({ editor, id }: { editor: Editor; id: TLShapeId }) => {
21
29
  const shape = useValue('shape for indicator', () => editor.store.get(id), [editor, id])
@@ -61,13 +69,14 @@ export const DefaultShapeIndicator = memo(function DefaultShapeIndicator({
61
69
  useQuickReactor(
62
70
  'indicator transform',
63
71
  () => {
72
+ if (hidden) return
64
73
  const elm = rIndicator.current
65
74
  if (!elm) return
66
75
  const pageTransform = editor.getShapePageTransform(shapeId)
67
76
  if (!pageTransform) return
68
77
  elm.style.setProperty('transform', pageTransform.toCssString())
69
78
  },
70
- [editor, shapeId]
79
+ [editor, shapeId, hidden]
71
80
  )
72
81
 
73
82
  useLayoutEffect(() => {
@@ -52,7 +52,7 @@ beforeEach(() => {
52
52
  shapeUtils: [CustomShape],
53
53
  bindingUtils: [],
54
54
  tools: [],
55
- store: createTLStore({ shapeUtils: [CustomShape] }),
55
+ store: createTLStore({ shapeUtils: [CustomShape], bindingUtils: [] }),
56
56
  getContainer: () => document.body,
57
57
  })
58
58
  editor.setCameraOptions({ isLocked: true })
@@ -129,6 +129,7 @@ import { Group2d } from '../primitives/geometry/Group2d'
129
129
  import { intersectPolygonPolygon } from '../primitives/intersect'
130
130
  import { PI, approximately, areAnglesCompatible, clamp, pointInPolygon } from '../primitives/utils'
131
131
  import { ReadonlySharedStyleMap, SharedStyle, SharedStyleMap } from '../utils/SharedStylesMap'
132
+ import { areShapesContentEqual } from '../utils/areShapesContentEqual'
132
133
  import { dataUrlToFile } from '../utils/assets'
133
134
  import { debugFlags } from '../utils/debug-flags'
134
135
  import {
@@ -325,7 +326,6 @@ export class Editor extends EventEmitter<TLEventMap> {
325
326
  this.options = { ...defaultTldrawOptions, ...options }
326
327
 
327
328
  this.store = store
328
- this.disposables.add(this.store.dispose.bind(this.store))
329
329
  this.history = new HistoryManager<TLRecord>({
330
330
  store,
331
331
  annotateError: (error) => {
@@ -955,6 +955,7 @@ export class Editor extends EventEmitter<TLEventMap> {
955
955
  dispose() {
956
956
  this.disposables.forEach((dispose) => dispose())
957
957
  this.disposables.clear()
958
+ this.store.dispose()
958
959
  this.isDisposed = true
959
960
  }
960
961
 
@@ -1814,9 +1815,28 @@ export class Editor extends EventEmitter<TLEventMap> {
1814
1815
  return this
1815
1816
  }
1816
1817
 
1818
+ /**
1819
+ * Select the next shape in the reading order or in cardinal order.
1820
+ *
1821
+ * @example
1822
+ * ```ts
1823
+ * editor.selectAdjacentShape('next')
1824
+ * ```
1825
+ *
1826
+ * @public
1827
+ */
1817
1828
  selectAdjacentShape(direction: TLAdjacentDirection) {
1818
- const readingOrderShapes = this.getCurrentPageShapesInReadingOrder()
1819
1829
  const selectedShapeIds = this.getSelectedShapeIds()
1830
+ const firstParentId = selectedShapeIds[0] ? this.getShape(selectedShapeIds[0])?.parentId : null
1831
+ const isSelectedWithinContainer =
1832
+ firstParentId &&
1833
+ selectedShapeIds.every((shapeId) => this.getShape(shapeId)?.parentId === firstParentId) &&
1834
+ !isPageId(firstParentId)
1835
+ const readingOrderShapes = isSelectedWithinContainer
1836
+ ? this._getShapesInReadingOrder(
1837
+ this.getCurrentPageShapes().filter((shape) => shape.parentId === firstParentId)
1838
+ )
1839
+ : this.getCurrentPageShapesInReadingOrder()
1820
1840
  const currentShapeId: TLShapeId | undefined =
1821
1841
  selectedShapeIds.length === 1
1822
1842
  ? selectedShapeIds[0]
@@ -1838,13 +1858,7 @@ export class Editor extends EventEmitter<TLEventMap> {
1838
1858
  const shape = this.getShape(adjacentShapeId)
1839
1859
  if (!shape) return
1840
1860
 
1841
- this.setSelectedShapes([shape.id])
1842
- this.zoomToSelectionIfOffscreen(256, {
1843
- animation: {
1844
- duration: this.options.animationMediumMs,
1845
- },
1846
- inset: 0,
1847
- })
1861
+ this._selectShapesAndZoom([shape.id])
1848
1862
  }
1849
1863
 
1850
1864
  /**
@@ -1854,10 +1868,14 @@ export class Editor extends EventEmitter<TLEventMap> {
1854
1868
  * @public
1855
1869
  */
1856
1870
  @computed getCurrentPageShapesInReadingOrder(): TLShape[] {
1871
+ const shapes = this.getCurrentPageShapes().filter((shape) => isPageId(shape.parentId))
1872
+ return this._getShapesInReadingOrder(shapes)
1873
+ }
1874
+
1875
+ private _getShapesInReadingOrder(shapes: TLShape[]): TLShape[] {
1857
1876
  const SHALLOW_ANGLE = 20
1858
1877
  const ROW_THRESHOLD = 100
1859
1878
 
1860
- const shapes = this.getCurrentPageShapes()
1861
1879
  const tabbableShapes = shapes.filter((shape) => this.getShapeUtil(shape).canTabTo(shape))
1862
1880
 
1863
1881
  if (tabbableShapes.length <= 1) return tabbableShapes
@@ -2003,6 +2021,36 @@ export class Editor extends EventEmitter<TLEventMap> {
2003
2021
  return lowestScoringShape!.shape.id
2004
2022
  }
2005
2023
 
2024
+ selectParentShape() {
2025
+ const selectedShape = this.getOnlySelectedShape()
2026
+ if (!selectedShape) return
2027
+ const parentShape = this.getShape(selectedShape.parentId)
2028
+ if (!parentShape) return
2029
+ this._selectShapesAndZoom([parentShape.id])
2030
+ }
2031
+
2032
+ selectFirstChildShape() {
2033
+ const selectedShapes = this.getSelectedShapes()
2034
+ if (!selectedShapes.length) return
2035
+ const selectedShape = selectedShapes[0]
2036
+ const children = this.getSortedChildIdsForParent(selectedShape.id)
2037
+ .map((id) => this.getShape(id))
2038
+ .filter((i) => i) as TLShape[]
2039
+ const sortedChildren = this._getShapesInReadingOrder(children)
2040
+ if (sortedChildren.length === 0) return
2041
+ this._selectShapesAndZoom([sortedChildren[0].id])
2042
+ }
2043
+
2044
+ private _selectShapesAndZoom(ids: TLShapeId[]) {
2045
+ this.setSelectedShapes(ids)
2046
+ this.zoomToSelectionIfOffscreen(256, {
2047
+ animation: {
2048
+ duration: this.options.animationMediumMs,
2049
+ },
2050
+ inset: 0,
2051
+ })
2052
+ }
2053
+
2006
2054
  /**
2007
2055
  * Clear the selection.
2008
2056
  *
@@ -2275,13 +2323,21 @@ export class Editor extends EventEmitter<TLEventMap> {
2275
2323
  setEditingShape(shape: TLShapeId | TLShape | null): this {
2276
2324
  const id = typeof shape === 'string' ? shape : (shape?.id ?? null)
2277
2325
  this.setRichTextEditor(null)
2278
- if (id !== this.getEditingShapeId()) {
2326
+ const prevEditingShapeId = this.getEditingShapeId()
2327
+ if (id !== prevEditingShapeId) {
2279
2328
  if (id) {
2280
2329
  const shape = this.getShape(id)
2281
2330
  if (shape && this.getShapeUtil(shape).canEdit(shape)) {
2282
2331
  this.run(
2283
2332
  () => {
2284
2333
  this._updateCurrentPageState({ editingShapeId: id })
2334
+ if (prevEditingShapeId) {
2335
+ const prevEditingShape = this.getShape(prevEditingShapeId)
2336
+ if (prevEditingShape) {
2337
+ this.getShapeUtil(prevEditingShape).onEditEnd?.(prevEditingShape)
2338
+ }
2339
+ }
2340
+ this.getShapeUtil(shape).onEditStart?.(shape)
2285
2341
  },
2286
2342
  { history: 'ignore' }
2287
2343
  )
@@ -2294,6 +2350,12 @@ export class Editor extends EventEmitter<TLEventMap> {
2294
2350
  () => {
2295
2351
  this._updateCurrentPageState({ editingShapeId: null })
2296
2352
  this._currentRichTextEditor.set(null)
2353
+ if (prevEditingShapeId) {
2354
+ const prevEditingShape = this.getShape(prevEditingShapeId)
2355
+ if (prevEditingShape) {
2356
+ this.getShapeUtil(prevEditingShape).onEditEnd?.(prevEditingShape)
2357
+ }
2358
+ }
2297
2359
  },
2298
2360
  { history: 'ignore' }
2299
2361
  )
@@ -4574,7 +4636,7 @@ export class Editor extends EventEmitter<TLEventMap> {
4574
4636
  this.fonts.trackFontsForShape(shape)
4575
4637
  return this.getShapeUtil(shape).getGeometry(shape, opts)
4576
4638
  },
4577
- { areRecordsEqual: (a, b) => a.props === b.props }
4639
+ { areRecordsEqual: areShapesContentEqual }
4578
4640
  )
4579
4641
  }
4580
4642
  return this._shapeGeometryCaches[context].get(
@@ -4622,9 +4684,15 @@ export class Editor extends EventEmitter<TLEventMap> {
4622
4684
 
4623
4685
  /** @internal */
4624
4686
  @computed private _getShapeHandlesCache(): ComputedCache<TLHandle[] | undefined, TLShape> {
4625
- return this.store.createComputedCache('handles', (shape) => {
4626
- return this.getShapeUtil(shape).getHandles?.(shape)
4627
- })
4687
+ return this.store.createComputedCache(
4688
+ 'handles',
4689
+ (shape) => {
4690
+ return this.getShapeUtil(shape).getHandles?.(shape)
4691
+ },
4692
+ {
4693
+ areRecordsEqual: areShapesContentEqual,
4694
+ }
4695
+ )
4628
4696
  }
4629
4697
 
4630
4698
  /**
@@ -5845,9 +5913,15 @@ export class Editor extends EventEmitter<TLEventMap> {
5845
5913
  @computed
5846
5914
  private _getBindingsIndexCache() {
5847
5915
  const index = bindingsIndex(this)
5848
- return this.store.createComputedCache<TLBinding[], TLShape>('bindingsIndex', (shape) => {
5849
- return index.get().get(shape.id)
5850
- })
5916
+ return this.store.createComputedCache<TLBinding[], TLShape>(
5917
+ 'bindingsIndex',
5918
+ (shape) => {
5919
+ return index.get().get(shape.id)
5920
+ },
5921
+ // we can ignore the shape equality check here because the index is
5922
+ // computed incrementally based on what bindings are in the store
5923
+ { areRecordsEqual: () => true }
5924
+ )
5851
5925
  }
5852
5926
 
5853
5927
  /**
@@ -10214,7 +10288,7 @@ export class Editor extends EventEmitter<TLEventMap> {
10214
10288
 
10215
10289
  // If the camera behavior is "zoom" and the ctrl key is pressed, then pan;
10216
10290
  // If the camera behavior is "pan" and the ctrl key is not pressed, then zoom
10217
- if (inputs.ctrlKey) behavior = wheelBehavior === 'pan' ? 'zoom' : 'pan'
10291
+ if (info.ctrlKey) behavior = wheelBehavior === 'pan' ? 'zoom' : 'pan'
10218
10292
 
10219
10293
  switch (behavior) {
10220
10294
  case 'zoom': {
@@ -10330,12 +10404,10 @@ export class Editor extends EventEmitter<TLEventMap> {
10330
10404
  if (this.inputs.isPanning && this.inputs.isPointing) {
10331
10405
  // Handle spacebar / middle mouse button panning
10332
10406
  const { currentScreenPoint, previousScreenPoint } = this.inputs
10333
- const { panSpeed } = cameraOptions
10334
10407
  const offset = Vec.Sub(currentScreenPoint, previousScreenPoint)
10335
- this.setCamera(
10336
- new Vec(cx + (offset.x * panSpeed) / cz, cy + (offset.y * panSpeed) / cz, cz),
10337
- { immediate: true }
10338
- )
10408
+ this.setCamera(new Vec(cx + offset.x / cz, cy + offset.y / cz, cz), {
10409
+ immediate: true,
10410
+ })
10339
10411
  this.maybeTrackPerformance('Panning')
10340
10412
  return
10341
10413
  }
@@ -45,7 +45,6 @@ export interface HandleSnapGeometry {
45
45
 
46
46
  const defaultGetSelfSnapOutline = () => null
47
47
  const defaultGetSelfSnapPoints = () => []
48
-
49
48
  /** @public */
50
49
  export class HandleSnaps {
51
50
  readonly editor: Editor