@tldraw/editor 3.16.0-canary.ffdf566dd0a8 → 3.16.0-internal.71f83a8a571b

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 (218) hide show
  1. package/dist-cjs/index.d.ts +137 -126
  2. package/dist-cjs/index.js +6 -6
  3. package/dist-cjs/index.js.map +2 -2
  4. package/dist-cjs/lib/TldrawEditor.js +8 -8
  5. package/dist-cjs/lib/TldrawEditor.js.map +2 -2
  6. package/dist-cjs/lib/components/MenuClickCapture.js +0 -5
  7. package/dist-cjs/lib/components/MenuClickCapture.js.map +2 -2
  8. package/dist-cjs/lib/components/Shape.js +7 -10
  9. package/dist-cjs/lib/components/Shape.js.map +2 -2
  10. package/dist-cjs/lib/components/default-components/DefaultCanvas.js +14 -23
  11. package/dist-cjs/lib/components/default-components/DefaultCanvas.js.map +2 -2
  12. package/dist-cjs/lib/components/default-components/DefaultCollaboratorHint.js +1 -1
  13. package/dist-cjs/lib/components/default-components/DefaultCollaboratorHint.js.map +1 -1
  14. package/dist-cjs/lib/components/default-components/DefaultErrorFallback.js +1 -1
  15. package/dist-cjs/lib/components/default-components/DefaultErrorFallback.js.map +2 -2
  16. package/dist-cjs/lib/components/default-components/DefaultScribble.js +1 -1
  17. package/dist-cjs/lib/components/default-components/DefaultScribble.js.map +2 -2
  18. package/dist-cjs/lib/components/default-components/DefaultShapeIndicator.js +9 -1
  19. package/dist-cjs/lib/components/default-components/DefaultShapeIndicator.js.map +2 -2
  20. package/dist-cjs/lib/config/TLUserPreferences.js +1 -1
  21. package/dist-cjs/lib/config/TLUserPreferences.js.map +2 -2
  22. package/dist-cjs/lib/editor/Editor.js +67 -128
  23. package/dist-cjs/lib/editor/Editor.js.map +2 -2
  24. package/dist-cjs/lib/editor/derivations/notVisibleShapes.js +4 -0
  25. package/dist-cjs/lib/editor/derivations/notVisibleShapes.js.map +2 -2
  26. package/dist-cjs/lib/editor/derivations/parentsToChildren.js.map +2 -2
  27. package/dist-cjs/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.js +1 -1
  28. package/dist-cjs/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.js.map +2 -2
  29. package/dist-cjs/lib/editor/shapes/BaseBoxShapeUtil.js.map +1 -1
  30. package/dist-cjs/lib/editor/shapes/ShapeUtil.js +23 -0
  31. package/dist-cjs/lib/editor/shapes/ShapeUtil.js.map +2 -2
  32. package/dist-cjs/lib/editor/tools/BaseBoxShapeTool/BaseBoxShapeTool.js.map +2 -2
  33. package/dist-cjs/lib/editor/types/misc-types.js.map +1 -1
  34. package/dist-cjs/lib/exports/getSvgJsx.js +35 -16
  35. package/dist-cjs/lib/exports/getSvgJsx.js.map +2 -2
  36. package/dist-cjs/lib/hooks/useCanvasEvents.js +44 -35
  37. package/dist-cjs/lib/hooks/useCanvasEvents.js.map +2 -2
  38. package/dist-cjs/lib/hooks/useDocumentEvents.js +5 -5
  39. package/dist-cjs/lib/hooks/useDocumentEvents.js.map +2 -2
  40. package/dist-cjs/lib/hooks/useFixSafariDoubleTapZoomPencilEvents.js +1 -2
  41. package/dist-cjs/lib/hooks/useFixSafariDoubleTapZoomPencilEvents.js.map +2 -2
  42. package/dist-cjs/lib/hooks/useGestureEvents.js +1 -1
  43. package/dist-cjs/lib/hooks/useGestureEvents.js.map +2 -2
  44. package/dist-cjs/lib/hooks/useHandleEvents.js +3 -3
  45. package/dist-cjs/lib/hooks/useHandleEvents.js.map +2 -2
  46. package/dist-cjs/lib/hooks/usePassThroughMouseOverEvents.js +4 -1
  47. package/dist-cjs/lib/hooks/usePassThroughMouseOverEvents.js.map +2 -2
  48. package/dist-cjs/lib/hooks/usePassThroughWheelEvents.js +4 -1
  49. package/dist-cjs/lib/hooks/usePassThroughWheelEvents.js.map +2 -2
  50. package/dist-cjs/lib/hooks/useSelectionEvents.js +4 -4
  51. package/dist-cjs/lib/hooks/useSelectionEvents.js.map +2 -2
  52. package/dist-cjs/lib/license/LicenseManager.js +140 -53
  53. package/dist-cjs/lib/license/LicenseManager.js.map +2 -2
  54. package/dist-cjs/lib/license/LicenseProvider.js +39 -1
  55. package/dist-cjs/lib/license/LicenseProvider.js.map +2 -2
  56. package/dist-cjs/lib/license/Watermark.js +73 -11
  57. package/dist-cjs/lib/license/Watermark.js.map +3 -3
  58. package/dist-cjs/lib/license/useLicenseManagerState.js.map +2 -2
  59. package/dist-cjs/lib/options.js +6 -0
  60. package/dist-cjs/lib/options.js.map +2 -2
  61. package/dist-cjs/lib/primitives/Box.js +3 -0
  62. package/dist-cjs/lib/primitives/Box.js.map +2 -2
  63. package/dist-cjs/lib/primitives/Vec.js +0 -4
  64. package/dist-cjs/lib/primitives/Vec.js.map +2 -2
  65. package/dist-cjs/lib/primitives/geometry/Geometry2d.js +50 -20
  66. package/dist-cjs/lib/primitives/geometry/Geometry2d.js.map +2 -2
  67. package/dist-cjs/lib/primitives/geometry/Group2d.js +8 -1
  68. package/dist-cjs/lib/primitives/geometry/Group2d.js.map +2 -2
  69. package/dist-cjs/lib/utils/dom.js +12 -1
  70. package/dist-cjs/lib/utils/dom.js.map +2 -2
  71. package/dist-cjs/lib/utils/getPointerInfo.js +2 -2
  72. package/dist-cjs/lib/utils/getPointerInfo.js.map +2 -2
  73. package/dist-cjs/lib/utils/reparenting.js +2 -35
  74. package/dist-cjs/lib/utils/reparenting.js.map +3 -3
  75. package/dist-cjs/version.js +3 -3
  76. package/dist-cjs/version.js.map +1 -1
  77. package/dist-esm/index.d.mts +137 -126
  78. package/dist-esm/index.mjs +9 -7
  79. package/dist-esm/index.mjs.map +2 -2
  80. package/dist-esm/lib/TldrawEditor.mjs +9 -9
  81. package/dist-esm/lib/TldrawEditor.mjs.map +2 -2
  82. package/dist-esm/lib/components/MenuClickCapture.mjs +0 -5
  83. package/dist-esm/lib/components/MenuClickCapture.mjs.map +2 -2
  84. package/dist-esm/lib/components/Shape.mjs +7 -10
  85. package/dist-esm/lib/components/Shape.mjs.map +2 -2
  86. package/dist-esm/lib/components/default-components/DefaultCanvas.mjs +15 -24
  87. package/dist-esm/lib/components/default-components/DefaultCanvas.mjs.map +2 -2
  88. package/dist-esm/lib/components/default-components/DefaultCollaboratorHint.mjs +1 -1
  89. package/dist-esm/lib/components/default-components/DefaultCollaboratorHint.mjs.map +1 -1
  90. package/dist-esm/lib/components/default-components/DefaultErrorFallback.mjs +1 -1
  91. package/dist-esm/lib/components/default-components/DefaultErrorFallback.mjs.map +2 -2
  92. package/dist-esm/lib/components/default-components/DefaultScribble.mjs +1 -1
  93. package/dist-esm/lib/components/default-components/DefaultScribble.mjs.map +2 -2
  94. package/dist-esm/lib/components/default-components/DefaultShapeIndicator.mjs +9 -1
  95. package/dist-esm/lib/components/default-components/DefaultShapeIndicator.mjs.map +2 -2
  96. package/dist-esm/lib/config/TLUserPreferences.mjs +1 -1
  97. package/dist-esm/lib/config/TLUserPreferences.mjs.map +2 -2
  98. package/dist-esm/lib/editor/Editor.mjs +67 -128
  99. package/dist-esm/lib/editor/Editor.mjs.map +2 -2
  100. package/dist-esm/lib/editor/derivations/notVisibleShapes.mjs +4 -0
  101. package/dist-esm/lib/editor/derivations/notVisibleShapes.mjs.map +2 -2
  102. package/dist-esm/lib/editor/derivations/parentsToChildren.mjs.map +2 -2
  103. package/dist-esm/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.mjs +1 -1
  104. package/dist-esm/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.mjs.map +2 -2
  105. package/dist-esm/lib/editor/shapes/BaseBoxShapeUtil.mjs.map +1 -1
  106. package/dist-esm/lib/editor/shapes/ShapeUtil.mjs +23 -0
  107. package/dist-esm/lib/editor/shapes/ShapeUtil.mjs.map +2 -2
  108. package/dist-esm/lib/editor/tools/BaseBoxShapeTool/BaseBoxShapeTool.mjs.map +2 -2
  109. package/dist-esm/lib/exports/getSvgJsx.mjs +36 -16
  110. package/dist-esm/lib/exports/getSvgJsx.mjs.map +2 -2
  111. package/dist-esm/lib/hooks/useCanvasEvents.mjs +47 -37
  112. package/dist-esm/lib/hooks/useCanvasEvents.mjs.map +2 -2
  113. package/dist-esm/lib/hooks/useDocumentEvents.mjs +11 -6
  114. package/dist-esm/lib/hooks/useDocumentEvents.mjs.map +2 -2
  115. package/dist-esm/lib/hooks/useFixSafariDoubleTapZoomPencilEvents.mjs +2 -3
  116. package/dist-esm/lib/hooks/useFixSafariDoubleTapZoomPencilEvents.mjs.map +2 -2
  117. package/dist-esm/lib/hooks/useGestureEvents.mjs +2 -2
  118. package/dist-esm/lib/hooks/useGestureEvents.mjs.map +2 -2
  119. package/dist-esm/lib/hooks/useHandleEvents.mjs +9 -4
  120. package/dist-esm/lib/hooks/useHandleEvents.mjs.map +2 -2
  121. package/dist-esm/lib/hooks/usePassThroughMouseOverEvents.mjs +4 -1
  122. package/dist-esm/lib/hooks/usePassThroughMouseOverEvents.mjs.map +2 -2
  123. package/dist-esm/lib/hooks/usePassThroughWheelEvents.mjs +4 -1
  124. package/dist-esm/lib/hooks/usePassThroughWheelEvents.mjs.map +2 -2
  125. package/dist-esm/lib/hooks/useSelectionEvents.mjs +6 -5
  126. package/dist-esm/lib/hooks/useSelectionEvents.mjs.map +2 -2
  127. package/dist-esm/lib/license/LicenseManager.mjs +141 -54
  128. package/dist-esm/lib/license/LicenseManager.mjs.map +2 -2
  129. package/dist-esm/lib/license/LicenseProvider.mjs +39 -2
  130. package/dist-esm/lib/license/LicenseProvider.mjs.map +2 -2
  131. package/dist-esm/lib/license/Watermark.mjs +74 -12
  132. package/dist-esm/lib/license/Watermark.mjs.map +3 -3
  133. package/dist-esm/lib/license/useLicenseManagerState.mjs.map +2 -2
  134. package/dist-esm/lib/options.mjs +6 -0
  135. package/dist-esm/lib/options.mjs.map +2 -2
  136. package/dist-esm/lib/primitives/Box.mjs +4 -1
  137. package/dist-esm/lib/primitives/Box.mjs.map +2 -2
  138. package/dist-esm/lib/primitives/Vec.mjs +0 -4
  139. package/dist-esm/lib/primitives/Vec.mjs.map +2 -2
  140. package/dist-esm/lib/primitives/geometry/Geometry2d.mjs +53 -21
  141. package/dist-esm/lib/primitives/geometry/Geometry2d.mjs.map +2 -2
  142. package/dist-esm/lib/primitives/geometry/Group2d.mjs +8 -1
  143. package/dist-esm/lib/primitives/geometry/Group2d.mjs.map +2 -2
  144. package/dist-esm/lib/utils/dom.mjs +12 -1
  145. package/dist-esm/lib/utils/dom.mjs.map +2 -2
  146. package/dist-esm/lib/utils/getPointerInfo.mjs +2 -2
  147. package/dist-esm/lib/utils/getPointerInfo.mjs.map +2 -2
  148. package/dist-esm/lib/utils/reparenting.mjs +3 -40
  149. package/dist-esm/lib/utils/reparenting.mjs.map +2 -2
  150. package/dist-esm/version.mjs +3 -3
  151. package/dist-esm/version.mjs.map +1 -1
  152. package/editor.css +308 -292
  153. package/package.json +14 -37
  154. package/src/index.ts +4 -9
  155. package/src/lib/TldrawEditor.tsx +14 -21
  156. package/src/lib/components/MenuClickCapture.tsx +0 -8
  157. package/src/lib/components/Shape.tsx +6 -12
  158. package/src/lib/components/default-components/DefaultCanvas.tsx +12 -23
  159. package/src/lib/components/default-components/DefaultCollaboratorHint.tsx +1 -1
  160. package/src/lib/components/default-components/DefaultErrorFallback.tsx +1 -1
  161. package/src/lib/components/default-components/DefaultScribble.tsx +1 -1
  162. package/src/lib/components/default-components/DefaultShapeIndicator.tsx +5 -1
  163. package/src/lib/config/TLUserPreferences.ts +1 -1
  164. package/src/lib/editor/Editor.test.ts +108 -11
  165. package/src/lib/editor/Editor.ts +112 -195
  166. package/src/lib/editor/derivations/notVisibleShapes.ts +6 -0
  167. package/src/lib/editor/derivations/parentsToChildren.ts +1 -1
  168. package/src/lib/editor/managers/ClickManager/ClickManager.test.ts +15 -14
  169. package/src/lib/editor/managers/EdgeScrollManager/EdgeScrollManager.test.ts +16 -15
  170. package/src/lib/editor/managers/FocusManager/FocusManager.test.ts +49 -48
  171. package/src/lib/editor/managers/FontManager/FontManager.test.ts +38 -27
  172. package/src/lib/editor/managers/HistoryManager/HistoryManager.test.ts +7 -6
  173. package/src/lib/editor/managers/ScribbleManager/ScribbleManager.test.ts +12 -11
  174. package/src/lib/editor/managers/SnapManager/SnapManager.test.ts +57 -50
  175. package/src/lib/editor/managers/TextManager/TextManager.test.ts +51 -26
  176. package/src/lib/editor/managers/TickManager/TickManager.test.ts +14 -13
  177. package/src/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.test.ts +21 -26
  178. package/src/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.ts +1 -1
  179. package/src/lib/editor/shapes/BaseBoxShapeUtil.tsx +2 -2
  180. package/src/lib/editor/shapes/ShapeUtil.ts +51 -8
  181. package/src/lib/editor/tools/BaseBoxShapeTool/BaseBoxShapeTool.ts +2 -1
  182. package/src/lib/editor/types/misc-types.ts +0 -6
  183. package/src/lib/exports/getSvgJsx.test.ts +874 -0
  184. package/src/lib/exports/getSvgJsx.tsx +78 -21
  185. package/src/lib/hooks/useCanvasEvents.ts +60 -47
  186. package/src/lib/hooks/useDocumentEvents.ts +11 -6
  187. package/src/lib/hooks/useFixSafariDoubleTapZoomPencilEvents.ts +2 -2
  188. package/src/lib/hooks/useGestureEvents.ts +2 -2
  189. package/src/lib/hooks/useHandleEvents.ts +9 -4
  190. package/src/lib/hooks/usePassThroughMouseOverEvents.ts +4 -1
  191. package/src/lib/hooks/usePassThroughWheelEvents.ts +6 -1
  192. package/src/lib/hooks/useSelectionEvents.ts +6 -5
  193. package/src/lib/license/LicenseManager.test.ts +724 -383
  194. package/src/lib/license/LicenseManager.ts +201 -58
  195. package/src/lib/license/LicenseProvider.tsx +74 -2
  196. package/src/lib/license/Watermark.test.tsx +2 -1
  197. package/src/lib/license/Watermark.tsx +79 -12
  198. package/src/lib/license/useLicenseManagerState.ts +2 -2
  199. package/src/lib/options.ts +6 -0
  200. package/src/lib/primitives/Box.test.ts +126 -0
  201. package/src/lib/primitives/Box.ts +10 -1
  202. package/src/lib/primitives/Vec.ts +0 -5
  203. package/src/lib/primitives/geometry/Geometry2d.test.ts +420 -0
  204. package/src/lib/primitives/geometry/Geometry2d.ts +78 -21
  205. package/src/lib/primitives/geometry/Group2d.ts +10 -1
  206. package/src/lib/test/InFrontOfTheCanvas.test.tsx +187 -0
  207. package/src/lib/utils/dom.test.ts +94 -0
  208. package/src/lib/utils/dom.ts +38 -1
  209. package/src/lib/utils/getPointerInfo.ts +2 -1
  210. package/src/lib/utils/reparenting.ts +3 -69
  211. package/src/lib/utils/sync/LocalIndexedDb.test.ts +2 -1
  212. package/src/lib/utils/sync/TLLocalSyncClient.test.ts +15 -15
  213. package/src/version.ts +3 -3
  214. package/dist-cjs/lib/utils/nearestMultiple.js +0 -34
  215. package/dist-cjs/lib/utils/nearestMultiple.js.map +0 -7
  216. package/dist-esm/lib/utils/nearestMultiple.mjs +0 -14
  217. package/dist-esm/lib/utils/nearestMultiple.mjs.map +0 -7
  218. package/src/lib/utils/nearestMultiple.ts +0 -13
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@tldraw/editor",
3
3
  "description": "tldraw infinite canvas SDK (editor).",
