Block Selection

Loading...
Files
components/demo.tsx
'use client';

import React from 'react';

import { Plate } from '@udecode/plate/react';

import { editorPlugins } from '@/components/editor/plugins/editor-plugins';
import { useCreateEditor } from '@/components/editor/use-create-editor';
import { Editor, EditorContainer } from '@/components/plate-ui/editor';

import { DEMO_VALUES } from './values/demo-values';

export default function Demo({ id }: { id: string }) {
  const editor = useCreateEditor({
    plugins: [...editorPlugins],
    value: DEMO_VALUES[id],
  });

  return (
    <Plate editor={editor}>
      <EditorContainer variant="demo">
        <Editor />
      </EditorContainer>
    </Plate>
  );
}

The Block Selection feature allows users to select and manipulate entire text blocks, as opposed to individual words or characters. This powerful functionality enhances the editing experience by providing efficient ways to manage large sections of content.

Features

  • Select entire blocks with a single action
  • Multi-block selection
  • Copy, cut, and delete operations on selected blocks
  • Keyboard shortcuts for quick selection:
    • Cmd+A:
      • First press: select the current block
      • Double press: select all blocks
    • Arrow keys: select the block above or below
  • Customizable styling for selected blocks

Installation

npm install @udecode/plate-selection @udecode/plate-node-id

Usage

import { NodeIdPlugin } from '@udecode/plate-node-id';
import { BlockSelectionPlugin } from '@udecode/plate-selection/react';
 
const plugins = [
  // ...otherPlugins,
  NodeIdPlugin,
  BlockSelectionPlugin,
];

Exclude blocks from selection

You can exclude certain plugins from block selection using:

BlockSelectionPlugin.configure({
  inject: {
    // Exclude blocks below table rows
    excludeBelowPlugins: ['tr'],
    // Exclude block types
    excludePlugins: ['table', 'code_line', 'column_group', 'column'],
  }
})
  • excludeBelowPlugins: Plugin keys of non-selectable block descendants. Use this to prevent selection below specific blocks. For example, excluding 'tr' prevents selecting individual cells while still allowing table row selection.

  • excludePlugins: Plugin keys of non-selectable blocks.

Set scrollable container

If you're using EditorContainer from Editor, you can skip this section.

To control the scrollable container, configure the boundaries and container options within areaOptions. These options accept CSS selectors, such as #selection-demo #${editor.uid}, which are used with document.querySelector().

For this to work effectively:

  1. Add an id or className to your scroll container. If you're not sure about the container, you can add it to the <Editor /> component. We recommend using id={editor.uid}.
  2. Use the appropriate selector in your configuration.
  3. Don't forget to set position: relative to the container.

Default configuration:

BlockSelectionPlugin.configure({
  options: {
    areaOptions: {
      boundaries: `#${editor.uid}`,
      container: `#${editor.uid}`,
      selectables: `#${editor.uid} .slate-selectable`,
    },
  },
});

Set scroll speed

Use options.areaOptions.behaviour.scrolling.speedDivider to set the scroll speed.

The value 0.8 is our recommended speed since it's near the browser-native speed.

