Back to Plate

Blockquote transforms must keep selection inside the new quote

docs/solutions/ui-bugs/2026-04-02-blockquote-transforms-must-keep-selection-inside-the-new-quote.md

53.0.53.4 KB
Original Source

Blockquote transforms must keep selection inside the new quote

Problem

After blockquote became a real container node, the app-level editor helpers in apps/www still treated it like a flat text block.

That mismatch broke the editing flow: inserting or converting a quote created the right shape eventually, but the caret landed in the previous block instead of inside the new quote.

Symptoms

  • /quote in /blocks/editor-ai inserted a blockquote, then typing went into the previous paragraph.
  • Converting a paragraph into a blockquote moved selection from [1, 0] to the previous block instead of the wrapped paragraph at [1, 0, 0].
  • editor.tf.blockquote.toggle() did not reproduce the bug by itself, so the broken seam was easy to misread.

What Didn't Work

  • Treating this like another BaseBlockquotePlugin regression. The core wrap transform already preserved selection.
  • Keeping setBlockType(...) on flat setNodes({ type: KEYS.blockquote }). That let normalization repair the node shape later, after selection had already drifted.
  • Relying on generic select: true after inserting a blockquote node. For a container block, that is not precise enough.

Solution

Fix the shared apps/www editor transform seam instead of patching one UI caller at a time.

  • Add a createBlockquote(...) helper that inserts a container quote with an inner paragraph.
  • Add selectBlockquoteStart(...) so quote insertion explicitly selects the nested paragraph start.
  • Special-case insertBlock(editor, KEYS.blockquote) to insert the container shape and select [path, 0, 0].
  • Special-case setBlockType(editor, KEYS.blockquote) to call editor.tf.toggleBlock(type, { wrap: true }) instead of flat setNodes.
  • Route block-context-menu quote conversion through setBlockType(...) so it uses the same fixed path.
  • Add regressions for quote insert and quote conversion selection behavior.

The important transform seams became:

ts
if (type === KEYS.blockquote) {
  const insertPath = PathApi.next(path);

  editor.tf.insertNodes(createBlockquote(editor), { at: insertPath });
  selectBlockquoteStart(editor, insertPath);

  return;
}
ts
if (type === KEYS.blockquote) {
  editor.tf.toggleBlock(type, {
    ...(at ? { at } : {}),
    wrap: true,
  });

  return;
}

Why This Works

blockquote is now a container element, so insertion and conversion must preserve two things together:

  • the nested paragraph child
  • the nested selection path inside that paragraph

The core wrap transform already knew how to do that. The app helpers did not. Once the app seam stopped creating flat quotes and stopped using flat block conversion, selection stayed anchored inside the new quote.

Prevention

  • When a node becomes a container element, audit app-level insert and convert helpers. They often lag behind package-level transforms.
  • Do not use generic setNodes or generic select: true for container-block insertion when the user must land inside a nested text block.
  • If a core transform path behaves correctly but the UI still breaks, inspect the caller helpers before reopening plugin internals.
  • Add one regression for conversion and one for insertion whenever selection behavior depends on nested paths.