11 $getBlockElementNodesInSelection,
12 $getNodeFromSelection,
13 $insertNewBlockNodeAtSelection, $selectionContainsNodeType, $selectSingleNode,
14 $toggleSelectionBlockNodeType,
17 import {$createCodeBlockNode, $isCodeBlockNode, $openCodeEditorForNode, CodeBlockNode} from "@lexical/rich-text/LexicalCodeBlockNode";
18 import {$createCalloutNode, $isCalloutNode, CalloutCategory} from "@lexical/rich-text/LexicalCalloutNode";
19 import {$isListNode, insertList, ListNode, ListType, removeList} from "@lexical/list";
20 import {$createLinkNode, $isLinkNode} from "@lexical/link";
21 import {$createHeadingNode, $isHeadingNode, HeadingTagType} from "@lexical/rich-text/LexicalHeadingNode";
22 import {$createQuoteNode, $isQuoteNode} from "@lexical/rich-text/LexicalQuoteNode";
24 const $isHeaderNodeOfTag = (node: LexicalNode | null | undefined, tag: HeadingTagType) => {
25 return $isHeadingNode(node) && node.getTag() === tag;
28 export function toggleSelectionAsHeading(editor: LexicalEditor, tag: HeadingTagType) {
30 $toggleSelectionBlockNodeType(
31 (node) => $isHeaderNodeOfTag(node, tag),
32 () => $createHeadingNode(tag),
37 export function toggleSelectionAsParagraph(editor: LexicalEditor) {
39 $toggleSelectionBlockNodeType($isParagraphNode, $createParagraphNode);
43 export function toggleSelectionAsBlockquote(editor: LexicalEditor) {
45 $toggleSelectionBlockNodeType($isQuoteNode, $createQuoteNode);
49 export function toggleSelectionAsList(editor: LexicalEditor, type: ListType) {
50 editor.getEditorState().read(() => {
51 const selection = $getSelection();
52 const listSelected = $selectionContainsNodeType(selection, (node: LexicalNode | null | undefined): boolean => {
53 return $isListNode(node) && (node as ListNode).getListType() === type;
59 insertList(editor, type);
64 export function formatCodeBlock(editor: LexicalEditor) {
65 editor.getEditorState().read(() => {
66 const selection = $getSelection();
67 const lastSelection = getLastSelection(editor);
68 const codeBlock = $getNodeFromSelection(lastSelection, $isCodeBlockNode) as (CodeBlockNode | null);
69 if (codeBlock === null) {
71 const codeBlock = $createCodeBlockNode();
72 codeBlock.setCode(selection?.getTextContent() || '');
74 const selectionNodes = $getBlockElementNodesInSelection(selection);
75 const firstSelectionNode = selectionNodes[0];
76 const extraNodes = selectionNodes.slice(1);
77 if (firstSelectionNode) {
78 firstSelectionNode.replace(codeBlock);
79 extraNodes.forEach(n => n.remove());
81 $insertNewBlockNodeAtSelection(codeBlock, true);
84 $openCodeEditorForNode(editor, codeBlock);
85 $selectSingleNode(codeBlock);
88 $openCodeEditorForNode(editor, codeBlock);
93 export function cycleSelectionCalloutFormats(editor: LexicalEditor) {
95 const selection = $getSelection();
96 const blocks = $getBlockElementNodesInSelection(selection);
99 for (const block of blocks) {
100 if (!$isCalloutNode(block)) {
101 block.replace($createCalloutNode('info'), true);
110 const types: CalloutCategory[] = ['info', 'warning', 'danger', 'success'];
111 for (const block of blocks) {
112 if ($isCalloutNode(block)) {
113 const type = block.getCategory();
114 const typeIndex = types.indexOf(type);
115 const newIndex = (typeIndex + 1) % types.length;
116 const newType = types[newIndex];
117 block.setCategory(newType);
123 export function insertOrUpdateLink(editor: LexicalEditor, linkDetails: {text: string, title: string, target: string, url: string}) {
124 editor.update(() => {
125 const selection = $getSelection();
126 let link = $getNodeFromSelection(selection, $isLinkNode);
127 if ($isLinkNode(link)) {
128 link.setURL(linkDetails.url);
129 link.setTarget(linkDetails.target);
130 link.setTitle(linkDetails.title);
132 link = $createLinkNode(linkDetails.url, {
133 title: linkDetails.title,
134 target: linkDetails.target,
137 $insertNodes([link]);
140 if ($isLinkNode(link)) {
141 for (const child of link.getChildren()) {
144 link.append($createTextNode(linkDetails.text));