Lexical Building Custom Features
Before you begin building custom features for Lexical, it is crucial to familiarize yourself with the Lexical docs, particularly the "Concepts" section. This foundation is necessary for understanding Lexical's core principles, such as nodes, editor state, and commands.
Lexical features are designed to be modular, meaning each piece of functionality is encapsulated within just two specific interfaces: one for server-side code and one for client-side code.
By convention, these are named feature.server.ts
for server-side functionality and feature.client.ts
for client-side functionality. The primary functionality is housed within feature.server.ts
, which users will import into their projects. The client-side feature, although defined separately, is integrated and rendered server-side through the server feature.
That way, we still maintain a clear boundary between server and client code, while also centralizing the code needed for a feature in basically one place. This approach is beneficial for managing all the bits and pieces which make up your feature as a whole, such as toolbar entries, buttons, or new nodes, allowing each feature to be neatly contained and managed independently.
Do I need a custom feature?
Before you start building a custom feature, consider whether you can achieve your desired functionality using the existing BlocksFeature
. The BlocksFeature
is a powerful feature that allows you to create custom blocks with a variety of options, including custom React components, markdown converters, and more. If you can achieve your desired functionality using the BlocksFeature
, it is recommended to use it instead of building a custom feature.
Using the BlocksFeature, you can add both inline blocks (= can be inserted into a paragraph, in between text) and block blocks (= take up the whole line) to the editor. If you simply want to bring custom react components into the editor, this is the way to go.
Example: Code Field Block with language picker
This example demonstrates how to create a custom code field block with a language picker using the BlocksFeature
. Make sure to manually install @payloadcms/ui
first.
Field config:
CodeComponent.tsx:
Server Feature
Custom Blocks are not enough? To start building a custom feature, you should start with the server feature, which is the entry-point.
Example myFeature/feature.server.ts:
createServerFeature
is a helper function which lets you create new features without boilerplate code.
Now, the feature is ready to be used in the editor:
By default, this server feature does nothing - you haven't added any functionality yet. Depending on what you want your feature to do, the ServerFeature type exposes various properties you can set to inject custom functionality into the lexical editor.
i18n
Each feature can register their own translations, which are automatically scoped to the feature key:
This allows you to add i18n translations scoped to your feature. This specific example translation will be available under lexical:myFeature:label
- myFeature
being your feature key.
Markdown Transformers
The Server Feature, just like the Client Feature, allows you to add markdown transformers. Markdown transformers on the server are used when converting the editor from or to markdown.
In this example, the node will be outputted as +++
in Markdown, and the markdown +++
will be converted to a MyNode
node in the editor.
Nodes
While nodes added to the server feature do not control how the node is rendered in the editor, they control other aspects of the node:
- HTML conversion
- Node Hooks
- Sub fields
- Behavior in a headless editor
The createNode
helper function is used to create nodes with proper typing. It is recommended to use this function to create nodes.
While nodes in the client feature are added by themselves to the nodes array, nodes in the server feature can be added together with the following sibling options:
Option | Description |
---|---|
| If a node includes sub-fields (e.g. block and link nodes), passing the subFields schema here will make Payload automatically populate & run hooks for them. |
| If a node includes sub-fields, the sub-fields data needs to be returned here, alongside |
| Allows you to run population logic when a node's data was requested from GraphQL. While |
| The actual lexical node needs to be provided here. This also supports lexical node replacements. |
| This allows you to provide node validations, which are run when your document is being validated, alongside other Payload fields. You can use it to throw a validation error for a specific node in case its data is incorrect. |
| Allows you to define how a node can be serialized into different formats. Currently, only supports HTML. Markdown converters are defined in |
| Just like Payload fields, you can provide hooks which are run for this specific node. These are called Node Hooks. |
Feature load order
Server features can also accept a function as the feature
property (useful for sanitizing props, as mentioned below). This function will be called when the feature is loaded during the Payload sanitization process:
"Loading" here means the process of calling this feature
function. By default, features are called in the order in which they are added to the editor. However, sometimes you might want to load a feature after another feature has been loaded, or require a different feature to be loaded, throwing an error if this is not the case.
Within lexical, one example where this is done are our list features. Both UnorderedListFeature
and OrderedListFeature
register the same ListItem
node. Within UnorderedListFeature
we register it normally, but within OrderedListFeature
we want to only register the ListItem
node if the UnorderedListFeature
is not present - otherwise, we would have two features registering the same node.
Here is how we do it:
featureProviderMap
will always be available and contain all the features, even yet-to-be-loaded ones, so we can check if a feature is loaded by checking if its key
present in the map.
If you wanted to make sure a feature is loaded before another feature, you can use the dependenciesPriority
property:
Option | Description |
---|---|
| Keys of soft-dependencies needed for this feature. These are optional. Payload will attempt to load them before this feature, but doesn't throw an error if that's not possible. |
| Keys of dependencies needed for this feature. These dependencies do not have to be loaded first, but they have to exist, otherwise an error will be thrown. |
| Keys of priority dependencies needed for this feature. These dependencies have to be loaded first AND have to exist, otherwise an error will be thrown. They will be available in the |
Client Feature
Most of the functionality which the user actually sees and interacts with, like toolbar items and React components for nodes, resides on the client-side.
To set up your client-side feature, follow these three steps:
- Create a Separate File: Start by creating a new file specifically for your client feature, such as
myFeature/feature.client.ts
. It's important to keep client and server features in separate files to maintain a clean boundary between server and client code. - 'use client': Mark that file with a 'use client' directive at the top of the file
- Register the Client Feature: Register the client feature within your server feature, by passing it to the
ClientFeature
prop. This is needed because the server feature is the sole entry-point of your feature. This also means you are not able to create a client feature without a server feature, as you will not be able to register it otherwise.
Example myFeature/feature.client.ts:
Explore the APIs available through ClientFeature to add the specific functionality you need. Remember, do not import directly from '@payloadcms/richtext-lexical'
when working on the client-side, as it will cause errors with webpack or turbopack. Instead, use '@payloadcms/richtext-lexical/client'
for all client-side imports. Type-imports are excluded from this rule and can always be imported.
Adding a client feature to the server feature
Inside of your server feature, you can provide an import path to the client feature like this:
Nodes
Add nodes to the nodes
array in both your client & server feature. On the server side, nodes are utilized for backend operations like HTML conversion in a headless editor. On the client side, these nodes are integral to how content is displayed and managed in the editor, influencing how they are rendered, behave, and saved in the database.
Example:
myFeature/feature.client.ts:
This also supports lexical node replacements.
myFeature/nodes/MyNode.tsx:
Here is a basic DecoratorNode example:
Please do not add any 'use client' directives to your nodes, as the node class can be used on the server.
Plugins
One small part of a feature are plugins. The name stems from the lexical playground plugins and is just a small part of a lexical feature. Plugins are simply React components which are added to the editor, within all the lexical context providers. They can be used to add any functionality to the editor, by utilizing the lexical API.
Most commonly, they are used to register lexical listeners, node transforms or commands. For example, you could add a drawer to your plugin and register a command which opens it. That command can then be called from anywhere within lexical, e.g. from within your custom lexical node.
To add a plugin, simply add it to the plugins
array in your client feature:
Example plugin.tsx:
In this example, we register a lexical command, which simply inserts a new MyNode into the editor. This command can be called from anywhere within lexical, e.g. from within a custom node.
Toolbar groups
Toolbar groups are visual containers which hold toolbar items. There are different toolbar group types which determine how a toolbar item is displayed: dropdown
and buttons
.
All the default toolbar groups are exported from @payloadcms/richtext-lexical/client
. You can use them to add your own toolbar items to the editor:
- Dropdown:
toolbarAddDropdownGroupWithItems
- Dropdown:
toolbarTextDropdownGroupWithItems
- Buttons:
toolbarFormatGroupWithItems
- Buttons:
toolbarFeatureButtonsGroupWithItems
Within dropdown groups, items are positioned vertically when the dropdown is opened and include the icon & label. Within button groups, items are positioned horizontally and only include the icon. If a toolbar group with the same key is declared twice, all its items will be merged into one group.
Custom buttons toolbar group
Option | Description |
---|---|
| All toolbar items part of this toolbar group need to be added here. |
| Each toolbar group needs to have a unique key. Groups with the same keys will have their items merged together. |
| Determines where the toolbar group will be. |
| Controls the toolbar group type. Set to |
Example:
Custom dropdown toolbar group
Option | Description |
---|---|
| All toolbar items part of this toolbar group need to be added here. |
| Each toolbar group needs to have a unique key. Groups with the same keys will have their items merged together. |
| Determines where the toolbar group will be. |
| Controls the toolbar group type. Set to |
| The dropdown toolbar ChildComponent allows you to pass in a React Component which will be displayed within the dropdown button. |
Example:
Toolbar items
Custom nodes and features on its own are pointless, if they can't be added to the editor. You will need to hook in one of our interfaces which allow the user to interact with the editor:
- Fixed toolbar which stays fixed at the top of the editor
- Inline, floating toolbar which appears when selecting text
- Slash menu which appears when typing
/
in the editor - Markdown transformers, which are triggered when a certain text pattern is typed in the editor
- Or any other interfaces which can be added via your own plugins. Our toolbars are a prime example of this - they are just plugins.
To add a toolbar item to either the floating or the inline toolbar, you can add a ToolbarGroup with a ToolbarItem to the toolbarFixed
or toolbarInline
props of your client feature:
You will have to provide a toolbar group first, and then the items for that toolbar group (more on that above).
A ToolbarItem
various props you can use to customize its behavior:
Option | Description |
---|---|
| A React component which is rendered within your toolbar item's default button component. Usually, you want this to be an icon. |
| A React component which is rendered in place of the toolbar item's default button component, thus completely replacing it. The |
| The label will be displayed in your toolbar item, if it's within a dropdown group. To make use of i18n, this can be a function. |
| Each toolbar item needs to have a unique key. |
| A function which is called when the toolbar item is clicked. |
| This is optional and controls if the toolbar item is clickable or not. If |
| This is optional and controls if the toolbar item is highlighted or not |
The API for adding an item to the floating inline toolbar (toolbarInline
) is identical. If you wanted to add an item to both the fixed and inline toolbar, you can extract it into its own variable (typed as ToolbarGroup[]
) and add it to both the toolbarFixed
and toolbarInline
props.
Slash Menu groups
We're exporting slashMenuBasicGroupWithItems
from @payloadcms/richtext-lexical/client
which you can use to add items to the slash menu labelled "Basic". If you want to create your own slash menu group, here is an example:
By creating a helper function like this, you can easily re-use it and add items to it. All Slash Menu groups with the same keys will have their items merged together.
Option | Description |
---|---|
| An array of |
| The label will be displayed before your Slash Menu group. In order to make use of i18n, this can be a function. |
| Used for class names and, if label is not provided, for display. Slash menus with the same key will have their items merged together. |
Slash Menu items
The API for adding items to the slash menu is similar. There are slash menu groups, and each slash menu groups has items. Here is an example:
Option | Description |
---|---|
| The icon which is rendered in your slash menu item. |
| The label will be displayed in your slash menu item. In order to make use of i18n, this can be a function. |
| Each slash menu item needs to have a unique key. The key will be matched when typing, displayed if no |
| A function which is called when the slash menu item is selected. |
| Keywords are used to match the item for different texts typed after the '/'. E.g. you might want to show a horizontal rule item if you type both /hr, /separator, /horizontal etc. In addition to the keywords, the label and key will be used to find the right slash menu item. |
Markdown Transformers
The Client Feature, just like the Server Feature, allows you to add markdown transformers. Markdown transformers on the client are used to create new nodes when a certain markdown pattern is typed in the editor.
In this example, a new MyNode
will be inserted into the editor when `+++ ` is typed.
Providers
You can add providers to your client feature, which will be nested below the EditorConfigProvider
. This can be useful if you want to provide some context to your nodes or other parts of your feature.
Props
To accept props in your feature, type them as a generic.
Server Feature:
Client Feature:
The unSanitized props are what the user will pass to the feature when they call its provider function and add it to their editor config. You then have an option to sanitize those props. To sanitize those in the server feature, you can pass a function to feature
instead of an object:
Keep in mind that any sanitized props then have to be returned in the sanitizedServerFeatureProps
property.
In the client feature, it works similarly:
Bringing props from the server to the client
By default, the client feature will never receive any props from the server feature. In order to pass props from the server to the client, you can need to return those props in the server feature:
The reason the client feature does not have the same props available as the server by default is because all client props need to be serializable. You can totally accept things like functions or Maps as props in your server feature, but you will not be able to send those to the client. In the end, those props are sent from the server to the client over the network, so they need to be serializable.
More information
Have a look at the features we've already built - understanding how they work will help you understand how to create your own. There is no difference between the features included by default and the ones you create yourself - since those features are all isolated from the "core", you have access to the same APIs, whether the feature is part of Payload or not!