4
- "version": "3.16.0-canary.ffdf566dd0a8",
4
+ "version": "3.16.0-internal.71f83a8a571b",
5
5
  "author": {
6
6
  "name": "tldraw Inc.",
7
7
  "email": "hello@tldraw.com"
@@ -34,27 +34,28 @@
34
34
  "src"
35
35
  ],
36
36
  "scripts": {
37
- "test-ci": "lazy inherit",
38
- "test": "yarn run -T jest",
37
+ "test-ci": "yarn run -T vitest run --passWithNoTests",
38
+ "test": "yarn run -T vitest --passWithNoTests",
39
39
  "benchmark": "yarn run -T tsx ./internal/scripts/benchmark.ts",
40
- "test-coverage": "lazy inherit",
40
+ "test-coverage": "yarn run -T vitest run --coverage --passWithNoTests",
41
41
  "build": "yarn run -T tsx ../../internal/scripts/build-package.ts",
42
42
  "build-api": "yarn run -T tsx ../../internal/scripts/build-api.ts",
43
43
  "prepack": "yarn run -T tsx ../../internal/scripts/prepack.ts",
44
44
  "postpack": "../../internal/scripts/postpack.sh",
45
45
  "pack-tarball": "yarn pack",
46
- "lint": "yarn run -T tsx ../../internal/scripts/lint.ts"
46
+ "lint": "yarn run -T tsx ../../internal/scripts/lint.ts",
47
+ "context": "yarn run -T tsx ../../internal/scripts/context.ts"
47
48
  },
