Gum gum-tool-undo
Reference guide for Gum's undo/redo system. Load this when working on undo/redo behavior, the History tab, UndoManager, UndoPlugin, UndoSnapshot, or stale reference issues after undo.
git clone https://github.com/vchelaru/Gum
T=$(mktemp -d) && git clone --depth=1 https://github.com/vchelaru/Gum "$T" && mkdir -p ~/.claude/skills && cp -r "$T/.claude/skills/gum-tool-undo" ~/.claude/skills/vchelaru-gum-gum-tool-undo && rm -rf "$T"
.claude/skills/gum-tool-undo/SKILL.mdGum Undo/Redo System Reference
Overview
Gum has a snapshot-based undo/redo system scoped per-element. Undo history is displayed in the History tab in the Gum UI tool.
Key Characteristics
Per-Element Scoping
Undo history is stored separately for each open element (Screen, Component, or StandardElement). Switching between elements does not share or merge history — each element maintains its own independent undo stack.
No Selection Tracking
Undos do not record or restore the user's selection state. After undoing or redoing an operation, the selected object in the tree view or canvas may not match what was selected when the change was originally made.
No Persistence
Undo history is entirely in-memory and is cleared when the project is loaded or Gum is closed. There is no way to undo changes made in a previous session.
Element Deletion Is Not Undoable
When an element (Screen, Component, or StandardElement) is deleted, its entire undo history is discarded along with it. Deleting an element cannot be undone.
Behaviors Are Not Currently Supported
Undo/redo does not currently work for behavior-related changes. Changes to behaviors (adding, removing, or modifying) on an element may not be correctly undoable.
History Tab
The History tab in the Gum UI tool displays a human-readable list of all recorded undo actions for the currently selected element. Each entry shows a description of what changed, such as:
Modify element variables: X=10Add instances: MySpriteRemove instances: MySpriteAdd behaviors: MyBehaviorExposed variables: MyVar
The list is built by working backwards through undo snapshots and diffing consecutive states, so descriptions reflect the actual change rather than raw data.
What Is Tracked
The undo system records changes to:
- Element-level variable values (position, size, color, etc.)
- Instance additions and removals
- Instance reordering (tracked as index changes)
- State additions, removals, and variable changes within states
- Category additions and removals
- Variable exposure and unexposure
How Recording Works
The system uses a two-phase record approach:
— Captures a snapshot of the element's current state before a change begins. Called automatically byRecordState()
on element selection, state selection, etc. Do NOT call this manually from feature code.UndoPlugin
— Compares the current state against the recorded snapshot; if anything changed, saves an undo action. Called automatically when anRecordUndo()
is disposed.UndoLock
Correct Pattern for Recording Undos
Always use
RequestLock() — never call RecordState() or RecordUndo() manually:
using var undoLock = _undoManager.RequestLock(); // make your changes here // lock disposal fires RecordUndo() automatically
RequestLock() adds an UndoLock to UndoLocks. When the lock is disposed (end of using block), it removes itself; when UndoLocks reaches 0, HandleUndoLockChanged fires RecordUndo(). The RecordState() baseline is already set by the framework when the user selected the element.
Why not
manually? RecordState()
RecordState() is a no-op when any locks are held, and calling it outside of that flow risks overwriting the correct baseline snapshot.
Snapshots Are Deep Copies
Both element and behavior snapshots use
CloneElement/CloneBehavior, so every saved snapshot contains new object instances with different references than the live data. When undo is applied, the restored instances replace the live ones — meaning any code holding a reference to the pre-undo instance now has a stale reference that no longer exists in the element or behavior.
Consequence: after an undo,
_selectedState.SelectedInstance may point to a stale object. Reference-based lookups (e.g. tree node searches using ==) will fail. Name-based fallback is required to re-locate the logically equivalent node. If undo also changes the instance's name, selection cannot be restored and is silently dropped — this is considered acceptable.
Implementation Files
| File | Purpose |
|---|---|
| Core undo/redo logic; per-element history with |
| Event handlers that call / |
| Snapshot structure and diff/comparison logic () |
| History tab display and description generation |
| WPF ListBox UI for the History tab |
| Individual history item (display text + undo/redo direction) |
| Unit tests for undo behavior |
Known Limitations Summary
| Limitation | Details |
|---|---|
| No global undo | Each element has its own undo stack; cross-element changes are not grouped |
| No selection restore | Selection state is not captured or restored on undo/redo |
| No persistence | History is cleared on project load or app close |
| No element-deletion undo | Deleting an element removes its history permanently |
| Behaviors not supported | Behavior changes are not reliably undoable |