@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
text/typescript
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();
}
}
}