48
49
  "dependencies": {
49
50
  "@tiptap/core": "^2.9.1",
50
51
  "@tiptap/pm": "^2.9.1",
51
52
  "@tiptap/react": "^2.9.1",
52
- "@tldraw/state": "3.16.0-canary.ffdf566dd0a8",
53
- "@tldraw/state-react": "3.16.0-canary.ffdf566dd0a8",
54
- "@tldraw/store": "3.16.0-canary.ffdf566dd0a8",
55
- "@tldraw/tlschema": "3.16.0-canary.ffdf566dd0a8",
56
- "@tldraw/utils": "3.16.0-canary.ffdf566dd0a8",
57
- "@tldraw/validate": "3.16.0-canary.ffdf566dd0a8",
53
+ "@tldraw/state": "3.16.0-internal.71f83a8a571b",
54
+ "@tldraw/state-react": "3.16.0-internal.71f83a8a571b",
55
+ "@tldraw/store": "3.16.0-internal.71f83a8a571b",
56
+ "@tldraw/tlschema": "3.16.0-internal.71f83a8a571b",
57
+ "@tldraw/utils": "3.16.0-internal.71f83a8a571b",
58
+ "@tldraw/validate": "3.16.0-internal.71f83a8a571b",
58
59
  "@types/core-js": "^2.5.8",
59
60
  "@use-gesture/react": "^10.3.1",
60
61
  "classnames": "^2.5.1",
@@ -69,41 +70,17 @@
69
70
  },
