diff --git a/apps/docs/content/getting-started/quick-start.mdx b/apps/docs/content/getting-started/quick-start.mdx index fbfa2e481..7cc374c82 100644 --- a/apps/docs/content/getting-started/quick-start.mdx +++ b/apps/docs/content/getting-started/quick-start.mdx @@ -31,7 +31,7 @@ To import fonts and CSS for tldraw: - Copy and paste this into the file: ```CSS -@import url("https://fonts.googleapis.com/css2?family=Inter:wght@500;700;&display=swap"); +@import url("https://fonts.googleapis.com/css2?family=Inter:wght@500;700&display=swap"); @import url("tldraw/tldraw.css"); body { diff --git a/apps/dotcom/src/utils/context-menu/CursorChatMenuItem.tsx b/apps/dotcom/src/utils/context-menu/CursorChatMenuItem.tsx index db0e25d19..5a6872d56 100644 --- a/apps/dotcom/src/utils/context-menu/CursorChatMenuItem.tsx +++ b/apps/dotcom/src/utils/context-menu/CursorChatMenuItem.tsx @@ -7,7 +7,7 @@ export function CursorChatMenuItem() { const shouldShow = useValue( 'show cursor chat', () => { - return editor.getInstanceState().isCoarsePointer && !editor.getSelectedShapes().length + return !editor.getInstanceState().isCoarsePointer }, [editor] ) diff --git a/apps/dotcom/src/utils/useFileSystem.tsx b/apps/dotcom/src/utils/useFileSystem.tsx index c76e1934d..c328bb138 100644 --- a/apps/dotcom/src/utils/useFileSystem.tsx +++ b/apps/dotcom/src/utils/useFileSystem.tsx @@ -108,34 +108,33 @@ export function getSaveFileCopyAction( readonlyOk: true, kbd: '$s', async onSelect(source) { + handleUiEvent('save-project-to-file', { source }) + const documentName = + editor.getDocumentSettings().name === '' + ? defaultDocumentName + : editor.getDocumentSettings().name + const defaultName = + saveFileNames.get(editor.store) || `${documentName}${TLDRAW_FILE_EXTENSION}` + + const blobToSave = serializeTldrawJsonBlob(editor.store) + let handle try { - handleUiEvent('save-project-to-file', { source }) - const documentName = - editor.getDocumentSettings().name === '' - ? defaultDocumentName - : editor.getDocumentSettings().name - const defaultName = - saveFileNames.get(editor.store) || `${documentName}${TLDRAW_FILE_EXTENSION}` - - const blobToSave = serializeTldrawJsonBlob(editor.store) - - const handle = await fileSave(blobToSave, { + handle = await fileSave(blobToSave, { fileName: defaultName, extensions: [TLDRAW_FILE_EXTENSION], description: 'tldraw project', }) - - if (handle) { - // we deliberately don't store the handle for re-use - // next time. we always want to save a copy, but to - // help the user out we'll remember the last name - // they used - saveFileNames.set(editor.store, handle.name) - } else { - throw Error('Could not save file.') - } } catch (e) { - console.error(e) + // user cancelled + return + } + + if (handle) { + // we deliberately don't store the handle for re-use + // next time. we always want to save a copy, but to + // help the user out we'll remember the last name + // they used + saveFileNames.set(editor.store, handle.name) } }, } diff --git a/apps/examples/src/examples/action-overrides/ActionOverridesExample.tsx b/apps/examples/src/examples/action-overrides/ActionOverridesExample.tsx new file mode 100644 index 000000000..90d21d1ee --- /dev/null +++ b/apps/examples/src/examples/action-overrides/ActionOverridesExample.tsx @@ -0,0 +1,27 @@ +import { Tldraw } from 'tldraw' +import 'tldraw/tldraw.css' + +export default function BasicExample() { + return ( +
+ { + const newActions = { + ...actions, + delete: { ...actions['delete'], kbd: 'x' }, + } + return newActions + }, + }} + /> +
+ ) +} + +/* +This example shows how you can override tldraw's actions object to change the keyboard shortcuts. +In this case we're changing the delete action's shortcut to 'x'. To customize the actions menu +please see the custom actions menu example. For more information on keyboard shortcuts see the +keyboard shortcuts example. +*/ diff --git a/apps/examples/src/examples/action-overrides/README.md b/apps/examples/src/examples/action-overrides/README.md new file mode 100644 index 000000000..87b182bab --- /dev/null +++ b/apps/examples/src/examples/action-overrides/README.md @@ -0,0 +1,12 @@ +--- +title: Action overrides +component: ./ActionOverridesExample.tsx +category: ui +priority: 2 +--- + +Override tldraw's actions + +--- + +This example shows how you can override tldraw's actions object to change the keyboard shortcuts. In this case we're changing the delete action's shortcut to 'x'. To customize the actions menu please see the custom actions menu example. For more information on keyboard shortcuts see the keyboard shortcuts example. diff --git a/apps/examples/src/examples/custom-renderer/CustomRenderer.tsx b/apps/examples/src/examples/custom-renderer/CustomRenderer.tsx index 7e20457ea..66cbe70a1 100644 --- a/apps/examples/src/examples/custom-renderer/CustomRenderer.tsx +++ b/apps/examples/src/examples/custom-renderer/CustomRenderer.tsx @@ -36,7 +36,8 @@ export function CustomRenderer() { const theme = getDefaultColorTheme({ isDarkMode: editor.user.getIsDarkMode() }) const currentPageId = editor.getCurrentPageId() - for (const { shape, maskedPageBounds, opacity } of renderingShapes) { + for (const { shape, opacity } of renderingShapes) { + const maskedPageBounds = editor.getShapeMaskedPageBounds(shape) if (!maskedPageBounds) continue ctx.save() diff --git a/apps/examples/src/examples/keyboard-shortcuts/KeyboardShortcuts.tsx b/apps/examples/src/examples/keyboard-shortcuts/KeyboardShortcuts.tsx index b7c01e4eb..c0b7c1436 100644 --- a/apps/examples/src/examples/keyboard-shortcuts/KeyboardShortcuts.tsx +++ b/apps/examples/src/examples/keyboard-shortcuts/KeyboardShortcuts.tsx @@ -1,6 +1,5 @@ import { TLUiActionsContextType, TLUiOverrides, TLUiToolsContextType, Tldraw } from 'tldraw' import 'tldraw/tldraw.css' -import jsonSnapshot from './snapshot.json' // There's a guide at the bottom of this file! @@ -8,13 +7,18 @@ import jsonSnapshot from './snapshot.json' const overrides: TLUiOverrides = { //[a] actions(_editor, actions): TLUiActionsContextType { - actions['toggle-grid'].kbd = 'x' - return actions + const newActions = { + ...actions, + 'toggle-grid': { ...actions['toggle-grid'], kbd: 'x' }, + 'copy-as-png': { ...actions['copy-as-png'], kbd: '$1' }, + } + + return newActions }, //[b] tools(_editor, tools): TLUiToolsContextType { - tools['draw'].kbd = 'p' - return tools + const newTools = { ...tools, draw: { ...tools.draw, kbd: 'p' } } + return newTools }, } @@ -22,7 +26,7 @@ const overrides: TLUiOverrides = { export default function KeyboardShortcuts() { return (
- +
) } diff --git a/apps/examples/src/examples/keyboard-shortcuts/README.md b/apps/examples/src/examples/keyboard-shortcuts/README.md index f726e110c..bd050528e 100644 --- a/apps/examples/src/examples/keyboard-shortcuts/README.md +++ b/apps/examples/src/examples/keyboard-shortcuts/README.md @@ -5,8 +5,14 @@ category: ui priority: 2 --- -Override default keyboard shortcuts. +How to replace tldraw's default keyboard shortcuts with your own. --- -How to replace tldraw's default keyboard shortcuts with your own. +This example shows how you can replace tldraw's default keyboard shortcuts with your own, +or add a shortcut for an action that doesn't have one. An example of how to add shortcuts +for custom tools can be found in the custom-config example. + +- Toggle show grid by pressing 'x' +- Select the Draw tool by pressing 'p' +- Copy as png by pressing 'ctrl/cmd + 1' diff --git a/apps/examples/src/examples/keyboard-shortcuts/snapshot.json b/apps/examples/src/examples/keyboard-shortcuts/snapshot.json deleted file mode 100644 index 65718fd4b..000000000 --- a/apps/examples/src/examples/keyboard-shortcuts/snapshot.json +++ /dev/null @@ -1,156 +0,0 @@ -{ - "store": { - "asset:2051922215": { - "meta": {}, - "type": "image", - "props": { - "name": "tldrawFile", - "src": "", - "w": 1000, - "h": 595.6175298804781, - "mimeType": "image/png", - "isAnimated": false - }, - "id": "asset:2051922215", - "typeName": "asset" - }, - "document:document": { - "gridSize": 10, - "name": "", - "meta": {}, - "id": "document:document", - "typeName": "document" - }, - "page:NXBP7PKuITv3tMvoIRxFz": { - "meta": {}, - "id": "page:NXBP7PKuITv3tMvoIRxFz", - "name": "Page 1", - "index": "a1", - "typeName": "page" - }, - "shape:0O-D3-H12T9edVMgA0cDh": { - "x": 224.28665129528747, - "y": 474.44055394139673, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "text", - "props": { - "color": "black", - "size": "m", - "w": 438.1640625, - "text": "Select the Draw tool by pressing 'p'", - "font": "draw", - "align": "middle", - "autoSize": true, - "scale": 1 - }, - "parentId": "page:NXBP7PKuITv3tMvoIRxFz", - "index": "a2", - "id": "shape:0O-D3-H12T9edVMgA0cDh", - "typeName": "shape" - }, - "shape:tAsz1L8N4kLDwIvpCcjqN": { - "x": 241.59357019335232, - "y": 367.6707932953341, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "text", - "props": { - "color": "black", - "size": "m", - "w": 397.3125, - "text": "Toggle show grid by pressing 'x'", - "font": "draw", - "align": "middle", - "autoSize": true, - "scale": 1 - }, - "parentId": "page:NXBP7PKuITv3tMvoIRxFz", - "index": "a1", - "id": "shape:tAsz1L8N4kLDwIvpCcjqN", - "typeName": "shape" - }, - "shape:Jv8vs9T77_Ojum2OGZd7D": { - "x": 204.13270568300845, - "y": 579.9171242617172, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "id": "shape:Jv8vs9T77_Ojum2OGZd7D", - "type": "text", - "props": { - "color": "black", - "size": "m", - "w": 462.7421875, - "text": "Copy as png by pressing 'ctrl/cmd + 1'", - "font": "draw", - "align": "middle", - "autoSize": true, - "scale": 1 - }, - "parentId": "page:NXBP7PKuITv3tMvoIRxFz", - "index": "a3", - "typeName": "shape" - } - }, - "schema": { - "schemaVersion": 1, - "storeVersion": 4, - "recordVersions": { - "asset": { - "version": 1, - "subTypeKey": "type", - "subTypeVersions": { - "image": 3, - "video": 3, - "bookmark": 1 - } - }, - "camera": { - "version": 1 - }, - "document": { - "version": 2 - }, - "instance": { - "version": 23 - }, - "instance_page_state": { - "version": 5 - }, - "page": { - "version": 1 - }, - "shape": { - "version": 3, - "subTypeKey": "type", - "subTypeVersions": { - "group": 0, - "text": 1, - "bookmark": 2, - "draw": 1, - "geo": 8, - "note": 5, - "line": 1, - "frame": 0, - "arrow": 2, - "highlight": 0, - "embed": 4, - "image": 3, - "video": 2 - } - }, - "instance_presence": { - "version": 5 - }, - "pointer": { - "version": 1 - } - } - } -} diff --git a/apps/examples/src/examples/rendering-shape-changes/useRenderingShapesChange.ts b/apps/examples/src/examples/rendering-shape-changes/useRenderingShapesChange.ts index ae181b28d..4b4ac137f 100644 --- a/apps/examples/src/examples/rendering-shape-changes/useRenderingShapesChange.ts +++ b/apps/examples/src/examples/rendering-shape-changes/useRenderingShapesChange.ts @@ -5,26 +5,34 @@ export function useChangedShapesReactor( cb: (info: { culled: TLShape[]; restored: TLShape[] }) => void ) { const editor = useEditor() - const rPrevShapes = useRef(editor.getRenderingShapes()) + const rPrevShapes = useRef({ + renderingShapes: editor.getRenderingShapes(), + culledShapes: editor.getCulledShapes(), + }) useEffect(() => { return react('when rendering shapes change', () => { - const after = editor.getRenderingShapes() + const after = { + culledShapes: editor.getCulledShapes(), + renderingShapes: editor.getRenderingShapes(), + } const before = rPrevShapes.current const culled: TLShape[] = [] const restored: TLShape[] = [] - const beforeToVisit = new Set(before) + const beforeToVisit = new Set(before.renderingShapes) - for (const afterInfo of after) { - const beforeInfo = before.find((s) => s.id === afterInfo.id) + for (const afterInfo of after.renderingShapes) { + const beforeInfo = before.renderingShapes.find((s) => s.id === afterInfo.id) if (!beforeInfo) { continue } else { - if (afterInfo.isCulled && !beforeInfo.isCulled) { + const isAfterCulled = after.culledShapes.has(afterInfo.id) + const isBeforeCulled = before.culledShapes.has(beforeInfo.id) + if (isAfterCulled && !isBeforeCulled) { culled.push(afterInfo.shape) - } else if (!afterInfo.isCulled && beforeInfo.isCulled) { + } else if (!isAfterCulled && isBeforeCulled) { restored.push(afterInfo.shape) } beforeToVisit.delete(beforeInfo) diff --git a/apps/vscode/extension/CHANGELOG.md b/apps/vscode/extension/CHANGELOG.md index 2713f418c..c40252b56 100644 --- a/apps/vscode/extension/CHANGELOG.md +++ b/apps/vscode/extension/CHANGELOG.md @@ -1,3 +1,7 @@ +## 2.0.27 + +- Bug fixes and performance improvements. + ## 2.0.26 - Bug fixes and performance improvements. diff --git a/apps/vscode/extension/package.json b/apps/vscode/extension/package.json index 35c3332bb..9558bbf95 100644 --- a/apps/vscode/extension/package.json +++ b/apps/vscode/extension/package.json @@ -1,7 +1,7 @@ { "name": "tldraw-vscode", "description": "The tldraw extension for VS Code.", - "version": "2.0.26", + "version": "2.0.27", "private": true, "author": { "name": "tldraw Inc.", diff --git a/packages/editor/api-report.md b/packages/editor/api-report.md index a95820545..01be034e1 100644 --- a/packages/editor/api-report.md +++ b/packages/editor/api-report.md @@ -516,7 +516,7 @@ export function degreesToRadians(d: number): number; export const DOUBLE_CLICK_DURATION = 450; // @internal (undocumented) -export const DRAG_DISTANCE = 4; +export const DRAG_DISTANCE = 16; // @public (undocumented) export const EASINGS: { @@ -677,6 +677,7 @@ export class Editor extends EventEmitter { // @internal getCrashingError(): unknown; getCroppingShapeId(): null | TLShapeId; + getCulledShapes(): Set; getCurrentPage(): TLPage; getCurrentPageBounds(): Box | undefined; getCurrentPageId(): TLPageId; @@ -714,7 +715,6 @@ export class Editor extends EventEmitter { getPointInParentSpace(shape: TLShape | TLShapeId, point: VecLike): Vec; getPointInShapeSpace(shape: TLShape | TLShapeId, point: VecLike): Vec; getRenderingBounds(): Box; - getRenderingBoundsExpanded(): Box; getRenderingShapes(): { id: TLShapeId; shape: TLShape; @@ -722,8 +722,6 @@ export class Editor extends EventEmitter { index: number; backgroundIndex: number; opacity: number; - isCulled: boolean; - maskedPageBounds: Box | undefined; }[]; getSelectedShapeAtPoint(point: VecLike): TLShape | undefined; getSelectedShapeIds(): TLShapeId[]; @@ -784,8 +782,6 @@ export class Editor extends EventEmitter { index: number; backgroundIndex: number; opacity: number; - isCulled: boolean; - maskedPageBounds: Box | undefined; }[]; getViewportPageBounds(): Box; getViewportPageCenter(): Vec; @@ -1676,7 +1672,6 @@ export abstract class ShapeUtil { canResize: TLShapeUtilFlag; canScroll: TLShapeUtilFlag; canSnap: TLShapeUtilFlag; - canUnmount: TLShapeUtilFlag; abstract component(shape: Shape): any; // (undocumented) editor: Editor; @@ -1888,6 +1883,8 @@ export abstract class StateNode implements Partial { // (undocumented) onKeyUp?: TLEventHandlers['onKeyUp']; // (undocumented) + onLongPress?: TLEventHandlers['onLongPress']; + // (undocumented) onMiddleClick?: TLEventHandlers['onMiddleClick']; // (undocumented) onPointerDown?: TLEventHandlers['onPointerDown']; @@ -2185,6 +2182,8 @@ export interface TLEventHandlers { // (undocumented) onKeyUp: TLKeyboardEvent; // (undocumented) + onLongPress: TLPointerEvent; + // (undocumented) onMiddleClick: TLPointerEvent; // (undocumented) onPointerDown: TLPointerEvent; @@ -2437,7 +2436,7 @@ export type TLPointerEventInfo = TLBaseEventInfo & { } & TLPointerEventTarget; // @public (undocumented) -export type TLPointerEventName = 'middle_click' | 'pointer_down' | 'pointer_move' | 'pointer_up' | 'right_click'; +export type TLPointerEventName = 'long_press' | 'middle_click' | 'pointer_down' | 'pointer_move' | 'pointer_up' | 'right_click'; // @public (undocumented) export type TLPointerEventTarget = { diff --git a/packages/editor/api/api.json b/packages/editor/api/api.json index 9faa05b3d..ac7faa3f9 100644 --- a/packages/editor/api/api.json +++ b/packages/editor/api/api.json @@ -10367,6 +10367,51 @@ "isAbstract": false, "name": "getCroppingShapeId" }, + { + "kind": "Method", + "canonicalReference": "@tldraw/editor!Editor#getCulledShapes:member(1)", + "docComment": "/**\n * Get culled shapes.\n *\n * @public\n */\n", + "excerptTokens": [ + { + "kind": "Content", + "text": "getCulledShapes(): " + }, + { + "kind": "Reference", + "text": "Set", + "canonicalReference": "!Set:interface" + }, + { + "kind": "Content", + "text": "<" + }, + { + "kind": "Reference", + "text": "TLShapeId", + "canonicalReference": "@tldraw/tlschema!TLShapeId:type" + }, + { + "kind": "Content", + "text": ">" + }, + { + "kind": "Content", + "text": ";" + } + ], + "isStatic": false, + "returnTypeTokenRange": { + "startIndex": 1, + "endIndex": 5 + }, + "releaseTag": "Public", + "isProtected": false, + "overloadIndex": 1, + "parameters": [], + "isOptional": false, + "isAbstract": false, + "name": "getCulledShapes" + }, { "kind": "Method", "canonicalReference": "@tldraw/editor!Editor#getCurrentPage:member(1)", @@ -11959,38 +12004,6 @@ "isAbstract": false, "name": "getRenderingBounds" }, - { - "kind": "Method", - "canonicalReference": "@tldraw/editor!Editor#getRenderingBoundsExpanded:member(1)", - "docComment": "/**\n * The current rendering bounds in the current page space, expanded slightly. Used for determining which shapes to render and which to \"cull\".\n *\n * @public\n */\n", - "excerptTokens": [ - { - "kind": "Content", - "text": "getRenderingBoundsExpanded(): " - }, - { - "kind": "Reference", - "text": "Box", - "canonicalReference": "@tldraw/editor!Box:class" - }, - { - "kind": "Content", - "text": ";" - } - ], - "isStatic": false, - "returnTypeTokenRange": { - "startIndex": 1, - "endIndex": 2 - }, - "releaseTag": "Public", - "isProtected": false, - "overloadIndex": 1, - "parameters": [], - "isOptional": false, - "isAbstract": false, - "name": "getRenderingBoundsExpanded" - }, { "kind": "Method", "canonicalReference": "@tldraw/editor!Editor#getRenderingShapes:member(1)", @@ -12038,16 +12051,7 @@ }, { "kind": "Content", - "text": ">;\n index: number;\n backgroundIndex: number;\n opacity: number;\n isCulled: boolean;\n maskedPageBounds: " - }, - { - "kind": "Reference", - "text": "Box", - "canonicalReference": "@tldraw/editor!Box:class" - }, - { - "kind": "Content", - "text": " | undefined;\n }[]" + "text": ">;\n index: number;\n backgroundIndex: number;\n opacity: number;\n }[]" }, { "kind": "Content", @@ -12057,7 +12061,7 @@ "isStatic": false, "returnTypeTokenRange": { "startIndex": 1, - "endIndex": 12 + "endIndex": 10 }, "releaseTag": "Public", "isProtected": false, @@ -31139,41 +31143,6 @@ "isProtected": false, "isAbstract": false }, - { - "kind": "Property", - "canonicalReference": "@tldraw/editor!ShapeUtil#canUnmount:member", - "docComment": "/**\n * Whether the shape should unmount when not visible in the editor. Consider keeping this to false if the shape's `component` has local state.\n *\n * @public\n */\n", - "excerptTokens": [ - { - "kind": "Content", - "text": "canUnmount: " - }, - { - "kind": "Reference", - "text": "TLShapeUtilFlag", - "canonicalReference": "@tldraw/editor!TLShapeUtilFlag:type" - }, - { - "kind": "Content", - "text": "" - }, - { - "kind": "Content", - "text": ";" - } - ], - "isReadonly": false, - "isOptional": false, - "releaseTag": "Public", - "name": "canUnmount", - "propertyTypeTokenRange": { - "startIndex": 1, - "endIndex": 3 - }, - "isStatic": false, - "isProtected": false, - "isAbstract": false - }, { "kind": "Method", "canonicalReference": "@tldraw/editor!ShapeUtil#component:member(1)", @@ -35319,6 +35288,41 @@ "isProtected": false, "isAbstract": false }, + { + "kind": "Property", + "canonicalReference": "@tldraw/editor!StateNode#onLongPress:member", + "docComment": "", + "excerptTokens": [ + { + "kind": "Content", + "text": "onLongPress?: " + }, + { + "kind": "Reference", + "text": "TLEventHandlers", + "canonicalReference": "@tldraw/editor!TLEventHandlers:interface" + }, + { + "kind": "Content", + "text": "['onLongPress']" + }, + { + "kind": "Content", + "text": ";" + } + ], + "isReadonly": false, + "isOptional": true, + "releaseTag": "Public", + "name": "onLongPress", + "propertyTypeTokenRange": { + "startIndex": 1, + "endIndex": 3 + }, + "isStatic": false, + "isProtected": false, + "isAbstract": false + }, { "kind": "Property", "canonicalReference": "@tldraw/editor!StateNode#onMiddleClick:member", @@ -38590,6 +38594,34 @@ "endIndex": 2 } }, + { + "kind": "PropertySignature", + "canonicalReference": "@tldraw/editor!TLEventHandlers#onLongPress:member", + "docComment": "", + "excerptTokens": [ + { + "kind": "Content", + "text": "onLongPress: " + }, + { + "kind": "Reference", + "text": "TLPointerEvent", + "canonicalReference": "@tldraw/editor!TLPointerEvent:type" + }, + { + "kind": "Content", + "text": ";" + } + ], + "isReadonly": false, + "isOptional": false, + "releaseTag": "Public", + "name": "onLongPress", + "propertyTypeTokenRange": { + "startIndex": 1, + "endIndex": 2 + } + }, { "kind": "PropertySignature", "canonicalReference": "@tldraw/editor!TLEventHandlers#onMiddleClick:member", @@ -41123,7 +41155,7 @@ }, { "kind": "Content", - "text": "'middle_click' | 'pointer_down' | 'pointer_move' | 'pointer_up' | 'right_click'" + "text": "'long_press' | 'middle_click' | 'pointer_down' | 'pointer_move' | 'pointer_up' | 'right_click'" }, { "kind": "Content", diff --git a/packages/editor/editor.css b/packages/editor/editor.css index 7ed918505..03bbb64f4 100644 --- a/packages/editor/editor.css +++ b/packages/editor/editor.css @@ -24,10 +24,12 @@ /* Z Index */ --layer-background: 100; --layer-grid: 150; + --layer-culled-shapes: 175; --layer-canvas: 200; --layer-shapes: 300; --layer-overlays: 400; --layer-following-indicator: 1000; + --layer-blocker: 10000; /* Misc */ --tl-zoom: 1; @@ -236,6 +238,20 @@ input, contain: strict; } +.tl-culled-shapes { + width: 100%; + height: 100%; + z-index: var(--layer-culled-shapes); + position: absolute; + pointer-events: none; + contain: size layout; +} + +.tl-culled-shapes__canvas { + width: 100%; + height: 100%; +} + .tl-shapes { position: relative; z-index: var(--layer-shapes); @@ -264,18 +280,25 @@ input, position: absolute; top: 0px; left: 0px; + width: 100%; + height: 100%; pointer-events: none; } /* ------------------- Background ------------------- */ -.tl-background { +.tl-background__wrapper { + z-index: var(--layer-background); position: absolute; - background-color: var(--color-background); inset: 0px; height: 100%; width: 100%; - z-index: var(--layer-background); +} + +.tl-background { + background-color: var(--color-background); + width: 100%; + height: 100%; } /* --------------------- Grid Layer --------------------- */ @@ -338,10 +361,13 @@ input, } .tl-shape__culled { - position: relative; + position: absolute; + pointer-events: none; + overflow: visible; + transform-origin: top left; + contain: size layout; background-color: var(--color-culled); - width: 100%; - height: 100%; + z-index: 0; } /* ---------------- Shape Containers ---------------- */ @@ -1480,3 +1506,18 @@ it from receiving any pointer events or affecting the cursor. */ font-size: 12px; font-family: monospace; } + +/* ---------------- Hit test blocker ---------------- */ + +.tl-hit-test-blocker { + position: absolute; + z-index: var(--layer-blocker); + inset: 0px; + width: 100%; + height: 100%; + pointer-events: all; +} + +.tl-hit-test-blocker__hidden { + display: none; +} diff --git a/packages/editor/src/lib/components/Shape.tsx b/packages/editor/src/lib/components/Shape.tsx index 3762290b3..909d51885 100644 --- a/packages/editor/src/lib/components/Shape.tsx +++ b/packages/editor/src/lib/components/Shape.tsx @@ -1,12 +1,10 @@ import { useQuickReactor, useStateTracking } from '@tldraw/state' -import { IdOf } from '@tldraw/store' import { TLShape, TLShapeId } from '@tldraw/tlschema' import { memo, useCallback, useRef } from 'react' import { ShapeUtil } from '../editor/shapes/ShapeUtil' import { useEditor } from '../hooks/useEditor' import { useEditorComponents } from '../hooks/useEditorComponents' import { Mat } from '../primitives/Mat' -import { toDomPrecision } from '../primitives/utils' import { setStyleProperty } from '../utils/dom' import { OptionalErrorBoundary } from './ErrorBoundary' @@ -28,7 +26,6 @@ export const Shape = memo(function Shape({ index, backgroundIndex, opacity, - isCulled, dprMultiple, }: { id: TLShapeId @@ -37,7 +34,6 @@ export const Shape = memo(function Shape({ index: number backgroundIndex: number opacity: number - isCulled: boolean dprMultiple: number }) { const editor = useEditor() @@ -52,6 +48,9 @@ export const Shape = memo(function Shape({ clipPath: 'none', width: 0, height: 0, + x: 0, + y: 0, + isCulled: false, }) useQuickReactor( @@ -71,7 +70,11 @@ export const Shape = memo(function Shape({ } // Page transform - const transform = Mat.toCssString(editor.getShapePageTransform(id)) + const pageTransform = editor.getShapePageTransform(id) + const transform = Mat.toCssString(pageTransform) + const bounds = editor.getShapeGeometry(shape).bounds + + // Update if the tranform has changed if (transform !== prev.transform) { setStyleProperty(containerRef.current, 'transform', transform) setStyleProperty(bgContainerRef.current, 'transform', transform) @@ -81,7 +84,6 @@ export const Shape = memo(function Shape({ // Width / Height // We round the shape width and height up to the nearest multiple of dprMultiple // to avoid the browser making miscalculations when applying the transform. - const bounds = editor.getShapeGeometry(shape).bounds const widthRemainder = bounds.w % dprMultiple const heightRemainder = bounds.h % dprMultiple const width = widthRemainder === 0 ? bounds.w : bounds.w + (dprMultiple - widthRemainder) @@ -117,6 +119,22 @@ export const Shape = memo(function Shape({ [opacity, index, backgroundIndex] ) + useQuickReactor( + 'set display', + () => { + const shape = editor.getShape(id) + if (!shape) return // probably the shape was just deleted + + const culledShapes = editor.getCulledShapes() + const isCulled = culledShapes.has(id) + if (isCulled !== memoizedStuffRef.current.isCulled) { + setStyleProperty(containerRef.current, 'display', isCulled ? 'none' : 'block') + setStyleProperty(bgContainerRef.current, 'display', isCulled ? 'none' : 'block') + memoizedStuffRef.current.isCulled = isCulled + } + }, + [editor] + ) const annotateError = useCallback( (error: any) => editor.annotateError(error, { origin: 'shape', willCrashApp: false }), [editor] @@ -133,21 +151,15 @@ export const Shape = memo(function Shape({ data-shape-type={shape.type} draggable={false} > - {isCulled ? null : ( - - - - )} + + + )}
- {isCulled ? ( - - ) : ( - - - - )} + + +
) @@ -172,23 +184,3 @@ const InnerShapeBackground = memo( }, (prev, next) => prev.shape.props === next.shape.props && prev.shape.meta === next.shape.meta ) - -const CulledShape = function CulledShape({ shapeId }: { shapeId: IdOf }) { - const editor = useEditor() - const culledRef = useRef(null) - - useQuickReactor( - 'set shape stuff', - () => { - const bounds = editor.getShapeGeometry(shapeId).bounds - setStyleProperty( - culledRef.current, - 'transform', - `translate(${toDomPrecision(bounds.minX)}px, ${toDomPrecision(bounds.minY)}px)` - ) - }, - [editor] - ) - - return
-} diff --git a/packages/editor/src/lib/components/default-components/DefaultCanvas.tsx b/packages/editor/src/lib/components/default-components/DefaultCanvas.tsx index 16a18da6a..a752edb54 100644 --- a/packages/editor/src/lib/components/default-components/DefaultCanvas.tsx +++ b/packages/editor/src/lib/components/default-components/DefaultCanvas.tsx @@ -3,9 +3,10 @@ import { TLHandle, TLShapeId } from '@tldraw/tlschema' import { dedupe, modulate, objectMapValues } from '@tldraw/utils' import classNames from 'classnames' import { Fragment, JSX, useEffect, useRef, useState } from 'react' -import { COARSE_HANDLE_RADIUS, HANDLE_RADIUS } from '../../constants' +import { COARSE_HANDLE_RADIUS, HANDLE_RADIUS, TEXT_SHADOW_LOD } from '../../constants' import { useCanvasEvents } from '../../hooks/useCanvasEvents' import { useCoarsePointer } from '../../hooks/useCoarsePointer' +import { useContainer } from '../../hooks/useContainer' import { useDocumentEvents } from '../../hooks/useDocumentEvents' import { useEditor } from '../../hooks/useEditor' import { useEditorComponents } from '../../hooks/useEditorComponents' @@ -36,6 +37,7 @@ export function DefaultCanvas({ className }: TLCanvasComponentProps) { const rCanvas = useRef(null) const rHtmlLayer = useRef(null) const rHtmlLayer2 = useRef(null) + const container = useContainer() useScreenBounds(rCanvas) useDocumentEvents() @@ -44,11 +46,37 @@ export function DefaultCanvas({ className }: TLCanvasComponentProps) { useGestureEvents(rCanvas) useFixSafariDoubleTapZoomPencilEvents(rCanvas) + const rMemoizedStuff = useRef({ lodDisableTextOutline: false, allowTextOutline: true }) + useQuickReactor( 'position layers', - () => { + function positionLayersWhenCameraMoves() { const { x, y, z } = editor.getCamera() + // This should only run once on first load + if (rMemoizedStuff.current.allowTextOutline && editor.environment.isSafari) { + container.style.setProperty('--tl-text-outline', 'none') + rMemoizedStuff.current.allowTextOutline = false + } + + // And this should only run if we're not in Safari; + // If we're below the lod distance for text shadows, turn them off + if ( + rMemoizedStuff.current.allowTextOutline && + z < TEXT_SHADOW_LOD !== rMemoizedStuff.current.lodDisableTextOutline + ) { + const lodDisableTextOutline = z < TEXT_SHADOW_LOD + container.style.setProperty( + '--tl-text-outline', + lodDisableTextOutline + ? 'none' + : `0 var(--b) 0 var(--color-background), 0 var(--a) 0 var(--color-background), + var(--b) var(--b) 0 var(--color-background), var(--a) var(--b) 0 var(--color-background), + var(--a) var(--a) 0 var(--color-background), var(--b) var(--a) 0 var(--color-background)` + ) + rMemoizedStuff.current.lodDisableTextOutline = lodDisableTextOutline + } + // Because the html container has a width/height of 1px, we // need to create a small offset when zoomed to ensure that // the html container and svg container are lined up exactly. @@ -61,7 +89,7 @@ export function DefaultCanvas({ className }: TLCanvasComponentProps) { setStyleProperty(rHtmlLayer.current, 'transform', transform) setStyleProperty(rHtmlLayer2.current, 'transform', transform) }, - [editor] + [editor, container] ) const events = useCanvasEvents() @@ -105,9 +133,12 @@ export function DefaultCanvas({ className }: TLCanvasComponentProps) { {SvgDefs && } - {Background && } + {Background && ( +
+ +
+ )} -
@@ -129,6 +160,7 @@ export function DefaultCanvas({ className }: TLCanvasComponentProps) {
+ ) } @@ -350,6 +382,30 @@ function ShapesWithSVGs() { ) } +function ReflowIfNeeded() { + const editor = useEditor() + const culledShapesRef = useRef>(new Set()) + useQuickReactor( + 'reflow for culled shapes', + () => { + const culledShapes = editor.getCulledShapes() + if ( + culledShapesRef.current.size === culledShapes.size && + [...culledShapes].every((id) => culledShapesRef.current.has(id)) + ) + return + + culledShapesRef.current = culledShapes + const canvas = document.getElementsByClassName('tl-canvas') + if (canvas.length === 0) return + // This causes a reflow + // https://gist.github.com/paulirish/5d52fb081b3570c81e3a + const _height = (canvas[0] as HTMLDivElement).offsetHeight + }, + [editor] + ) + return null +} function ShapesToDisplay() { const editor = useEditor() @@ -370,6 +426,7 @@ function ShapesToDisplay() { {renderingShapes.map((result) => ( ))} + {editor.environment.isSafari && } ) } @@ -581,3 +638,16 @@ function InFrontOfTheCanvasWrapper() { if (!InFrontOfTheCanvas) return null return } + +function MovingCameraHitTestBlocker() { + const editor = useEditor() + const cameraState = useValue('camera state', () => editor.getCameraState(), [editor]) + + return ( +
+ ) +} diff --git a/packages/editor/src/lib/constants.ts b/packages/editor/src/lib/constants.ts index fbdb79814..418491770 100644 --- a/packages/editor/src/lib/constants.ts +++ b/packages/editor/src/lib/constants.ts @@ -34,10 +34,10 @@ export const DOUBLE_CLICK_DURATION = 450 export const MULTI_CLICK_DURATION = 200 /** @internal */ -export const COARSE_DRAG_DISTANCE = 6 +export const COARSE_DRAG_DISTANCE = 36 // 6 squared /** @internal */ -export const DRAG_DISTANCE = 4 +export const DRAG_DISTANCE = 16 // 4 squared /** @internal */ export const SVG_PADDING = 32 @@ -104,3 +104,9 @@ export const COARSE_HANDLE_RADIUS = 20 /** @internal */ export const HANDLE_RADIUS = 12 + +/** @internal */ +export const LONG_PRESS_DURATION = 500 + +/** @internal */ +export const TEXT_SHADOW_LOD = 0.35 diff --git a/packages/editor/src/lib/editor/Editor.ts b/packages/editor/src/lib/editor/Editor.ts index 8729d3fa3..db09c3a1a 100644 --- a/packages/editor/src/lib/editor/Editor.ts +++ b/packages/editor/src/lib/editor/Editor.ts @@ -78,6 +78,7 @@ import { FOLLOW_CHASE_ZOOM_UNSNAP, HIT_TEST_MARGIN, INTERNAL_POINTER_IDS, + LONG_PRESS_DURATION, MAX_PAGES, MAX_SHAPES_PER_PAGE, MAX_ZOOM, @@ -100,6 +101,7 @@ import { getReorderingShapesChanges } from '../utils/reorderShapes' import { applyRotationToSnapshotShapes, getRotationSnapshot } from '../utils/rotation' import { uniqueId } from '../utils/uniqueId' import { arrowBindingsIndex } from './derivations/arrowBindingsIndex' +import { notVisibleShapes } from './derivations/notVisibleShapes' import { parentsToChildren } from './derivations/parentsToChildren' import { deriveShapeIdsInCurrentPage } from './derivations/shapeIdsInCurrentPage' import { getSvgJsx } from './getSvgJsx' @@ -3071,50 +3073,26 @@ export class Editor extends EventEmitter { index: number backgroundIndex: number opacity: number - isCulled: boolean - maskedPageBounds: Box | undefined }[] = [] let nextIndex = MAX_SHAPES_PER_PAGE * 2 let nextBackgroundIndex = MAX_SHAPES_PER_PAGE - // We only really need these if we're using editor state, but that's ok - const editingShapeId = this.getEditingShapeId() - const selectedShapeIds = this.getSelectedShapeIds() const erasingShapeIds = this.getErasingShapeIds() - const renderingBoundsExpanded = this.getRenderingBoundsExpanded() - - // If renderingBoundsMargin is set to Infinity, then we won't cull offscreen shapes - const isCullingOffScreenShapes = Number.isFinite(this.renderingBoundsMargin) const addShapeById = (id: TLShapeId, opacity: number, isAncestorErasing: boolean) => { const shape = this.getShape(id) if (!shape) return opacity *= shape.opacity - let isCulled = false let isShapeErasing = false const util = this.getShapeUtil(shape) - const maskedPageBounds = this.getShapeMaskedPageBounds(id) if (useEditorState) { isShapeErasing = !isAncestorErasing && erasingShapeIds.includes(id) if (isShapeErasing) { opacity *= 0.32 } - - isCulled = - isCullingOffScreenShapes && - // only cull shapes that allow unmounting, i.e. not stateful components - util.canUnmount(shape) && - // never cull editingg shapes - editingShapeId !== id && - // if the shape is fully outside of its parent's clipping bounds... - (maskedPageBounds === undefined || - // ...or if the shape is outside of the expanded viewport bounds... - (!renderingBoundsExpanded.includes(maskedPageBounds) && - // ...and if it's not selected... then cull it - !selectedShapeIds.includes(id))) } renderingShapes.push({ @@ -3124,8 +3102,6 @@ export class Editor extends EventEmitter { index: nextIndex, backgroundIndex: nextBackgroundIndex, opacity, - isCulled, - maskedPageBounds, }) nextIndex += 1 @@ -3195,19 +3171,6 @@ export class Editor extends EventEmitter { /** @internal */ private readonly _renderingBounds = atom('rendering viewport', new Box()) - /** - * The current rendering bounds in the current page space, expanded slightly. Used for determining which shapes - * to render and which to "cull". - * - * @public - */ - getRenderingBoundsExpanded() { - return this._renderingBoundsExpanded.get() - } - - /** @internal */ - private readonly _renderingBoundsExpanded = atom('rendering viewport expanded', new Box()) - /** * Update the rendering bounds. This should be called when the viewport has stopped changing, such * as at the end of a pan, zoom, or animation. @@ -3225,13 +3188,6 @@ export class Editor extends EventEmitter { if (viewportPageBounds.equals(this._renderingBounds.__unsafe__getWithoutCapture())) return this this._renderingBounds.set(viewportPageBounds.clone()) - if (Number.isFinite(this.renderingBoundsMargin)) { - this._renderingBoundsExpanded.set( - viewportPageBounds.clone().expandBy(this.renderingBoundsMargin / this.getZoomLevel()) - ) - } else { - this._renderingBoundsExpanded.set(viewportPageBounds) - } return this } @@ -3272,7 +3228,7 @@ export class Editor extends EventEmitter { * * @public */ - getCurrentPageId(): TLPageId { + @computed getCurrentPageId(): TLPageId { return this.getInstanceState().currentPageId } @@ -4029,6 +3985,33 @@ export class Editor extends EventEmitter { return this.isShapeOrAncestorLocked(this.getShapeParent(shape)) } + @computed + private _notVisibleShapes() { + return notVisibleShapes(this) + } + + /** + * Get culled shapes. + * + * @public + */ + @computed + getCulledShapes() { + const notVisibleShapes = this._notVisibleShapes().get() + const selectedShapeIds = this.getSelectedShapeIds() + const editingId = this.getEditingShapeId() + const culledShapes = new Set(notVisibleShapes) + // we don't cull the shape we are editing + if (editingId) { + culledShapes.delete(editingId) + } + // we also don't cull selected shapes + selectedShapeIds.forEach((id) => { + culledShapes.delete(id) + }) + return culledShapes + } + /** * The bounds of the current page (the common bounds of all of the shapes on the page). * @@ -4111,7 +4094,6 @@ export class Editor extends EventEmitter { if (filter) return filter(shape) return true }) - for (let i = shapesToCheck.length - 1; i >= 0; i--) { const shape = shapesToCheck[i] const geometry = this.getShapeGeometry(shape) @@ -4400,10 +4382,8 @@ export class Editor extends EventEmitter { * @public */ @computed getCurrentPageRenderingShapesSorted(): TLShape[] { - return this.getRenderingShapes() - .filter(({ isCulled }) => !isCulled) - .sort((a, b) => a.index - b.index) - .map(({ shape }) => shape) + const culledShapes = this.getCulledShapes() + return this.getCurrentPageShapesSorted().filter(({ id }) => !culledShapes.has(id)) } /** @@ -7987,6 +7967,9 @@ export class Editor extends EventEmitter { /** @internal */ private _selectedShapeIdsAtPointerDown: TLShapeId[] = [] + /** @internal */ + private _longPressTimeout = -1 as any + /** @internal */ capturedPointerId: number | null = null @@ -8004,7 +7987,13 @@ export class Editor extends EventEmitter { */ dispatch = (info: TLEventInfo): this => { this._pendingEventsForNextTick.push(info) - if (!(info.type === 'pointer' || info.type === 'wheel' || info.type === 'pinch')) { + if ( + !( + (info.type === 'pointer' && info.name === 'pointer_move') || + info.type === 'wheel' || + info.type === 'pinch' + ) + ) { this._flushEventsForTick(0) } return this @@ -8023,8 +8012,8 @@ export class Editor extends EventEmitter { } if (elapsed > 0) { this.root.handleEvent({ type: 'misc', name: 'tick', elapsed }) - this.scribbles.tick(elapsed) } + this.scribbles.tick(elapsed) }) } @@ -8084,6 +8073,7 @@ export class Editor extends EventEmitter { switch (type) { case 'pinch': { if (!this.getInstanceState().canMoveCamera) return + clearTimeout(this._longPressTimeout) this._updateInputsFromEvent(info) switch (info.name) { @@ -8204,10 +8194,11 @@ export class Editor extends EventEmitter { if ( !inputs.isDragging && inputs.isPointing && - originPagePoint.dist(currentPagePoint) > + Vec.Dist2(originPagePoint, currentPagePoint) > (this.getInstanceState().isCoarsePointer ? COARSE_DRAG_DISTANCE : DRAG_DISTANCE) / this.getZoomLevel() ) { + clearTimeout(this._longPressTimeout) inputs.isDragging = true } } @@ -8225,6 +8216,10 @@ export class Editor extends EventEmitter { case 'pointer_down': { this.clearOpenMenus() + this._longPressTimeout = setTimeout(() => { + this.dispatch({ ...info, name: 'long_press' }) + }, LONG_PRESS_DURATION) + this._selectedShapeIdsAtPointerDown = this.getSelectedShapeIds() // Firefox bug fix... @@ -8289,10 +8284,11 @@ export class Editor extends EventEmitter { if ( !inputs.isDragging && inputs.isPointing && - originPagePoint.dist(currentPagePoint) > + Vec.Dist2(originPagePoint, currentPagePoint) > (this.getInstanceState().isCoarsePointer ? COARSE_DRAG_DISTANCE : DRAG_DISTANCE) / this.getZoomLevel() ) { + clearTimeout(this._longPressTimeout) inputs.isDragging = true } break @@ -8435,6 +8431,8 @@ export class Editor extends EventEmitter { break } case 'pointer_up': { + clearTimeout(this._longPressTimeout) + const otherEvent = this._clickManager.transformPointerUpEvent(info) if (info.name !== otherEvent.name) { this.root.handleEvent(info) diff --git a/packages/editor/src/lib/editor/derivations/notVisibleShapes.ts b/packages/editor/src/lib/editor/derivations/notVisibleShapes.ts new file mode 100644 index 000000000..461835500 --- /dev/null +++ b/packages/editor/src/lib/editor/derivations/notVisibleShapes.ts @@ -0,0 +1,105 @@ +import { RESET_VALUE, computed, isUninitialized } from '@tldraw/state' +import { TLPageId, TLShapeId, isShape, isShapeId } from '@tldraw/tlschema' +import { Box } from '../../primitives/Box' +import { Editor } from '../Editor' + +function isShapeNotVisible(editor: Editor, id: TLShapeId, viewportPageBounds: Box): boolean { + const maskedPageBounds = editor.getShapeMaskedPageBounds(id) + // if the shape is fully outside of its parent's clipping bounds... + if (maskedPageBounds === undefined) return true + + // if the shape is fully outside of the viewport page bounds... + return !viewportPageBounds.includes(maskedPageBounds) +} + +/** + * Incremental derivation of not visible shapes. + * Non visible shapes are shapes outside of the viewport page bounds and shapes outside of parent's clipping bounds. + * + * @param editor - Instance of the tldraw Editor. + * @returns Incremental derivation of non visible shapes. + */ +export const notVisibleShapes = (editor: Editor) => { + const isCullingOffScreenShapes = Number.isFinite(editor.renderingBoundsMargin) + const shapeHistory = editor.store.query.filterHistory('shape') + let lastPageId: TLPageId | null = null + let prevViewportPageBounds: Box + + function fromScratch(editor: Editor): Set { + const shapes = editor.getCurrentPageShapeIds() + lastPageId = editor.getCurrentPageId() + const viewportPageBounds = editor.getViewportPageBounds() + prevViewportPageBounds = viewportPageBounds.clone() + const notVisibleShapes = new Set() + shapes.forEach((id) => { + if (isShapeNotVisible(editor, id, viewportPageBounds)) { + notVisibleShapes.add(id) + } + }) + return notVisibleShapes + } + return computed>('getCulledShapes', (prevValue, lastComputedEpoch) => { + if (!isCullingOffScreenShapes) return new Set() + + if (isUninitialized(prevValue)) { + return fromScratch(editor) + } + const diff = shapeHistory.getDiffSince(lastComputedEpoch) + + if (diff === RESET_VALUE) { + return fromScratch(editor) + } + + const currentPageId = editor.getCurrentPageId() + if (lastPageId !== currentPageId) { + return fromScratch(editor) + } + const viewportPageBounds = editor.getViewportPageBounds() + if (!prevViewportPageBounds || !viewportPageBounds.equals(prevViewportPageBounds)) { + return fromScratch(editor) + } + + let nextValue = null as null | Set + const addId = (id: TLShapeId) => { + // Already added + if (prevValue.has(id)) return + if (!nextValue) nextValue = new Set(prevValue) + nextValue.add(id) + } + const deleteId = (id: TLShapeId) => { + // No need to delete since it's not there + if (!prevValue.has(id)) return + if (!nextValue) nextValue = new Set(prevValue) + nextValue.delete(id) + } + + for (const changes of diff) { + for (const record of Object.values(changes.added)) { + if (isShape(record)) { + const isCulled = isShapeNotVisible(editor, record.id, viewportPageBounds) + if (isCulled) { + addId(record.id) + } + } + } + + for (const [_from, to] of Object.values(changes.updated)) { + if (isShape(to)) { + const isCulled = isShapeNotVisible(editor, to.id, viewportPageBounds) + if (isCulled) { + addId(to.id) + } else { + deleteId(to.id) + } + } + } + for (const id of Object.keys(changes.removed)) { + if (isShapeId(id)) { + deleteId(id) + } + } + } + + return nextValue ?? prevValue + }) +} diff --git a/packages/editor/src/lib/editor/getSvgJsx.tsx b/packages/editor/src/lib/editor/getSvgJsx.tsx index 34965135c..1f16b0ce1 100644 --- a/packages/editor/src/lib/editor/getSvgJsx.tsx +++ b/packages/editor/src/lib/editor/getSvgJsx.tsx @@ -38,7 +38,8 @@ export async function getSvgJsx( if (opts.bounds) { bbox = opts.bounds } else { - for (const { maskedPageBounds } of renderingShapes) { + for (const { id } of renderingShapes) { + const maskedPageBounds = editor.getShapeMaskedPageBounds(id) if (!maskedPageBounds) continue if (bbox) { bbox.union(maskedPageBounds) diff --git a/packages/editor/src/lib/editor/managers/ClickManager.ts b/packages/editor/src/lib/editor/managers/ClickManager.ts index ef2c11111..ef5e53b10 100644 --- a/packages/editor/src/lib/editor/managers/ClickManager.ts +++ b/packages/editor/src/lib/editor/managers/ClickManager.ts @@ -227,7 +227,7 @@ export class ClickManager { if ( this._clickState !== 'idle' && this._clickScreenPoint && - this._clickScreenPoint.dist(this.editor.inputs.currentScreenPoint) > + Vec.Dist2(this._clickScreenPoint, this.editor.inputs.currentScreenPoint) > (this.editor.getInstanceState().isCoarsePointer ? COARSE_DRAG_DISTANCE : DRAG_DISTANCE) ) { this.cancelDoubleClickTimeout() diff --git a/packages/editor/src/lib/editor/shapes/ShapeUtil.ts b/packages/editor/src/lib/editor/shapes/ShapeUtil.ts index d2c3c9553..fcdd38ce4 100644 --- a/packages/editor/src/lib/editor/shapes/ShapeUtil.ts +++ b/packages/editor/src/lib/editor/shapes/ShapeUtil.ts @@ -89,13 +89,6 @@ export abstract class ShapeUtil { */ canScroll: TLShapeUtilFlag = () => false - /** - * Whether the shape should unmount when not visible in the editor. Consider keeping this to false if the shape's `component` has local state. - * - * @public - */ - canUnmount: TLShapeUtilFlag = () => true - /** * Whether the shape can be bound to by an arrow. * diff --git a/packages/editor/src/lib/editor/tools/StateNode.ts b/packages/editor/src/lib/editor/tools/StateNode.ts index 5c5378b63..f170fe90e 100644 --- a/packages/editor/src/lib/editor/tools/StateNode.ts +++ b/packages/editor/src/lib/editor/tools/StateNode.ts @@ -198,6 +198,7 @@ export abstract class StateNode implements Partial { onWheel?: TLEventHandlers['onWheel'] onPointerDown?: TLEventHandlers['onPointerDown'] onPointerMove?: TLEventHandlers['onPointerMove'] + onLongPress?: TLEventHandlers['onLongPress'] onPointerUp?: TLEventHandlers['onPointerUp'] onDoubleClick?: TLEventHandlers['onDoubleClick'] onTripleClick?: TLEventHandlers['onTripleClick'] diff --git a/packages/editor/src/lib/editor/types/event-types.ts b/packages/editor/src/lib/editor/types/event-types.ts index 5fa41deb8..89ab5725e 100644 --- a/packages/editor/src/lib/editor/types/event-types.ts +++ b/packages/editor/src/lib/editor/types/event-types.ts @@ -16,6 +16,7 @@ export type TLPointerEventTarget = export type TLPointerEventName = | 'pointer_down' | 'pointer_move' + | 'long_press' | 'pointer_up' | 'right_click' | 'middle_click' @@ -152,6 +153,7 @@ export type TLExitEventHandler = (info: any, to: string) => void export interface TLEventHandlers { onPointerDown: TLPointerEvent onPointerMove: TLPointerEvent + onLongPress: TLPointerEvent onRightClick: TLPointerEvent onDoubleClick: TLClickEvent onTripleClick: TLClickEvent @@ -176,6 +178,7 @@ export const EVENT_NAME_MAP: Record< wheel: 'onWheel', pointer_down: 'onPointerDown', pointer_move: 'onPointerMove', + long_press: 'onLongPress', pointer_up: 'onPointerUp', right_click: 'onRightClick', middle_click: 'onMiddleClick', diff --git a/packages/editor/src/lib/primitives/Vec.ts b/packages/editor/src/lib/primitives/Vec.ts index 82bf47395..e0ee4b717 100644 --- a/packages/editor/src/lib/primitives/Vec.ts +++ b/packages/editor/src/lib/primitives/Vec.ts @@ -308,19 +308,20 @@ export class Vec { static Per(A: VecLike): Vec { return new Vec(A.y, -A.x) } - - static Dist2(A: VecLike, B: VecLike): number { - return Vec.Sub(A, B).len2() - } - static Abs(A: VecLike): Vec { return new Vec(Math.abs(A.x), Math.abs(A.y)) } + // Get the distance between two points. static Dist(A: VecLike, B: VecLike): number { return Math.hypot(A.y - B.y, A.x - B.x) } + // Get the squared distance between two points. This is faster to calculate (no square root) so useful for "minimum distance" checks where the actual measurement does not matter. + static Dist2(A: VecLike, B: VecLike): number { + return (A.x - B.x) * (A.x - B.x) + (A.y - B.y) * (A.y - B.y) + } + /** * Dot product of two vectors which is used to calculate the angle between them. */ diff --git a/packages/editor/src/lib/primitives/geometry/Arc2d.ts b/packages/editor/src/lib/primitives/geometry/Arc2d.ts index 5feee17ef..3ca170a47 100644 --- a/packages/editor/src/lib/primitives/geometry/Arc2d.ts +++ b/packages/editor/src/lib/primitives/geometry/Arc2d.ts @@ -52,15 +52,16 @@ export class Arc2d extends Geometry2d { // Get the point (P) on the arc, then pick the nearest of A, B, and P const P = _center.clone().add(point.clone().sub(_center).uni().mul(radius)) - let distance = Infinity let nearest: Vec | undefined - for (const pt of [A, B, P]) { - if (point.dist(pt) < distance) { - nearest = pt - distance = point.dist(pt) + let dist = Infinity + let d: number + for (const p of [A, B, P]) { + d = Vec.Dist2(point, p) + if (d < dist) { + nearest = p + dist = d } } - if (!nearest) throw Error('nearest point not found') return nearest } diff --git a/packages/editor/src/lib/primitives/geometry/CubicBezier2d.ts b/packages/editor/src/lib/primitives/geometry/CubicBezier2d.ts index 9923891b7..912144378 100644 --- a/packages/editor/src/lib/primitives/geometry/CubicBezier2d.ts +++ b/packages/editor/src/lib/primitives/geometry/CubicBezier2d.ts @@ -55,9 +55,11 @@ export class CubicBezier2d extends Polyline2d { nearestPoint(A: Vec): Vec { let nearest: Vec | undefined let dist = Infinity + let d: number + let p: Vec for (const edge of this.segments) { - const p = edge.nearestPoint(A) - const d = p.dist(A) + p = edge.nearestPoint(A) + d = Vec.Dist2(p, A) if (d < dist) { nearest = p dist = d diff --git a/packages/editor/src/lib/primitives/geometry/CubicSpline2d.ts b/packages/editor/src/lib/primitives/geometry/CubicSpline2d.ts index 3c362b91b..8ff99bb0a 100644 --- a/packages/editor/src/lib/primitives/geometry/CubicSpline2d.ts +++ b/packages/editor/src/lib/primitives/geometry/CubicSpline2d.ts @@ -67,15 +67,16 @@ export class CubicSpline2d extends Geometry2d { nearestPoint(A: Vec): Vec { let nearest: Vec | undefined let dist = Infinity + let d: number + let p: Vec for (const segment of this.segments) { - const p = segment.nearestPoint(A) - const d = p.dist(A) + p = segment.nearestPoint(A) + d = Vec.Dist2(p, A) if (d < dist) { nearest = p dist = d } } - if (!nearest) throw Error('nearest point not found') return nearest } diff --git a/packages/editor/src/lib/primitives/geometry/Ellipse2d.ts b/packages/editor/src/lib/primitives/geometry/Ellipse2d.ts index d15b864af..a2340c298 100644 --- a/packages/editor/src/lib/primitives/geometry/Ellipse2d.ts +++ b/packages/editor/src/lib/primitives/geometry/Ellipse2d.ts @@ -76,15 +76,16 @@ export class Ellipse2d extends Geometry2d { nearestPoint(A: Vec): Vec { let nearest: Vec | undefined let dist = Infinity + let d: number + let p: Vec for (const edge of this.edges) { - const p = edge.nearestPoint(A) - const d = p.dist(A) + p = edge.nearestPoint(A) + d = Vec.Dist2(p, A) if (d < dist) { nearest = p dist = d } } - if (!nearest) throw Error('nearest point not found') return nearest } diff --git a/packages/editor/src/lib/primitives/geometry/Geometry2d.ts b/packages/editor/src/lib/primitives/geometry/Geometry2d.ts index e028de31e..b7df7a84d 100644 --- a/packages/editor/src/lib/primitives/geometry/Geometry2d.ts +++ b/packages/editor/src/lib/primitives/geometry/Geometry2d.ts @@ -55,14 +55,17 @@ export abstract class Geometry2d { } nearestPointOnLineSegment(A: Vec, B: Vec): Vec { - let distance = Infinity + const { vertices } = this let nearest: Vec | undefined - for (let i = 0; i < this.vertices.length; i++) { - const point = this.vertices[i] - const d = Vec.DistanceToLineSegment(A, B, point) - if (d < distance) { - distance = d - nearest = point + let dist = Infinity + let d: number + let p: Vec + for (let i = 0; i < vertices.length; i++) { + p = vertices[i] + d = Vec.DistanceToLineSegment(A, B, p) + if (d < dist) { + dist = d + nearest = p } } if (!nearest) throw Error('nearest point not found') diff --git a/packages/editor/src/lib/primitives/geometry/Group2d.ts b/packages/editor/src/lib/primitives/geometry/Group2d.ts index 3caeabfc8..b98f1ea99 100644 --- a/packages/editor/src/lib/primitives/geometry/Group2d.ts +++ b/packages/editor/src/lib/primitives/geometry/Group2d.ts @@ -30,8 +30,8 @@ export class Group2d extends Geometry2d { } override nearestPoint(point: Vec): Vec { - let d = Infinity - let p: Vec | undefined + let dist = Infinity + let nearest: Vec | undefined const { children } = this @@ -39,16 +39,18 @@ export class Group2d extends Geometry2d { throw Error('no children') } + let p: Vec + let d: number for (const child of children) { - const nearest = child.nearestPoint(point) - const dist = nearest.dist(point) - if (dist < d) { - d = dist - p = nearest + p = child.nearestPoint(point) + d = Vec.Dist2(p, point) + if (d < dist) { + dist = d + nearest = p } } - if (!p) throw Error('nearest point not found') - return p + if (!nearest) throw Error('nearest point not found') + return nearest } override distanceToPoint(point: Vec, hitInside = false) { diff --git a/packages/editor/src/lib/primitives/geometry/Polyline2d.ts b/packages/editor/src/lib/primitives/geometry/Polyline2d.ts index ffa541cb2..84c8e7471 100644 --- a/packages/editor/src/lib/primitives/geometry/Polyline2d.ts +++ b/packages/editor/src/lib/primitives/geometry/Polyline2d.ts @@ -51,18 +51,17 @@ export class Polyline2d extends Geometry2d { const { segments } = this let nearest = this.points[0] let dist = Infinity - let p: Vec // current point on segment let d: number // distance from A to p for (let i = 0; i < segments.length; i++) { p = segments[i].nearestPoint(A) - d = p.dist(A) + d = Vec.Dist2(p, A) if (d < dist) { nearest = p dist = d } } - + if (!nearest) throw Error('nearest point not found') return nearest } diff --git a/packages/store/api/api.json b/packages/store/api/api.json index 4b35dd4ba..668ba8f04 100644 --- a/packages/store/api/api.json +++ b/packages/store/api/api.json @@ -4201,7 +4201,7 @@ { "kind": "Property", "canonicalReference": "@tldraw/store!Store#onBeforeChange:member", - "docComment": "/**\n * A callback before after each record's change.\n *\n * @param prev - The previous value, if any.\n *\n * @param next - The next value.\n */\n", + "docComment": "/**\n * A callback fired before each record's change.\n *\n * @param prev - The previous value, if any.\n *\n * @param next - The next value.\n */\n", "excerptTokens": [ { "kind": "Content", diff --git a/packages/store/src/lib/Store.ts b/packages/store/src/lib/Store.ts index 968996247..ceb24785c 100644 --- a/packages/store/src/lib/Store.ts +++ b/packages/store/src/lib/Store.ts @@ -311,7 +311,7 @@ export class Store { onAfterCreate?: (record: R, source: 'remote' | 'user') => void /** - * A callback before after each record's change. + * A callback fired before each record's change. * * @param prev - The previous value, if any. * @param next - The next value. diff --git a/packages/store/src/lib/devFreeze.ts b/packages/store/src/lib/devFreeze.ts index d1c9622d4..d481ad37e 100644 --- a/packages/store/src/lib/devFreeze.ts +++ b/packages/store/src/lib/devFreeze.ts @@ -15,6 +15,8 @@ import { STRUCTURED_CLONE_OBJECT_PROTOTYPE } from '@tldraw/utils' * @public */ export function devFreeze(object: T): T { + return object + if (process.env.NODE_ENV === 'production') { return object } diff --git a/packages/tldraw/api-report.md b/packages/tldraw/api-report.md index 608ceb74d..2a861e466 100644 --- a/packages/tldraw/api-report.md +++ b/packages/tldraw/api-report.md @@ -536,8 +536,6 @@ export class EmbedShapeUtil extends BaseBoxShapeUtil { // (undocumented) canResize: (shape: TLEmbedShape) => boolean; // (undocumented) - canUnmount: TLShapeUtilFlag; - // (undocumented) component(shape: TLEmbedShape): JSX_2.Element; // (undocumented) getDefaultProps(): TLEmbedShape['props']; diff --git a/packages/tldraw/api/api.json b/packages/tldraw/api/api.json index 24a04a7c8..4d0d710d6 100644 --- a/packages/tldraw/api/api.json +++ b/packages/tldraw/api/api.json @@ -5705,50 +5705,6 @@ "isProtected": false, "isAbstract": false }, - { - "kind": "Property", - "canonicalReference": "tldraw!EmbedShapeUtil#canUnmount:member", - "docComment": "", - "excerptTokens": [ - { - "kind": "Content", - "text": "canUnmount: " - }, - { - "kind": "Reference", - "text": "TLShapeUtilFlag", - "canonicalReference": "@tldraw/editor!TLShapeUtilFlag:type" - }, - { - "kind": "Content", - "text": "<" - }, - { - "kind": "Reference", - "text": "TLEmbedShape", - "canonicalReference": "@tldraw/tlschema!TLEmbedShape:type" - }, - { - "kind": "Content", - "text": ">" - }, - { - "kind": "Content", - "text": ";" - } - ], - "isReadonly": false, - "isOptional": false, - "releaseTag": "Public", - "name": "canUnmount", - "propertyTypeTokenRange": { - "startIndex": 1, - "endIndex": 5 - }, - "isStatic": false, - "isProtected": false, - "isAbstract": false - }, { "kind": "Method", "canonicalReference": "tldraw!EmbedShapeUtil#component:member(1)", diff --git a/packages/tldraw/src/lib/shapes/draw/toolStates/Drawing.ts b/packages/tldraw/src/lib/shapes/draw/toolStates/Drawing.ts index 049223951..804144248 100644 --- a/packages/tldraw/src/lib/shapes/draw/toolStates/Drawing.ts +++ b/packages/tldraw/src/lib/shapes/draw/toolStates/Drawing.ts @@ -309,7 +309,7 @@ export class Drawing extends StateNode { } const hasMovedFarEnough = - Vec.Dist(pagePointWhereNextSegmentChanged, inputs.currentPagePoint) > DRAG_DISTANCE + Vec.Dist2(pagePointWhereNextSegmentChanged, inputs.currentPagePoint) > DRAG_DISTANCE // Find the distance from where the pointer was when shift was released and // where it is now; if it's far enough away, then update the page point where @@ -380,7 +380,7 @@ export class Drawing extends StateNode { } const hasMovedFarEnough = - Vec.Dist(pagePointWhereNextSegmentChanged, inputs.currentPagePoint) > DRAG_DISTANCE + Vec.Dist2(pagePointWhereNextSegmentChanged, inputs.currentPagePoint) > DRAG_DISTANCE // Find the distance from where the pointer was when shift was released and // where it is now; if it's far enough away, then update the page point where diff --git a/packages/tldraw/src/lib/shapes/embed/EmbedShapeUtil.tsx b/packages/tldraw/src/lib/shapes/embed/EmbedShapeUtil.tsx index 80f11a83e..470fb3e0f 100644 --- a/packages/tldraw/src/lib/shapes/embed/EmbedShapeUtil.tsx +++ b/packages/tldraw/src/lib/shapes/embed/EmbedShapeUtil.tsx @@ -34,9 +34,6 @@ export class EmbedShapeUtil extends BaseBoxShapeUtil { override hideSelectionBoundsFg: TLShapeUtilFlag = (shape) => !this.canResize(shape) override canEdit: TLShapeUtilFlag = () => true - override canUnmount: TLShapeUtilFlag = (shape: TLEmbedShape) => { - return !!getEmbedInfo(shape.props.url)?.definition?.canUnmount - } override canResize = (shape: TLEmbedShape) => { return !!getEmbedInfo(shape.props.url)?.definition?.doesResize } diff --git a/packages/tldraw/src/lib/shapes/shared/defaultStyleDefs.tsx b/packages/tldraw/src/lib/shapes/shared/defaultStyleDefs.tsx index 8a6ec0787..16e4495be 100644 --- a/packages/tldraw/src/lib/shapes/shared/defaultStyleDefs.tsx +++ b/packages/tldraw/src/lib/shapes/shared/defaultStyleDefs.tsx @@ -179,6 +179,11 @@ function usePattern() { const [backgroundUrls, setBackgroundUrls] = useState(defaultPatterns) useEffect(() => { + if (process.env.NODE_ENV === 'test') { + setIsReady(true) + return + } + const promises: Promise<{ zoom: number; url: string; darkMode: boolean }>[] = [] for (let i = 1; i <= Math.ceil(MAX_ZOOM); i++) { diff --git a/packages/tldraw/src/lib/tools/EraserTool/childStates/Pointing.ts b/packages/tldraw/src/lib/tools/EraserTool/childStates/Pointing.ts index 8d3da33aa..bf46a8c01 100644 --- a/packages/tldraw/src/lib/tools/EraserTool/childStates/Pointing.ts +++ b/packages/tldraw/src/lib/tools/EraserTool/childStates/Pointing.ts @@ -4,6 +4,7 @@ import { TLEventHandlers, TLFrameShape, TLGroupShape, + TLPointerEventInfo, TLShapeId, } from '@tldraw/editor' @@ -52,9 +53,13 @@ export class Pointing extends StateNode { this.editor.setErasingShapes([...erasing]) } + override onLongPress: TLEventHandlers['onLongPress'] = (info) => { + this.startErasing(info) + } + override onPointerMove: TLEventHandlers['onPointerMove'] = (info) => { if (this.editor.inputs.isDragging) { - this.parent.transition('erasing', info) + this.startErasing(info) } } @@ -74,6 +79,10 @@ export class Pointing extends StateNode { this.cancel() } + private startErasing(info: TLPointerEventInfo) { + this.parent.transition('erasing', info) + } + complete() { const erasingShapeIds = this.editor.getErasingShapeIds() diff --git a/packages/tldraw/src/lib/tools/HandTool/childStates/Dragging.ts b/packages/tldraw/src/lib/tools/HandTool/childStates/Dragging.ts index 8d6e60553..a0138ef0e 100644 --- a/packages/tldraw/src/lib/tools/HandTool/childStates/Dragging.ts +++ b/packages/tldraw/src/lib/tools/HandTool/childStates/Dragging.ts @@ -3,7 +3,10 @@ import { CAMERA_SLIDE_FRICTION, StateNode, TLEventHandlers, Vec } from '@tldraw/ export class Dragging extends StateNode { static override id = 'dragging' + initialCamera = new Vec() + override onEnter = () => { + this.initialCamera = Vec.From(this.editor.getCamera()) this.update() } @@ -16,7 +19,7 @@ export class Dragging extends StateNode { } override onCancel: TLEventHandlers['onCancel'] = () => { - this.complete() + this.parent.transition('idle') } override onComplete = () => { @@ -24,21 +27,27 @@ export class Dragging extends StateNode { } private update() { - const { currentScreenPoint, previousScreenPoint } = this.editor.inputs + const { initialCamera, editor } = this + const { currentScreenPoint, originScreenPoint } = editor.inputs - const delta = Vec.Sub(currentScreenPoint, previousScreenPoint) - - if (Math.abs(delta.x) > 0 || Math.abs(delta.y) > 0) { - this.editor.pan(delta) - } + const delta = Vec.Sub(currentScreenPoint, originScreenPoint).div(editor.getZoomLevel()) + if (delta.len2() === 0) return + editor.setCamera(initialCamera.clone().add(delta)) } private complete() { - this.editor.slideCamera({ - speed: Math.min(2, this.editor.inputs.pointerVelocity.len()), - direction: this.editor.inputs.pointerVelocity, - friction: CAMERA_SLIDE_FRICTION, - }) + const { editor } = this + const { pointerVelocity } = editor.inputs + + const velocityAtPointerUp = Math.min(pointerVelocity.len(), 2) + + if (velocityAtPointerUp > 0.1) { + this.editor.slideCamera({ + speed: velocityAtPointerUp, + direction: pointerVelocity, + friction: CAMERA_SLIDE_FRICTION, + }) + } this.parent.transition('idle') } diff --git a/packages/tldraw/src/lib/tools/HandTool/childStates/Pointing.ts b/packages/tldraw/src/lib/tools/HandTool/childStates/Pointing.ts index e28f74462..e5746c578 100644 --- a/packages/tldraw/src/lib/tools/HandTool/childStates/Pointing.ts +++ b/packages/tldraw/src/lib/tools/HandTool/childStates/Pointing.ts @@ -8,12 +8,20 @@ export class Pointing extends StateNode { this.editor.setCursor({ type: 'grabbing', rotation: 0 }) } - override onPointerMove: TLEventHandlers['onPointerMove'] = (info) => { + override onLongPress: TLEventHandlers['onLongPress'] = () => { + this.startDragging() + } + + override onPointerMove: TLEventHandlers['onPointerMove'] = () => { if (this.editor.inputs.isDragging) { - this.parent.transition('dragging', info) + this.startDragging() } } + private startDragging() { + this.parent.transition('dragging') + } + override onPointerUp: TLEventHandlers['onPointerUp'] = () => { this.complete() } diff --git a/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingCropHandle.ts b/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingCropHandle.ts index 5e8414ebc..603aaa471 100644 --- a/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingCropHandle.ts +++ b/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingCropHandle.ts @@ -29,16 +29,23 @@ export class PointingCropHandle extends StateNode { } override onPointerMove: TLEventHandlers['onPointerMove'] = () => { - const isDragging = this.editor.inputs.isDragging - - if (isDragging) { - this.parent.transition('cropping', { - ...this.info, - onInteractionEnd: this.info.onInteractionEnd, - }) + if (this.editor.inputs.isDragging) { + this.startCropping() } } + override onLongPress: TLEventHandlers['onLongPress'] = () => { + this.startCropping() + } + + private startCropping() { + if (this.editor.getInstanceState().isReadonly) return + this.parent.transition('cropping', { + ...this.info, + onInteractionEnd: this.info.onInteractionEnd, + }) + } + override onPointerUp: TLEventHandlers['onPointerUp'] = () => { if (this.info.onInteractionEnd) { this.editor.setCurrentTool(this.info.onInteractionEnd, this.info) diff --git a/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingHandle.ts b/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingHandle.ts index 910280191..5ba8b0579 100644 --- a/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingHandle.ts +++ b/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingHandle.ts @@ -31,10 +31,19 @@ export class PointingHandle extends StateNode { override onPointerMove: TLEventHandlers['onPointerMove'] = () => { if (this.editor.inputs.isDragging) { - this.parent.transition('dragging_handle', this.info) + this.startDraggingHandle() } } + override onLongPress: TLEventHandlers['onLongPress'] = () => { + this.startDraggingHandle() + } + + private startDraggingHandle() { + if (this.editor.getInstanceState().isReadonly) return + this.parent.transition('dragging_handle', this.info) + } + override onCancel: TLEventHandlers['onCancel'] = () => { this.cancel() } diff --git a/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingResizeHandle.ts b/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingResizeHandle.ts index 8633629c6..28e5ee041 100644 --- a/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingResizeHandle.ts +++ b/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingResizeHandle.ts @@ -46,13 +46,20 @@ export class PointingResizeHandle extends StateNode { } override onPointerMove: TLEventHandlers['onPointerMove'] = () => { - const isDragging = this.editor.inputs.isDragging - - if (isDragging) { - this.parent.transition('resizing', this.info) + if (this.editor.inputs.isDragging) { + this.startResizing() } } + override onLongPress: TLEventHandlers['onLongPress'] = () => { + this.startResizing() + } + + private startResizing() { + if (this.editor.getInstanceState().isReadonly) return + this.parent.transition('resizing', this.info) + } + override onPointerUp: TLEventHandlers['onPointerUp'] = () => { this.complete() } diff --git a/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingRotateHandle.ts b/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingRotateHandle.ts index fbd95219d..d5d9388c1 100644 --- a/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingRotateHandle.ts +++ b/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingRotateHandle.ts @@ -28,14 +28,21 @@ export class PointingRotateHandle extends StateNode { this.editor.setCursor({ type: 'default', rotation: 0 }) } - override onPointerMove = () => { - const { isDragging } = this.editor.inputs - - if (isDragging) { - this.parent.transition('rotating', this.info) + override onPointerMove: TLEventHandlers['onPointerMove'] = () => { + if (this.editor.inputs.isDragging) { + this.startRotating() } } + override onLongPress: TLEventHandlers['onLongPress'] = () => { + this.startRotating() + } + + private startRotating() { + if (this.editor.getInstanceState().isReadonly) return + this.parent.transition('rotating', this.info) + } + override onPointerUp = () => { this.complete() } diff --git a/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingSelection.ts b/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingSelection.ts index 14ce18f57..8ac57a1e1 100644 --- a/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingSelection.ts +++ b/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingSelection.ts @@ -25,11 +25,19 @@ export class PointingSelection extends StateNode { override onPointerMove: TLEventHandlers['onPointerMove'] = (info) => { if (this.editor.inputs.isDragging) { - if (this.editor.getInstanceState().isReadonly) return - this.parent.transition('translating', info) + this.startTranslating(info) } } + override onLongPress: TLEventHandlers['onLongPress'] = (info) => { + this.startTranslating(info) + } + + private startTranslating(info: TLPointerEventInfo) { + if (this.editor.getInstanceState().isReadonly) return + this.parent.transition('translating', info) + } + override onDoubleClick?: TLClickEvent | undefined = (info) => { const hoveredShape = this.editor.getHoveredShape() const hitShape = diff --git a/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingShape.ts b/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingShape.ts index 4dad8d5e4..7a08ee826 100644 --- a/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingShape.ts +++ b/packages/tldraw/src/lib/tools/SelectTool/childStates/PointingShape.ts @@ -195,11 +195,19 @@ export class PointingShape extends StateNode { override onPointerMove: TLEventHandlers['onPointerMove'] = (info) => { if (this.editor.inputs.isDragging) { - if (this.editor.getInstanceState().isReadonly) return - this.parent.transition('translating', info) + this.startTranslating(info) } } + override onLongPress: TLEventHandlers['onLongPress'] = (info) => { + this.startTranslating(info) + } + + private startTranslating(info: TLPointerEventInfo) { + if (this.editor.getInstanceState().isReadonly) return + this.parent.transition('translating', info) + } + override onCancel: TLEventHandlers['onCancel'] = () => { this.cancel() } diff --git a/packages/tldraw/src/lib/tools/SelectTool/childStates/ScribbleBrushing.ts b/packages/tldraw/src/lib/tools/SelectTool/childStates/ScribbleBrushing.ts index 477dfe59a..34b9c585e 100644 --- a/packages/tldraw/src/lib/tools/SelectTool/childStates/ScribbleBrushing.ts +++ b/packages/tldraw/src/lib/tools/SelectTool/childStates/ScribbleBrushing.ts @@ -42,9 +42,7 @@ export class ScribbleBrushing extends StateNode { this.updateScribbleSelection(true) - requestAnimationFrame(() => { - this.editor.updateInstanceState({ brush: null }) - }) + this.editor.updateInstanceState({ brush: null }) } override onExit = () => { diff --git a/packages/tldraw/src/lib/tools/selection-logic/updateHoveredId.ts b/packages/tldraw/src/lib/tools/selection-logic/updateHoveredId.ts index 9f0d44202..f03d333c3 100644 --- a/packages/tldraw/src/lib/tools/selection-logic/updateHoveredId.ts +++ b/packages/tldraw/src/lib/tools/selection-logic/updateHoveredId.ts @@ -1,6 +1,6 @@ -import { Editor, HIT_TEST_MARGIN, TLShape } from '@tldraw/editor' +import { Editor, HIT_TEST_MARGIN, TLShape, throttle } from '@tldraw/editor' -export function updateHoveredId(editor: Editor) { +function _updateHoveredId(editor: Editor) { // todo: consider replacing `get hoveredShapeId` with this; it would mean keeping hoveredShapeId in memory rather than in the store and possibly re-computing it more often than necessary const hitShape = editor.getShapeAtPoint(editor.inputs.currentPagePoint, { hitInside: false, @@ -30,3 +30,6 @@ export function updateHoveredId(editor: Editor) { return editor.setHoveredShape(shapeToHover.id) } + +export const updateHoveredId = + process.env.NODE_ENV === 'test' ? _updateHoveredId : throttle(_updateHoveredId, 32) diff --git a/packages/tldraw/src/lib/ui/components/HelperButtons/BackToContent.tsx b/packages/tldraw/src/lib/ui/components/HelperButtons/BackToContent.tsx index f2968a86f..004dedfdf 100644 --- a/packages/tldraw/src/lib/ui/components/HelperButtons/BackToContent.tsx +++ b/packages/tldraw/src/lib/ui/components/HelperButtons/BackToContent.tsx @@ -17,13 +17,12 @@ export function BackToContent() { const renderingShapes = editor.getRenderingShapes() const renderingBounds = editor.getRenderingBounds() - // renderingShapes will also include shapes that have the canUnmount flag - // set to true. These shapes will be on the canvas but may not be in the - // viewport... so we also need to narrow down the list to only shapes that - // are ALSO in the viewport. - const visibleShapes = renderingShapes.filter( - (s) => s.maskedPageBounds && renderingBounds.includes(s.maskedPageBounds) - ) + // Rendering shapes includes all the shapes in the current page. + // We have to filter them down to just the shapes that are inside the renderingBounds. + const visibleShapes = renderingShapes.filter((s) => { + const maskedPageBounds = editor.getShapeMaskedPageBounds(s.id) + return maskedPageBounds && renderingBounds.includes(maskedPageBounds) + }) const showBackToContentNow = visibleShapes.length === 0 && editor.getCurrentPageShapes().length > 0 diff --git a/packages/tldraw/src/lib/ui/components/Minimap/MinimapManager.ts b/packages/tldraw/src/lib/ui/components/Minimap/MinimapManager.ts index 993030791..eeef0fd7f 100644 --- a/packages/tldraw/src/lib/ui/components/Minimap/MinimapManager.ts +++ b/packages/tldraw/src/lib/ui/components/Minimap/MinimapManager.ts @@ -189,7 +189,7 @@ export class MinimapManager { this const { width: cw, height: ch } = canvasScreenBounds - const selectedShapeIds = editor.getSelectedShapeIds() + const selectedShapeIds = new Set(editor.getSelectedShapeIds()) const viewportPageBounds = editor.getViewportPageBounds() if (!cvs || !pageBounds) { @@ -215,16 +215,6 @@ export class MinimapManager { ctx.scale(sx, sy) ctx.translate(-contentPageBounds.minX, -contentPageBounds.minY) - // Default radius for rounded rects - const rx = 8 / sx - const ry = 8 / sx - // Min radius - const ax = 1 / sx - const ay = 1 / sx - // Max radius factor - const bx = rx / 4 - const by = ry / 4 - // shapes const shapesPath = new Path2D() const selectedPath = new Path2D() @@ -237,14 +227,11 @@ export class MinimapManager { let pb: Box & { id: TLShapeId } for (let i = 0, n = pageBounds.length; i < n; i++) { pb = pageBounds[i] - MinimapManager.roundedRect( - selectedShapeIds.includes(pb.id) ? selectedPath : shapesPath, + ;(selectedShapeIds.has(pb.id) ? selectedPath : shapesPath).rect( pb.minX, pb.minY, pb.width, - pb.height, - clamp(rx, ax, pb.width / bx), - clamp(ry, ay, pb.height / by) + pb.height ) } diff --git a/packages/tldraw/src/test/getCulledShapes.test.tsx b/packages/tldraw/src/test/getCulledShapes.test.tsx new file mode 100644 index 000000000..53c99bd35 --- /dev/null +++ b/packages/tldraw/src/test/getCulledShapes.test.tsx @@ -0,0 +1,138 @@ +import { Box, TLShapeId, createShapeId } from '@tldraw/editor' +import { TestEditor } from './TestEditor' +import { TL } from './test-jsx' + +let editor: TestEditor + +beforeEach(() => { + editor = new TestEditor() + editor.setScreenBounds({ x: 0, y: 0, w: 1800, h: 900 }) + editor.renderingBoundsMargin = 100 +}) + +function createShapes() { + return editor.createShapesFromJsx([ + , + + + {/* this is outside of the frames clipping bounds, so it should never be rendered */} + + , + ]) +} + +it('lists shapes in viewport', () => { + const ids = createShapes() + editor.selectNone() + // D is clipped and so should always be culled / outside of viewport + expect(editor.getCulledShapes()).toStrictEqual(new Set([ids.D])) + + // Move the camera 201 pixels to the right and 201 pixels down + editor.pan({ x: -201, y: -201 }) + jest.advanceTimersByTime(500) + + // A is now outside of the viewport + expect(editor.getCulledShapes()).toStrictEqual(new Set([ids.A, ids.D])) + + editor.pan({ x: -900, y: -900 }) + jest.advanceTimersByTime(500) + // Now all shapes are outside of the viewport + expect(editor.getCulledShapes()).toStrictEqual(new Set([ids.A, ids.B, ids.C, ids.D])) + + editor.select(ids.B) + // We don't cull selected shapes + expect(editor.getCulledShapes()).toStrictEqual(new Set([ids.A, ids.C, ids.D])) + + editor.setEditingShape(ids.C) + // or shapes being edited + expect(editor.getCulledShapes()).toStrictEqual(new Set([ids.A, ids.D])) +}) + +const shapeSize = 100 +const numberOfShapes = 100 + +function getChangeOutsideBounds(viewportSize: number) { + const changeDirection = Math.random() > 0.5 ? 1 : -1 + const maxChange = 1000 + const changeAmount = 1 + Math.random() * maxChange + if (changeDirection === 1) { + // We need to get past the viewport size and then add a bit more + return viewportSize + changeAmount + } else { + // We also need to take the shape size into account + return -changeAmount - shapeSize + } +} + +function getChangeInsideBounds(viewportSize: number) { + // We can go from -shapeSize to viewportSize + return -shapeSize + Math.random() * (viewportSize + shapeSize) +} + +function createFuzzShape(viewport: Box) { + const id = createShapeId() + if (Math.random() > 0.5) { + const positionChange = Math.random() + // Should x, or y, or both go outside the bounds? + const dimensionChange = positionChange < 0.33 ? 'x' : positionChange < 0.66 ? 'y' : 'both' + const xOutsideBounds = dimensionChange === 'x' || dimensionChange === 'both' + const yOutsideBounds = dimensionChange === 'y' || dimensionChange === 'both' + + // Create a shape outside the viewport + editor.createShape({ + id, + type: 'geo', + x: + viewport.x + + (xOutsideBounds ? getChangeOutsideBounds(viewport.w) : getChangeInsideBounds(viewport.w)), + y: + viewport.y + + (yOutsideBounds ? getChangeOutsideBounds(viewport.h) : getChangeInsideBounds(viewport.h)), + props: { w: shapeSize, h: shapeSize }, + }) + return { isCulled: true, id } + } else { + // Create a shape inside the viewport + editor.createShape({ + id, + type: 'geo', + x: viewport.x + getChangeInsideBounds(viewport.w), + y: viewport.y + getChangeInsideBounds(viewport.h), + props: { w: shapeSize, h: shapeSize }, + }) + return { isCulled: false, id } + } +} + +it('correctly calculates the culled shapes when adding and deleting shapes', () => { + const viewport = editor.getViewportPageBounds() + const shapes: Array = [] + for (let i = 0; i < numberOfShapes; i++) { + const { isCulled, id } = createFuzzShape(viewport) + shapes.push(id) + if (isCulled) { + expect(editor.getCulledShapes()).toContain(id) + } else { + expect(editor.getCulledShapes()).not.toContain(id) + } + } + const numberOfShapesToDelete = Math.floor((Math.random() * numberOfShapes) / 2) + for (let i = 0; i < numberOfShapesToDelete; i++) { + const index = Math.floor(Math.random() * (shapes.length - 1)) + const id = shapes[index] + if (id) { + editor.deleteShape(id) + shapes[index] = undefined + expect(editor.getCulledShapes()).not.toContain(id) + } + } + + const culledShapesIncremental = editor.getCulledShapes() + + // force full refresh + editor.pan({ x: -1, y: 0 }) + editor.pan({ x: 1, y: 0 }) + + const culledShapeFromScratch = editor.getCulledShapes() + expect(culledShapesIncremental).toEqual(culledShapeFromScratch) +}) diff --git a/packages/tldraw/src/test/renderingShapes.test.tsx b/packages/tldraw/src/test/renderingShapes.test.tsx index 322dfc34f..8b774da5b 100644 --- a/packages/tldraw/src/test/renderingShapes.test.tsx +++ b/packages/tldraw/src/test/renderingShapes.test.tsx @@ -60,47 +60,6 @@ it('updates the rendering viewport when the camera stops moving', () => { expect(editor.getShapePageBounds(ids.A)).toMatchObject({ x: 100, y: 100, w: 100, h: 100 }) }) -it('lists shapes in viewport', () => { - const ids = createShapes() - editor.selectNone() - expect(editor.getRenderingShapes().map(({ id, isCulled }) => [id, isCulled])).toStrictEqual([ - [ids.A, false], // A is within the expanded rendering bounds, so should not be culled; and it's in the regular viewport too, so it's on screen. - [ids.B, false], - [ids.C, false], - [ids.D, true], // D is clipped and so should always be culled / outside of viewport - ]) - - // Move the camera 201 pixels to the right and 201 pixels down - editor.pan({ x: -201, y: -201 }) - jest.advanceTimersByTime(500) - - expect(editor.getRenderingShapes().map(({ id, isCulled }) => [id, isCulled])).toStrictEqual([ - [ids.A, false], // A should not be culled, even though it's no longer in the viewport (because it's still in the EXPANDED viewport) - [ids.B, false], - [ids.C, false], - [ids.D, true], // D is clipped and so should always be culled / outside of viewport - ]) - - editor.pan({ x: -100, y: -100 }) - jest.advanceTimersByTime(500) - - expect(editor.getRenderingShapes().map(({ id, isCulled }) => [id, isCulled])).toStrictEqual([ - [ids.A, true], // A should be culled now that it's outside of the expanded viewport too - [ids.B, false], - [ids.C, false], - [ids.D, true], // D is clipped and so should always be culled, even if it's in the viewport - ]) - - editor.pan({ x: -900, y: -900 }) - jest.advanceTimersByTime(500) - expect(editor.getRenderingShapes().map(({ id, isCulled }) => [id, isCulled])).toStrictEqual([ - [ids.A, true], - [ids.B, true], - [ids.C, true], - [ids.D, true], - ]) -}) - it('lists shapes in viewport sorted by id with correct indexes & background indexes', () => { const ids = createShapes() // Expect the results to be sorted correctly by id diff --git a/packages/tlschema/api-report.md b/packages/tlschema/api-report.md index c272c8d22..d3e74a98a 100644 --- a/packages/tlschema/api-report.md +++ b/packages/tlschema/api-report.md @@ -218,7 +218,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly width: 720; readonly height: 500; readonly doesResize: true; - readonly canUnmount: true; readonly overridePermissions: { readonly 'allow-top-navigation': true; }; @@ -231,7 +230,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly width: 720; readonly height: 500; readonly doesResize: true; - readonly canUnmount: true; readonly toEmbedUrl: (url: string) => string | undefined; readonly fromEmbedUrl: (url: string) => string | undefined; }, { @@ -241,7 +239,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly width: 720; readonly height: 500; readonly doesResize: true; - readonly canUnmount: false; readonly toEmbedUrl: (url: string) => string | undefined; readonly fromEmbedUrl: (url: string) => string | undefined; }, { @@ -253,7 +250,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly width: 720; readonly height: 500; readonly doesResize: true; - readonly canUnmount: false; readonly toEmbedUrl: (url: string) => string | undefined; readonly fromEmbedUrl: (url: string) => string | undefined; }, { @@ -265,7 +261,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly width: 720; readonly height: 500; readonly doesResize: true; - readonly canUnmount: false; readonly toEmbedUrl: (url: string) => string | undefined; readonly fromEmbedUrl: (url: string) => string | undefined; }, { @@ -277,7 +272,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly width: 520; readonly height: 400; readonly doesResize: true; - readonly canUnmount: false; readonly toEmbedUrl: (url: string) => string | undefined; readonly fromEmbedUrl: (url: string) => string | undefined; }, { @@ -287,7 +281,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly width: 520; readonly height: 400; readonly doesResize: false; - readonly canUnmount: false; readonly toEmbedUrl: (url: string) => string | undefined; readonly fromEmbedUrl: (url: string) => string | undefined; }, { @@ -297,7 +290,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly width: 800; readonly height: 450; readonly doesResize: true; - readonly canUnmount: false; readonly overridePermissions: { readonly 'allow-presentation': true; }; @@ -313,7 +305,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly minWidth: 460; readonly minHeight: 360; readonly doesResize: true; - readonly canUnmount: false; readonly instructionLink: "https://support.google.com/calendar/answer/41207?hl=en"; readonly toEmbedUrl: (url: string) => string | undefined; readonly fromEmbedUrl: (url: string) => string | undefined; @@ -326,7 +317,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly minWidth: 460; readonly minHeight: 360; readonly doesResize: true; - readonly canUnmount: false; readonly toEmbedUrl: (url: string) => string | undefined; readonly fromEmbedUrl: (url: string) => string | undefined; }, { @@ -336,7 +326,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly width: 720; readonly height: 500; readonly doesResize: true; - readonly canUnmount: true; readonly toEmbedUrl: (url: string) => string | undefined; readonly fromEmbedUrl: (url: string) => string | undefined; }, { @@ -346,7 +335,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly width: 720; readonly height: 500; readonly doesResize: true; - readonly canUnmount: false; readonly toEmbedUrl: (url: string) => string | undefined; readonly fromEmbedUrl: (url: string) => string | undefined; }, { @@ -356,7 +344,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly width: 720; readonly height: 500; readonly doesResize: true; - readonly canUnmount: false; readonly toEmbedUrl: (url: string) => string | undefined; readonly fromEmbedUrl: (url: string) => string | undefined; }, { @@ -368,7 +355,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly minHeight: 500; readonly overrideOutlineRadius: 12; readonly doesResize: true; - readonly canUnmount: false; readonly toEmbedUrl: (url: string) => string | undefined; readonly fromEmbedUrl: (url: string) => string | undefined; }, { @@ -378,7 +364,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly width: 640; readonly height: 360; readonly doesResize: true; - readonly canUnmount: false; readonly isAspectRatioLocked: true; readonly toEmbedUrl: (url: string) => string | undefined; readonly fromEmbedUrl: (url: string) => string | undefined; @@ -389,7 +374,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly width: 720; readonly height: 500; readonly doesResize: true; - readonly canUnmount: false; readonly isAspectRatioLocked: true; readonly toEmbedUrl: (url: string) => string | undefined; readonly fromEmbedUrl: (url: string) => string | undefined; @@ -400,7 +384,6 @@ export const EMBED_DEFINITIONS: readonly [{ readonly width: 720; readonly height: 500; readonly doesResize: true; - readonly canUnmount: false; readonly isAspectRatioLocked: false; readonly backgroundColor: "#fff"; readonly toEmbedUrl: (url: string) => string | undefined; @@ -417,7 +400,6 @@ export type EmbedDefinition = { readonly width: number; readonly height: number; readonly doesResize: boolean; - readonly canUnmount: boolean; readonly isAspectRatioLocked?: boolean; readonly overridePermissions?: TLEmbedShapePermissions; readonly instructionLink?: string; diff --git a/packages/tlschema/api/api.json b/packages/tlschema/api/api.json index 3ecd1144d..30ee2f5f4 100644 --- a/packages/tlschema/api/api.json +++ b/packages/tlschema/api/api.json @@ -1801,7 +1801,7 @@ }, { "kind": "Content", - "text": "readonly [{\n readonly type: \"tldraw\";\n readonly title: \"tldraw\";\n readonly hostnames: readonly [\"beta.tldraw.com\", \"tldraw.com\", \"localhost:3000\"];\n readonly minWidth: 300;\n readonly minHeight: 300;\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly canUnmount: true;\n readonly overridePermissions: {\n readonly 'allow-top-navigation': true;\n };\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"figma\";\n readonly title: \"Figma\";\n readonly hostnames: readonly [\"figma.com\"];\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly canUnmount: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"google_maps\";\n readonly title: \"Google Maps\";\n readonly hostnames: readonly [\"google.*\"];\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly canUnmount: false;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"val_town\";\n readonly title: \"Val Town\";\n readonly hostnames: readonly [\"val.town\"];\n readonly minWidth: 260;\n readonly minHeight: 100;\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly canUnmount: false;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"codesandbox\";\n readonly title: \"CodeSandbox\";\n readonly hostnames: readonly [\"codesandbox.io\"];\n readonly minWidth: 300;\n readonly minHeight: 300;\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly canUnmount: false;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"codepen\";\n readonly title: \"Codepen\";\n readonly hostnames: readonly [\"codepen.io\"];\n readonly minWidth: 300;\n readonly minHeight: 300;\n readonly width: 520;\n readonly height: 400;\n readonly doesResize: true;\n readonly canUnmount: false;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"scratch\";\n readonly title: \"Scratch\";\n readonly hostnames: readonly [\"scratch.mit.edu\"];\n readonly width: 520;\n readonly height: 400;\n readonly doesResize: false;\n readonly canUnmount: false;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"youtube\";\n readonly title: \"YouTube\";\n readonly hostnames: readonly [\"*.youtube.com\", \"youtube.com\", \"youtu.be\"];\n readonly width: 800;\n readonly height: 450;\n readonly doesResize: true;\n readonly canUnmount: false;\n readonly overridePermissions: {\n readonly 'allow-presentation': true;\n };\n readonly isAspectRatioLocked: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"google_calendar\";\n readonly title: \"Google Calendar\";\n readonly hostnames: readonly [\"calendar.google.*\"];\n readonly width: 720;\n readonly height: 500;\n readonly minWidth: 460;\n readonly minHeight: 360;\n readonly doesResize: true;\n readonly canUnmount: false;\n readonly instructionLink: \"https://support.google.com/calendar/answer/41207?hl=en\";\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"google_slides\";\n readonly title: \"Google Slides\";\n readonly hostnames: readonly [\"docs.google.*\"];\n readonly width: 720;\n readonly height: 500;\n readonly minWidth: 460;\n readonly minHeight: 360;\n readonly doesResize: true;\n readonly canUnmount: false;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"github_gist\";\n readonly title: \"GitHub Gist\";\n readonly hostnames: readonly [\"gist.github.com\"];\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly canUnmount: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"replit\";\n readonly title: \"Replit\";\n readonly hostnames: readonly [\"replit.com\"];\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly canUnmount: false;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"felt\";\n readonly title: \"Felt\";\n readonly hostnames: readonly [\"felt.com\"];\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly canUnmount: false;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"spotify\";\n readonly title: \"Spotify\";\n readonly hostnames: readonly [\"open.spotify.com\"];\n readonly width: 720;\n readonly height: 500;\n readonly minHeight: 500;\n readonly overrideOutlineRadius: 12;\n readonly doesResize: true;\n readonly canUnmount: false;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"vimeo\";\n readonly title: \"Vimeo\";\n readonly hostnames: readonly [\"vimeo.com\", \"player.vimeo.com\"];\n readonly width: 640;\n readonly height: 360;\n readonly doesResize: true;\n readonly canUnmount: false;\n readonly isAspectRatioLocked: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"excalidraw\";\n readonly title: \"Excalidraw\";\n readonly hostnames: readonly [\"excalidraw.com\"];\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly canUnmount: false;\n readonly isAspectRatioLocked: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"observable\";\n readonly title: \"Observable\";\n readonly hostnames: readonly [\"observablehq.com\"];\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly canUnmount: false;\n readonly isAspectRatioLocked: false;\n readonly backgroundColor: \"#fff\";\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}]" + "text": "readonly [{\n readonly type: \"tldraw\";\n readonly title: \"tldraw\";\n readonly hostnames: readonly [\"beta.tldraw.com\", \"tldraw.com\", \"localhost:3000\"];\n readonly minWidth: 300;\n readonly minHeight: 300;\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly overridePermissions: {\n readonly 'allow-top-navigation': true;\n };\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"figma\";\n readonly title: \"Figma\";\n readonly hostnames: readonly [\"figma.com\"];\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"google_maps\";\n readonly title: \"Google Maps\";\n readonly hostnames: readonly [\"google.*\"];\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"val_town\";\n readonly title: \"Val Town\";\n readonly hostnames: readonly [\"val.town\"];\n readonly minWidth: 260;\n readonly minHeight: 100;\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"codesandbox\";\n readonly title: \"CodeSandbox\";\n readonly hostnames: readonly [\"codesandbox.io\"];\n readonly minWidth: 300;\n readonly minHeight: 300;\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"codepen\";\n readonly title: \"Codepen\";\n readonly hostnames: readonly [\"codepen.io\"];\n readonly minWidth: 300;\n readonly minHeight: 300;\n readonly width: 520;\n readonly height: 400;\n readonly doesResize: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"scratch\";\n readonly title: \"Scratch\";\n readonly hostnames: readonly [\"scratch.mit.edu\"];\n readonly width: 520;\n readonly height: 400;\n readonly doesResize: false;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"youtube\";\n readonly title: \"YouTube\";\n readonly hostnames: readonly [\"*.youtube.com\", \"youtube.com\", \"youtu.be\"];\n readonly width: 800;\n readonly height: 450;\n readonly doesResize: true;\n readonly overridePermissions: {\n readonly 'allow-presentation': true;\n };\n readonly isAspectRatioLocked: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"google_calendar\";\n readonly title: \"Google Calendar\";\n readonly hostnames: readonly [\"calendar.google.*\"];\n readonly width: 720;\n readonly height: 500;\n readonly minWidth: 460;\n readonly minHeight: 360;\n readonly doesResize: true;\n readonly instructionLink: \"https://support.google.com/calendar/answer/41207?hl=en\";\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"google_slides\";\n readonly title: \"Google Slides\";\n readonly hostnames: readonly [\"docs.google.*\"];\n readonly width: 720;\n readonly height: 500;\n readonly minWidth: 460;\n readonly minHeight: 360;\n readonly doesResize: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"github_gist\";\n readonly title: \"GitHub Gist\";\n readonly hostnames: readonly [\"gist.github.com\"];\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"replit\";\n readonly title: \"Replit\";\n readonly hostnames: readonly [\"replit.com\"];\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"felt\";\n readonly title: \"Felt\";\n readonly hostnames: readonly [\"felt.com\"];\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"spotify\";\n readonly title: \"Spotify\";\n readonly hostnames: readonly [\"open.spotify.com\"];\n readonly width: 720;\n readonly height: 500;\n readonly minHeight: 500;\n readonly overrideOutlineRadius: 12;\n readonly doesResize: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"vimeo\";\n readonly title: \"Vimeo\";\n readonly hostnames: readonly [\"vimeo.com\", \"player.vimeo.com\"];\n readonly width: 640;\n readonly height: 360;\n readonly doesResize: true;\n readonly isAspectRatioLocked: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"excalidraw\";\n readonly title: \"Excalidraw\";\n readonly hostnames: readonly [\"excalidraw.com\"];\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly isAspectRatioLocked: true;\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}, {\n readonly type: \"observable\";\n readonly title: \"Observable\";\n readonly hostnames: readonly [\"observablehq.com\"];\n readonly width: 720;\n readonly height: 500;\n readonly doesResize: true;\n readonly isAspectRatioLocked: false;\n readonly backgroundColor: \"#fff\";\n readonly toEmbedUrl: (url: string) => string | undefined;\n readonly fromEmbedUrl: (url: string) => string | undefined;\n}]" } ], "fileUrlPath": "packages/tlschema/src/shapes/TLEmbedShape.ts", @@ -1824,7 +1824,7 @@ }, { "kind": "Content", - "text": "{\n readonly type: string;\n readonly title: string;\n readonly hostnames: readonly string[];\n readonly minWidth?: number;\n readonly minHeight?: number;\n readonly width: number;\n readonly height: number;\n readonly doesResize: boolean;\n readonly canUnmount: boolean;\n readonly isAspectRatioLocked?: boolean;\n readonly overridePermissions?: " + "text": "{\n readonly type: string;\n readonly title: string;\n readonly hostnames: readonly string[];\n readonly minWidth?: number;\n readonly minHeight?: number;\n readonly width: number;\n readonly height: number;\n readonly doesResize: boolean;\n readonly isAspectRatioLocked?: boolean;\n readonly overridePermissions?: " }, { "kind": "Reference", diff --git a/packages/tlschema/src/migrations.test.ts b/packages/tlschema/src/migrations.test.ts index 377920f2d..f9a6d9548 100644 --- a/packages/tlschema/src/migrations.test.ts +++ b/packages/tlschema/src/migrations.test.ts @@ -2030,6 +2030,34 @@ describe('Fractional indexing for line points', () => { }) }) +describe('add white', () => { + const { up, down } = rootShapeMigrations.migrators[rootShapeVersions.AddWhite] + + test('up works as expected', () => { + expect( + up({ + props: {}, + }) + ).toEqual({ + props: {}, + }) + }) + + test('down works as expected', () => { + expect( + down({ + props: { + color: 'white', + }, + }) + ).toEqual({ + props: { + color: 'black', + }, + }) + }) +}) + /* --- PUT YOUR MIGRATIONS TESTS ABOVE HERE --- */ for (const migrator of allMigrators) { diff --git a/packages/tlschema/src/records/TLShape.ts b/packages/tlschema/src/records/TLShape.ts index 5821ae09e..300c5fad4 100644 --- a/packages/tlschema/src/records/TLShape.ts +++ b/packages/tlschema/src/records/TLShape.ts @@ -87,11 +87,12 @@ export const rootShapeVersions = { AddIsLocked: 1, HoistOpacity: 2, AddMeta: 3, + AddWhite: 4, } as const /** @internal */ export const rootShapeMigrations = defineMigrations({ - currentVersion: rootShapeVersions.AddMeta, + currentVersion: rootShapeVersions.AddWhite, migrators: { [rootShapeVersions.AddIsLocked]: { up: (record) => { @@ -147,6 +148,22 @@ export const rootShapeMigrations = defineMigrations({ } }, }, + [rootShapeVersions.AddWhite]: { + up: (record) => { + return { + ...record, + } + }, + down: (record) => { + return { + ...record, + props: { + ...record.props, + color: record.props.color === 'white' ? 'black' : record.props.color, + }, + } + }, + }, }, }) diff --git a/packages/tlschema/src/shapes/TLEmbedShape.ts b/packages/tlschema/src/shapes/TLEmbedShape.ts index 6037f2aad..416ab83a3 100644 --- a/packages/tlschema/src/shapes/TLEmbedShape.ts +++ b/packages/tlschema/src/shapes/TLEmbedShape.ts @@ -24,7 +24,6 @@ export const EMBED_DEFINITIONS = [ width: 720, height: 500, doesResize: true, - canUnmount: true, overridePermissions: { 'allow-top-navigation': true, }, @@ -50,7 +49,6 @@ export const EMBED_DEFINITIONS = [ width: 720, height: 500, doesResize: true, - canUnmount: true, toEmbedUrl: (url) => { if ( !!url.match( @@ -81,7 +79,6 @@ export const EMBED_DEFINITIONS = [ width: 720, height: 500, doesResize: true, - canUnmount: false, toEmbedUrl: (url) => { if (url.includes('/maps/')) { const match = url.match(/@(.*),(.*),(.*)z/) @@ -120,7 +117,6 @@ export const EMBED_DEFINITIONS = [ width: 720, height: 500, doesResize: true, - canUnmount: false, toEmbedUrl: (url) => { const urlObj = safeParseUrl(url) // e.g. extract "steveruizok.mathFact" from https://www.val.town/v/steveruizok.mathFact @@ -149,7 +145,6 @@ export const EMBED_DEFINITIONS = [ width: 720, height: 500, doesResize: true, - canUnmount: false, toEmbedUrl: (url) => { const urlObj = safeParseUrl(url) const matches = urlObj && urlObj.pathname.match(/\/s\/([^/]+)\/?/) @@ -176,7 +171,6 @@ export const EMBED_DEFINITIONS = [ width: 520, height: 400, doesResize: true, - canUnmount: false, toEmbedUrl: (url) => { const CODEPEN_URL_REGEXP = /https:\/\/codepen.io\/([^/]+)\/pen\/([^/]+)/ const matches = url.match(CODEPEN_URL_REGEXP) @@ -203,7 +197,6 @@ export const EMBED_DEFINITIONS = [ width: 520, height: 400, doesResize: false, - canUnmount: false, toEmbedUrl: (url) => { const SCRATCH_URL_REGEXP = /https?:\/\/scratch.mit.edu\/projects\/([^/]+)/ const matches = url.match(SCRATCH_URL_REGEXP) @@ -230,7 +223,6 @@ export const EMBED_DEFINITIONS = [ width: 800, height: 450, doesResize: true, - canUnmount: false, overridePermissions: { 'allow-presentation': true, }, @@ -275,7 +267,6 @@ export const EMBED_DEFINITIONS = [ minWidth: 460, minHeight: 360, doesResize: true, - canUnmount: false, instructionLink: 'https://support.google.com/calendar/answer/41207?hl=en', toEmbedUrl: (url) => { const urlObj = safeParseUrl(url) @@ -318,7 +309,6 @@ export const EMBED_DEFINITIONS = [ minWidth: 460, minHeight: 360, doesResize: true, - canUnmount: false, toEmbedUrl: (url) => { const urlObj = safeParseUrl(url) @@ -353,7 +343,6 @@ export const EMBED_DEFINITIONS = [ width: 720, height: 500, doesResize: true, - canUnmount: true, toEmbedUrl: (url) => { const urlObj = safeParseUrl(url) if (urlObj && urlObj.pathname.match(/\/([^/]+)\/([^/]+)/)) { @@ -378,7 +367,6 @@ export const EMBED_DEFINITIONS = [ width: 720, height: 500, doesResize: true, - canUnmount: false, toEmbedUrl: (url) => { const urlObj = safeParseUrl(url) if (urlObj && urlObj.pathname.match(/\/@([^/]+)\/([^/]+)/)) { @@ -406,7 +394,6 @@ export const EMBED_DEFINITIONS = [ width: 720, height: 500, doesResize: true, - canUnmount: false, toEmbedUrl: (url) => { const urlObj = safeParseUrl(url) if (urlObj && urlObj.pathname.match(/^\/map\//)) { @@ -432,7 +419,6 @@ export const EMBED_DEFINITIONS = [ minHeight: 500, overrideOutlineRadius: 12, doesResize: true, - canUnmount: false, toEmbedUrl: (url) => { const urlObj = safeParseUrl(url) if (urlObj && urlObj.pathname.match(/^\/(artist|album)\//)) { @@ -455,7 +441,6 @@ export const EMBED_DEFINITIONS = [ width: 640, height: 360, doesResize: true, - canUnmount: false, isAspectRatioLocked: true, toEmbedUrl: (url) => { const urlObj = safeParseUrl(url) @@ -486,7 +471,6 @@ export const EMBED_DEFINITIONS = [ width: 720, height: 500, doesResize: true, - canUnmount: false, isAspectRatioLocked: true, toEmbedUrl: (url) => { const urlObj = safeParseUrl(url) @@ -510,7 +494,6 @@ export const EMBED_DEFINITIONS = [ width: 720, height: 500, doesResize: true, - canUnmount: false, isAspectRatioLocked: false, backgroundColor: '#fff', toEmbedUrl: (url) => { @@ -619,7 +602,6 @@ export type EmbedDefinition = { readonly width: number readonly height: number readonly doesResize: boolean - readonly canUnmount: boolean readonly isAspectRatioLocked?: boolean readonly overridePermissions?: TLEmbedShapePermissions readonly instructionLink?: string diff --git a/packages/utils/api-report.md b/packages/utils/api-report.md index e1379999c..022651e9a 100644 --- a/packages/utils/api-report.md +++ b/packages/utils/api-report.md @@ -176,6 +176,15 @@ export function mapObjectMapValues( [K in Key]: ValueAfter; }; +// @internal (undocumented) +export function measureAverageDuration(_target: any, propertyKey: string, descriptor: PropertyDescriptor): PropertyDescriptor; + +// @internal (undocumented) +export function measureCbDuration(name: string, cb: () => any): any; + +// @internal (undocumented) +export function measureDuration(_target: any, propertyKey: string, descriptor: PropertyDescriptor): PropertyDescriptor; + // @public export class MediaHelpers { static getImageSize(blob: Blob): Promise<{ diff --git a/packages/utils/src/index.ts b/packages/utils/src/index.ts index b008b9753..90273933b 100644 --- a/packages/utils/src/index.ts +++ b/packages/utils/src/index.ts @@ -36,6 +36,7 @@ export { objectMapKeys, objectMapValues, } from './lib/object' +export { measureAverageDuration, measureCbDuration, measureDuration } from './lib/perf' export { PngHelpers } from './lib/png' export { type IndexKey } from './lib/reordering/IndexKey' export { diff --git a/packages/utils/src/lib/perf.ts b/packages/utils/src/lib/perf.ts new file mode 100644 index 000000000..e6ac86450 --- /dev/null +++ b/packages/utils/src/lib/perf.ts @@ -0,0 +1,50 @@ +/** @internal */ +export function measureCbDuration(name: string, cb: () => any) { + const now = performance.now() + const result = cb() + // eslint-disable-next-line no-console + console.log(`${name} took`, performance.now() - now, 'ms') + return result +} + +/** @internal */ +export function measureDuration(_target: any, propertyKey: string, descriptor: PropertyDescriptor) { + const originalMethod = descriptor.value + descriptor.value = function (...args: any[]) { + const start = performance.now() + const result = originalMethod.apply(this, args) + const end = performance.now() + // eslint-disable-next-line no-console + console.log(`${propertyKey} took ${end - start}ms `) + return result + } + return descriptor +} + +const averages = new Map() + +/** @internal */ +export function measureAverageDuration( + _target: any, + propertyKey: string, + descriptor: PropertyDescriptor +) { + const originalMethod = descriptor.value + descriptor.value = function (...args: any[]) { + const start = performance.now() + const result = originalMethod.apply(this, args) + const end = performance.now() + const value = averages.get(descriptor.value)! + const length = end - start + const total = value.total + length + const count = value.count + 1 + averages.set(descriptor.value, { total, count }) + // eslint-disable-next-line no-console + console.log( + `${propertyKey} took ${(end - start).toFixed(2)}ms | average ${(total / count).toFixed(2)}ms` + ) + return result + } + averages.set(descriptor.value, { total: 0, count: 0 }) + return descriptor +}