@tldraw/editor 3.14.0-canary.d8a1c8c23469 → 3.14.0-canary.db789786fb06

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 (242) hide show
  1. package/dist-cjs/index.d.ts +220 -117
  2. package/dist-cjs/index.js +11 -8
  3. package/dist-cjs/index.js.map +2 -2
  4. package/dist-cjs/lib/config/TLSessionStateSnapshot.js +1 -12
  5. package/dist-cjs/lib/config/TLSessionStateSnapshot.js.map +3 -3
  6. package/dist-cjs/lib/editor/Editor.js +132 -101
  7. package/dist-cjs/lib/editor/Editor.js.map +2 -2
  8. package/dist-cjs/lib/editor/bindings/BindingUtil.js.map +2 -2
  9. package/dist-cjs/lib/editor/derivations/bindingsIndex.js +22 -22
  10. package/dist-cjs/lib/editor/derivations/bindingsIndex.js.map +2 -2
  11. package/dist-cjs/lib/editor/derivations/notVisibleShapes.js +16 -20
  12. package/dist-cjs/lib/editor/derivations/notVisibleShapes.js.map +3 -3
  13. package/dist-cjs/lib/editor/derivations/parentsToChildren.js +16 -16
  14. package/dist-cjs/lib/editor/derivations/parentsToChildren.js.map +2 -2
  15. package/dist-cjs/lib/editor/managers/{ClickManager.js → ClickManager/ClickManager.js} +1 -1
  16. package/dist-cjs/lib/editor/managers/ClickManager/ClickManager.js.map +7 -0
  17. package/dist-cjs/lib/editor/managers/{EdgeScrollManager.js → EdgeScrollManager/EdgeScrollManager.js} +2 -2
  18. package/dist-cjs/lib/editor/managers/EdgeScrollManager/EdgeScrollManager.js.map +7 -0
  19. package/dist-cjs/lib/editor/managers/FocusManager/FocusManager.js.map +7 -0
  20. package/dist-cjs/lib/editor/managers/{FontManager.js → FontManager/FontManager.js} +4 -1
  21. package/dist-cjs/lib/editor/managers/FontManager/FontManager.js.map +7 -0
  22. package/dist-cjs/lib/editor/managers/{HistoryManager.js → HistoryManager/HistoryManager.js} +67 -7
  23. package/dist-cjs/lib/editor/managers/HistoryManager/HistoryManager.js.map +7 -0
  24. package/dist-cjs/lib/editor/managers/{ScribbleManager.js → ScribbleManager/ScribbleManager.js} +1 -1
  25. package/dist-cjs/lib/editor/managers/ScribbleManager/ScribbleManager.js.map +7 -0
  26. package/dist-cjs/lib/editor/managers/{TextManager.js → TextManager/TextManager.js} +73 -42
  27. package/dist-cjs/lib/editor/managers/TextManager/TextManager.js.map +7 -0
  28. package/dist-cjs/lib/editor/managers/{TickManager.js → TickManager/TickManager.js} +1 -1
  29. package/dist-cjs/lib/editor/managers/TickManager/TickManager.js.map +7 -0
  30. package/dist-cjs/lib/editor/managers/{UserPreferencesManager.js → UserPreferencesManager/UserPreferencesManager.js} +1 -1
  31. package/dist-cjs/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.js.map +7 -0
  32. package/dist-cjs/lib/editor/shapes/ShapeUtil.js +8 -10
  33. package/dist-cjs/lib/editor/shapes/ShapeUtil.js.map +2 -2
  34. package/dist-cjs/lib/editor/shapes/group/GroupShapeUtil.js +6 -0
  35. package/dist-cjs/lib/editor/shapes/group/GroupShapeUtil.js.map +2 -2
  36. package/dist-cjs/lib/editor/tools/BaseBoxShapeTool/children/Pointing.js +10 -6
  37. package/dist-cjs/lib/editor/tools/BaseBoxShapeTool/children/Pointing.js.map +3 -3
  38. package/dist-cjs/lib/editor/tools/StateNode.js +3 -3
  39. package/dist-cjs/lib/editor/tools/StateNode.js.map +2 -2
  40. package/dist-cjs/lib/editor/types/emit-types.js.map +1 -1
  41. package/dist-cjs/lib/editor/types/external-content.js.map +1 -1
  42. package/dist-cjs/lib/exports/getSvgJsx.js.map +1 -1
  43. package/dist-cjs/lib/hooks/useCanvasEvents.js +1 -2
  44. package/dist-cjs/lib/hooks/useCanvasEvents.js.map +2 -2
  45. package/dist-cjs/lib/primitives/Box.js +33 -33
  46. package/dist-cjs/lib/primitives/Box.js.map +2 -2
  47. package/dist-cjs/lib/primitives/Vec.js +13 -8
  48. package/dist-cjs/lib/primitives/Vec.js.map +2 -2
  49. package/dist-cjs/lib/primitives/geometry/Arc2d.js +41 -21
  50. package/dist-cjs/lib/primitives/geometry/Arc2d.js.map +2 -2
  51. package/dist-cjs/lib/primitives/geometry/Circle2d.js +11 -11
  52. package/dist-cjs/lib/primitives/geometry/Circle2d.js.map +2 -2
  53. package/dist-cjs/lib/primitives/geometry/CubicBezier2d.js +13 -16
  54. package/dist-cjs/lib/primitives/geometry/CubicBezier2d.js.map +2 -2
  55. package/dist-cjs/lib/primitives/geometry/CubicSpline2d.js +4 -4
  56. package/dist-cjs/lib/primitives/geometry/CubicSpline2d.js.map +2 -2
  57. package/dist-cjs/lib/primitives/geometry/Edge2d.js +14 -17
  58. package/dist-cjs/lib/primitives/geometry/Edge2d.js.map +2 -2
  59. package/dist-cjs/lib/primitives/geometry/Ellipse2d.js +10 -10
  60. package/dist-cjs/lib/primitives/geometry/Ellipse2d.js.map +2 -2
  61. package/dist-cjs/lib/primitives/geometry/Geometry2d.js +6 -2
  62. package/dist-cjs/lib/primitives/geometry/Geometry2d.js.map +2 -2
  63. package/dist-cjs/lib/primitives/geometry/Point2d.js +6 -6
  64. package/dist-cjs/lib/primitives/geometry/Point2d.js.map +2 -2
  65. package/dist-cjs/lib/primitives/geometry/Polygon2d.js +3 -0
  66. package/dist-cjs/lib/primitives/geometry/Polygon2d.js.map +2 -2
  67. package/dist-cjs/lib/primitives/geometry/Polyline2d.js +8 -5
  68. package/dist-cjs/lib/primitives/geometry/Polyline2d.js.map +2 -2
  69. package/dist-cjs/lib/primitives/geometry/Rectangle2d.js +22 -11
  70. package/dist-cjs/lib/primitives/geometry/Rectangle2d.js.map +2 -2
  71. package/dist-cjs/lib/primitives/geometry/Stadium2d.js +22 -22
  72. package/dist-cjs/lib/primitives/geometry/Stadium2d.js.map +2 -2
  73. package/dist-cjs/lib/utils/reorderShapes.js +11 -10
  74. package/dist-cjs/lib/utils/reorderShapes.js.map +2 -2
  75. package/dist-cjs/lib/utils/reparenting.js +232 -0
  76. package/dist-cjs/lib/utils/reparenting.js.map +7 -0
  77. package/dist-cjs/lib/utils/richText.js +7 -2
  78. package/dist-cjs/lib/utils/richText.js.map +2 -2
  79. package/dist-cjs/version.js +3 -3
  80. package/dist-cjs/version.js.map +1 -1
  81. package/dist-esm/index.d.mts +220 -117
  82. package/dist-esm/index.mjs +15 -8
  83. package/dist-esm/index.mjs.map +2 -2
  84. package/dist-esm/lib/config/TLSessionStateSnapshot.mjs +1 -1
  85. package/dist-esm/lib/config/TLSessionStateSnapshot.mjs.map +2 -2
  86. package/dist-esm/lib/editor/Editor.mjs +132 -101
  87. package/dist-esm/lib/editor/Editor.mjs.map +2 -2
  88. package/dist-esm/lib/editor/bindings/BindingUtil.mjs.map +2 -2
  89. package/dist-esm/lib/editor/derivations/bindingsIndex.mjs +22 -22
  90. package/dist-esm/lib/editor/derivations/bindingsIndex.mjs.map +2 -2
  91. package/dist-esm/lib/editor/derivations/notVisibleShapes.mjs +16 -20
  92. package/dist-esm/lib/editor/derivations/notVisibleShapes.mjs.map +3 -3
  93. package/dist-esm/lib/editor/derivations/parentsToChildren.mjs +16 -16
  94. package/dist-esm/lib/editor/derivations/parentsToChildren.mjs.map +2 -2
  95. package/dist-esm/lib/editor/managers/{ClickManager.mjs → ClickManager/ClickManager.mjs} +1 -1
  96. package/dist-esm/lib/editor/managers/ClickManager/ClickManager.mjs.map +7 -0
  97. package/dist-esm/lib/editor/managers/{EdgeScrollManager.mjs → EdgeScrollManager/EdgeScrollManager.mjs} +2 -2
  98. package/dist-esm/lib/editor/managers/EdgeScrollManager/EdgeScrollManager.mjs.map +7 -0
  99. package/dist-esm/lib/editor/managers/FocusManager/FocusManager.mjs.map +7 -0
  100. package/dist-esm/lib/editor/managers/{FontManager.mjs → FontManager/FontManager.mjs} +4 -1
  101. package/dist-esm/lib/editor/managers/FontManager/FontManager.mjs.map +7 -0
  102. package/dist-esm/lib/editor/managers/{HistoryManager.mjs → HistoryManager/HistoryManager.mjs} +63 -3
  103. package/dist-esm/lib/editor/managers/HistoryManager/HistoryManager.mjs.map +7 -0
  104. package/dist-esm/lib/editor/managers/{ScribbleManager.mjs → ScribbleManager/ScribbleManager.mjs} +1 -1
  105. package/dist-esm/lib/editor/managers/ScribbleManager/ScribbleManager.mjs.map +7 -0
  106. package/dist-esm/lib/editor/managers/{TextManager.mjs → TextManager/TextManager.mjs} +73 -42
  107. package/dist-esm/lib/editor/managers/TextManager/TextManager.mjs.map +7 -0
  108. package/dist-esm/lib/editor/managers/{TickManager.mjs → TickManager/TickManager.mjs} +1 -1
  109. package/dist-esm/lib/editor/managers/TickManager/TickManager.mjs.map +7 -0
  110. package/dist-esm/lib/editor/managers/{UserPreferencesManager.mjs → UserPreferencesManager/UserPreferencesManager.mjs} +1 -1
  111. package/dist-esm/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.mjs.map +7 -0
  112. package/dist-esm/lib/editor/shapes/ShapeUtil.mjs +8 -10
  113. package/dist-esm/lib/editor/shapes/ShapeUtil.mjs.map +2 -2
  114. package/dist-esm/lib/editor/shapes/group/GroupShapeUtil.mjs +6 -0
  115. package/dist-esm/lib/editor/shapes/group/GroupShapeUtil.mjs.map +2 -2
  116. package/dist-esm/lib/editor/tools/BaseBoxShapeTool/children/Pointing.mjs +10 -6
  117. package/dist-esm/lib/editor/tools/BaseBoxShapeTool/children/Pointing.mjs.map +3 -3
  118. package/dist-esm/lib/editor/tools/StateNode.mjs +3 -3
  119. package/dist-esm/lib/editor/tools/StateNode.mjs.map +2 -2
  120. package/dist-esm/lib/exports/getSvgJsx.mjs.map +1 -1
  121. package/dist-esm/lib/hooks/useCanvasEvents.mjs +1 -2
  122. package/dist-esm/lib/hooks/useCanvasEvents.mjs.map +2 -2
  123. package/dist-esm/lib/primitives/Box.mjs +33 -33
  124. package/dist-esm/lib/primitives/Box.mjs.map +2 -2
  125. package/dist-esm/lib/primitives/Vec.mjs +13 -8
  126. package/dist-esm/lib/primitives/Vec.mjs.map +2 -2
  127. package/dist-esm/lib/primitives/geometry/Arc2d.mjs +41 -21
  128. package/dist-esm/lib/primitives/geometry/Arc2d.mjs.map +2 -2
  129. package/dist-esm/lib/primitives/geometry/Circle2d.mjs +11 -11
  130. package/dist-esm/lib/primitives/geometry/Circle2d.mjs.map +2 -2
  131. package/dist-esm/lib/primitives/geometry/CubicBezier2d.mjs +13 -16
  132. package/dist-esm/lib/primitives/geometry/CubicBezier2d.mjs.map +2 -2
  133. package/dist-esm/lib/primitives/geometry/CubicSpline2d.mjs +4 -4
  134. package/dist-esm/lib/primitives/geometry/CubicSpline2d.mjs.map +2 -2
  135. package/dist-esm/lib/primitives/geometry/Edge2d.mjs +14 -17
  136. package/dist-esm/lib/primitives/geometry/Edge2d.mjs.map +2 -2
  137. package/dist-esm/lib/primitives/geometry/Ellipse2d.mjs +11 -11
  138. package/dist-esm/lib/primitives/geometry/Ellipse2d.mjs.map +2 -2
  139. package/dist-esm/lib/primitives/geometry/Geometry2d.mjs +6 -2
  140. package/dist-esm/lib/primitives/geometry/Geometry2d.mjs.map +2 -2
  141. package/dist-esm/lib/primitives/geometry/Point2d.mjs +6 -6
  142. package/dist-esm/lib/primitives/geometry/Point2d.mjs.map +2 -2
  143. package/dist-esm/lib/primitives/geometry/Polygon2d.mjs +3 -0
  144. package/dist-esm/lib/primitives/geometry/Polygon2d.mjs.map +2 -2
  145. package/dist-esm/lib/primitives/geometry/Polyline2d.mjs +8 -5
  146. package/dist-esm/lib/primitives/geometry/Polyline2d.mjs.map +2 -2
  147. package/dist-esm/lib/primitives/geometry/Rectangle2d.mjs +22 -11
  148. package/dist-esm/lib/primitives/geometry/Rectangle2d.mjs.map +2 -2
  149. package/dist-esm/lib/primitives/geometry/Stadium2d.mjs +22 -22
  150. package/dist-esm/lib/primitives/geometry/Stadium2d.mjs.map +2 -2
  151. package/dist-esm/lib/utils/reorderShapes.mjs +11 -10
  152. package/dist-esm/lib/utils/reorderShapes.mjs.map +2 -2
  153. package/dist-esm/lib/utils/reparenting.mjs +216 -0
  154. package/dist-esm/lib/utils/reparenting.mjs.map +7 -0
  155. package/dist-esm/lib/utils/richText.mjs +8 -3
  156. package/dist-esm/lib/utils/richText.mjs.map +2 -2
  157. package/dist-esm/version.mjs +3 -3
  158. package/dist-esm/version.mjs.map +1 -1
  159. package/editor.css +442 -492
  160. package/package.json +8 -9
  161. package/src/index.ts +20 -7
  162. package/src/lib/config/TLSessionStateSnapshot.ts +1 -1
  163. package/src/lib/editor/Editor.test.ts +252 -3
  164. package/src/lib/editor/Editor.ts +150 -109
  165. package/src/lib/editor/bindings/BindingUtil.ts +6 -0
  166. package/src/lib/editor/derivations/bindingsIndex.ts +27 -26
  167. package/src/lib/editor/derivations/notVisibleShapes.ts +24 -25
  168. package/src/lib/editor/derivations/parentsToChildren.ts +28 -25
  169. package/src/lib/editor/managers/ClickManager/ClickManager.test.ts +442 -0
  170. package/src/lib/editor/managers/{ClickManager.ts → ClickManager/ClickManager.ts} +3 -3
  171. package/src/lib/editor/managers/EdgeScrollManager/EdgeScrollManager.test.ts +374 -0
  172. package/src/lib/editor/managers/{EdgeScrollManager.ts → EdgeScrollManager/EdgeScrollManager.ts} +3 -3
  173. package/src/lib/editor/managers/FocusManager/FocusManager.test.ts +455 -0
  174. package/src/lib/editor/managers/{FocusManager.ts → FocusManager/FocusManager.ts} +1 -1
  175. package/src/lib/editor/managers/FontManager/FontManager.test.ts +263 -0
  176. package/src/lib/editor/managers/{FontManager.ts → FontManager/FontManager.ts} +5 -2
  177. package/src/lib/editor/managers/{HistoryManager.test.ts → HistoryManager/HistoryManager.test.ts} +388 -1
  178. package/src/lib/editor/managers/{HistoryManager.ts → HistoryManager/HistoryManager.ts} +76 -3
  179. package/src/lib/editor/managers/ScribbleManager/ScribbleManager.test.ts +624 -0
  180. package/src/lib/editor/managers/{ScribbleManager.ts → ScribbleManager/ScribbleManager.ts} +2 -2
  181. package/src/lib/editor/managers/SnapManager/SnapManager.test.ts +485 -0
  182. package/src/lib/editor/managers/TextManager/TextManager.test.ts +407 -0
  183. package/src/lib/editor/managers/{TextManager.ts → TextManager/TextManager.ts} +119 -87
  184. package/src/lib/editor/managers/TickManager/TickManager.test.ts +314 -0
  185. package/src/lib/editor/managers/{TickManager.ts → TickManager/TickManager.ts} +2 -2
  186. package/src/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.test.ts +591 -0
  187. package/src/lib/editor/managers/{UserPreferencesManager.ts → UserPreferencesManager/UserPreferencesManager.ts} +2 -2
  188. package/src/lib/editor/shapes/ShapeUtil.ts +57 -16
  189. package/src/lib/editor/shapes/group/GroupShapeUtil.tsx +8 -0
  190. package/src/lib/editor/tools/BaseBoxShapeTool/children/Pointing.ts +22 -17
  191. package/src/lib/editor/tools/StateNode.ts +3 -3
  192. package/src/lib/editor/types/emit-types.ts +4 -0
  193. package/src/lib/editor/types/external-content.ts +11 -2
  194. package/src/lib/exports/getSvgJsx.tsx +1 -1
  195. package/src/lib/hooks/useCanvasEvents.ts +0 -1
  196. package/src/lib/primitives/Box.test.ts +588 -7
  197. package/src/lib/primitives/Box.ts +33 -33
  198. package/src/lib/primitives/Vec.test.ts +2 -2
  199. package/src/lib/primitives/Vec.ts +13 -8
  200. package/src/lib/primitives/geometry/Arc2d.ts +42 -23
  201. package/src/lib/primitives/geometry/Circle2d.ts +12 -12
  202. package/src/lib/primitives/geometry/CubicBezier2d.test.ts +5 -0
  203. package/src/lib/primitives/geometry/CubicBezier2d.ts +13 -17
  204. package/src/lib/primitives/geometry/CubicSpline2d.ts +5 -5
  205. package/src/lib/primitives/geometry/Edge2d.ts +14 -18
  206. package/src/lib/primitives/geometry/Ellipse2d.ts +12 -13
  207. package/src/lib/primitives/geometry/Geometry2d.ts +7 -2
  208. package/src/lib/primitives/geometry/Point2d.ts +6 -6
  209. package/src/lib/primitives/geometry/Polygon2d.ts +4 -0
  210. package/src/lib/primitives/geometry/Polyline2d.ts +10 -7
  211. package/src/lib/primitives/geometry/Rectangle2d.ts +24 -11
  212. package/src/lib/primitives/geometry/Stadium2d.ts +22 -23
  213. package/src/lib/utils/reorderShapes.ts +10 -13
  214. package/src/lib/utils/reparenting.ts +383 -0
  215. package/src/lib/utils/richText.ts +10 -4
  216. package/src/version.ts +3 -3
  217. package/dist-cjs/lib/editor/managers/ClickManager.js.map +0 -7
  218. package/dist-cjs/lib/editor/managers/EdgeScrollManager.js.map +0 -7
  219. package/dist-cjs/lib/editor/managers/FocusManager.js.map +0 -7
  220. package/dist-cjs/lib/editor/managers/FontManager.js.map +0 -7
  221. package/dist-cjs/lib/editor/managers/HistoryManager.js.map +0 -7
  222. package/dist-cjs/lib/editor/managers/ScribbleManager.js.map +0 -7
  223. package/dist-cjs/lib/editor/managers/Stack.js +0 -82
  224. package/dist-cjs/lib/editor/managers/Stack.js.map +0 -7
  225. package/dist-cjs/lib/editor/managers/TextManager.js.map +0 -7
  226. package/dist-cjs/lib/editor/managers/TickManager.js.map +0 -7
  227. package/dist-cjs/lib/editor/managers/UserPreferencesManager.js.map +0 -7
  228. package/dist-esm/lib/editor/managers/ClickManager.mjs.map +0 -7
  229. package/dist-esm/lib/editor/managers/EdgeScrollManager.mjs.map +0 -7
  230. package/dist-esm/lib/editor/managers/FocusManager.mjs.map +0 -7
  231. package/dist-esm/lib/editor/managers/FontManager.mjs.map +0 -7
  232. package/dist-esm/lib/editor/managers/HistoryManager.mjs.map +0 -7
  233. package/dist-esm/lib/editor/managers/ScribbleManager.mjs.map +0 -7
  234. package/dist-esm/lib/editor/managers/Stack.mjs +0 -62
  235. package/dist-esm/lib/editor/managers/Stack.mjs.map +0 -7
  236. package/dist-esm/lib/editor/managers/TextManager.mjs.map +0 -7
  237. package/dist-esm/lib/editor/managers/TickManager.mjs.map +0 -7
  238. package/dist-esm/lib/editor/managers/UserPreferencesManager.mjs.map +0 -7
  239. package/src/lib/editor/managers/ScribbleManager.test.ts +0 -32
  240. package/src/lib/editor/managers/Stack.ts +0 -71
  241. /package/dist-cjs/lib/editor/managers/{FocusManager.js → FocusManager/FocusManager.js} +0 -0
  242. /package/dist-esm/lib/editor/managers/{FocusManager.mjs → FocusManager/FocusManager.mjs} +0 -0
