@tldraw/editor 4.1.0-canary.9c36de6e611c → 4.1.0-canary.a152954244d2

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 (75) hide show
  1. package/dist-cjs/index.d.ts +16 -1
  2. package/dist-cjs/index.js +1 -1
  3. package/dist-cjs/lib/TldrawEditor.js +2 -2
  4. package/dist-cjs/lib/components/ErrorBoundary.js +1 -1
  5. package/dist-cjs/lib/components/HTMLContainer.js +1 -1
  6. package/dist-cjs/lib/components/MenuClickCapture.js +30 -15
  7. package/dist-cjs/lib/components/MenuClickCapture.js.map +2 -2
  8. package/dist-cjs/lib/components/SVGContainer.js +1 -1
  9. package/dist-cjs/lib/components/default-components/DefaultCanvas.js +1 -1
  10. package/dist-cjs/lib/components/default-components/DefaultCollaboratorHint.js +1 -1
  11. package/dist-cjs/lib/components/default-components/DefaultCursor.js +1 -1
  12. package/dist-cjs/lib/components/default-components/DefaultErrorFallback.js +1 -1
  13. package/dist-cjs/lib/components/default-components/DefaultHandle.js +1 -1
  14. package/dist-cjs/lib/components/default-components/DefaultScribble.js +1 -1
  15. package/dist-cjs/lib/components/default-components/DefaultSelectionBackground.js +1 -1
  16. package/dist-cjs/lib/components/default-components/DefaultSelectionForeground.js +1 -1
  17. package/dist-cjs/lib/components/default-components/DefaultShapeIndicator.js +1 -1
  18. package/dist-cjs/lib/components/default-components/DefaultShapeWrapper.js +1 -1
  19. package/dist-cjs/lib/components/default-components/DefaultSnapIndictor.js +2 -2
  20. package/dist-cjs/lib/components/default-components/DefaultSpinner.js +1 -1
  21. package/dist-cjs/lib/editor/Editor.js +5 -3
  22. package/dist-cjs/lib/editor/Editor.js.map +2 -2
  23. package/dist-cjs/lib/editor/derivations/parentsToChildren.js +3 -7
  24. package/dist-cjs/lib/editor/derivations/parentsToChildren.js.map +2 -2
  25. package/dist-cjs/lib/editor/managers/SnapManager/HandleSnaps.js +67 -2
  26. package/dist-cjs/lib/editor/managers/SnapManager/HandleSnaps.js.map +2 -2
  27. package/dist-cjs/lib/editor/shapes/ShapeUtil.js.map +2 -2
  28. package/dist-cjs/lib/hooks/useCanvasEvents.js +2 -1
  29. package/dist-cjs/lib/hooks/useCanvasEvents.js.map +2 -2
  30. package/dist-cjs/lib/hooks/useDarkMode.js +1 -1
  31. package/dist-cjs/lib/hooks/useEditor.js +1 -1
  32. package/dist-cjs/lib/hooks/useGestureEvents.js +1 -1
  33. package/dist-cjs/lib/hooks/useHandleEvents.js +1 -1
  34. package/dist-cjs/lib/hooks/useZoomCss.js +1 -1
  35. package/dist-cjs/lib/license/LicenseManager.js +16 -6
  36. package/dist-cjs/lib/license/LicenseManager.js.map +2 -2
  37. package/dist-cjs/lib/license/Watermark.js +4 -4
  38. package/dist-cjs/lib/license/Watermark.js.map +1 -1
  39. package/dist-cjs/lib/primitives/geometry/Geometry2d.js +5 -0
  40. package/dist-cjs/lib/primitives/geometry/Geometry2d.js.map +2 -2
  41. package/dist-cjs/version.js +3 -3
  42. package/dist-cjs/version.js.map +1 -1
  43. package/dist-esm/index.d.mts +16 -1
  44. package/dist-esm/index.mjs +1 -1
  45. package/dist-esm/lib/components/MenuClickCapture.mjs +30 -15
  46. package/dist-esm/lib/components/MenuClickCapture.mjs.map +2 -2
  47. package/dist-esm/lib/editor/Editor.mjs +5 -2
  48. package/dist-esm/lib/editor/Editor.mjs.map +2 -2
  49. package/dist-esm/lib/editor/derivations/parentsToChildren.mjs +3 -7
  50. package/dist-esm/lib/editor/derivations/parentsToChildren.mjs.map +2 -2
  51. package/dist-esm/lib/editor/managers/SnapManager/HandleSnaps.mjs +67 -2
  52. package/dist-esm/lib/editor/managers/SnapManager/HandleSnaps.mjs.map +2 -2
  53. package/dist-esm/lib/editor/shapes/ShapeUtil.mjs.map +2 -2
  54. package/dist-esm/lib/hooks/useCanvasEvents.mjs +2 -1
  55. package/dist-esm/lib/hooks/useCanvasEvents.mjs.map +2 -2
  56. package/dist-esm/lib/license/LicenseManager.mjs +16 -6
  57. package/dist-esm/lib/license/LicenseManager.mjs.map +2 -2
  58. package/dist-esm/lib/license/Watermark.mjs +4 -4
  59. package/dist-esm/lib/license/Watermark.mjs.map +1 -1
  60. package/dist-esm/lib/primitives/geometry/Geometry2d.mjs +5 -0
  61. package/dist-esm/lib/primitives/geometry/Geometry2d.mjs.map +2 -2
  62. package/dist-esm/version.mjs +3 -3
  63. package/dist-esm/version.mjs.map +1 -1
  64. package/editor.css +7 -2
  65. package/package.json +7 -7
  66. package/src/lib/components/MenuClickCapture.tsx +35 -17
  67. package/src/lib/editor/Editor.ts +5 -2
  68. package/src/lib/editor/derivations/parentsToChildren.ts +4 -10
  69. package/src/lib/editor/managers/SnapManager/HandleSnaps.ts +91 -4
  70. package/src/lib/editor/shapes/ShapeUtil.ts +10 -0
  71. package/src/lib/hooks/useCanvasEvents.ts +8 -1
  72. package/src/lib/license/LicenseManager.ts +22 -6
  73. package/src/lib/license/Watermark.tsx +4 -4
  74. package/src/lib/primitives/geometry/Geometry2d.ts +6 -0
  75. package/src/version.ts +3 -3
