@tldraw/editor 3.12.1 → 3.13.0-canary.064d79cae9fb

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 (94) hide show
  1. package/CHANGELOG.md +0 -20
  2. package/dist-cjs/index.d.ts +30 -14
  3. package/dist-cjs/index.js +1 -1
  4. package/dist-cjs/lib/TldrawEditor.js +2 -1
  5. package/dist-cjs/lib/TldrawEditor.js.map +2 -2
  6. package/dist-cjs/lib/components/Shape.js +12 -8
  7. package/dist-cjs/lib/components/Shape.js.map +2 -2
  8. package/dist-cjs/lib/components/default-components/DefaultCanvas.js +27 -2
  9. package/dist-cjs/lib/components/default-components/DefaultCanvas.js.map +2 -2
  10. package/dist-cjs/lib/components/default-components/DefaultErrorFallback.js +14 -12
  11. package/dist-cjs/lib/components/default-components/DefaultErrorFallback.js.map +2 -2
  12. package/dist-cjs/lib/components/default-components/DefaultShapeIndicator.js +17 -11
  13. package/dist-cjs/lib/components/default-components/DefaultShapeIndicator.js.map +2 -2
  14. package/dist-cjs/lib/components/default-components/DefaultSpinner.js +1 -1
  15. package/dist-cjs/lib/components/default-components/DefaultSpinner.js.map +2 -2
  16. package/dist-cjs/lib/editor/Editor.js +46 -28
  17. package/dist-cjs/lib/editor/Editor.js.map +3 -3
  18. package/dist-cjs/lib/editor/shapes/ShapeUtil.js.map +2 -2
  19. package/dist-cjs/lib/exports/getSvgJsx.js +12 -3
  20. package/dist-cjs/lib/exports/getSvgJsx.js.map +2 -2
  21. package/dist-cjs/lib/hooks/useDocumentEvents.js +3 -2
  22. package/dist-cjs/lib/hooks/useDocumentEvents.js.map +2 -2
  23. package/dist-cjs/lib/hooks/useEditorComponents.js +16 -15
  24. package/dist-cjs/lib/hooks/useEditorComponents.js.map +2 -2
  25. package/dist-cjs/lib/license/LicenseManager.js +8 -1
  26. package/dist-cjs/lib/license/LicenseManager.js.map +2 -2
  27. package/dist-cjs/lib/options.js.map +2 -2
  28. package/dist-cjs/lib/utils/areShapesContentEqual.js +25 -0
  29. package/dist-cjs/lib/utils/areShapesContentEqual.js.map +7 -0
  30. package/dist-cjs/lib/utils/dom.js +3 -3
  31. package/dist-cjs/lib/utils/dom.js.map +2 -2
  32. package/dist-cjs/lib/utils/nearestMultiple.js +34 -0
  33. package/dist-cjs/lib/utils/nearestMultiple.js.map +7 -0
  34. package/dist-cjs/lib/utils/rotation.js +5 -5
  35. package/dist-cjs/lib/utils/rotation.js.map +2 -2
  36. package/dist-cjs/version.js +3 -3
  37. package/dist-cjs/version.js.map +1 -1
  38. package/dist-esm/index.d.mts +30 -14
  39. package/dist-esm/index.mjs +1 -1
  40. package/dist-esm/lib/TldrawEditor.mjs +2 -1
  41. package/dist-esm/lib/TldrawEditor.mjs.map +2 -2
  42. package/dist-esm/lib/components/Shape.mjs +12 -8
  43. package/dist-esm/lib/components/Shape.mjs.map +2 -2
  44. package/dist-esm/lib/components/default-components/DefaultCanvas.mjs +27 -2
  45. package/dist-esm/lib/components/default-components/DefaultCanvas.mjs.map +2 -2
  46. package/dist-esm/lib/components/default-components/DefaultErrorFallback.mjs +14 -12
  47. package/dist-esm/lib/components/default-components/DefaultErrorFallback.mjs.map +2 -2
  48. package/dist-esm/lib/components/default-components/DefaultShapeIndicator.mjs +17 -11
  49. package/dist-esm/lib/components/default-components/DefaultShapeIndicator.mjs.map +2 -2
  50. package/dist-esm/lib/components/default-components/DefaultSpinner.mjs +1 -1
  51. package/dist-esm/lib/components/default-components/DefaultSpinner.mjs.map +2 -2
  52. package/dist-esm/lib/editor/Editor.mjs +46 -28
  53. package/dist-esm/lib/editor/Editor.mjs.map +3 -3
  54. package/dist-esm/lib/editor/shapes/ShapeUtil.mjs.map +2 -2
  55. package/dist-esm/lib/exports/getSvgJsx.mjs +12 -3
  56. package/dist-esm/lib/exports/getSvgJsx.mjs.map +2 -2
  57. package/dist-esm/lib/hooks/useDocumentEvents.mjs +3 -2
  58. package/dist-esm/lib/hooks/useDocumentEvents.mjs.map +2 -2
  59. package/dist-esm/lib/hooks/useEditorComponents.mjs +16 -15
  60. package/dist-esm/lib/hooks/useEditorComponents.mjs.map +2 -2
  61. package/dist-esm/lib/license/LicenseManager.mjs +8 -1
  62. package/dist-esm/lib/license/LicenseManager.mjs.map +2 -2
  63. package/dist-esm/lib/options.mjs.map +2 -2
  64. package/dist-esm/lib/utils/areShapesContentEqual.mjs +5 -0
  65. package/dist-esm/lib/utils/areShapesContentEqual.mjs.map +7 -0
  66. package/dist-esm/lib/utils/dom.mjs +3 -3
  67. package/dist-esm/lib/utils/dom.mjs.map +2 -2
  68. package/dist-esm/lib/utils/nearestMultiple.mjs +14 -0
  69. package/dist-esm/lib/utils/nearestMultiple.mjs.map +7 -0
  70. package/dist-esm/lib/utils/rotation.mjs +5 -5
  71. package/dist-esm/lib/utils/rotation.mjs.map +2 -2
  72. package/dist-esm/version.mjs +3 -3
  73. package/dist-esm/version.mjs.map +1 -1
  74. package/editor.css +11 -0
  75. package/package.json +7 -7
  76. package/src/lib/TldrawEditor.tsx +6 -1
  77. package/src/lib/components/Shape.tsx +14 -10
  78. package/src/lib/components/default-components/DefaultCanvas.tsx +32 -2
  79. package/src/lib/components/default-components/DefaultErrorFallback.tsx +25 -14
  80. package/src/lib/components/default-components/DefaultShapeIndicator.tsx +17 -8
  81. package/src/lib/components/default-components/DefaultSpinner.tsx +1 -1
  82. package/src/lib/editor/Editor.ts +43 -27
  83. package/src/lib/editor/shapes/ShapeUtil.ts +13 -1
  84. package/src/lib/exports/getSvgJsx.tsx +16 -7
  85. package/src/lib/hooks/useDocumentEvents.ts +7 -2
  86. package/src/lib/hooks/useEditorComponents.tsx +32 -28
  87. package/src/lib/license/LicenseManager.test.ts +40 -0
  88. package/src/lib/license/LicenseManager.ts +13 -1
  89. package/src/lib/options.ts +4 -0
  90. package/src/lib/utils/areShapesContentEqual.ts +4 -0
  91. package/src/lib/utils/dom.ts +4 -4
  92. package/src/lib/utils/nearestMultiple.ts +13 -0
  93. package/src/lib/utils/rotation.ts +8 -6
  94. package/src/version.ts +3 -3