@@ -7,5 +7,9 @@ export class Polygon2d extends Polyline2d {
7
7
  constructor(config: Omit<Geometry2dOptions, 'isClosed'> & { points: Vec[] }) {
8
8
  super({ ...config })
9
9
  this.isClosed = true
10
+
11
+ if (config.points.length < 3) {
12
+ throw new Error('Polygon2d: points must be an array of at least 3 points')
13
+ }
10
14
  }
11
15
  }
@@ -4,18 +4,21 @@ import { Geometry2d, Geometry2dOptions } from './Geometry2d'
4
4
 
5
5
  /** @public */
6
6
  export class Polyline2d extends Geometry2d {
7
- points: Vec[]
7
+ private _points: Vec[]
8
+ private _segments?: Edge2d[]
8
9
 
9
10
  constructor(config: Omit<Geometry2dOptions, 'isFilled' | 'isClosed'> & { points: Vec[] }) {
10
11
  super({ isClosed: false, isFilled: false, ...config })
11
12
  const { points } = config
12
- this.points = points
13
- }
13
+ this._points = points
14
14
 
15
- _segments?: Edge2d[]
15
+ if (points.length < 2) {
16
+ throw new Error('Polyline2d: points must be an array of at least 2 points')
17
+ }
18
+ }
16
19
 
