Unit Testing Plate
Learn how to unit test Plate editor and plugins.
This guide outlines best practices for unit testing Plate plugins and components using @udecode/plate-test-utils
.
Installation
npm install @udecode/plate-test-utils
Setting Up Tests
Add the JSX pragma at the top of your test file:
/** @jsx jsx */
import { jsx } from '@udecode/plate-test-utils';
jsx; // so ESLint doesn't complain
This allows you to use JSX syntax for creating editor values.
Creating Test Cases
Editor State Representation
Use JSX to represent editor states:
const input = (
<editor>
<hp>
Hello<cursor /> world
</hp>
</editor>
) as any as PlateEditor;
Node elements like <hp />
, <hul />
, <hli />
represent different types of nodes.
Special elements like <cursor />
, <anchor />
, and <focus />
represent selection states.
Testing Transforms
- Create an input state
- Define the expected output state
- Use
createPlateEditor
orcreatePlateTestEditor
to set up the editor - Apply the transform(s)
- Assert the editor's new state
Example testing bold formatting:
it('should use custom hotkey for bold', async () => {
const input = (
<editor>
<hp>
Hello <anchor />
world
<focus />
</hp>
</editor>
) as any as PlateEditor;
const output = (
<editor>
<hp>
Hello <htext bold>world</htext>
</hp>
</editor>
) as any as PlateEditor;
const [editor, { triggerKeyboardEvent }] = await createPlateTestEditor({
value: input.children,
selection: input.selection,
plugins: [
BoldPlugin.configure({
handlers: {
onKeyDown: ({ editor, event }) => {
if (event.key === 'b' && event.ctrlKey) {
editor.tf.toggle.mark({ key: 'bold' });
}
},
},
}),
],
});
await triggerKeyboardEvent('mod+b');
expect(editor.children).toEqual(output.children);
});
Testing Selection
Test how operations affect the editor's selection:
it('should collapse selection on backspace', async () => {
const input = (
<editor>
<hp>
He<anchor />llo wor<focus />ld
</hp>
</editor>
) as any as PlateEditor;
const output = (
<editor>
<hp>
He<cursor />ld
</hp>
</editor>
) as any as PlateEditor;
const [editor] = await createPlateTestEditor({
value: input.children,
selection: input.selection,
});
editor.tf.deleteBackward();
expect(editor.children).toEqual(output.children);
expect(editor.selection).toEqual(output.selection);
});
Testing Key Events
Use triggerKeyboardEvent
from createPlateTestEditor
:
it('should extend selection on shift+ArrowRight', async () => {
const input = (
<editor>
<hp>
Hello <cursor />world
</hp>
</editor>
) as any as PlateEditor;
const output = (
<editor>
<hp>
Hello <anchor />wor<focus />ld
</hp>
</editor>
) as any as PlateEditor;
const [editor, { triggerKeyboardEvent }] = await createPlateTestEditor({
value: input.children,
selection: input.selection,
});
await triggerKeyboardEvent('shift+ArrowRight');
await triggerKeyboardEvent('shift+ArrowRight');
await triggerKeyboardEvent('shift+ArrowRight');
expect(editor.selection).toEqual(output.selection);
});
Testing Complex Scenarios
For complex plugins like tables, test various scenarios:
describe('Table plugin', () => {
it('should add a row below on Tab in last cell', async () => {
const input = (
<editor>
<htable>
<htr>
<htd>Cell 1</htd>
<htd>
Cell 2<cursor />
</htd>
</htr>
</htable>
</editor>
) as any as PlateEditor;
const output = (
<editor>
<htable>
<htr>
<htd>Cell 1</htd>
<htd>Cell 2</htd>
</htr>
<htr>
<htd>
<cursor />
</htd>
<htd></htd>
</htr>
</htable>
</editor>
) as any as PlateEditor;
const [editor, { triggerKeyboardEvent }] = await createPlateTestEditor({
value: input.children,
selection: input.selection,
plugins: [TablePlugin],
});
await triggerKeyboardEvent('Tab');
expect(editor.children).toEqual(output.children);
expect(editor.selection).toEqual(output.selection);
});
});
Testing Plugins with Options
Test how different plugin options affect behavior:
it('should use custom hotkey for bold', async () => {
const handler = jest.fn();
const input = (
<editor>
<hp>
Hello <cursor />world
</hp>
</editor>
) as any as PlateEditor;
const output = (
<editor>
<hp>
Hello <htext bold>world</htext>
</hp>
</editor>
) as any as PlateEditor;
const [editor, { triggerKeyboardEvent }] = await createPlateTestEditor({
value: input.children,
selection: input.selection,
plugins: [
BoldPlugin.configure({
shortcuts: {
toggleBold: {
handler,
keys: 'mod+shift+b',
},
},
}),
],
});
await triggerKeyboardEvent('mod+shift+b');
expect(editor.children).toEqual(output.children);
});
Mocking vs. Real Transforms
While mocking can be useful for isolating specific behaviors, Plate tests often assess actual editor children and selection after transforms. This approach ensures that plugins work correctly with the entire editor state.