BlockSelectionPlugin.configure({
  options: {
    areaOptions: {
      behaviour: {
      scrolling: {
        // You can slow down the scroll speed by setting a bigger value.
        speedDivider: 1.5,
      },
      // The distance needed to move for the selection area to appear.
      // If it’s too small, it may cause the mouse click event to be blocked. 10 is a good default.
      startThreshold: 4,
    },
  },
}

Add selectable element

Add data-plate-selectable="true" to any element you want to start block selection.

Prevent unselect

To prevent unselecting blocks when clicking on certain elements, add the data-plate-prevent-unselect attribute to those components

For example:

  <YourSpecialButtoon data-plate-prevent-unselect />

Full Page Selection

Making Elements Selectable

You can enable block selection for elements outside the <Editor /> component, similar to the Potion template. Add the data-plate-selectable attribute to any component you want to make selectable:

<Cover data-plate-selectable />
<Sidebar data-plate-selectable />

This works for any element, even those outside the editor's DOM tree.

Resetting Selection

There are two ways to handle resetting selection across the full page:

  1. Direct API call:
editor.api.blockSelection.unselect();
  1. Click outside handler:
const handleClickOutside = (event: MouseEvent) => {
  if (!(event.target as HTMLElement).closest('[data-plate-selectable]')) {
    editor.api.blockSelection.unselect();
  }
};

Styling

Selection area

Style the selection area by adding the .slate-selection-area class to your editor container component. For example:

'[&_.slate-selection-area]:border [&_.slate-selection-area]:border-primary [&_.slate-selection-area]:bg-primary/10'

Selected element

To determine if an element is selected, use useBlockSelected hook. You can render a visual indicator around selected blocks using our BlockSelection component or create your own.

This component should be rendered inside each block element for consistent selection feedback. Plate UI is doing it in PlateElement.

Plugins

BlockSelectionPlugin

Options

Collapse all

    Options for the selection area. Example:

    {
      boundaries: [`#${editor.uid}`],
      container: [`#${editor.uid}`],
      selectables: [`#${editor.uid} .slate-selectable`],
      selectionAreaClass: 'slate-selection-area',
    }

    The padding-right of the editor.

    Enables or disables the context menu for block selection.

    • Default: false

    Indicates whether block selection is currently active.

    • Default: false

    A function to handle the keydown event when selecting.

    Options for querying nodes during block selection.

    • Default: { maxLevel: 1 }

    A set of IDs for the currently selected blocks.

    • Default: new Set()

    (Internal) The ID of the anchor block in the current selection. Used for shift-based selection.

    • Default: null

    Function to determine if a block element is selectable.

    • Default: () => true

BlockMenuPlugin

API

editor.api.blockSelection.focus

Focuses the block selection shadow input. This input handles copy, delete, and paste events for selected blocks.

editor.api.blockSelection.addSelectedRow

Adds a selected row to the block selection.

Parameters

Collapse all

    The ID of the row to be selected.

editor.api.blockSelection.getNodes

Gets the selected blocks in the editor.

Returns

Collapse all

    An array of selected block entries.

editor.api.blockSelection.resetSelectedIds

Resets the set of selected IDs to an empty set.

editor.api.blockSelection.selectedAll

Selects all selectable blocks in the editor.

editor.api.blockSelection.setSelectedIds

Sets the selected IDs based on added and removed elements.

Parameters

Collapse all

editor.api.blockSelection.unselect

Unselects all blocks and sets the isSelecting flag to false.

Transforms

editor.tf.blockSelection.duplicate

Duplicates the selected blocks.

editor.tf.blockSelection.removeNodes

Removes the selected nodes from the editor.

editor.tf.blockSelection.select

Selects the nodes returned by getNodes() and resets the selected IDs.

editor.tf.blockSelection.setNodes

Sets properties on the selected nodes.

Parameters

Collapse all

    The properties to set on the selected nodes.

    Options for setting nodes.

editor.tf.blockSelection.setTexts

Sets text properties on the selected nodes.

Parameters

Collapse all

    The text properties to set on the selected nodes.

    Options for setting text nodes, excluding the 'at' property.

Hooks

useBlockSelectable

A hook that provides props for making a block element selectable, including context menu behavior.

Returns

Collapse all

    Props to be spread on the block element:

useBlockSelected

Returns true if context block is selected.

useBlockSelectionNodes

Returns an array of node entries for the currently selected blocks.

useBlockSelectionFragment

Returns an array of nodes for the currently selected blocks.

useBlockSelectionFragmentProp

Returns fragment prop for the currently selected blocks.

useSelectionArea

A hook that initializes and manages the selection area functionality.

editor.api.blockSelection.isSelectable

Checks if a block element is selectable.

Parameters

Collapse all

    The block element to check.

    The path to the block element.

Returns

Collapse all

    Returns true if the block is selectable.

editor.api.blockSelection.moveSelection

Moves the selection up or down to the next selectable block.

Parameters

Collapse all

    The direction to move the selection.

When moving up:

  • Gets the previous selectable block from the top-most selected block
  • Sets it as the new anchor
  • Clears previous selection and selects only this block

When moving down:

  • Gets the next selectable block from the bottom-most selected block
  • Sets it as the new anchor
  • Clears previous selection and selects only this block

editor.api.blockSelection.shiftSelection

Expands or shrinks the selection based on the anchor block.

Parameters

Collapse all

    The direction to expand/shrink the selection.

For SHIFT + DOWN:

  • If anchor is top-most: Expands down by adding block below bottom-most
  • Otherwise: Shrinks from top-most (unless top-most is the anchor)

For SHIFT + UP:

  • If anchor is bottom-most: Expands up by adding block above top-most
  • Otherwise: Shrinks from bottom-most (unless bottom-most is the anchor)

The anchor block always remains selected. If no anchor is set, it defaults to:

  • Bottom-most block for SHIFT + UP
  • Top-most block for SHIFT + DOWN