Plate keeps Slate's document model and moves editor setup, rendering, handlers, and command wiring into plugins. Migrate the editor shell first, then move custom rendering and behavior into plugins.
Install
pnpm add platejspnpm add platejsUse feature packages only for the nodes, marks, or behavior you add to the editor. Plate UI users should start with Plate UI instead of rebuilding every component by hand.
Migration Map
| Slate surface | Plate surface |
|---|---|
createEditor() plus withReact() | usePlateEditor() in React components, or createPlateEditor() in factories and tests. |
<Slate> plus <Editable> | <Plate> plus <PlateContent>. |
renderElement / renderLeaf switch statements | Plugin components through .withComponent() or node.component. |
withX(editor) plugin functions | .overrideEditor() for wrappers, .extend*() for new APIs and transforms. |
Top-level event handlers on Editable | Plugin handlers or shortcuts. |
Transforms.* imports | editor.tf.* transforms. |
Editor.* imports | editor.api.* queries. |
Editor Shell
Move the editor value into the editor creation call and render the editable with PlateContent.
'use client';
import { Plate, PlateContent, usePlateEditor } from 'platejs/react';
const initialValue = [
{
children: [{ text: 'Hello Plate.' }],
type: 'p',
},
];
export function Editor() {
const editor = usePlateEditor({
value: initialValue,
});
return (
<Plate editor={editor}>
<PlateContent className="p-4" />
</Plate>
);
}'use client';
import { Plate, PlateContent, usePlateEditor } from 'platejs/react';
const initialValue = [
{
children: [{ text: 'Hello Plate.' }],
type: 'p',
},
];
export function Editor() {
const editor = usePlateEditor({
value: initialValue,
});
return (
<Plate editor={editor}>
<PlateContent className="p-4" />
</Plate>
);
}Use createPlateEditor() when the editor is created outside React memoization.
import { createPlateEditor } from 'platejs/react';
export const editor = createPlateEditor({
value: [
{
children: [{ text: 'Draft' }],
type: 'p',
},
],
});import { createPlateEditor } from 'platejs/react';
export const editor = createPlateEditor({
value: [
{
children: [{ text: 'Draft' }],
type: 'p',
},
],
});Custom Elements
Replace renderElement branches with node plugins. Use .withComponent() when the only change is the React component.
import {
ParagraphPlugin,
PlateElement,
type PlateElementProps,
} from 'platejs/react';
export function ParagraphElement({
children,
...props
}: PlateElementProps) {
return (
<PlateElement className="m-0 px-0 py-1" {...props}>
{children}
</PlateElement>
);
}
export const AppParagraphPlugin = ParagraphPlugin.withComponent(
ParagraphElement
);import {
ParagraphPlugin,
PlateElement,
type PlateElementProps,
} from 'platejs/react';
export function ParagraphElement({
children,
...props
}: PlateElementProps) {
return (
<PlateElement className="m-0 px-0 py-1" {...props}>
{children}
</PlateElement>
);
}
export const AppParagraphPlugin = ParagraphPlugin.withComponent(
ParagraphElement
);If your Slate document stores a custom type like paragraph, keep that type on the plugin.
export const AppParagraphPlugin = ParagraphPlugin.configure({
node: { type: 'paragraph' },
}).withComponent(ParagraphElement);export const AppParagraphPlugin = ParagraphPlugin.configure({
node: { type: 'paragraph' },
}).withComponent(ParagraphElement);Custom Behavior
Use .overrideEditor() when the Slate plugin wrapped an existing editor method.
import { createPlatePlugin } from 'platejs/react';
export const LimitExclamationPlugin = createPlatePlugin({
key: 'limitExclamation',
}).overrideEditor(({ tf: { insertText } }) => ({
transforms: {
insertText(text, options) {
insertText(text === '!' ? '.' : text, options);
},
},
}));import { createPlatePlugin } from 'platejs/react';
export const LimitExclamationPlugin = createPlatePlugin({
key: 'limitExclamation',
}).overrideEditor(({ tf: { insertText } }) => ({
transforms: {
insertText(text, options) {
insertText(text === '!' ? '.' : text, options);
},
},
}));Use .extendEditorApi() or .extendEditorTransforms() when the plugin adds a new method.
import { createPlatePlugin } from 'platejs/react';
export const SignaturePlugin = createPlatePlugin({
key: 'signature',
}).extendEditorTransforms(({ editor }) => ({
insertSignature() {
editor.tf.insertText(' - Plate');
},
}));import { createPlatePlugin } from 'platejs/react';
export const SignaturePlugin = createPlatePlugin({
key: 'signature',
}).extendEditorTransforms(({ editor }) => ({
insertSignature() {
editor.tf.insertText(' - Plate');
},
}));Handlers And Shortcuts
Move editor events into the plugin that owns the behavior.
import { createPlatePlugin } from 'platejs/react';
export const TabPlugin = createPlatePlugin({
key: 'tab',
handlers: {
onKeyDown: ({ event }) => {
if (event.key !== 'Tab') return false;
event.preventDefault();
return true;
},
},
});import { createPlatePlugin } from 'platejs/react';
export const TabPlugin = createPlatePlugin({
key: 'tab',
handlers: {
onKeyDown: ({ event }) => {
if (event.key !== 'Tab') return false;
event.preventDefault();
return true;
},
},
});Use shortcuts when the key combination should call a plugin API, transform, or explicit handler.
import { createPlatePlugin } from 'platejs/react';
export const SavePlugin = createPlatePlugin({
key: 'save',
}).extend({
shortcuts: {
draft: {
keys: 'mod+s',
handler: ({ event }) => {
event.preventDefault();
return true;
},
},
},
});import { createPlatePlugin } from 'platejs/react';
export const SavePlugin = createPlatePlugin({
key: 'save',
}).extend({
shortcuts: {
draft: {
keys: 'mod+s',
handler: ({ event }) => {
event.preventDefault();
return true;
},
},
},
});API Calls
Plate keeps Slate-style direct methods for compatibility, but plugin code should use the namespaced API and transform surfaces.
editor.tf.toggleMark('bold');
editor.tf.insertText('Hello');
const text = editor.api.string([]);
if (editor.selection) {
const isStart = editor.api.isStart(editor.selection.anchor, []);
}editor.tf.toggleMark('bold');
editor.tf.insertText('Hello');
const text = editor.api.string([]);
if (editor.selection) {
const isStart = editor.api.isStart(editor.selection.anchor, []);
}Headless Code
Use createSlateEditor from platejs for non-React importers, serializers, transforms, and tests.
import { createSlateEditor } from 'platejs';
export const editor = createSlateEditor({
value: [
{
children: [{ text: 'Headless document.' }],
type: 'p',
},
],
});import { createSlateEditor } from 'platejs';
export const editor = createSlateEditor({
value: [
{
children: [{ text: 'Headless document.' }],
type: 'p',
},
],
});Related
- Editor for editor creation options.
- Plugin Components for replacing
renderElementandrenderLeaf. - Plugin Methods for
.configure(),.extend*(), and.overrideEditor(). - Plugin Shortcuts for keyboard command wiring.