17
20
  // eslint-disable-next-line no-restricted-syntax
18
- get segments() {
21
+ protected get segments() {
19
22
  if (!this._segments) {
20
23
  this._segments = []
21
24
  const { vertices } = this
@@ -38,12 +41,12 @@ export class Polyline2d extends Geometry2d {
38
41
  }
39
42
 
40
43
  getVertices() {
41
- return this.points
44
+ return this._points
42
45
  }
43
46
 
44
47
  nearestPoint(A: VecLike): Vec {
45
48
  const { segments } = this
46
- let nearest = this.points[0]
49
+ let nearest = this._points[0]
47
50
  let dist = Infinity
48
51
  let p: Vec // current point on segment
49
52
  let d: number // distance from A to p
@@ -5,10 +5,10 @@ import { Polygon2d } from './Polygon2d'
5
5
 
6
6
  /** @public */
7
7
  export class Rectangle2d extends Polygon2d {
8
- x: number
9
- y: number
10
- w: number
11
- h: number
8
+ private _x: number
9
+ private _y: number
10
+ private _w: number
11
+ private _h: number
12
12
 
13
13
  constructor(
14
14
  config: Omit<Geometry2dOptions, 'isClosed'> & {
@@ -28,18 +28,31 @@ export class Rectangle2d extends Polygon2d {
28
28
  new Vec(x, y + height),
29
29
  ],
30
30
  })
31
- this.x = x
32
- this.y = y
33
- this.w = width
34
- this.h = height
31
+ this._x = x
32
+ this._y = y
33
+ this._w = width
34
+ this._h = height
35
35
  }
36
36
 
37
37
  getBounds() {
38
- return new Box(this.x, this.y, this.w, this.h)
38
+ return new Box(this._x, this._y, this._w, this._h)
39
39
  }
40
40
 
41
41
  getSvgPathData(): string {
42
- const { x, y, w, h } = this
43
- return `M${x},${y} h${w} v${h} h-${w}z`
42
+ const { _x: x, _y: y, _w: w, _h: h } = this
43
+ this.negativeZeroFix()
44
+ return `M${x},${y} h${w} v${h} h${-w}z`
44
45
  }
46
+
47
+ private negativeZeroFix() {
48
+ this._x = zeroFix(this._x)
49
+ this._y = zeroFix(this._y)
50
+ this._w = zeroFix(this._w)
51
+ this._h = zeroFix(this._h)
52
+ }
53
+ }
54
+
55
+ function zeroFix(value: number) {
56
+ if (Object.is(value, -0)) return 0
57
+ return value
45
58
  }
@@ -7,13 +7,12 @@ import { Geometry2d, Geometry2dOptions } from './Geometry2d'
7
7
 
8
8
  /** @public */
9
9
  export class Stadium2d extends Geometry2d {
10
- w: number
11
- h: number
12
-
13
- a: Arc2d
14
- b: Edge2d
15
- c: Arc2d
16
- d: Edge2d
10
+ private _w: number
11
+ private _h: number
12
+ private _a: Arc2d
13
+ private _b: Edge2d
14
+ private _c: Arc2d
15
+ private _d: Edge2d
17
16
 
18
17
  constructor(
19
18
  public config: Omit<Geometry2dOptions, 'isClosed'> & {
@@ -23,45 +22,45 @@ export class Stadium2d extends Geometry2d {
23
22
  ) {
24
23
  super({ ...config, isClosed: true })
25
24
  const { width: w, height: h } = config
26
- this.w = w
27
- this.h = h
25
+ this._w = w
26
+ this._h = h
28
27
 
29
28
  if (h > w) {
30
29
  const r = w / 2
31
- this.a = new Arc2d({
30
+ this._a = new Arc2d({
32
31
  start: new Vec(0, r),
33
32
  end: new Vec(w, r),
34
33
  center: new Vec(w / 2, r),
35
34
  sweepFlag: 1,
36
35
  largeArcFlag: 1,
37
36
  })
38
- this.b = new Edge2d({ start: new Vec(w, r), end: new Vec(w, h - r) })
39
- this.c = new Arc2d({
37
+ this._b = new Edge2d({ start: new Vec(w, r), end: new Vec(w, h - r) })
38
+ this._c = new Arc2d({
40
39
  start: new Vec(w, h - r),
41
40
  end: new Vec(0, h - r),
42
41
  center: new Vec(w / 2, h - r),
43
42
  sweepFlag: 1,
44
43
  largeArcFlag: 1,
45
44
  })
46
- this.d = new Edge2d({ start: new Vec(0, h - r), end: new Vec(0, r) })
45
+ this._d = new Edge2d({ start: new Vec(0, h - r), end: new Vec(0, r) })
47
46
  } else {
48
47
  const r = h / 2
49
- this.a = new Arc2d({
48
+ this._a = new Arc2d({
50
49
  start: new Vec(r, h),
51
50
  end: new Vec(r, 0),
52
51
  center: new Vec(r, r),
53
52
  sweepFlag: 1,
54
53
  largeArcFlag: 1,
55
54
  })
56
- this.b = new Edge2d({ start: new Vec(r, 0), end: new Vec(w - r, 0) })
57
- this.c = new Arc2d({
55
+ this._b = new Edge2d({ start: new Vec(r, 0), end: new Vec(w - r, 0) })
56
+ this._c = new Arc2d({
58
57
  start: new Vec(w - r, 0),
59
58
  end: new Vec(w - r, h),
60
59
  center: new Vec(w - r, r),
61
60
  sweepFlag: 1,
62
61
  largeArcFlag: 1,
63
62
  })
64
- this.d = new Edge2d({ start: new Vec(w - r, h), end: new Vec(r, h) })
63
+ this._d = new Edge2d({ start: new Vec(w - r, h), end: new Vec(r, h) })
65
64
  }
66
65
  }
67
66
 
@@ -71,7 +70,7 @@ export class Stadium2d extends Geometry2d {
71
70
  let _d: number
72
71
  let p: Vec
73
72
 
74
- const { a, b, c, d } = this
73
+ const { _a: a, _b: b, _c: c, _d: d } = this
75
74
  for (const part of [a, b, c, d]) {
76
75
  p = part.nearestPoint(A)
77
76
  _d = Vec.Dist2(p, A)
@@ -85,12 +84,12 @@ export class Stadium2d extends Geometry2d {
85
84
  }
86
85
 
87
86
  hitTestLineSegment(A: VecLike, B: VecLike): boolean {
88
- const { a, b, c, d } = this
87
+ const { _a: a, _b: b, _c: c, _d: d } = this
89
88
  return [a, b, c, d].some((edge) => edge.hitTestLineSegment(A, B))
90
89
  }
91
90
 
92
91
  getVertices() {
93
- const { a, b, c, d } = this
92
+ const { _a: a, _b: b, _c: c, _d: d } = this
94
93
  return [a, b, c, d].reduce<Vec[]>((a, p) => {
95
94
  a.push(...p.vertices)
96
95
  return a
@@ -98,17 +97,17 @@ export class Stadium2d extends Geometry2d {
98
97
  }
99
98
 
100
99
  getBounds() {
101
- return new Box(0, 0, this.w, this.h)
100
+ return new Box(0, 0, this._w, this._h)
102
101
  }
103
102
 
104
103
  getLength() {
105
- const { h, w } = this
104
+ const { _h: h, _w: w } = this
106
105
  if (h > w) return (PI * (w / 2) + (h - w)) * 2
107
106
  else return (PI * (h / 2) + (w - h)) * 2
108
107
  }
109
108
 
110
109
  getSvgPathData() {
111
- const { a, b, c, d } = this
110
+ const { _a: a, _b: b, _c: c, _d: d } = this
112
111
  return [a, b, c, d].map((p, i) => p.getSvgPathData(i === 0)).join(' ') + ' Z'
113
112
  }
114
113
  }
@@ -1,8 +1,6 @@
1
1
  import { TLParentId, TLShape, TLShapeId, TLShapePartial } from '@tldraw/tlschema'
2
2
  import { IndexKey, compact, getIndicesBetween, sortByIndex } from '@tldraw/utils'
3
3
  import { Editor } from '../editor/Editor'
4
- import { Vec } from '../primitives/Vec'
5
- import { polygonsIntersect } from '../primitives/intersect'
6
4
 
7
5
  export function getReorderingShapesChanges(
8
6
  editor: Editor,
@@ -155,19 +153,18 @@ function reorderToFront(moving: Set<TLShape>, children: TLShape[], changes: TLSh
155
153
  }
156
154
 
157
155
  function getOverlapChecker(editor: Editor, moving: Set<TLShape>) {
158
- const movingVertices = Array.from(moving)
159
- .map((shape) => {
160
- const vertices = editor.getShapePageGeometry(shape).vertices
161
- if (!vertices) return null
162
- return { shape, vertices }
156
+ const movingBounds = compact(
157
+ Array.from(moving).map((shape) => {
158
+ const bounds = editor.getShapePageBounds(shape)
159
+ if (!bounds) return null
160
+ return { shape, bounds }
163
161
  })
164
- .filter(Boolean) as { shape: TLShape; vertices: Vec[] }[]
165
-
162
+ )
166
163
  const isOverlapping = (child: TLShape) => {
167
- const vertices = editor.getShapePageGeometry(child).vertices
168
- if (!vertices) return false
169
- return movingVertices.some((other) => {
170
- return polygonsIntersect(other.vertices, vertices)
164
+ const bounds = editor.getShapePageBounds(child)
165
+ if (!bounds) return false
166
+ return movingBounds.some((other) => {
167
+ return other.bounds.includes(bounds)
171
168
  })
172
169
  }
173
170
 
@@ -0,0 +1,383 @@
1
+ import { EMPTY_ARRAY } from '@tldraw/state'
2
+ import { TLGroupShape, TLParentId, TLShape, TLShapeId } from '@tldraw/tlschema'
3
+ import { IndexKey, compact, getIndexAbove, getIndexBetween } from '@tldraw/utils'
4
+ import { Editor } from '../editor/Editor'
5
+ import { Vec } from '../primitives/Vec'
6
+ import { Geometry2d } from '../primitives/geometry/Geometry2d'
7
+ import { Group2d } from '../primitives/geometry/Group2d'
8
+ import {
9
+ intersectPolygonPolygon,
10
+ polygonIntersectsPolyline,
11
+ polygonsIntersect,
12
+ } from '../primitives/intersect'
13
+ import { pointInPolygon } from '../primitives/utils'
14
+
15
+ /**
16
+ * Reparents shapes that are no longer contained within their parent shapes.
17
+ * todo: rename me to something more descriptive, like `reparentOccludedShapes` or `reparentAutoDroppedShapes`
18
+ *
19
+ * @param editor - The editor instance.
20
+ * @param shapeIds - The IDs of the shapes to reparent.
21
+ * @param opts - Optional options, including a callback to filter out certain parents, such as when removing a frame.
22
+ *
23
+ * @public
24
+ */
25
+ export function kickoutOccludedShapes(
26
+ editor: Editor,
27
+ shapeIds: TLShapeId[],
28
+ opts?: { filter?(parent: TLShape): boolean }
29
+ ) {
30
+ const parentsToCheck = new Set<TLShape>()
31
+
32
+ for (const id of shapeIds) {
33
+ const shape = editor.getShape(id)
34
+
35
+ if (!shape) continue
36
+ parentsToCheck.add(shape)
37
+
38
+ const parent = editor.getShape(shape.parentId)
39
+ if (!parent) continue
40
+ parentsToCheck.add(parent)
41
+ }
42
+
43
+ // Check all of the parents and gather up parents who have lost children
44
+ const parentsToLostChildren = new Map<TLShape, TLShapeId[]>()
45
+
46
+ for (const parent of parentsToCheck) {
47
+ const childIds = editor.getSortedChildIdsForParent(parent)
48
+ if (opts?.filter && !opts.filter(parent)) {
49
+ // If the shape is filtered out, we kick out all of its children
50
+ parentsToLostChildren.set(parent, childIds)
51
+ } else {
52
+ const overlappingChildren = getOverlappingShapes(editor, parent.id, childIds)
53
+ if (overlappingChildren.length < childIds.length) {
54
+ parentsToLostChildren.set(
55
+ parent,
56
+ childIds.filter((id) => !overlappingChildren.includes(id))
57
+ )
58
+ }
59
+ }
60
+ }
61
+
62
+ // Get all of the shapes on the current page, sorted by their index
63
+ const sortedShapeIds = editor.getCurrentPageShapesSorted().map((s) => s.id)
64
+
65
+ const parentsToNewChildren: Record<
66
+ TLParentId,
67
+ { parentId: TLParentId; shapeIds: TLShapeId[]; index?: IndexKey }
68
+ > = {}
69
+
70
+ for (const [prevParent, lostChildrenIds] of parentsToLostChildren) {
71
+ const lostChildren = compact(lostChildrenIds.map((id) => editor.getShape(id)))
72
+
73
+ // Don't fall "up" into frames in front of the shape
74
+ // if (pageShapes.indexOf(shape) < frameSortPosition) continue shapeCheck
75
+
76
+ // Otherwise, we have no next dropping shape under the cursor, so go find
77
+ // all the frames on the page where the moving shapes will fall into
78
+ const { reparenting, remainingShapesToReparent } = getDroppedShapesToNewParents(
79
+ editor,
80
+ lostChildren,
81
+ (shape, maybeNewParent) => {
82
+ // If we're filtering out a potential parent, don't reparent shapes to the filtered out shape
83
+ if (opts?.filter && !opts.filter(maybeNewParent)) return false
84
+ return (
85
+ maybeNewParent.id !== prevParent.id &&
86
+ sortedShapeIds.indexOf(maybeNewParent.id) < sortedShapeIds.indexOf(shape.id)
87
+ )
88
+ }
89
+ )
90
+
91
+ reparenting.forEach((childrenToReparent, newParentId) => {
92
+ if (childrenToReparent.length === 0) return
93
+ if (!parentsToNewChildren[newParentId]) {
94
+ parentsToNewChildren[newParentId] = {
95
+ parentId: newParentId,
96
+ shapeIds: [],
97
+ }
98
+ }
99
+ parentsToNewChildren[newParentId].shapeIds.push(...childrenToReparent.map((s) => s.id))
100
+ })
101
+
102
+ // Reparent the rest to the page (or containing group)
103
+ if (remainingShapesToReparent.size > 0) {
104
+ // The remaining shapes are going to be reparented to the old parent's containing group, if there was one, or else to the page
105
+ const newParentId =
106
+ editor.findShapeAncestor(prevParent, (s) => editor.isShapeOfType<TLGroupShape>(s, 'group'))
107
+ ?.id ?? editor.getCurrentPageId()
108
+
109
+ remainingShapesToReparent.forEach((shape) => {
110
+ if (!parentsToNewChildren[newParentId]) {
111
+ let insertIndexKey: IndexKey | undefined
112
+
113
+ const oldParentSiblingIds = editor.getSortedChildIdsForParent(newParentId)
114
+ const oldParentIndex = oldParentSiblingIds.indexOf(prevParent.id)
115
+ if (oldParentIndex > -1) {
116
+ // If the old parent is a direct child of the new parent, then we'll add them above the old parent but below the next sibling.
117
+ const siblingsIndexAbove = oldParentSiblingIds[oldParentIndex + 1]
118
+ const indexKeyAbove = siblingsIndexAbove
119
+ ? editor.getShape(siblingsIndexAbove)!.index
120
+ : getIndexAbove(prevParent.index)
121
+ insertIndexKey = getIndexBetween(prevParent.index, indexKeyAbove)
122
+ } else {
123
+ // If the old parent is not a direct child of the new parent, then we'll add them to the "top" of the new parent's children.
124
+ // This is done automatically if we leave the index undefined, so let's do that.
125
+ }
126
+
127
+ parentsToNewChildren[newParentId] = {
128
+ parentId: newParentId,
129
+ shapeIds: [],
130
+ index: insertIndexKey,
131
+ }
132
+ }
133
+
134
+ parentsToNewChildren[newParentId].shapeIds.push(shape.id)
135
+ })
136
+ }
137
+ }
138
+
139
+ editor.run(() => {
140
+ Object.values(parentsToNewChildren).forEach(({ parentId, shapeIds, index }) => {
141
+ if (shapeIds.length === 0) return
142
+ // Before we reparent, sort the new shape ids by their place in the original absolute order on the page
143
+ shapeIds.sort((a, b) => (sortedShapeIds.indexOf(a) < sortedShapeIds.indexOf(b) ? -1 : 1))
144
+ editor.reparentShapes(shapeIds, parentId, index)
145
+ })
146
+ })
147
+ }
148
+
149
+ /**
150
+ * Get the shapes that overlap with a given shape.
151
+ *
152
+ * @param editor - The editor instance.
153
+ * @param shape - The shapes or shape IDs to check against.
154
+ * @param otherShapes - The shapes or shape IDs to check for overlap.
155
+ * @returns An array of shapes or shape IDs that overlap with the given shape.
156
+ */
157
+ function getOverlappingShapes<T extends TLShape[] | TLShapeId[]>(
158
+ editor: Editor,
159
+ shape: T[number],
160
+ otherShapes: T
161
+ ) {
162
+ if (otherShapes.length === 0) {
163
+ return EMPTY_ARRAY
164
+ }
165
+
166
+ const parentPageBounds = editor.getShapePageBounds(shape)
167
+ if (!parentPageBounds) return EMPTY_ARRAY
168
+
169
+ const parentGeometry = editor.getShapeGeometry(shape)
170
+ const parentPageTransform = editor.getShapePageTransform(shape)
171
+ const parentPageCorners = parentPageTransform.applyToPoints(parentGeometry.vertices)
172
+
173
+ const parentPageMaskVertices = editor.getShapeMask(shape)
174
+ const parentPagePolygon = parentPageMaskVertices
175
+ ? intersectPolygonPolygon(parentPageMaskVertices, parentPageCorners)
176
+ : parentPageCorners
177
+
178
+ if (!parentPagePolygon) return EMPTY_ARRAY
179
+
180
+ return otherShapes.filter((childId) => {
181
+ const shapePageBounds = editor.getShapePageBounds(childId)
182
+ if (!shapePageBounds || !parentPageBounds.includes(shapePageBounds)) return false
183
+
184
+ const parentPolygonInShapeShape = editor
185
+ .getShapePageTransform(childId)
186
+ .clone()
187
+ .invert()
188
+ .applyToPoints(parentPagePolygon)
189
+
190
+ const geometry = editor.getShapeGeometry(childId)
191
+
192
+ return doesGeometryOverlapPolygon(geometry, parentPolygonInShapeShape)
193
+ })
194
+ }
195
+
196
+ /**
197
+ * @public
198
+ */
199
+ export function doesGeometryOverlapPolygon(
200
+ geometry: Geometry2d,
201
+ parentCornersInShapeSpace: Vec[]
202
+ ): boolean {
203
+ // If the child is a group, check if any of its children overlap the box
204
+ if (geometry instanceof Group2d) {
205
+ return geometry.children.some((childGeometry) =>
206
+ doesGeometryOverlapPolygon(childGeometry, parentCornersInShapeSpace)
207
+ )
208
+ }
209
+
210
+ // Otherwise, check if the geometry overlaps the box
211
+ const { vertices, center, isFilled, isEmptyLabel, isClosed } = geometry
212
+
213
+ // We'll do things in order of cheapest to most expensive checks
214
+
215
+ // Skip empty labels
216
+ if (isEmptyLabel) return false
217
+
218
+ // If any of the shape's vertices are inside the occluder, it's inside
219
+ if (vertices.some((v) => pointInPolygon(v, parentCornersInShapeSpace))) {
220
+ return true
221
+ }
222
+
223
+ // If the shape is filled and closed and its center is inside the parent, it's inside
224
+ if (isClosed) {
225
+ if (isFilled) {
226
+ // If closed and filled, check if the center is inside the parent
227
+ if (pointInPolygon(center, parentCornersInShapeSpace)) {
228
+ return true
229
+ }
230
+
231
+ // ..then, slightly more expensive check, see the shape covers the entire parent but not its center
232
+ if (parentCornersInShapeSpace.every((v) => pointInPolygon(v, vertices))) {
233
+ return true
234
+ }
235
+ }
236
+
237
+ // If any the shape's vertices intersect the edge of the occluder, it's inside.
238
+ // for example when a rotated rectangle is moved over the corner of a parent rectangle
239
+ // If the child shape is closed, intersect as a polygon
240
+ if (polygonsIntersect(parentCornersInShapeSpace, vertices)) {
241
+ return true
242
+ }
243
+ } else {
244
+ // if the child shape is not closed, intersect as a polyline
245
+ if (polygonIntersectsPolyline(parentCornersInShapeSpace, vertices)) {
246
+ return true
247
+ }
248
+ }
249
+
250
+ // If none of the above checks passed, the shape is outside the parent
251
+ return false
252
+ }
253
+
254
+ /**
255
+ * Get the shapes that will be reparented to new parents when the shapes are dropped.
256
+ *
257
+ * @param editor - The editor instance.
258
+ * @param shapes - The shapes to check.
259
+ * @param cb - A callback to filter out certain shapes.
260
+ * @returns An object with the shapes that will be reparented to new parents and the shapes that will be reparented to the page or their ancestral group.
261
+ *
262
+ * @public
263
+ */
264
+ export function getDroppedShapesToNewParents(
265
+ editor: Editor,
266
+ shapes: Set<TLShape> | TLShape[],
267
+ cb?: (shape: TLShape, parent: TLShape) => boolean
268
+ ) {
269
+ const shapesToActuallyCheck = new Set<TLShape>(shapes)
270
+ const movingGroups = new Set<TLGroupShape>()
271
+
272
+ for (const shape of shapes) {
273
+ const parent = editor.getShapeParent(shape)
274
+ if (parent && editor.isShapeOfType<TLGroupShape>(parent, 'group')) {
275
+ if (!movingGroups.has(parent)) {
276
+ movingGroups.add(parent)
277
+ }
278
+ }
279
+ }
280
+
281
+ // If all of a group's children are moving, then move the group instead
282
+ for (const movingGroup of movingGroups) {
283
+ const children = compact(
284
+ editor.getSortedChildIdsForParent(movingGroup).map((id) => editor.getShape(id))
285
+ )
286
+ for (const child of children) {
287
+ shapesToActuallyCheck.delete(child)
288
+ }
289
+ shapesToActuallyCheck.add(movingGroup)
290
+ }
291
+
292
+ // this could be cached and passed in
293
+ const shapeGroupIds = new Map<TLShapeId, TLShapeId | undefined>()
294
+
295
+ const reparenting = new Map<TLShapeId, TLShape[]>()
296
+
297
+ const remainingShapesToReparent = new Set(shapesToActuallyCheck)
298
+
299
+ const potentialParentShapes = editor
300
+ .getCurrentPageShapesSorted()
301
+ // filter out any shapes that aren't frames or that are included among the provided shapes
302
+ .filter(
303
+ (s) =>
304
+ editor.getShapeUtil(s).canReceiveNewChildrenOfType?.(s, s.type) &&
305
+ !remainingShapesToReparent.has(s)
306
+ )
307
+
308
+ parentCheck: for (let i = potentialParentShapes.length - 1; i >= 0; i--) {
309
+ const parentShape = potentialParentShapes[i]
310
+ const parentShapeContainingGroupId = editor.findShapeAncestor(parentShape, (s) =>
311
+ editor.isShapeOfType<TLGroupShape>(s, 'group')
312
+ )?.id
313
+
314
+ const parentGeometry = editor.getShapeGeometry(parentShape)
315
+ const parentPageTransform = editor.getShapePageTransform(parentShape)
316
+ const parentPageMaskVertices = editor.getShapeMask(parentShape)
317
+ const parentPageCorners = parentPageTransform.applyToPoints(parentGeometry.vertices)
318
+ const parentPagePolygon = parentPageMaskVertices
319
+ ? intersectPolygonPolygon(parentPageMaskVertices, parentPageCorners)
320
+ : parentPageCorners
321
+
322
+ if (!parentPagePolygon) continue parentCheck
323
+
324
+ const childrenToReparent = []
325
+
326
+ // For each of the dropping shapes...
327
+ shapeCheck: for (const shape of remainingShapesToReparent) {
328
+ // Don't reparent a frame to itself
329
+ if (parentShape.id === shape.id) continue shapeCheck
330
+
331
+ // Use the callback to filter out certain shapes
332
+ if (cb && !cb(shape, parentShape)) continue shapeCheck
333
+
334
+ if (!shapeGroupIds.has(shape.id)) {
335
+ shapeGroupIds.set(
336
+ shape.id,
337
+ editor.findShapeAncestor(shape, (s) => editor.isShapeOfType<TLGroupShape>(s, 'group'))?.id
338
+ )
339
+ }
340
+
341
+ const shapeGroupId = shapeGroupIds.get(shape.id)
342
+
343
+ // Are the shape and the parent part of different groups?
344
+ if (shapeGroupId !== parentShapeContainingGroupId) continue shapeCheck
345
+
346
+ // Is the shape is actually the ancestor of the parent?
347
+ if (editor.findShapeAncestor(parentShape, (s) => shape.id === s.id)) continue shapeCheck
348
+
349
+ // Convert the parent polygon to the shape's space
350
+ const parentPolygonInShapeSpace = editor
351
+ .getShapePageTransform(shape)
352
+ .clone()
353
+ .invert()
354
+ .applyToPoints(parentPagePolygon)
355
+
356
+ // If the shape overlaps the parent polygon, reparent it to that parent
357
+ if (doesGeometryOverlapPolygon(editor.getShapeGeometry(shape), parentPolygonInShapeSpace)) {
358
+ // Use the util to check if the shape can be reparented to the parent
359
+ if (
360
+ !editor.getShapeUtil(parentShape).canReceiveNewChildrenOfType?.(parentShape, shape.type)
361
+ )
362
+ continue shapeCheck
363
+
364
+ if (shape.parentId !== parentShape.id) {
365
+ childrenToReparent.push(shape)
366
+ }
367
+ remainingShapesToReparent.delete(shape)
368
+ continue shapeCheck
369
+ }
370
+ }
371
+
372
+ if (childrenToReparent.length) {
373
+ reparenting.set(parentShape.id, childrenToReparent)
374
+ }
375
+ }
376
+
377
+ return {
378
+ // these are the shapes that will be reparented to new parents
379
+ reparenting,
380
+ // these are the shapes that will be reparented to the page or their ancestral group
381
+ remainingShapesToReparent,
382
+ }
383
+ }
@@ -1,10 +1,10 @@
1
1
  import { getSchema, JSONContent, Editor as TTEditor } from '@tiptap/core'
2
- import { Node } from '@tiptap/pm/model'
2
+ import { Node, Schema } from '@tiptap/pm/model'
3
3
  import { EditorProviderProps } from '@tiptap/react'
4
4
  import { TLRichText } from '@tldraw/tlschema'
5
- import { assert } from '@tldraw/utils'
5
+ import { assert, WeakCache } from '@tldraw/utils'
6
6
  import { Editor } from '../editor/Editor'
7
- import { TLFontFace } from '../editor/managers/FontManager'
7
+ import { TLFontFace } from '../editor/managers/FontManager/FontManager'
8
8
 
9
9
  /**
10
10
  * This is the TipTap editor! Docs are {@link https://tiptap.dev/docs}.
@@ -39,6 +39,11 @@ export type RichTextFontVisitor = (
39
39
  addFont: (font: TLFontFace) => void
40
40
  ) => RichTextFontVisitorState
41
41
 
42
+ const schemaCache = new WeakCache<EditorProviderProps, Schema>()
43
+ export function getTipTapSchema(tipTapConfig: EditorProviderProps) {
44
+ return schemaCache.get(tipTapConfig, () => getSchema(tipTapConfig.extensions ?? []))
45
+ }
46
+
42
47
  /** @public */
43
48
  export function getFontsFromRichText(
44
49
  editor: Editor,
@@ -49,7 +54,8 @@ export function getFontsFromRichText(
49
54
  assert(tipTapConfig, 'textOptions.tipTapConfig must be set to use rich text')
50
55
  assert(addFontsFromNode, 'textOptions.addFontsFromNode must be set to use rich text')
51
56
 
52
- const schema = getSchema(tipTapConfig.extensions ?? [])
57
+ const schema = getTipTapSchema(tipTapConfig)
58
+
53
59
  const rootNode = Node.fromJSON(schema, richText as JSONContent)
54
60
 
55
61
  const fonts = new Set<TLFontFace>()