diff --git a/examples/03-ui-components/13-equation/.bnexample.json b/examples/03-ui-components/13-equation/.bnexample.json new file mode 100644 index 0000000000..1e4b6d5d3e --- /dev/null +++ b/examples/03-ui-components/13-equation/.bnexample.json @@ -0,0 +1,11 @@ +{ + "playground": true, + "docs": false, + "author": "matthewlipski", + "tags": ["Equation", "Inline Equation", "Custom Schemas", "Latex", "Slash Menu"], + "dependencies": { + "katex": "^0.16.11", + "@types/katex": "^0.16.7", + "react-icons": "^5.2.1" + } +} diff --git a/examples/03-ui-components/13-equation/App.tsx b/examples/03-ui-components/13-equation/App.tsx new file mode 100644 index 0000000000..42c3fcc292 --- /dev/null +++ b/examples/03-ui-components/13-equation/App.tsx @@ -0,0 +1,91 @@ +import { + BlockNoteSchema, + defaultInlineContentSpecs, + filterSuggestionItems, +} from "@blocknote/core"; +import "@blocknote/core/fonts/inter.css"; +import { + SuggestionMenuController, + getDefaultReactSlashMenuItems, + useCreateBlockNote, +} from "@blocknote/react"; +import { BlockNoteView } from "@blocknote/mantine"; +import { RiFormula } from "react-icons/ri"; +import "@blocknote/mantine/style.css"; + +import { InlineEquation } from "./Equation"; + +// Our schema with block specs, which contain the configs and implementations for blocks +// that we want our editor to use. +const schema = BlockNoteSchema.create({ + inlineContentSpecs: { + ...defaultInlineContentSpecs, + inlineEquation: InlineEquation, + }, +}); + +// Slash menu item to insert an Alert block +const insertLaTex = (editor: typeof schema.BlockNoteEditor) => ({ + icon: , + title: "Inline Equation", + key: "inlineEquation", + subtext: "Insert mathematical symbols in text.", + aliases: ["equation", "latex", "katex"], + group: "Other", + onItemClick: () => { + const view = editor._tiptapEditor.view; + const pos = editor._tiptapEditor.state.selection.from; + const tr = view.state.tr.insert( + pos, + view.state.schema.nodes.inlineEquation.create(), + ); + view.dispatch(tr); + }, +}); + +export default function App() { + // Creates a new editor instance. + const editor = useCreateBlockNote({ + schema, + initialContent: [ + { + type: "paragraph", + content: [ + "This is an example inline equation ", + { + type: "inlineEquation", + content: "c = \\pm\\sqrt{a^2 + b^2}", + }, + ], + }, + { + type: "paragraph", + content: "Press the '/' key to open the Slash Menu and add another", + }, + { + type: "paragraph", + }, + { + type: "paragraph", + }, + { + type: "paragraph", + }, + ], + }); + + // Renders the editor instance. + return ( + + + filterSuggestionItems( + [...getDefaultReactSlashMenuItems(editor), insertLaTex(editor)], + query + ) + } + /> + + ); +} diff --git a/examples/03-ui-components/13-equation/Equation.tsx b/examples/03-ui-components/13-equation/Equation.tsx new file mode 100644 index 0000000000..7eace7beb3 --- /dev/null +++ b/examples/03-ui-components/13-equation/Equation.tsx @@ -0,0 +1,326 @@ +import { useComponentsContext,useEditorContentOrSelectionChange } from "@blocknote/react"; +import { NodeView } from "prosemirror-view"; +import { BlockNoteEditor } from "@blocknote/core"; +import { + NodeViewProps, + NodeViewRenderer, + NodeViewWrapper, + ReactNodeViewRenderer, +} from "@tiptap/react"; +import { + createStronglyTypedTiptapNode, + createInternalInlineContentSpec, +} from "@blocknote/core"; +import { mergeAttributes } from "@tiptap/core"; +import {ChangeEvent, useEffect, useState, useRef, useCallback, forwardRef, MouseEvent as ReactMouseEvent} from "react"; +import katex from "katex"; +import { AiOutlineEnter } from "react-icons/ai"; +import "katex/dist/katex.min.css"; +import "./styles.css"; + + +const TextareaView = forwardRef((props: any, ref: any) => { + const { autofocus, ...rest } = props; + useEffect(() => { + if (autofocus && ref.current) { + ref.current.setSelectionRange(0, ref.current.value.length); + ref.current.focus() + } + }, [autofocus, ref]); + + return ( +