Accessing the Editor
Getting a reference to the editor instance
To do most things with Plate, you'll need to access the editor
instance at one point or another. The way you'll do this depends on the context in which you need to access the editor
.
From Inside a Plugin
Most often, when you want to extend the functionality of your editor, you'll create a custom Plate plugin. Luckily, plugins are one of the easiest places to access the editor
instance.
Inside Event Handlers
Use the first argument of the handler function.
const createMyPlugin = createPluginFactory({
key: KEY_MY_PLUGIN,
handlers: {
onKeyDown: (editor) => (event) => {
// Do something with editor
},
onChange: (editor) => (value) => {
// Do something with editor
},
},
});
Using the Then Option
The purpose of the then
option is to access the editor
instance in plugin options that normally don't have access to it. Pass a function that takes an editor
and returns an object to be merged with the top-level plugin options.
For example, suppose you had this code and wanted to access the editor
instance inside deserializeHtml
:
const createMyPlugin = createPluginFactory({
key: KEY_MY_PLUGIN,
deserializeHtml: {
// Need editor here
},
});
You would wrap the deserializeHtml
option inside then
.
const createMyPlugin = createPluginFactory({
key: KEY_MY_PLUGIN,
then: (editor) => ({
deserializeHtml: {
// Do something with editor
},
}),
});
From a Child of Plate
Use the usePlateEditorRef
or usePlateEditorState
hooks.
Internally, usePlateEditorState
is a wrapper for usePlateEditorRef
. The only difference is that usePlateEditorState
causes React to re-render whenever the editor
state changes, whereas usePlateEditorRef
does not cause a re-render. Since editor
is mutable and is updated by reference, usePlateEditorRef
will be sufficient (and more efficient) in most situations.
You can call these hooks from any React component that is rendered as a descendant of the Plate
(or PlateProvider
) component, including Plugin Components.
const ParagraphElement = ({
className,
children,
...props
}: PlateElementProps) => {
const editor = usePlateEditorRef();
const handleClick = useCallback(() => {
console.log('You clicked on a paragraph, and the editor is ', editor);
}, [editor]);
return (
<PlateElement asChild className={className} {...props}>
<p onClick={handleClick}>
{children}
</p>
</PlateElement>
);
};
One common pattern is to add an effect component as a child of Plate
that consumes editor
and returns null
.
const CustomEffect = () => {
const editor = usePlateEditorRef();
useEffect(() => {
const interval = setInterval(() => {
console.log('The editor is ', editor);
}, 1000);
return () => clearInterval(interval);
}, [editor]);
return null;
};
export default () => (
<Plate>
<CustomEffect />
</Plate>
);
If editor
is modified by reference, why include it in dependency lists?
Good question! Even though editor
is usually modified by reference, there are some situations in which it's replaced with a fresh instance, such as when the editor is reset.
From a Sibling of Plate
Wrap Plate
and the sibling in PlateProvider
, and then use usePlateEditorRef
or usePlateEditorState
from within the sibling.
Note that certain Plate
props, such as plugins
and initialValue
, must be lifted up to the PlateProvider
. See From an Ancestor if this is impractical or impossible for you to achieve.
const Toolbar = () => {
const editor = usePlateEditorState();
// Do something with editor
// ...
};
export default () => (
<PlateProvider>
<Toolbar />
<Plate />
</PlateProvider>
);
From an Ancestor
If you need to access the editor
instance from an ancestor of Plate
, wrapping the relevant components in a PlateProvider
is the preferred solution. If this is not an option, you can instead use the editorRef
prop to pass a reference to the editor
instance up the React component tree to where it is needed.
The editorRef
prop can be used with useRef
, useState
, or a custom ref callback. Regardless of which you use, you'll need to handle the case where editor
is null. This happens when the editor hasn't had a chance to render yet or has unmounted.
With a Ref Object
const App = () => {
const editorRef = useRef<PlateEditor | null>(null);
const handleSomeEvent = useCallback(() => {
// editor has type PlateEditor | null
const editor = editorRef.current;
if (editor) {
// Do something with editor
}
}, []);
// Pass editorRef and handleSomeEvent down to where they're needed
// ...
};
const MyEditor = ({
editorRef,
}: {
editorRef: MutableRefObject<PlateEditor | null>;
}) => <Plate editorRef={editorRef} />;
With State
If you want your ancestor component to re-render when the editor content changes, you may want to use useState
to store your editor
instance. Since the editorRef
callback is only called once when the editor first mounts, you'll also need to manually trigger a re-render by updating a counter whenever the onChange
handler of Plate
is called.
Using editorRef
with useState
without a counter is equivalent to using usePlateEditorRef
instead of usePlateEditorState
(the difference is discussed above). Most of the time, if you don't need the ancestor component to re-render on every change, you should be using useRef
instead.
const App = () => {
const [editor, setEditor] = useState<PlateEditor | null>(null);
const [, handleUpdateEditor] = useReducer((x) => x + 1, 0);
// Pass editor, setEditor and handleUpdateEditor down to where they're needed
// ...
};
const EditorPreview = ({
editor,
}: {
editor: PlateEditor | null;
}) => {
// html has type string | null
const html = useMemo(
() => editor && serializeHtml(editor, {
nodes: editor.children,
}),
[editor, editor?.children]
);
if (!html) return null;
return <div dangerouslySetInnerHTML={{ __html: html }} />;
};
const MyEditor = ({
setEditor,
handleUpdateEditor,
}: {
setEditor: (editor: PlateEditor | null) => void;
handleUpdateEditor: () => void;
}) => (
<Plate
editorRef={setEditor}
onChange={handleUpdateEditor}
/>
);
Temporary Editor Instance
Sometimes, you'll need to access an editor
instance, but not necessarily the same editor
instance that is used by the Plate editor itself. Such cases include serializing a Plate value to HTML (either on the client or on the server) and deserializing HTML to produce an initial value for Plate.
In these cases, you can create a temporary editor
instance using createPlateEditor({ plugins })
. The only requirement is that you pass the same set of plugins to createPlateEditor
as you pass to the Plate editor itself.
See the following example to deserialize a HTML value and use it as the initial value of the Plate editor.
// Alternatively, define the plugins inside the React component using useMemo
const plugins = createPlugins([
// ...
]);
export default ({ initialHtml }: { initialHtml: string }) => {
/**
* Changing the initialValue after render is not supported, so initialHtml
* is not included in the useMemo deps.
*/
const initialValue = useMemo(() => {
const tmpEditor = createPlateEditor({ plugins });
return deserializeHtml(tmpEditor, {
element: initialHtml,
});
}, []);
return (
<Plate
plugins={plugins}
initialValue={initialValue}
/>
);
};