UNPKG

@blocknote/core

Version:

A "Notion-style" block-based extensible text editor built on top of Prosemirror and Tiptap.

114 lines (101 loc) 3.7 kB
import { blockToNode } from "../api/nodeConversions/blockToNode.js"; import type { BlockNoteEditor } from "../editor/BlockNoteEditor.js"; import type { BlockNoDefaults, BlockSchema, InlineContentSchema, StyleSchema, } from "../schema/index.js"; import { mergeCSSClasses } from "../util/browser.js"; // Function that creates a ProseMirror `DOMOutputSpec` for a default block. // Since all default blocks have the same structure (`blockContent` div with a // `inlineContent` element inside), this function only needs the block's name // for the `data-content-type` attribute of the `blockContent` element and the // HTML tag of the `inlineContent` element, as well as any HTML attributes to // add to those. export function createDefaultBlockDOMOutputSpec( blockName: string, htmlTag: string, blockContentHTMLAttributes: Record<string, string>, inlineContentHTMLAttributes: Record<string, string>, ) { const blockContent = document.createElement("div"); blockContent.className = mergeCSSClasses( "bn-block-content", blockContentHTMLAttributes.class, ); blockContent.setAttribute("data-content-type", blockName); for (const [attribute, value] of Object.entries(blockContentHTMLAttributes)) { if (attribute !== "class") { blockContent.setAttribute(attribute, value); } } const inlineContent = document.createElement(htmlTag); inlineContent.className = mergeCSSClasses( "bn-inline-content", inlineContentHTMLAttributes.class, ); for (const [attribute, value] of Object.entries( inlineContentHTMLAttributes, )) { if (attribute !== "class") { inlineContent.setAttribute(attribute, value); } } blockContent.appendChild(inlineContent); return { dom: blockContent, contentDOM: inlineContent, }; } // Function used to convert default blocks to HTML. It uses the corresponding // node's `renderHTML` method to do the conversion by using a default // `DOMSerializer`. export const defaultBlockToHTML = < BSchema extends BlockSchema, I extends InlineContentSchema, S extends StyleSchema, >( block: BlockNoDefaults<BSchema, I, S>, editor: BlockNoteEditor<BSchema, I, S>, ): { dom: HTMLElement; contentDOM?: HTMLElement; } => { let node = blockToNode(block, editor.pmSchema); if (node.type.name === "blockContainer") { // for regular blocks, get the toDOM spec from the blockContent node node = node.firstChild!; } const toDOM = editor.pmSchema.nodes[node.type.name].spec.toDOM; if (toDOM === undefined) { throw new Error( "This block has no default HTML serialization as its corresponding TipTap node doesn't implement `renderHTML`.", ); } const renderSpec = toDOM(node); if (typeof renderSpec !== "object" || !("dom" in renderSpec)) { throw new Error( "Cannot use this block's default HTML serialization as its corresponding TipTap node's `renderHTML` function does not return an object with the `dom` property.", ); } return renderSpec as { dom: HTMLElement; contentDOM?: HTMLElement; }; }; // Function that merges all paragraphs into a single one separated by line breaks. // This is used when parsing blocks like list items and table cells, as they may // contain multiple paragraphs that ProseMirror will not be able to handle // properly. export function mergeParagraphs(element: HTMLElement) { const paragraphs = element.querySelectorAll("p"); if (paragraphs.length > 1) { const firstParagraph = paragraphs[0]; for (let i = 1; i < paragraphs.length; i++) { const paragraph = paragraphs[i]; firstParagraph.innerHTML += "<br>" + paragraph.innerHTML; paragraph.remove(); } } }