content/docs/migration/slate-to-plate.mdx
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.
npm install platejs
Use 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.
| 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. |
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 createPlateEditor() when the editor is created outside React memoization.
import { createPlateEditor } from 'platejs/react';
export const editor = createPlateEditor({
value: [
{
children: [{ text: 'Draft' }],
type: 'p',
},
],
});
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
);
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);
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);
},
},
}));
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');
},
}));
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;
},
},
});
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;
},
},
},
});
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, []);
}
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',
},
],
});
renderElement and renderLeaf..configure(), .extend*(), and .overrideEditor().