@@ -42,6 +42,7 @@ import {
42
42
  TLImageAsset,
43
43
  TLInstance,
44
44
  TLInstancePageState,
45
+ TLInstancePresence,
45
46
  TLNoteShape,
46
47
  TLPOINTER_ID,
47
48
  TLPage,
@@ -128,6 +129,7 @@ import { Group2d } from '../primitives/geometry/Group2d'
128
129
  import { intersectPolygonPolygon } from '../primitives/intersect'
129
130
  import { PI, approximately, areAnglesCompatible, clamp, pointInPolygon } from '../primitives/utils'
130
131
  import { ReadonlySharedStyleMap, SharedStyle, SharedStyleMap } from '../utils/SharedStylesMap'
132
+ import { areShapesContentEqual } from '../utils/areShapesContentEqual'
131
133
  import { dataUrlToFile } from '../utils/assets'
132
134
  import { debugFlags } from '../utils/debug-flags'
133
135
  import {
@@ -1704,8 +1706,7 @@ export class Editor extends EventEmitter<TLEventMap> {
1704
1706
  * @readonly
1705
1707
  */
1706
1708
  @computed getSelectedShapes(): TLShape[] {
1707
- const { selectedShapeIds } = this.getCurrentPageState()
1708
- return compact(selectedShapeIds.map((id) => this.store.get(id)))
1709
+ return compact(this.getSelectedShapeIds().map((id) => this.store.get(id)))
1709
1710
  }
1710
1711
 
1711
1712
  /**
@@ -2575,14 +2576,25 @@ export class Editor extends EventEmitter<TLEventMap> {
2575
2576
  return baseCamera
2576
2577
  }
2577
2578
 
2579
+ private _getFollowingPresence(targetUserId: string | null) {
2580
+ const visited = [this.user.getId()]
2581
+ const collaborators = this.getCollaborators()
2582
+ let leaderPresence = null as null | TLInstancePresence
2583
+ while (targetUserId && !visited.includes(targetUserId)) {
2584
+ leaderPresence = collaborators.find((c) => c.userId === targetUserId) ?? null
2585
+ targetUserId = leaderPresence?.followingUserId ?? null
2586
+ if (leaderPresence) {
2587
+ visited.push(leaderPresence.userId)
2588
+ }
2589
+ }
2590
+ return leaderPresence
2591
+ }
2592
+
2578
2593
  @computed
2579
2594
  private getViewportPageBoundsForFollowing(): null | Box {
2580
- const followingUserId = this.getInstanceState().followingUserId
2581
- if (!followingUserId) return null
2582
- const leaderPresence = this.getCollaborators().find((c) => c.userId === followingUserId)
2583
- if (!leaderPresence) return null
2595
+ const leaderPresence = this._getFollowingPresence(this.getInstanceState().followingUserId)
2584
2596
 
2585
- if (!leaderPresence.camera || !leaderPresence.screenBounds) return null
2597
+ if (!leaderPresence?.camera || !leaderPresence?.screenBounds) return null
2586
2598
 
2587
2599
  // Fit their viewport inside of our screen bounds
2588
2600
  // 1. calculate their viewport in page space
@@ -3781,15 +3793,6 @@ export class Editor extends EventEmitter<TLEventMap> {
3781
3793
  // if we were already following someone, stop following them
3782
3794
  this.stopFollowingUser()
3783
3795
 
3784
- const leaderPresences = this._getCollaboratorsQuery()
3785
- .get()
3786
- .filter((p) => p.userId === userId)
3787
-
3788
- if (!leaderPresences.length) {
3789
- console.warn('User not found')
3790
- return this
3791
- }
3792
-
3793
3796
  const thisUserId = this.user.getId()
3794
3797
 
3795
3798
  if (!thisUserId) {
@@ -3797,13 +3800,14 @@ export class Editor extends EventEmitter<TLEventMap> {
3797
3800
  // allow to continue since it's probably fine most of the time.
3798
3801
  }
3799
3802
 
3800
- // If the leader is following us, then we can't follow them
3801
- if (leaderPresences.some((p) => p.followingUserId === thisUserId)) {
3803
+ const leaderPresence = this._getFollowingPresence(userId)
3804
+
3805
+ if (!leaderPresence) {
3802
3806
  return this
3803
3807
  }
3804
3808
 
3805
3809
  const latestLeaderPresence = computed('latestLeaderPresence', () => {
3806
- return this.getCollaborators().find((p) => p.userId === userId)
3810
+ return this._getFollowingPresence(userId)
3807
3811
  })
3808
3812
 
3809
3813
  transact(() => {
@@ -4571,7 +4575,7 @@ export class Editor extends EventEmitter<TLEventMap> {
4571
4575
  this.fonts.trackFontsForShape(shape)
4572
4576
  return this.getShapeUtil(shape).getGeometry(shape, opts)
4573
4577
  },
4574
- { areRecordsEqual: (a, b) => a.props === b.props }
4578
+ { areRecordsEqual: areShapesContentEqual }
4575
4579
  )
4576
4580
  }
4577
4581
  return this._shapeGeometryCaches[context].get(
@@ -4619,9 +4623,15 @@ export class Editor extends EventEmitter<TLEventMap> {
4619
4623
 
4620
4624
  /** @internal */
4621
4625
  @computed private _getShapeHandlesCache(): ComputedCache<TLHandle[] | undefined, TLShape> {
4622
- return this.store.createComputedCache('handles', (shape) => {
4623
- return this.getShapeUtil(shape).getHandles?.(shape)
4624
- })
4626
+ return this.store.createComputedCache(
4627
+ 'handles',
4628
+ (shape) => {
4629
+ return this.getShapeUtil(shape).getHandles?.(shape)
4630
+ },
4631
+ {
4632
+ areRecordsEqual: areShapesContentEqual,
4633
+ }
4634
+ )
4625
4635
  }
4626
4636
 
4627
4637
  /**
@@ -5842,9 +5852,15 @@ export class Editor extends EventEmitter<TLEventMap> {
5842
5852
  @computed
5843
5853
  private _getBindingsIndexCache() {
5844
5854
  const index = bindingsIndex(this)
5845
- return this.store.createComputedCache<TLBinding[], TLShape>('bindingsIndex', (shape) => {
5846
- return index.get().get(shape.id)
5847
- })
5855
+ return this.store.createComputedCache<TLBinding[], TLShape>(
5856
+ 'bindingsIndex',
5857
+ (shape) => {
5858
+ return index.get().get(shape.id)
5859
+ },
5860
+ // we can ignore the shape equality check here because the index is
5861
+ // computed incrementally based on what bindings are in the store
5862
+ { areRecordsEqual: () => true }
5863
+ )
5848
5864
  }
5849
5865
 
5850
5866
  /**
@@ -10211,7 +10227,7 @@ export class Editor extends EventEmitter<TLEventMap> {
10211
10227
 
10212
10228
  // If the camera behavior is "zoom" and the ctrl key is pressed, then pan;
10213
10229
  // If the camera behavior is "pan" and the ctrl key is not pressed, then zoom
10214
- if (inputs.ctrlKey) behavior = wheelBehavior === 'pan' ? 'zoom' : 'pan'
10230
+ if (info.ctrlKey) behavior = wheelBehavior === 'pan' ? 'zoom' : 'pan'
10215
10231
 
10216
10232
  switch (behavior) {
10217
10233
  case 'zoom': {
@@ -19,6 +19,7 @@ import { TLFontFace } from '../managers/FontManager'
19
19
  import { BoundsSnapGeometry } from '../managers/SnapManager/BoundsSnaps'
20
20
  import { HandleSnapGeometry } from '../managers/SnapManager/HandleSnaps'
21
21
  import { SvgExportContext } from '../types/SvgExportContext'
22
+ import { TLClickEventInfo } from '../types/event-types'
22
23
  import { TLResizeHandle } from '../types/selection-types'
23
24
 
24
25
  /** @public */
@@ -671,10 +672,21 @@ export abstract class ShapeUtil<Shape extends TLUnknownShape = TLUnknownShape> {
671
672
  * A callback called when a shape's edge is double clicked.
672
673
  *
673
674
  * @param shape - The shape.
675
+ * @param info - Info about the edge.
674
676
  * @returns A change to apply to the shape, or void.
675
677
  * @public
676
678
  */
677
- onDoubleClickEdge?(shape: Shape): TLShapePartial<Shape> | void
679
+ onDoubleClickEdge?(shape: Shape, info: TLClickEventInfo): TLShapePartial<Shape> | void
680
+
681
+ /**
682
+ * A callback called when a shape's corner is double clicked.
683
+ *
684
+ * @param shape - The shape.
685
+ * @param info - Info about the corner.
686
+ * @returns A change to apply to the shape, or void.
687
+ * @public
688
+ */
689
+ onDoubleClickCorner?(shape: Shape, info: TLClickEventInfo): TLShapePartial<Shape> | void
678
690
 
679
691
  /**
680
692
  * A callback called when a shape is double clicked.
@@ -365,6 +365,21 @@ function SvgExport({
365
365
  onMount()
366
366
  }, [onMount, shapeElements])
367
367
 
368
+ let backgroundColor = background ? theme.background : 'transparent'
369
+
370
+ if (singleFrameShapeId && background) {
371
+ const frameShapeUtil = editor.getShapeUtil('frame') as any as
372
+ | undefined
373
+ | { options: { showColors: boolean } }
374
+ if (frameShapeUtil?.options.showColors) {
375
+ const shape = editor.getShape(singleFrameShapeId)! as TLFrameShape
376
+ const color = theme[shape.props.color]
377
+ backgroundColor = color.frame.fill
378
+ } else {
379
+ backgroundColor = theme.solid
380
+ }
381
+ }
382
+
368
383
  return (
369
384
  <SvgExportContextProvider editor={editor} context={exportContext}>
370
385
  <svg
@@ -375,13 +390,7 @@ function SvgExport({
375
390
  viewBox={`${bbox.minX} ${bbox.minY} ${bbox.width} ${bbox.height}`}
376
391
  strokeLinecap="round"
377
392
  strokeLinejoin="round"
378
- style={{
379
- backgroundColor: background
380
- ? singleFrameShapeId
381
- ? theme.solid
382
- : theme.background
383
- : 'transparent',
384
- }}
393
+ style={{ backgroundColor }}
385
394
  data-color-mode={isDarkMode ? 'dark' : 'light'}
386
395
  className={`tl-container tl-theme__force-sRGB ${isDarkMode ? 'tl-theme__dark' : 'tl-theme__light'}`}
387
396
  >
@@ -11,6 +11,7 @@ export function useDocumentEvents() {
11
11
  const editor = useEditor()
12
12
  const container = useContainer()
13
13
 
14
+ const isEditing = useValue('isEditing', () => editor.getEditingShapeId(), [editor])
14
15
  const isAppFocused = useValue('isFocused', () => editor.getIsFocused(), [editor])
15
16
 
16
17
  // Prevent the browser's default drag and drop behavior on our container (UI, etc)
@@ -125,7 +126,11 @@ export function useDocumentEvents() {
125
126
  if (areShortcutsDisabled(editor)) {
126
127
  return
127
128
  }
128
- if (hasSelectedShapes) {
129
+ // isEditing here sounds like it's about text editing
130
+ // but more specifically, this is so you can tab into an
131
+ // embed that's being 'edited'. In our world,
132
+ // editing an embed, means it's interactive.
133
+ if (hasSelectedShapes && !isEditing) {
129
134
  // This is used in tandem with shape navigation.
130
135
  preventDefault(e)
131
136
  }
@@ -289,7 +294,7 @@ export function useDocumentEvents() {
289
294
  container.removeEventListener('keydown', handleKeyDown)
290
295
  container.removeEventListener('keyup', handleKeyUp)
291
296
  }
292
- }, [editor, container, isAppFocused])
297
+ }, [editor, container, isAppFocused, isEditing])
293
298
  }
294
299
 
295
300
  function areShortcutsDisabled(editor: Editor) {
@@ -51,29 +51,30 @@ import { useShallowObjectIdentity } from './useIdentity'
51
51
  /** @public */
52
52
  export interface TLEditorComponents {
53
53
  Background?: ComponentType | null
54
- SvgDefs?: ComponentType | null
55
54
  Brush?: ComponentType<TLBrushProps> | null
56
- ZoomBrush?: ComponentType<TLBrushProps> | null
57
- ShapeIndicators?: ComponentType | null
58
- ShapeIndicator?: ComponentType<TLShapeIndicatorProps> | null
59
- Cursor?: ComponentType<TLCursorProps> | null
60
55
  Canvas?: ComponentType<TLCanvasComponentProps> | null
61
56
  CollaboratorBrush?: ComponentType<TLBrushProps> | null
62
57
  CollaboratorCursor?: ComponentType<TLCursorProps> | null
63
58
  CollaboratorHint?: ComponentType<TLCollaboratorHintProps> | null
59
+ CollaboratorScribble?: ComponentType<TLScribbleProps> | null
64
60
  CollaboratorShapeIndicator?: ComponentType<TLShapeIndicatorProps> | null
61
+ Cursor?: ComponentType<TLCursorProps> | null
65
62
  Grid?: ComponentType<TLGridProps> | null
66
- Scribble?: ComponentType<TLScribbleProps> | null
67
- CollaboratorScribble?: ComponentType<TLScribbleProps> | null
68
- SnapIndicator?: ComponentType<TLSnapIndicatorProps> | null
69
- Handles?: ComponentType<TLHandlesProps> | null
70
63
  Handle?: ComponentType<TLHandleProps> | null
71
- Spinner?: ComponentType | null
72
- SelectionForeground?: ComponentType<TLSelectionForegroundProps> | null
73
- SelectionBackground?: ComponentType<TLSelectionBackgroundProps> | null
74
- OnTheCanvas?: ComponentType | null
64
+ Handles?: ComponentType<TLHandlesProps> | null
75
65
  InFrontOfTheCanvas?: ComponentType | null
76
66
  LoadingScreen?: ComponentType | null
67
+ OnTheCanvas?: ComponentType | null
68
+ Overlays?: ComponentType | null
69
+ Scribble?: ComponentType<TLScribbleProps> | null
70
+ SelectionBackground?: ComponentType<TLSelectionBackgroundProps> | null
71
+ SelectionForeground?: ComponentType<TLSelectionForegroundProps> | null
72
+ ShapeIndicator?: ComponentType<TLShapeIndicatorProps> | null
73
+ ShapeIndicators?: ComponentType | null
74
+ SnapIndicator?: ComponentType<TLSnapIndicatorProps> | null
75
+ Spinner?: ComponentType | null
76
+ SvgDefs?: ComponentType | null
77
+ ZoomBrush?: ComponentType<TLBrushProps> | null
77
78
 
78
79
  // These will always have defaults
79
80
  ErrorFallback?: TLErrorFallbackComponent
@@ -96,32 +97,35 @@ export function EditorComponentsProvider({
96
97
  const value = useMemo(
97
98
  (): Required<TLEditorComponents> => ({
98
99
  Background: DefaultBackground,
99
- SvgDefs: DefaultSvgDefs,
100
100
  Brush: DefaultBrush,
101
- ZoomBrush: DefaultBrush,
101
+ Canvas: DefaultCanvas,
102
102
  CollaboratorBrush: DefaultBrush,
103
- Cursor: DefaultCursor,
104
103
  CollaboratorCursor: DefaultCursor,
105
104
  CollaboratorHint: DefaultCollaboratorHint,
105
+ CollaboratorScribble: DefaultScribble,
106
106
  CollaboratorShapeIndicator: DefaultShapeIndicator,
107
+ Cursor: DefaultCursor,
107
108
  Grid: DefaultGrid,
109
+ Handle: DefaultHandle,
110
+ Handles: DefaultHandles,
111
+ InFrontOfTheCanvas: null,
112
+ LoadingScreen: DefaultLoadingScreen,
113
+ OnTheCanvas: null,
114
+ Overlays: null,
108
115
  Scribble: DefaultScribble,
116
+ SelectionBackground: DefaultSelectionBackground,
117
+ SelectionForeground: DefaultSelectionForeground,
118
+ ShapeIndicator: DefaultShapeIndicator,
119
+ ShapeIndicators: DefaultShapeIndicators,
109
120
  SnapIndicator: DefaultSnapIndicator,
110
- Handles: DefaultHandles,
111
- Handle: DefaultHandle,
112
- CollaboratorScribble: DefaultScribble,
121
+ Spinner: DefaultSpinner,
122
+ SvgDefs: DefaultSvgDefs,
123
+ ZoomBrush: DefaultBrush,
124
+
113
125
  ErrorFallback: DefaultErrorFallback,
114
126
  ShapeErrorFallback: DefaultShapeErrorFallback,
115
127
  ShapeIndicatorErrorFallback: DefaultShapeIndicatorErrorFallback,
116
- Spinner: DefaultSpinner,
117
- SelectionBackground: DefaultSelectionBackground,
118
- SelectionForeground: DefaultSelectionForeground,
119
- ShapeIndicators: DefaultShapeIndicators,
120
- ShapeIndicator: DefaultShapeIndicator,
121
- OnTheCanvas: null,
122
- InFrontOfTheCanvas: null,
123
- Canvas: DefaultCanvas,
124
- LoadingScreen: DefaultLoadingScreen,
128
+
125
129
  ..._overrides,
126
130
  }),
127
131
  [_overrides]
@@ -317,6 +317,46 @@ describe('LicenseManager', () => {
317
317
  expect(result.isDomainValid).toBe(false)
318
318
  })
319
319
 
320
+ it('Succeeds if it is a vscode extension', async () => {
321
+ // @ts-ignore
322
+ delete window.location
323
+ // @ts-ignore
324
+ window.location = new URL(
325
+ 'vscode-webview:vscode-webview://1ipd8pun8ud7nd7hv9d112g7evi7m10vak9vviuvia66ou6aibp3/index.html?id=6ec2dc7a-afe9-45d9-bd71-1749f9568d28&origin=955b256f-37e1-4a72-a2f4-ad633e88239c&swVersion=4&extensionId=tldraw-org.tldraw-vscode&platform=electron&vscode-resource-base-authority=vscode-resource.vscode-cdn.net&parentOrigin=vscode-file%3A%2F%2Fvscode-app'
326
+ )
327
+
328
+ const permissiveHostsInfo = JSON.parse(STANDARD_LICENSE_INFO)
329
+ permissiveHostsInfo[PROPERTIES.HOSTS] = ['tldraw-org.tldraw-vscode']
330
+ const permissiveLicenseKey = await generateLicenseKey(
331
+ JSON.stringify(permissiveHostsInfo),
332
+ keyPair
333
+ )
334
+ const result = (await licenseManager.getLicenseFromKey(
335
+ permissiveLicenseKey
336
+ )) as ValidLicenseKeyResult
337
+ expect(result.isDomainValid).toBe(true)
338
+ })
339
+
340
+ it('Fails if it is a vscode extension with the wrong id', async () => {
341
+ // @ts-ignore
342
+ delete window.location
343
+ // @ts-ignore
344
+ window.location = new URL(
345
+ 'vscode-webview:vscode-webview://1ipd8pun8ud7nd7hv9d112g7evi7m10vak9vviuvia66ou6aibp3/index.html?id=6ec2dc7a-afe9-45d9-bd71-1749f9568d28&origin=955b256f-37e1-4a72-a2f4-ad633e88239c&swVersion=4&extensionId=tldraw-org.tldraw-vscode&platform=electron&vscode-resource-base-authority=vscode-resource.vscode-cdn.net&parentOrigin=vscode-file%3A%2F%2Fvscode-app'
346
+ )
347
+
348
+ const permissiveHostsInfo = JSON.parse(STANDARD_LICENSE_INFO)
349
+ permissiveHostsInfo[PROPERTIES.HOSTS] = ['blah-org.blah-vscode']
350
+ const permissiveLicenseKey = await generateLicenseKey(
351
+ JSON.stringify(permissiveHostsInfo),
352
+ keyPair
353
+ )
354
+ const result = (await licenseManager.getLicenseFromKey(
355
+ permissiveLicenseKey
356
+ )) as ValidLicenseKeyResult
357
+ expect(result.isDomainValid).toBe(false)
358
+ })
359
+
320
360
  it('Checks for internal license', async () => {
321
361
  const internalLicenseInfo = JSON.parse(STANDARD_LICENSE_INFO)
322
362
  internalLicenseInfo[PROPERTIES.FLAGS] = FLAGS.INTERNAL_LICENSE
@@ -111,7 +111,10 @@ export class LicenseManager {
111
111
  if (testEnvironment === 'production') return false
112
112
 
113
113
  // If we are using https on a non-localhost domain we assume it's a production env and a development one otherwise
114
- return window.location.protocol !== 'https:' || window.location.hostname === 'localhost'
114
+ return (
115
+ !['https:', 'vscode-webview:'].includes(window.location.protocol) ||
116
+ window.location.hostname === 'localhost'
117
+ )
115
118
  }
116
119
 
117
120
  private async extractLicenseKey(licenseKey: string): Promise<LicenseInfo> {
@@ -250,6 +253,15 @@ export class LicenseManager {
250
253
  return globToRegex.test(currentHostname) || globToRegex.test(`www.${currentHostname}`)
251
254
  }
252
255
 
256
+ // VSCode support
257
+ if (window.location.protocol === 'vscode-webview:') {
258
+ const currentUrl = new URL(window.location.href)
259
+ const extensionId = currentUrl.searchParams.get('extensionId')
260
+ if (normalizedHost === extensionId) {
261
+ return true
262
+ }
263
+ }
264
+
253
265
  return false
254
266
  })
255
267
  }
@@ -80,6 +80,10 @@ export interface TldrawOptions {
80
80
  * nonce to use in the editor's styles.
81
81
  */
82
82
  readonly nonce: string | undefined
83
+ /**
84
+ * Branding name of the app, currently only used for adding aria-label for the application.
85
+ */
86
+ readonly branding?: string
83
87
  }
84
88
 
85
89
  /** @public */
@@ -0,0 +1,4 @@
1
+ import { TLShape } from '@tldraw/tlschema'
2
+
3
+ export const areShapesContentEqual = (a: TLShape, b: TLShape) =>
4
+ a.props === b.props && a.meta === b.meta
@@ -91,14 +91,14 @@ export const setStyleProperty = (
91
91
  elm.style.setProperty(property, value as string)
92
92
  }
93
93
 
94
- const INPUTS = ['input', 'select', 'button', 'textarea']
95
-
96
94
  /** @internal */
97
- export function activeElementShouldCaptureKeys() {
95
+ export function activeElementShouldCaptureKeys(allowButtons = false) {
98
96
  const { activeElement } = document
97
+ const elements = allowButtons ? ['input', 'textarea'] : ['input', 'select', 'button', 'textarea']
99
98
  return !!(
100
99
  activeElement &&
101
100
  ((activeElement as HTMLElement).isContentEditable ||
102
- INPUTS.indexOf(activeElement.tagName.toLowerCase()) > -1)
101
+ elements.indexOf(activeElement.tagName.toLowerCase()) > -1 ||
102
+ activeElement.classList.contains('tlui-slider__thumb'))
103
103
  )
104
104
  }
@@ -0,0 +1,13 @@
1
+ // Euclidean algorithm to find the GCD
2
+ function gcd(a: number, b: number): number {
3
+ return b === 0 ? a : gcd(b, a % b)
4
+ }
5
+
6
+ // Returns the lowest value that the given number can be multiplied by to reach an integer
7
+ export function nearestMultiple(float: number) {
8
+ const decimal = float.toString().split('.')[1]
9
+ if (!decimal) return 1
10
+ const denominator = Math.pow(10, decimal.length)
11
+ const numerator = parseInt(decimal, 10)
12
+ return denominator / gcd(numerator, denominator)
13
+ }
@@ -26,11 +26,13 @@ export function getRotationSnapshot({
26
26
  return null
27
27
  }
28
28
 
29
- const pageCenter = rotatedPageBounds.center.clone().rotWith(rotatedPageBounds.point, rotation)
29
+ const initialPageCenter = rotatedPageBounds.center
30
+ .clone()
31
+ .rotWith(rotatedPageBounds.point, rotation)
30
32
 
31
33
  return {
32
- pageCenter,
33
- initialCursorAngle: pageCenter.angle(editor.inputs.originPagePoint),
34
+ initialPageCenter,
35
+ initialCursorAngle: initialPageCenter.angle(editor.inputs.originPagePoint),
34
36
  initialShapesRotation: rotation,
35
37
  shapeSnapshots: shapes.map((shape) => ({
36
38
  shape,
@@ -43,7 +45,7 @@ export function getRotationSnapshot({
43
45
  * @internal
44
46
  **/
45
47
  export interface TLRotationSnapshot {
46
- pageCenter: Vec
48
+ initialPageCenter: Vec
47
49
  initialCursorAngle: number
48
50
  initialShapesRotation: number
49
51
  shapeSnapshots: {
@@ -66,7 +68,7 @@ export function applyRotationToSnapshotShapes({
66
68
  stage: 'start' | 'update' | 'end' | 'one-off'
67
69
  centerOverride?: VecLike
68
70
  }) {
69
- const { pageCenter, shapeSnapshots } = snapshot
71
+ const { initialPageCenter, shapeSnapshots } = snapshot
70
72
 
71
73
  editor.updateShapes(
72
74
  shapeSnapshots.map(({ shape, initialPagePoint }) => {
@@ -77,7 +79,7 @@ export function applyRotationToSnapshotShapes({
77
79
  ? editor.getShapePageTransform(shape.parentId)!
78
80
  : Mat.Identity()
79
81
 
80
- const newPagePoint = Vec.RotWith(initialPagePoint, centerOverride ?? pageCenter, delta)
82
+ const newPagePoint = Vec.RotWith(initialPagePoint, centerOverride ?? initialPageCenter, delta)
81
83
 
82
84
  const newLocalPoint = Mat.applyToPoint(
83
85
  // use the current parent transform in case it has moved/resized since the start
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 = '3.12.1'
4
+ export const version = '3.13.0-canary.064d79cae9fb'
5
5
  export const publishDates = {
6
6
  major: '2024-09-13T14:36:29.063Z',
7
- minor: '2025-04-15T13:47:42.642Z',
8
- patch: '2025-04-22T10:42:34.949Z',
7
+ minor: '2025-05-05T18:38:06.966Z',
8
+ patch: '2025-05-05T18:38:06.966Z',
9
9
  }