70
71
  "devDependencies": {
71
72
  "@peculiar/webcrypto": "^1.5.0",
72
- "@testing-library/jest-dom": "^5.17.0",
73
73
  "@testing-library/react": "^15.0.7",
74
74
  "@types/benchmark": "^2.1.5",
75
75
  "@types/react": "^18.3.18",
76
76
  "@types/wicg-file-system-access": "^2020.9.8",
77
77
  "benchmark": "^2.1.4",
78
78
  "fake-indexeddb": "^4.0.2",
79
- "jest-canvas-mock": "^2.5.2",
80
- "jest-environment-jsdom": "^29.7.0",
81
79
  "lazyrepo": "0.0.0-alpha.27",
82
80
  "react": "^18.3.1",
83
81
  "react-dom": "^18.3.1",
84
- "resize-observer-polyfill": "^1.5.1"
85
- },
86
- "jest": {
87
- "preset": "../../internal/config/jest/node/jest-preset.js",
88
- "testEnvironment": "../../../packages/utils/patchedJestJsDom.js",
89
- "fakeTimers": {
90
- "enableGlobally": true
91
- },
92
- "testPathIgnorePatterns": [
93
- "^.+\\.*.css$"
94
- ],
95
- "moduleNameMapper": {
96
- "^~(.*)": "<rootDir>/src/$1",
97
- "\\.(css|less|scss|sass)$": "identity-obj-proxy"
98
- },
99
- "setupFiles": [
100
- "raf/polyfill",
101
- "jest-canvas-mock",
102
- "<rootDir>/setupTests.js"
103
- ],
104
- "setupFilesAfterEnv": [
105
- "../../internal/config/setupJest.ts"
106
- ]
82
+ "resize-observer-polyfill": "^1.5.1",
83
+ "vitest": "^3.2.4"
107
84
  },
108
85
  "module": "dist-esm/index.mjs",
109
86
  "source": "src/index.ts",
package/src/index.ts CHANGED
@@ -268,7 +268,6 @@ export {
268
268
  type TLGetShapeAtPointOptions,
269
269
  type TLImageExportOptions,
270
270
  type TLSvgExportOptions,
271
- type TLSvgOptions,
272
271
  type TLUpdatePointerOptions,
273
272
  } from './lib/editor/types/misc-types'
