Block Selection
'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:
- Add an
id
orclassName
to your scroll container. If you're not sure about the container, you can add it to the<Editor />
component. We recommend usingid={editor.uid}
. - Use the appropriate selector in your configuration.
- 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:
- Direct API call:
editor.api.blockSelection.deselect();
- Click outside handler:
const handleClickOutside = (event: MouseEvent) => {
if (!(event.target as HTMLElement).closest('[data-plate-selectable]')) {
editor.api.blockSelection.deselect();
}
};
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
- Default:
false
- Default:
false
- Default:
{ maxLevel: 1 }
- Default:
new Set()
- Default:
null
- Default:
() => true
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.
Indicates whether block selection is currently active.
A function to handle the keydown
event when selecting.
Options for querying nodes during block selection.
A set of IDs for the currently selected blocks.
(Internal) The ID of the anchor block in the current selection. Used for shift-based selection.
Function to determine if a block element is selectable.
BlockMenuPlugin
API
editor.api.blockSelection.add
Adds one or more blocks to the selection.
Parameters
The ID(s) of the block(s) to be selected.
editor.api.blockSelection.clear
Resets the set of selected IDs to an empty set.
editor.api.blockSelection.delete
Removes one or more blocks from the selection.
Parameters
The ID(s) of the block(s) to remove from selection.
editor.api.blockSelection.deselect
Deselects all blocks and sets the isSelecting
flag to false.
editor.api.blockSelection.focus
Focuses the block selection shadow input. This input handles copy, delete, and paste events for selected blocks.
editor.api.blockSelection.getNodes
Gets the selected blocks in the editor.
Returns
An array of selected block entries.
editor.api.blockSelection.has
Checks if one or more blocks are selected.
Parameters
The ID(s) of the block(s) to check.
Returns
Returns true if all specified blocks are selected.
editor.api.blockSelection.selectAll
Selects all selectable blocks in the editor.
editor.api.blockSelection.set
Sets the selection to one or more blocks, clearing any existing selection.
Parameters
The ID(s) of the block(s) to be selected.
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
The properties to set on the selected nodes.
Options for setting nodes.
editor.tf.blockSelection.setTexts
Sets text properties on the selected nodes.
Parameters
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
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
The block element to check.
The path to the block element.
Returns
Returns true if the block is selectable.
editor.api.blockSelection.moveSelection
Moves the selection up or down to the next selectable block.
Parameters
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
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