@@ -4,7 +4,7 @@ import { assertExists, uniqueId } from '@tldraw/utils'
4
4
  import { Vec } from '../../../primitives/Vec'
5
5
  import { Geometry2d } from '../../../primitives/geometry/Geometry2d'
6
6
  import { Editor } from '../../Editor'
7
- import { SnapData, SnapManager } from './SnapManager'
7
+ import { PointsSnapIndicator, SnapData, SnapManager } from './SnapManager'
8
8
 
9
9
  /**
10
10
  * When dragging a handle, users can snap the handle to key geometry on other nearby shapes.
@@ -43,6 +43,11 @@ export interface HandleSnapGeometry {
43
43
  getSelfSnapPoints?(handle: TLHandle): VecModel[]
44
44
  }
45
45
 
46
+ interface AlignPointsSnap {
47
+ snaps: PointsSnapIndicator[]
48
+ nudge: Vec
49
+ }
50
+
46
51
  const defaultGetSelfSnapOutline = () => null
47
52
  const defaultGetSelfSnapPoints = () => []
48
53
  /** @public */
@@ -171,6 +176,67 @@ export class HandleSnaps {
171
176
  return null
172
177
  }
173
178
 
179
+ private getHandleSnapData({
180
+ handle,
181
+ currentShapeId,
182
+ }: {
183
+ handle: TLHandle
184
+ currentShapeId: TLShapeId
185
+ }): AlignPointsSnap | null {
186
+ const snapThreshold = this.manager.getSnapThreshold()
187
+ const currentShapeTransform = assertExists(this.editor.getShapePageTransform(currentShapeId))
188
+ const handleInPageSpace = currentShapeTransform.applyToPoint(handle)
189
+
190
+ let nearestXSnap: Vec | null = null
191
+ let nearestYSnap: Vec | null = null
192
+ let minOffsetX = snapThreshold
193
+ let minOffsetY = snapThreshold
194
+
195
+ for (const snapPoint of this.iterateSnapPointsInPageSpace(currentShapeId, handle)) {
196
+ const offsetX = Math.abs(handleInPageSpace.x - snapPoint.x)
197
+ const offsetY = Math.abs(handleInPageSpace.y - snapPoint.y)
198
+ if (offsetX < minOffsetX) {
199
+ minOffsetX = offsetX
200
+ nearestXSnap = snapPoint
201
+ }
202
+ if (offsetY < minOffsetY) {
203
+ minOffsetY = offsetY
204
+ nearestYSnap = snapPoint
205
+ }
206
+ }
207
+
208
+ if (!nearestXSnap && !nearestYSnap) {
209
+ return null
210
+ }
211
+
212
+ const nudge = new Vec(
213
+ nearestXSnap ? nearestXSnap.x - handleInPageSpace.x : 0,
214
+ nearestYSnap ? nearestYSnap.y - handleInPageSpace.y : 0
215
+ )
216
+
217
+ const snappedHandle = Vec.Add(handleInPageSpace, nudge)
218
+ const snaps: PointsSnapIndicator[] = []
219
+
220
+ if (nearestXSnap) {
221
+ const snappedHandleOnX = new Vec(nearestXSnap.x, snappedHandle.y)
222
+ snaps.push({
223
+ id: uniqueId(),
224
+ type: 'points',
225
+ points: [nearestXSnap, snappedHandleOnX],
226
+ })
227
+ }
228
+ if (nearestYSnap) {
229
+ const snappedHandleOnY = new Vec(snappedHandle.x, nearestYSnap.y)
230
+ snaps.push({
231
+ id: uniqueId(),
232
+ type: 'points',
233
+ points: [nearestYSnap, snappedHandleOnY],
234
+ })
235
+ }
236
+
237
+ return { snaps, nudge }
238
+ }
239
+
174
240
  snapHandle({
175
241
  currentShapeId,
176
242
  handle,
@@ -180,10 +246,16 @@ export class HandleSnaps {
180
246
  }): SnapData | null {
181
247
  const currentShapeTransform = assertExists(this.editor.getShapePageTransform(currentShapeId))
182
248
  const handleInPageSpace = currentShapeTransform.applyToPoint(handle)
183
- const snapPosition = this.getHandleSnapPosition({ currentShapeId, handle, handleInPageSpace })
249
+ // eslint-disable-next-line @typescript-eslint/no-deprecated
250
+ const snapType = handle.canSnap ? 'point' : handle.snapType
251
+
252
+ if (snapType === 'point') {
253
+ const snapPosition = this.getHandleSnapPosition({ currentShapeId, handle, handleInPageSpace })
254
+
255
+ if (!snapPosition) {
256
+ return null
257
+ }
184
258
 
185
- // If we found a point, display snap lines, and return the nudge
186
- if (snapPosition) {
187
259
  this.manager.setIndicators([
188
260
  {
189
261
  id: uniqueId(),
@@ -195,6 +267,21 @@ export class HandleSnaps {
195
267
  return { nudge: Vec.Sub(snapPosition, handleInPageSpace) }
196
268
  }
197
269
 
270
+ if (snapType === 'align') {
271
+ const snapData = this.getHandleSnapData({
272
+ handle,
273
+ currentShapeId,
274
+ })
275
+
276
+ if (!snapData) {
277
+ return null
278
+ }
279
+
280
+ this.manager.setIndicators(snapData.snaps)
281
+
282
+ return { nudge: snapData.nudge }
283
+ }
284
+
198
285
  return null
199
286
  }
200
287
  }
@@ -328,6 +328,16 @@ export abstract class ShapeUtil<Shape extends TLUnknownShape = TLUnknownShape> {
328
328
  */
329
329
  shouldClipChild?(child: TLShape): boolean
330
330
 
331
+ /**
332
+ * Whether a specific shape should hide in the minimap.
333
+ *
334
+ * If not defined, the default behavior is to show all shapes in the minimap.
335
+ *
336
+ * @returns boolean indicating if this shape should hide in the minimap
337
+ * @public
338
+ */
339
+ hideInMinimap?(_shape: Shape): boolean
340
+
331
341
  /**
332
342
  * Whether the shape should hide its resize handles when selected.
333
343
  *
@@ -1,6 +1,7 @@
1
1
  import { useValue } from '@tldraw/state-react'
2
2
  import React, { useEffect, useMemo } from 'react'
3
3
  import { RIGHT_MOUSE_BUTTON } from '../constants'
4
+ import { tlenv } from '../globals/environment'
4
5
  import { preventDefault, releasePointerCapture, setPointerCapture } from '../utils/dom'
5
6
  import { getPointerInfo } from '../utils/getPointerInfo'
6
7
  import { useEditor } from './useEditor'
@@ -161,8 +162,14 @@ export function useCanvasEvents() {
161
162
  // For tools that benefit from a higher fidelity of events,
162
163
  // we dispatch the coalesced events.
163
164
  // N.B. Sometimes getCoalescedEvents isn't present on iOS, ugh.
165
+ // Specifically, in local mode (non-https) mode, iOS does not `useCoalescedEvents`
166
+ // so it appears like the ink is working locally, when really it's just that `useCoalescedEvents`
167
+ // is disabled. The intent here is to have `useCoalescedEvents` disabled for iOS.
164
168
  const events =
165
- currentTool.useCoalescedEvents && e.getCoalescedEvents ? e.getCoalescedEvents() : [e]
169
+ !tlenv.isIos && currentTool.useCoalescedEvents && e.getCoalescedEvents
170
+ ? e.getCoalescedEvents()
171
+ : [e]
172
+
166
173
  for (const singleEvent of events) {
167
174
  editor.dispatch({
168
175
  type: 'pointer',
@@ -171,6 +171,17 @@ export class LicenseManager {
171
171
  url.searchParams.set('license_type', trackType)
172
172
  if ('license' in result) {
173
173
  url.searchParams.set('license_id', result.license.id)
174
+ const sku = this.isFlagEnabled(result.license.flags, FLAGS.EVALUATION_LICENSE)
175
+ ? 'evaluation'
176
+ : this.isFlagEnabled(result.license.flags, FLAGS.ANNUAL_LICENSE)
177
+ ? 'annual'
178
+ : this.isFlagEnabled(result.license.flags, FLAGS.PERPETUAL_LICENSE)
179
+ ? 'perpetual'
180
+ : 'unknown'
181
+ url.searchParams.set('sku', sku)
182
+ }
183
+ if (process.env.NODE_ENV) {
184
+ url.searchParams.set('environment', process.env.NODE_ENV)
174
185
  }
175
186
 
176
187
  // eslint-disable-next-line no-restricted-globals
@@ -406,22 +417,27 @@ export class LicenseManager {
406
417
  // If we added a new flag it will be twice the value of the currently highest flag.
407
418
  // And if all the current flags are on we would get the `HIGHEST_FLAG * 2 - 1`, so anything higher than that means there are new flags.
408
419
  if (result.license.flags >= HIGHEST_FLAG * 2) {
409
- this.outputMessages([
410
- 'This tldraw license contains some unknown flags.',
411
- 'You may want to update tldraw packages to a newer version to get access to new functionality.',
412
- ])
420
+ this.outputMessages(
421
+ [
422
+ 'Warning: This tldraw license contains some unknown flags.',
423
+ 'This will still work, however, you may want to update tldraw packages to a newer version to get access to new functionality.',
424
+ ],
425
+ 'warning'
426
+ )
413
427
  }
414
428
  }
415
429
 
416
- private outputMessages(messages: string[]) {
430
+ private outputMessages(messages: string[], type: 'warning' | 'error' = 'error') {
417
431
  if (this.isTest) return
418
432
  if (this.verbose) {
419
433
  this.outputDelimiter()
420
434
  for (const message of messages) {
435
+ const color = type === 'warning' ? 'orange' : 'crimson'
436
+ const bgColor = type === 'warning' ? 'orange' : 'crimson'
421
437
  // eslint-disable-next-line no-console
422
438
  console.log(
423
439
  `%c${message}`,
424
- `color: white; background: crimson; padding: 2px; border-radius: 3px;`
440
+ `color: ${color}; background: ${bgColor}; padding: 2px; border-radius: 3px;`
425
441
  )
426
442
  }
427
443
  this.outputDelimiter()
@@ -208,22 +208,22 @@ To remove the watermark, please purchase a license at tldraw.dev.
208
208
  }
209
209
 
210
210
  @media (hover: hover) {
211
- .${className}[data-licensed='false'] > button {
211
+ .${className} > button {
212
212
  pointer-events: none;
213
213
  }
214
214
 
215
- .${className}[data-licensed='false']:hover {
215
+ .${className}:hover {
216
216
  background-color: var(--tl-color-background);
217
217
  transition: background-color 0.2s ease-in-out;
218
218
  transition-delay: 0.32s;
219
219
  }
220
220
 
221
- .${className}[data-licensed='false']:hover > button {
221
+ .${className}:hover > button {
222
222
  animation: ${className}_delayed_link 0.2s forwards ease-in-out;
223
223
  animation-delay: 0.32s;
224
224
  }
225
225
 
226
- .${className}[data-licensed='false'] > button:focus-visible {
226
+ .${className} > button:focus-visible {
227
227
  opacity: 1;
228
228
  }
229
229
  }
@@ -120,6 +120,8 @@ export abstract class Geometry2d {
120
120
  distanceToLineSegment(A: VecLike, B: VecLike, filters?: Geometry2dFilters) {
121
121
  if (Vec.Equals(A, B)) return this.distanceToPoint(A, false, filters)
122
122
  const { vertices } = this
123
+ if (vertices.length === 0) throw Error('nearest point not found')
124
+ if (vertices.length === 1) return Vec.Dist(A, vertices[0])
123
125
  let nearest: Vec | undefined
124
126
  let dist = Infinity
125
127
  let d: number, p: Vec, q: Vec
@@ -175,6 +177,8 @@ export abstract class Geometry2d {
175
177
  interpolateAlongEdge(t: number, _filters?: Geometry2dFilters): Vec {
176
178
  const { vertices } = this
177
179
 
180
+ if (vertices.length === 0) return new Vec(0, 0)
181
+ if (vertices.length === 1) return vertices[0]
178
182
  if (t <= 0) return vertices[0]
179
183
 
180
184
  const distanceToTravel = t * this.length
@@ -209,6 +213,8 @@ export abstract class Geometry2d {
209
213
  let closestDistance = Infinity
210
214
  let distanceTraveled = 0
211
215
 
216
+ if (vertices.length === 0 || vertices.length === 1) return 0
217
+
212
218
  for (let i = 0; i < (this.isClosed ? vertices.length : vertices.length - 1); i++) {
213
219
  const curr = vertices[i]
214
220
  const next = vertices[(i + 1) % vertices.length]
package/src/version.ts CHANGED
@@ -1,9 +1,9 @@
1
1
  // This file is automatically generated by internal/scripts/refresh-assets.ts.
2
2
  // Do not edit manually. Or do, I'm a comment, not a cop.
3
3
 
4
- export const version = '4.1.0-canary.9c36de6e611c'
4
+ export const version = '4.1.0-canary.a152954244d2'
5
5
  export const publishDates = {
6
6
  major: '2025-09-18T14:39:22.803Z',
7
- minor: '2025-09-18T16:02:18.336Z',
8
- patch: '2025-09-18T16:02:18.336Z',
7
+ minor: '2025-10-15T10:12:21.273Z',
8
+ patch: '2025-10-15T10:12:21.273Z',
9
9
  }