TypeScript
Slate supports typing of one Slate document model (ie. one set of custom Editor, Element and Text types). If you need to support more than one document model, see the section Multiple Document Models.
Warning: You must define CustomTypes when using TypeScript or Slate will display typing errors.
When migrating from 0.47.x, read the guide below first. Also keep in mind these common migration issues:
- When referring to node.type, you may see the error Property 'type' does not exist on type 'Node'. To fix this, you need to add code like Element.isElement(node) && node.type === 'paragraph'. This is necessary because a Node can be an Element or Text and Text does not have a type property.
- Be careful when you define the CustomType for Editor. Make sure to define the CustomType for Editor as BaseEditor & .... It should not be Editor & ...
To define a custom Element or Text type, extend the CustomTypes interface in the slate module like this.
While you can define types directly in the CustomTypes interface, best practice is to define and export each type separately so that you can reference individual types like a ParagraphElement.
Using best practices, the custom types might look something like:
In this example, CustomText is equal to FormattedText but in a real editor, there can be more types of text like text in a code block which may not allow formatting for example.
Because it gets asked often, this section explains why Slate's type definition is atypical.
Slate needs to support a feature called type discrimination which is available when using union types (e.g. ParagraphElement | HeadingElement). This allows a user to narrow a type. If presented with code like if (node.type === 'paragraph') { ... } the inside of the block, will narrow the type of node to ParagraphElement.
Slate also needs a way to allow developers to get their custom types into Slate. This is done through declaration merging which is a feature of an interface.
Slate combines a union type and an interface to deliver this feature.
For more information see Proposal: Add Custom TypeScript Types to Slate
At the moment, Slate supports types for a single document model at a time. For example, it cannot support a full Rich Text Editor for editing documents while also having a different Editor for editing comments.
Slate's TypeScript support was designed this way because some improved typing support was better than none. The goal is to support typing for multiple editor definitions in the future but this will depend on community input.
One workaround for supporting multiple document models is to create each editor in a separate package and then import them. This hasn't been tested but should work.
Currently there is also support for extending other types but these haven't been tested as thoroughly as the ones documented above:
- Selection
- Range
- Point
Feel free to extend these types but extending these types should be considered experimental.
For some examples of how to use types, see packages/slate-react/src/custom-types.ts in the slate repository.