v2.2.0
Bindings
Bindings allow you to create relationships between shapes. Our default arrow shapes are a great example of this: each end of the arrow can bind to the shape it's pointing to. When that shape moves, so does the arrow. Before this change, it wasn't possible to build things like arrows on top of the tldraw sdk - arrows were hard-coded into the library. Now, with bindings, you can create arrows, constraint systems, visual programming environments, and much more.
Check out the bindings guide for more information. (#3326, #3780, #3797, #3800, #3871)
Camera constraints
You can now limit the camera in tldraw to a certain fixed area of the canvas. This is useful for creating experiences that don't quite fit the "infinite canvas" paradigm: document annotators, image editor, slideshow creators, etc.
See the camera constraints guide for more information. (#3282, #3747, #3814, #3828, #3844, #3863)
Configurable options prop
You can now override many options which were previously hard-coded constants. Pass an options
prop into the tldraw component to change the maximum number of pages, grid steps, or other previously hard-coded values.
See TldrawOptions
for details. (#3799, #3900)
Breaking changes
- The
canBind
flag now accepts an options object instead of just the shape in question. If you're relying on its arguments, check outTLShapeUtilCanBindOpts
for its replacement. editor.sideEffects.registerBatchCompleteHandler
has been replaced witheditor.sideEffects.registerOperationCompleteHandler
(#3748)editor.getArrowInfo(shape)
has been replaced withgetArrowInfo(editor, shape)
editor.getArrowsBoundTo(shape)
has been removed. Instead, useeditor.getBindingsToShape(shape, 'arrow')
and follow thefromId
of each binding to the corresponding arrow shape- These types have moved from
@tldraw/editor
totldraw
:TLArcInfo
TLArrowInfo
TLArrowPoint
- The
start
andend
properties onTLArrowShape
no longer havetype: point | binding
. Instead, they're always a point, which may be out of date if a binding exists. To check for & retrieve arrow bindings, usegetArrowBindings(editor, shape)
instead. getArrowTerminalsInArrowSpace
must be passed aTLArrowBindings
as a third argument:getArrowTerminalsInArrowSpace(editor, shape, getArrowBindings(editor, shape))
- The following types have been renamed:
ShapeProps
->RecordProps
ShapePropsType
->RecordPropsType
TLShapePropsMigrations
->TLPropsMigrations
SchemaShapeInfo
->SchemaPropsInfo
Undo/redo
1. History Options
Previously, some (not all!) commands accepted a history options object with squashing
, ephemeral
, and preserveRedoStack
flags. Squashing enabled/disabled a memory optimisation (storing individual commands vs squashing them together). Ephemeral stopped a command from affecting the undo/redo stack at all. Preserve redo stack stopped commands from wiping the redo stack. These flags were never available consistently - some commands had them and others didn't.
In this version, most of these flags have been removed. squashing
is gone entirely (everything squashes & does so much faster than before). There were a couple of commands that had a special default - for example, updateInstanceState
used to default to being ephemeral
. Those maintain the defaults, but the options look a little different now - {ephemeral: true}
is now {history: 'ignore'}
and {preserveRedoStack: true}
is now {history: 'record-preserveRedoStack'}
.
If you were previously using these options in places where they've now been removed, you can use wrap them with editor.history.ignore(fn)
or editor.history.batch(fn, {history: 'record-preserveRedoStack'})
. For example,
editor.nudgeShapes(..., { ephemeral: true })
can now be written as
editor.history.ignore(() => {
editor.nudgeShapes(...)
})
2. Automatic recording
Previously, only commands (e.g. editor.updateShapes
and things that use it) were added to the undo/redo stack. Everything else (e.g. editor.store.put
) wasn't. Now, everything that touches the store is recorded in the undo/redo stack (unless it's part of mergeRemoteChanges
). You can use editor.history.ignore(fn)
as above if you want to make other changes to the store that aren't recorded - this is short for editor.history.batch(fn, {history: 'ignore'})
When upgrading to this version of tldraw, you shouldn't need to change anything unless you're using store.put
, store.remove
, or store.applyDiff
outside of store.mergeRemoteChanges
. If you are, you can preserve the functionality of those not being recorded by wrapping them either in mergeRemoteChanges
(if they're multiplayer-related) or history.ignore
as appropriate.
3. Side effects
Before this diff, any changes in side-effects weren't captured by the undo-redo stack. This was actually the motivation for this change in the first place! But it's a pretty big change, and if you're using side effects we recommend you double-check how they interact with undo/redo before/after this change. To get the old behaviour back, wrap your side effects in editor.history.ignore
.
4. Mark options
Previously, editor.mark(id)
accepted two additional boolean parameters: onUndo
and onRedo
. If these were set to false, then when undoing or redoing we'd skip over that mark and keep going until we found one with those values set to true. We've removed those options - if you're using them, let us know and we'll figure out an alternative!
Improvements
- Nicer rendering for bookmarks without preview images. (#3856)
- Improve undo/redo UX around image cropping. (#3891)
- Disable toolbar items that don't work when not in select mode. (#3819)
- ❤️ We've added a heart shape to the geo shape set. (#3787)
- Detect coarse pointers (ie touch) more reliably. (#3572, #3656, #3795)
- Reduce padding when zooming to fit. (#3798)
- Increase the default limit of shapes per page from 2000 to 4000 (#3716)
- Unify list of accepted image types and expand to include webp, webm, apng, & avif. (#3730)
- Prunes unused assets when loading a
.tldr
file. (#3689) - Improve handling of mouse-type devices that support pressure, e.g. wacom tablets. They now use the same freehand options as true pen-type inputs. (#3639)
- Separate the text align property for text shapes and labels. Text shapes are now left-aligned by default. (#3627)
- Add desmos graph embed type. (#3608)
- Tweak default gap value to be consistent with sticky note gaps. (#3606)
API changes
- Add
editor.blur
method. (#3875) - Better defaults for
createTLStore
. (#3886) - Add
getSnapshot
andloadSnapshot
for easier loading/saving of tldraw documents. Read more here. (#3811) - Add
select
option toEditor.groupShapes
andEditor.ungroupShapes
. (#3690) InFrontOfTheCanvas
now has access to the editor's UI context (#3782)useEditor
and other context-based hooks will now throw an error when used out-of-context, instead of returning a fake value. (#3750)- Expose migrations, validators, and versions from tlschema. Previously, we weren't exporting migrations & validators for our default shapes. This meant that it wasn't possible to make your own tlschema with both our default shapes and some of your own (e.g. for custom multiplayer). This fixes that by exposing all the migrations, validators, and versions from tlschema, plus
defaultShapeSchemas
which can be passed directly tocreateTLSchema
.(#3613)
Bug fixes
- Fix 'insert media' undo removing other changes. (#3910)
- Fix referrer being sent for bookmarks and images. (#3881)
- Prevent stale shape data sometimes being used in render. (#3882)
- Fix an issue with pen pressure. (#3877)
- Fixed a bug where the minimum distance for a drag was wrong when zoomed in or out. (#3873)
- Make sure timers/animation frames are disposed along with the editor. (#3852)
- Fix some inconsistencies with text label rendering. (#3830)
- Fixed cropped images not exporting properly. (#3837)
- Fix bug with spacebar & middle mousenbutton panning. (#3791, #3792)
- Make sure any in-progress interactions are cancelled when switching page/ (#3771
- Fixes a bug that caused the cursor & shapes to wiggle around when following someone else's viewport. (#3695)
- Fix some long-stanging cross-browser issues with focus management. (#3718)
- Fix bug preventing imports in Astro. (#3742)
- Fixes an issue with copy pasting shapes as svg and png not correctly working for patterned shapes. (#3708)
- Fix RTL text layout for SVG exports. (#3680)
- Fixes a rare crash effecting text shapes on mobile. (#3672)
- Fix textbox direction when it contains both RTL and LTR languages /(#3188)
- Fix an links in embeds that open the embedded site (e.g. YouTube). (#3609)
- Fix pasting not working from Edit menu. (#3623)
- Fixed a bug with resizing text shapes from the left and right while holding alt. (#3632)
- Fix a bug where locked shapes could still be hovered. (#3575)
- Fix clicking on the minimap sometimes not changing the viewport. (#3617)
- Fix an issue with the minimap bugging out after you change the window's height. (#3621)
Translations
- Update French, Hungarian, & Korean translations.
- Add Bahasa Indonesia translation. (#3649)
Authors: 14
- alex (@SomeHats)
- CodeTorso (@CodeTorso)
- David Sheldrick (@ds300)
- Eric Mika (@kitschpatrol)
- Eswar Prasad Clinton. A (@eswarclynn)
- fakerr (@not-first)
- Lorenzo Lewis (@lorenzolewis)
- Lu Wilson (@TodePond)
- Mime Čuvalo (@mimecuvalo)
- Mitja Bezenšek (@MitjaBezensek)
- Mohammad Kazemi (@mokazemi)
- Steve Ruiz (@steveruizok)
- Taha (@Taha-Hassan-Git)
- Trevor Dobbertin (@Trevato)