Migrating from Slate to Plate

Learn how to migrate from Slate to Plate.

Plate is built on top of Slate, so migrating from a pure Slate implementation to Plate is relatively straightforward. This guide will help you transition your Slate-based editor to Plate.

1. Install Plate

First, install the necessary Plate packages. If you're new to Plate, you might want to start by reading the Introduction to get an overview of the library.

npm install @udecode/plate

2. Replace Slate Imports

Replace your Slate imports with Plate imports. Plate re-exports most Slate types and functions:

// Before
import { createEditor } from 'slate';
import { Slate, Editable, withReact } from 'slate-react';
 
// After
import { createPlateEditor, Plate, PlateContent } from '@udecode/plate/react';

3. Create a Plate Editor

Replace createEditor, withHistory and withReact with createPlateEditor:

// Before
const editor = useMemo(() => withReact(withHistory(createEditor()))), []);
 
// After
const editor = createPlateEditor({
  value,
  plugins: [
    // Additional plugins here
  ],
});

For more details on editor configuration, check out the Editor Configuration guide.

4. Replace Slate and Editable Components

Replace the Slate and Editable components with Plate's Plate component:

// Before
<Slate editor={editor} value={value}>
  <Editable className="p-4" />
</Slate>
 
// After
<Plate editor={editor}>
  <PlateContent className="p-4" />
</Plate>

5. Convert Custom Elements and Leaves

For custom elements and leaves, create Plate plugins:

// Before
const renderElement = useCallback(({ attributes, children, element }) => {
  switch (element.type) {
    case 'paragraph':
      return <p {...attributes}>{children}</p>;
    // ... other cases
  }
}, []);
 
// After
import { withCn, type PlateElement } from '@udecode/plate/react';
 
const ParagraphElement = withRef<typeof PlateElement>(
  ({ children, className, ...props }, ref) => {
    return (
      <PlateElement
        asChild
        className={cn('py-1', className)}
        ref={ref}
        {...props}
      >
        <p>{children}</p>
      </PlateElement>
    );
  }
);
 
const ParagraphPlugin = createPlatePlugin({
  key: 'p',
  node: {
    isElement: true,
    type: 'paragraph',
    component: ParagraphElement,
  },
});

Learn more about creating plugins in the Plugin Configuration guide and Plugin Components guide.

6. Convert Slate Plugins to Plate Plugins

If you have custom Slate plugins, convert them to Plate plugins:

// Before
const withMyPlugin = (editor) => {
  const { insertText } = editor;
  editor.insertText = (text) => {
    // Custom logic
    insertText(text);
  };
  return editor;
};
 
// After
const MyPlugin = createPlatePlugin({
  key: 'myPlugin',
}).overrideEditor(({ editor, tf: { insertText } }) => ({
  transforms: {
    insertText(text, options) {
      // Custom logic
      insertText(text, options);
    },
  }
}));
 
// For adding new methods:
const MyOtherPlugin = createPlatePlugin({
  key: 'myOtherPlugin',
}).extendEditorTransforms(({ editor }) => ({
  newMethod(text) {
    // Add new functionality
  }
}));

For more information on working with the plugin context, see the Plugin Context guide.

7. Update Event Handlers

Update your event handlers to use Plate's plugin system:

// Before
const onKeyDown = (event) => {
  if (event.key === 'Tab') {
    // Handle tab
  }
};
 
// After
const TabPlugin = createPlatePlugin({
  key: 'tab',
  handlers: {
    onKeyDown: ({ editor, event }) => {
      if (event.key === 'Tab') {
        // Handle tab
      }
    },
  },
});

Alternatively, you can use Plate's powerful shortcuts system:

const TabPlugin = createPlatePlugin({
  key: 'tab',
  shortcuts: {
    indent: {
      handler: ({ editor }) => {
        // Handle tab
      },
      keys: ['Tab'],
    },
  },
});

For more details on using shortcuts, check out the Plugin Shortcuts guide.

8. Adapt to Plate's API

Familiarize yourself with Plate's API and use its utilities and hooks:

// Using Plate's transforms
editor.tf.toggle.mark({ key: 'bold' });
 
// Using Plate's debug API
editor.api.debug.log('Hello, Plate!');

For a comprehensive list of editor methods, see the Editor Methods guide.

9. Leverage Plate's Built-in Plugins

Plate comes with many built-in plugins that you can see in the sidebar. Use them to quickly add functionality:

import { BoldPlugin, ItalicPlugin, UnderlinePlugin } from '@udecode/plate/react';
 
const plugins = [
  BoldPlugin,
  ItalicPlugin,
  UnderlinePlugin,
  // ... other plugins
];
 
const editor = createPlateEditor({ plugins });

10. Testing and Refinement

After migrating, thoroughly test your editor to ensure all functionality works as expected. Refine and optimize your implementation using Plate's features and best practices.

For debugging tips and strategies, check out our Debugging guide.