Docs
Typescript
Typescript
Generic types for Plate.
The core types will be the default of Slate in the near future, see this PR.
Plate delivers a structured typing layer built on top of Slate, which primarily employs generic types.
Differences with Slate types:
Editor
type: Plate introducesTEditor<V>
, whereV
symbolizes the "value" that Slate edits. In a generic editor,V
would correspond toTElement[]
, as this is what the editor accepts as children. However, in a custom editor,V
might be defined asTEditor<Array<Paragraph | Quote>>
.- Methods related to
TEditor
andTNode
: These methods have also been made generic. For instance, when you callgetLeafNode(editor, path)
, it is recognized that the return value is aTText
node. More specifically, it recognizes the text node of the type defined in your custom elements, along with any marks you've defined. TheTEditor
type in Plate does not correspond to theEditor
type in Slate, hence, Plate has introduced type-only forks of all Slate methods that useEditor
, which you should utilize. See the full list in Slate and Slate React. - Declaration merging approach: Plate's use of generic types supersedes the declaration merging approach, offering several benefits. One issue with declaration merging was the inability to distinguish between an "unknown" and a "known" element as the underlying type was changed. Similarly, managing two editors on a page with different schemas was not possible. The use of generics in Plate, however, can smoothly replace the declaration merging approach while facilitating an easy migration. You can continue to pass the same custom element definitions into
TEditor
.
Example of Defining Types in plate-types.ts
:
The following example outlines the types in plate-types.ts
:
- The
MyValue
type, which will serve as the type of oureditor.children
. MyValue
is a crucial type since most core types derive from it.- Types following
MyValue
are optional, yet highly recommended. The process of writing generic types can be repetitive, so aim to do it just once. - A set of typed functions are also included for your use.
Naming Convention:
- We use
T...Element
because...Element
is already utilized by Plate UI components, and we wish to avoid naming conflicts among imports. TheT...
prefix also differentiates Plate types from Slate types. My
is used for consumer types of the library. It's succinct and unambiguous. You could substitute it with the tag name of your project.
Usage
import type React from 'react';
import type { ELEMENT_BLOCKQUOTE } from '@udecode/plate-block-quote';
import type {
ELEMENT_CODE_BLOCK,
ELEMENT_CODE_LINE,
} from '@udecode/plate-code-block';
import type { TCommentText } from '@udecode/plate-comments';
import type {
ELEMENT_EXCALIDRAW,
TExcalidrawElement,
} from '@udecode/plate-excalidraw';
import type {
ELEMENT_H1,
ELEMENT_H2,
ELEMENT_H3,
ELEMENT_H4,
ELEMENT_H5,
ELEMENT_H6,
} from '@udecode/plate-heading';
import type { ELEMENT_HR } from '@udecode/plate-horizontal-rule';
import type { ELEMENT_LINK, TLinkElement } from '@udecode/plate-link';
import type {
ELEMENT_LI,
ELEMENT_OL,
ELEMENT_TODO_LI,
ELEMENT_UL,
TTodoListItemElement,
} from '@udecode/plate-list';
import type {
ELEMENT_IMAGE,
ELEMENT_MEDIA_EMBED,
TImageElement,
TMediaEmbedElement,
} from '@udecode/plate-media';
import type {
ELEMENT_MENTION,
ELEMENT_MENTION_INPUT,
TMentionElement,
TMentionInputElement,
} from '@udecode/plate-mention';
import type { ELEMENT_PARAGRAPH } from '@udecode/plate-paragraph';
import type {
ELEMENT_TABLE,
ELEMENT_TD,
ELEMENT_TR,
TTableElement,
} from '@udecode/plate-table';
import type { ELEMENT_TOGGLE, TToggleElement } from '@udecode/plate-toggle';
import type { TText } from '@udecode/slate';
import {
type EElement,
type PlateEditor,
type PlateId,
type TElement,
useEditorRef,
} from '@udecode/plate-common';
/** Text */
export type EmptyText = {
text: '';
};
export type PlainText = {
text: string;
};
export interface RichText extends TText, TCommentText {
backgroundColor?: React.CSSProperties['backgroundColor'];
bold?: boolean;
code?: boolean;
color?: React.CSSProperties['color'];
fontFamily?: React.CSSProperties['fontFamily'];
fontSize?: React.CSSProperties['fontSize'];
fontWeight?: React.CSSProperties['fontWeight'];
italic?: boolean;
kbd?: boolean;
strikethrough?: boolean;
subscript?: boolean;
underline?: boolean;
}
/** Inline Elements */
export interface MyLinkElement extends TLinkElement {
children: RichText[];
type: typeof ELEMENT_LINK;
}
export interface MyMentionInputElement extends TMentionInputElement {
children: [PlainText];
type: typeof ELEMENT_MENTION_INPUT;
}
export interface MyMentionElement extends TMentionElement {
children: [EmptyText];
type: typeof ELEMENT_MENTION;
}
export type MyInlineElement =
| MyLinkElement
| MyMentionElement
| MyMentionInputElement;
export type MyInlineDescendant = MyInlineElement | RichText;
export type MyInlineChildren = MyInlineDescendant[];
/** Block props */
export interface MyIndentProps {
indent?: number;
}
export interface MyIndentListProps extends MyIndentProps {
listRestart?: number;
listStart?: number;
listStyleType?: string;
}
export interface MyLineHeightProps {
lineHeight?: React.CSSProperties['lineHeight'];
}
export interface MyAlignProps {
align?: React.CSSProperties['textAlign'];
}
export interface MyBlockElement
extends TElement,
MyIndentListProps,
MyLineHeightProps {
id?: PlateId;
}
/** Blocks */
export interface MyParagraphElement extends MyBlockElement {
children: MyInlineChildren;
type: typeof ELEMENT_PARAGRAPH;
}
export interface MyH1Element extends MyBlockElement {
children: MyInlineChildren;
type: typeof ELEMENT_H1;
}
export interface MyH2Element extends MyBlockElement {
children: MyInlineChildren;
type: typeof ELEMENT_H2;
}
export interface MyH3Element extends MyBlockElement {
children: MyInlineChildren;
type: typeof ELEMENT_H3;
}
export interface MyH4Element extends MyBlockElement {
children: MyInlineChildren;
type: typeof ELEMENT_H4;
}
export interface MyH5Element extends MyBlockElement {
children: MyInlineChildren;
type: typeof ELEMENT_H5;
}
export interface MyH6Element extends MyBlockElement {
children: MyInlineChildren;
type: typeof ELEMENT_H6;
}
export interface MyBlockquoteElement extends MyBlockElement {
children: MyInlineChildren;
type: typeof ELEMENT_BLOCKQUOTE;
}
export interface MyCodeBlockElement extends MyBlockElement {
children: MyCodeLineElement[];
type: typeof ELEMENT_CODE_BLOCK;
}
export interface MyCodeLineElement extends TElement {
children: PlainText[];
type: typeof ELEMENT_CODE_LINE;
}
export interface MyTableElement extends TTableElement, MyBlockElement {
children: MyTableRowElement[];
type: typeof ELEMENT_TABLE;
}
export interface MyTableRowElement extends TElement {
children: MyTableCellElement[];
type: typeof ELEMENT_TR;
}
export interface MyTableCellElement extends TElement {
children: MyNestableBlock[];
type: typeof ELEMENT_TD;
}
export interface MyBulletedListElement extends TElement, MyBlockElement {
children: MyListItemElement[];
type: typeof ELEMENT_UL;
}
export interface MyNumberedListElement extends TElement, MyBlockElement {
children: MyListItemElement[];
type: typeof ELEMENT_OL;
}
export interface MyListItemElement extends TElement, MyBlockElement {
children: MyInlineChildren;
type: typeof ELEMENT_LI;
}
export interface MyTodoListElement
extends TTodoListItemElement,
MyBlockElement {
children: MyInlineChildren;
type: typeof ELEMENT_TODO_LI;
}
export interface MyToggleElement extends TToggleElement, MyBlockElement {
children: MyInlineChildren;
type: typeof ELEMENT_TOGGLE;
}
export interface MyImageElement extends TImageElement, MyBlockElement {
children: [EmptyText];
type: typeof ELEMENT_IMAGE;
}
export interface MyMediaEmbedElement
extends TMediaEmbedElement,
MyBlockElement {
children: [EmptyText];
type: typeof ELEMENT_MEDIA_EMBED;
}
export interface MyHrElement extends MyBlockElement {
children: [EmptyText];
type: typeof ELEMENT_HR;
}
export interface MyExcalidrawElement
extends TExcalidrawElement,
MyBlockElement {
children: [EmptyText];
type: typeof ELEMENT_EXCALIDRAW;
}
export type MyNestableBlock = MyParagraphElement;
export type MyElement = EElement<MyValue>;
export type MyBlock = Exclude<MyElement, MyInlineElement>;
export type MyRootBlock =
| MyBlockquoteElement
| MyBulletedListElement
| MyCodeBlockElement
| MyExcalidrawElement
| MyH1Element
| MyH2Element
| MyH3Element
| MyH4Element
| MyH5Element
| MyH6Element
| MyHrElement
| MyImageElement
| MyMediaEmbedElement
| MyNumberedListElement
| MyParagraphElement
| MyTableElement
| MyTodoListElement
| MyToggleElement;
/** Editor types */
export type MyValue = MyRootBlock[];
export type MyEditor = { isDragging?: boolean } & PlateEditor<MyValue>;
export const useMyEditorRef = () => useEditorRef<MyEditor>();