274
273
  export {
@@ -331,9 +330,11 @@ export {
331
330
  type InvalidLicenseReason,
332
331
  type LicenseFromKeyResult,
333
332
  type LicenseInfo,
333
+ type LicenseState,
334
334
  type TestEnvironment,
335
335
  type ValidLicenseKeyResult,
336
336
  } from './lib/license/LicenseManager'
337
+ export { LICENSE_TIMEOUT } from './lib/license/LicenseProvider'
337
338
  export { defaultTldrawOptions, type TldrawOptions } from './lib/options'
338
339
  export {
339
340
  Box,
@@ -446,10 +447,12 @@ export {
446
447
  export {
447
448
  activeElementShouldCaptureKeys,
448
449
  loopToHtmlElement,
450
+ markEventAsHandled,
449
451
  preventDefault,
450
452
  releasePointerCapture,
451
453
  setPointerCapture,
452
454
  stopEventPropagation,
455
+ wasEventAlreadyHandled,
453
456
  } from './lib/utils/dom'
454
457
  export { EditorAtom } from './lib/utils/EditorAtom'
455
458
  export { getIncrementedName } from './lib/utils/getIncrementedName'
@@ -485,14 +488,6 @@ export { type TLStoreWithStatus } from './lib/utils/sync/StoreWithStatus'
485
488
  export { uniq } from './lib/utils/uniq'
486
489
  export { openWindow } from './lib/utils/window-open'
487
490
 
488
- /**
489
- * @deprecated Licensing is now enabled in the tldraw SDK.
490
- * @public */
491
- export function debugEnableLicensing() {
492
- // noop
493
- return
494
- }
495
-
496
491
  registerTldrawLibraryVersion(
497
492
  (globalThis as any).TLDRAW_LIBRARY_NAME,
498
493
  (globalThis as any).TLDRAW_LIBRARY_VERSION,
@@ -1,9 +1,10 @@
1
1
  import { MigrationSequence, Store } from '@tldraw/store'
2
2
  import { TLShape, TLStore, TLStoreSnapshot } from '@tldraw/tlschema'
3
- import { Required, annotateError } from '@tldraw/utils'
3
+ import { annotateError, Required } from '@tldraw/utils'
4
+ import classNames from 'classnames'
4
5
  import React, {
5
- ReactNode,
6
6
  memo,
7
+ ReactNode,
7
8
  useCallback,
8
9
  useEffect,
9
10
  useLayoutEffect,
@@ -12,16 +13,14 @@ import React, {
12
13
  useState,
13
14
  useSyncExternalStore,
14
15
  } from 'react'
15
-
16
- import classNames from 'classnames'
17
16
  import { version } from '../version'
18
- import { OptionalErrorBoundary } from './components/ErrorBoundary'
19
17
  import { DefaultErrorFallback } from './components/default-components/DefaultErrorFallback'
20
- import { TLEditorSnapshot } from './config/TLEditorSnapshot'
18
+ import { OptionalErrorBoundary } from './components/ErrorBoundary'
21
19
  import { TLStoreBaseOptions } from './config/createTLStore'
22
- import { TLUser, createTLUser } from './config/createTLUser'
20
+ import { createTLUser, TLUser } from './config/createTLUser'
23
21
  import { TLAnyBindingUtilConstructor } from './config/defaultBindings'
24
22
  import { TLAnyShapeUtilConstructor } from './config/defaultShapes'
23
+ import { TLEditorSnapshot } from './config/TLEditorSnapshot'
25
24
  import { Editor } from './editor/Editor'
26
25
  import { TLStateNodeConstructor } from './editor/tools/StateNode'
27
26
  import { TLCameraOptions } from './editor/types/misc-types'
@@ -45,7 +44,7 @@ import { LicenseProvider } from './license/LicenseProvider'
45
44
  import { Watermark } from './license/Watermark'
46
45
  import { TldrawOptions } from './options'
47
46
  import { TLDeepLinkOptions } from './utils/deepLinks'
48
- import { stopEventPropagation } from './utils/dom'
47
+ import { markEventAsHandled } from './utils/dom'
49
48
  import { TLTextOptions } from './utils/richText'
50
49
  import { TLStoreWithStatus } from './utils/sync/StoreWithStatus'
51
50
 
@@ -189,13 +188,6 @@ export interface TldrawEditorBaseProps {
189
188
  */
190
189
  deepLinks?: true | TLDeepLinkOptions
191
190
 
192
- /**
193
- * Predicate for whether or not a shape should be hidden.
194
- *
195
- * @deprecated Use {@link TldrawEditorBaseProps#getShapeVisibility} instead.
196
- */
197
- isShapeHidden?(shape: TLShape, editor: Editor): boolean
198
-
199
191
  /**
200
192
  * Provides a way to hide shapes.
201
193
  *
@@ -283,7 +275,7 @@ export const TldrawEditor = memo(function TldrawEditor({
283
275
  data-tldraw={version}
284
276
  draggable={false}
285
277
  className={classNames(`${TL_CONTAINER_CLASS} tl-theme__light`, className)}
286
- onPointerDown={stopEventPropagation}
278
+ onPointerDown={markEventAsHandled}
287
279
  tabIndex={-1}
288
280
  role="application"
289
281
  aria-label={_options?.branding ?? 'tldraw'}
@@ -412,8 +404,6 @@ function TldrawEditorWithReadyStore({
412
404
  options,
413
405
  licenseKey,
414
406
  deepLinks: _deepLinks,
415
- // eslint-disable-next-line @typescript-eslint/no-deprecated
416
- isShapeHidden,
417
407
  getShapeVisibility,
418
408
  assetUrls,
419
409
  }: Required<
@@ -473,7 +463,6 @@ function TldrawEditorWithReadyStore({
473
463
  textOptions,
474
464
  options,
475
465
  licenseKey,
476
- isShapeHidden,
477
466
  getShapeVisibility,
478
467
  fontAssetUrls: assetUrls?.fonts,
479
468
  })
@@ -509,7 +498,6 @@ function TldrawEditorWithReadyStore({
509
498
  user,
510
499
  setEditor,
511
500
  licenseKey,
512
- isShapeHidden,
513
501
  getShapeVisibility,
514
502
  textOptions,
515
503
  assetUrls,
@@ -586,8 +574,13 @@ function TldrawEditorWithReadyStore({
586
574
  if (editor !== fontLoadingState?.editor) {
587
575
  fontLoadingState = null
588
576
  }
589
- useEffect(() => {
577
+ useLayoutEffect(() => {
590
578
  if (!editor) return
579
+ if (editor.options.maxFontsToLoadBeforeRender === 0) {
580
+ setFontLoadingState({ editor, isLoaded: true })
581
+ return
582
+ }
583
+
591
584
  let isCancelled = false
592
585
 
593
586
  setFontLoadingState({ editor, isLoaded: false })
@@ -50,12 +50,6 @@ export function MenuClickCapture() {
50
50
  // Do nothing unless we're pointing
51
51
  if (!rPointerState.current.isDown) return
52
52
 
53
- // If we're already dragging, pass on the event as it is
54
- if (rPointerState.current.isDragging) {
55
- canvasEvents.onPointerMove?.(e)
56
- return
57
- }
58
-
59
53
  if (
60
54
  // We're pointing, but are we dragging?
61
55
  Vec.Dist2(rPointerState.current.start, new Vec(e.clientX, e.clientY)) >
@@ -75,8 +69,6 @@ export function MenuClickCapture() {
75
69
  clientY: y,
76
70
  button: 0,
77
71
  })
78
- // call the pointer move with the current pointer position
79
- canvasEvents.onPointerMove?.(e)
80
72
  }
81
73
  },
82
74
  [canvasEvents, editor]
@@ -28,7 +28,6 @@ export const Shape = memo(function Shape({
28
28
  index,
29
29
  backgroundIndex,
30
30
  opacity,
31
- dprMultiple,
32
31
  }: {
33
32
  id: TLShapeId
34
33
  shape: TLShape
@@ -36,7 +35,6 @@ export const Shape = memo(function Shape({
36
35
  index: number
37
36
  backgroundIndex: number
38
37
  opacity: number
39
- dprMultiple: number
40
38
  }) {
41
39
  const editor = useEditor()
42
40
 
@@ -91,18 +89,14 @@ export const Shape = memo(function Shape({
91
89
  }
92
90
 
93
91
  // Width / Height
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)
92
+ const width = Math.max(bounds.width, 1)
93
+ const height = Math.max(bounds.height, 1)
100
94
 
101
95
  if (width !== prev.width || height !== prev.height) {
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')
96
+ setStyleProperty(containerRef.current, 'width', width + 'px')
97
+ setStyleProperty(containerRef.current, 'height', height + 'px')
98
+ setStyleProperty(bgContainerRef.current, 'width', width + 'px')
99
+ setStyleProperty(bgContainerRef.current, 'height', height + 'px')
106
100
  prev.width = width
107
101
  prev.height = height
108
102
  }
@@ -21,8 +21,7 @@ import { Mat } from '../../primitives/Mat'
21
21
  import { Vec } from '../../primitives/Vec'
22
22
  import { toDomPrecision } from '../../primitives/utils'
23
23
  import { debugFlags } from '../../utils/debug-flags'
24
- import { setStyleProperty } from '../../utils/dom'
25
- import { nearestMultiple } from '../../utils/nearestMultiple'
24
+ import { markEventAsHandled, setStyleProperty } from '../../utils/dom'
26
25
  import { GeometryDebuggingView } from '../GeometryDebuggingView'
27
26
  import { LiveCollaborators } from '../LiveCollaborators'
28
27
  import { MenuClickCapture } from '../MenuClickCapture'
@@ -173,10 +172,18 @@ export function DefaultCanvas({ className }: TLCanvasComponentProps) {
173
172
  <LiveCollaborators />
174
173
  </div>
175
174
  </div>
175
+ <div
176
+ className="tl-canvas__in-front"
177
+ onPointerDown={markEventAsHandled}
178
+ onPointerUp={markEventAsHandled}
179
+ onTouchStart={markEventAsHandled}
180
+ onTouchEnd={markEventAsHandled}
181
+ >
182
+ <InFrontOfTheCanvasWrapper />
183
+ </div>
176
184
  <MovingCameraHitTestBlocker />
177
185
  </div>
178
186
  <MenuClickCapture />
179
- <InFrontOfTheCanvasWrapper />
180
187
  </>
181
188
  )
182
189
  }
@@ -390,18 +397,9 @@ function ShapesWithSVGs() {
390
397
 
391
398
  const renderingShapes = useValue('rendering shapes', () => editor.getRenderingShapes(), [editor])
392
399
 
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
-
402
400
  return renderingShapes.map((result) => (
403
401
  <Fragment key={result.id + '_fragment'}>
404
- <Shape {...result} dprMultiple={dprMultiple} />
402
+ <Shape {...result} />
405
403
  <DebugSvgCopy id={result.id} mode="iframe" />
406
404
  </Fragment>
407
405
  ))
@@ -436,19 +434,10 @@ function ShapesToDisplay() {
436
434
 
437
435
  const renderingShapes = useValue('rendering shapes', () => editor.getRenderingShapes(), [editor])
438
436
 
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
-
448
437
  return (
449
438
  <>
450
439
  {renderingShapes.map((result) => (
451
- <Shape key={result.id + '_shape'} {...result} dprMultiple={dprMultiple} />
440
+ <Shape key={result.id + '_shape'} {...result} />
452
441
  ))}
453
442
  {tlenv.isSafari && <ReflowIfNeeded />}
454
443
  </>
@@ -44,7 +44,7 @@ export function DefaultCollaboratorHint({
44
44
  href={`#${cursorHintId}`}
45
45
  color={color}
46
46
  strokeWidth={3}
47
- stroke="var(--color-background)"
47
+ stroke="var(--tl-color-background)"
48
48
  />
49
49
  <use href={`#${cursorHintId}`} color={color} opacity={opacity} />
50
50
  </svg>
@@ -75,7 +75,7 @@ export const DefaultErrorFallback: TLErrorFallbackComponent = ({ error, editor }
75
75
 
76
76
  // if we can't find a theme class from the app or from a parent, we have
77
77
  // to fall back on using a media query:
78
- if (typeof window !== 'undefined' && 'matchMedia' in window) {
78
+ if (typeof window !== 'undefined' && window.matchMedia) {
79
79
  setIsDarkMode(window.matchMedia('(prefers-color-scheme: dark)').matches)
80
80
  }
81
81
  }, [isDarkModeFromApp])
@@ -21,7 +21,7 @@ export function DefaultScribble({ scribble, zoom, color, opacity, className }: T
21
21
  <path
22
22
  className="tl-scribble"
23
23
  d={getSvgPathFromPoints(scribble.points, false)}
24
- stroke={color ?? `var(--color-${scribble.color})`}
24
+ stroke={color ?? `var(--tl-color-${scribble.color})`}
25
25
  fill="none"
26
26
  strokeWidth={8 / zoom}
27
27
  opacity={opacity ?? scribble.opacity}
@@ -87,7 +87,11 @@ export const DefaultShapeIndicator = memo(function DefaultShapeIndicator({
87
87
 
88
88
  return (
89
89
  <svg ref={rIndicator} className={classNames('tl-overlays__item', className)} aria-hidden="true">
90
- <g className="tl-shape-indicator" stroke={color ?? 'var(--color-selected)'} opacity={opacity}>
90
+ <g
91
+ className="tl-shape-indicator"
92
+ stroke={color ?? 'var(--tl-color-selected)'}
93
+ opacity={opacity}
94
+ >
91
95
  <InnerIndicator editor={editor} id={shapeId} />
92
96
  </g>
93
97
  </svg>
@@ -135,7 +135,7 @@ function getRandomColor() {
135
135
 
136
136
  /** @internal */
137
137
  export function userPrefersReducedMotion() {
138
- if (typeof window !== 'undefined' && 'matchMedia' in window) {
138
+ if (typeof window !== 'undefined' && window.matchMedia) {
139
139
  return window.matchMedia?.('(prefers-reduced-motion: reduce)')?.matches ?? false
140
140
  }
141
141
 
@@ -1,3 +1,4 @@
1
+ import { vi } from 'vitest'
1
2
  import {
2
3
  Box,
3
4
  Geometry2d,
@@ -11,6 +12,12 @@ import {
11
12
  } from '../..'
12
13
  import { Editor } from './Editor'
13
14
 
15
+ declare module '@tldraw/tlschema' {
16
+ export interface GlobalShapePropsMap {
17
+ 'my-custom-shape': ICustomShape
18
+ }
19
+ }
20
+
14
21
  type ICustomShape = TLBaseShape<
15
22
  'my-custom-shape',
16
23
  {
@@ -59,8 +66,8 @@ beforeEach(() => {
59
66
  getContainer: () => document.body,
60
67
  })
61
68
  editor.setCameraOptions({ isLocked: true })
62
- editor.setCamera = jest.fn()
63
- editor.user.getAnimationSpeed = jest.fn()
69
+ editor.setCamera = vi.fn()
70
+ editor.user.getAnimationSpeed = vi.fn()
64
71
  })
65
72
 
66
73
  describe('centerOnPoint', () => {
@@ -94,13 +101,13 @@ describe('updateShape', () => {
94
101
 
95
102
  describe('zoomToFit', () => {
96
103
  it('no-op when isLocked is set', () => {
97
- editor.getCurrentPageShapeIds = jest.fn(() => new Set([createShapeId('box1')]))
104
+ editor.getCurrentPageShapeIds = vi.fn(() => new Set([createShapeId('box1')]))
98
105
  editor.zoomToFit()
99
106
  expect(editor.setCamera).not.toHaveBeenCalled()
100
107
  })
101
108
 
102
109
  it('sets camera when isLocked is set and force flag is set', () => {
103
- editor.getCurrentPageShapeIds = jest.fn(() => new Set([createShapeId('box1')]))
110
+ editor.getCurrentPageShapeIds = vi.fn(() => new Set([createShapeId('box1')]))
104
111
  editor.zoomToFit({ force: true })
105
112
  expect(editor.setCamera).toHaveBeenCalled()
106
113
  })
@@ -144,13 +151,13 @@ describe('zoomOut', () => {
144
151
 
145
152
  describe('zoomToSelection', () => {
146
153
  it('no-op when isLocked is set', () => {
147
- editor.getSelectionPageBounds = jest.fn(() => Box.From({ x: 0, y: 0, w: 100, h: 100 }))
154
+ editor.getSelectionPageBounds = vi.fn(() => Box.From({ x: 0, y: 0, w: 100, h: 100 }))
148
155
  editor.zoomToSelection()
149
156
  expect(editor.setCamera).not.toHaveBeenCalled()
150
157
  })
151
158
 
152
159
  it('sets camera when isLocked is set and force flag is set', () => {
153
- editor.getSelectionPageBounds = jest.fn(() => Box.From({ x: 0, y: 0, w: 100, h: 100 }))
160
+ editor.getSelectionPageBounds = vi.fn(() => Box.From({ x: 0, y: 0, w: 100, h: 100 }))
154
161
  editor.zoomToSelection({ force: true })
155
162
  expect(editor.setCamera).toHaveBeenCalled()
156
163
  })
@@ -286,7 +293,7 @@ describe('getShapesAtPoint', () => {
286
293
 
287
294
  it('filters out hidden shapes', () => {
288
295
  // Create a spy to mock isShapeHidden
289
- const isShapeHiddenSpy = jest.spyOn(editor, 'isShapeHidden')
296
+ const isShapeHiddenSpy = vi.spyOn(editor, 'isShapeHidden')
290
297
  isShapeHiddenSpy.mockImplementation((shape) => {
291
298
  return typeof shape === 'string' ? shape === ids.shape3 : shape.id === ids.shape3
292
299
  })
@@ -352,7 +359,7 @@ describe('getShapesAtPoint', () => {
352
359
 
353
360
  it('returns empty array when all shapes are hidden', () => {
354
361
  // Mock all shapes as hidden
355
- const isShapeHiddenSpy = jest.spyOn(editor, 'isShapeHidden')
362
+ const isShapeHiddenSpy = vi.spyOn(editor, 'isShapeHidden')
356
363
  isShapeHiddenSpy.mockReturnValue(true)
357
364
 
358
365
  const shapes = editor.getShapesAtPoint({ x: 50, y: 50 })
@@ -692,7 +699,7 @@ describe('selectAll', () => {
692
699
  const initialSelectedIds = editor.getSelectedShapeIds()
693
700
 
694
701
  // Spy on setSelectedShapes to verify it's not called
695
- const setSelectedShapesSpy = jest.spyOn(editor, 'setSelectedShapes')
702
+ const setSelectedShapesSpy = vi.spyOn(editor, 'setSelectedShapes')
696
703
 
697
704
  // Call selectAll
698
705
  editor.selectAll()
@@ -713,7 +720,7 @@ describe('selectAll', () => {
713
720
  const initialSelectedIds = editor.getSelectedShapeIds()
714
721
 
715
722
  // Spy on setSelectedShapes to verify it's not called
716
- const setSelectedShapesSpy = jest.spyOn(editor, 'setSelectedShapes')
723
+ const setSelectedShapesSpy = vi.spyOn(editor, 'setSelectedShapes')
717
724
 
718
725
  // Call selectAll
719
726
  editor.selectAll()
@@ -818,7 +825,7 @@ describe('selectAll', () => {
818
825
  const initialSelectedIds = Array.from(editor.getSelectedShapeIds())
819
826
 
820
827
  // Spy on setSelectedShapes to verify it's not called
821
- const setSelectedShapesSpy = jest.spyOn(editor, 'setSelectedShapes')
828
+ const setSelectedShapesSpy = vi.spyOn(editor, 'setSelectedShapes')
822
829
 
823
830
  // Call selectAll
824
831
  editor.selectAll()
@@ -832,3 +839,93 @@ describe('selectAll', () => {
832
839
  setSelectedShapesSpy.mockRestore()
833
840
  })
834
841
  })
842
+
843
+ describe('putExternalContent', () => {
844
+ let mockHandler: any
845
+
846
+ beforeEach(() => {
847
+ mockHandler = vi.fn()
848
+ editor.registerExternalContentHandler('text', mockHandler)
849
+ })
850
+
851
+ it('calls external content handler when not readonly', async () => {
852
+ vi.spyOn(editor, 'getIsReadonly').mockReturnValue(false)
853
+
854
+ const info = { type: 'text' as const, text: 'test-data' }
855
+ await editor.putExternalContent(info)
856
+
857
+ expect(mockHandler).toHaveBeenCalledWith(info)
858
+ })
859
+
860
+ it('does not call external content handler when readonly', async () => {
861
+ vi.spyOn(editor, 'getIsReadonly').mockReturnValue(true)
862
+
863
+ const info = { type: 'text' as const, text: 'test-data' }
864
+ await editor.putExternalContent(info)
865
+
866
+ expect(mockHandler).not.toHaveBeenCalled()
867
+ })
868
+
869
+ it('calls external content handler when readonly but force is true', async () => {
870
+ vi.spyOn(editor, 'getIsReadonly').mockReturnValue(true)
871
+
872
+ const info = { type: 'text' as const, text: 'test-data' }
873
+ await editor.putExternalContent(info, { force: true })
874
+
875
+ expect(mockHandler).toHaveBeenCalledWith(info)
876
+ })
877
+
878
+ it('calls external content handler when force is false and not readonly', async () => {
879
+ vi.spyOn(editor, 'getIsReadonly').mockReturnValue(false)
880
+
881
+ const info = { type: 'text' as const, text: 'test-data' }
882
+ await editor.putExternalContent(info, { force: false })
883
+
884
+ expect(mockHandler).toHaveBeenCalledWith(info)
885
+ })
886
+ })
887
+
888
+ describe('replaceExternalContent', () => {
889
+ let mockHandler: any
890
+
891
+ beforeEach(() => {
892
+ mockHandler = vi.fn()
893
+ editor.registerExternalContentHandler('text', mockHandler)
894
+ })
895
+
896
+ it('calls external content handler when not readonly', async () => {
897
+ vi.spyOn(editor, 'getIsReadonly').mockReturnValue(false)
898
+
899
+ const info = { type: 'text' as const, text: 'test-data' }
900
+ await editor.replaceExternalContent(info)
901
+
902
+ expect(mockHandler).toHaveBeenCalledWith(info)
903
+ })
904
+
905
+ it('does not call external content handler when readonly', async () => {
906
+ vi.spyOn(editor, 'getIsReadonly').mockReturnValue(true)
907
+
908
+ const info = { type: 'text' as const, text: 'test-data' }
909
+ await editor.replaceExternalContent(info)
910
+
911
+ expect(mockHandler).not.toHaveBeenCalled()
912
+ })
913
+
914
+ it('calls external content handler when readonly but force is true', async () => {
915
+ vi.spyOn(editor, 'getIsReadonly').mockReturnValue(true)
916
+
917
+ const info = { type: 'text' as const, text: 'test-data' }
918
+ await editor.replaceExternalContent(info, { force: true })
919
+
920
+ expect(mockHandler).toHaveBeenCalledWith(info)
921
+ })
922
+
923
+ it('calls external content handler when force is false and not readonly', async () => {
924
+ vi.spyOn(editor, 'getIsReadonly').mockReturnValue(false)
925
+
926
+ const info = { type: 'text' as const, text: 'test-data' }
927
+ await editor.replaceExternalContent(info, { force: false })
928
+
929
+ expect(mockHandler).toHaveBeenCalledWith(info)
930
+ })
931
+ })