@blocknote/core
Version:
A "Notion-style" block-based extensible text editor built on top of Prosemirror and Tiptap.
134 lines (121 loc) • 4.03 kB
text/typescript
import { InputRule } from "@tiptap/core";
import { updateBlockCommand } from "../../../api/blockManipulation/commands/updateBlock/updateBlock.js";
import { getBlockInfoFromSelection } from "../../../api/getBlockInfoFromPos.js";
import {
PropSchema,
createBlockSpecFromStronglyTypedTiptapNode,
createStronglyTypedTiptapNode,
} from "../../../schema/index.js";
import { createDefaultBlockDOMOutputSpec } from "../../defaultBlockHelpers.js";
import { defaultProps } from "../../defaultProps.js";
import { getListItemContent } from "../getListItemContent.js";
import { handleEnter } from "../ListItemKeyboardShortcuts.js";
export const bulletListItemPropSchema = {
...defaultProps,
} satisfies PropSchema;
const BulletListItemBlockContent = createStronglyTypedTiptapNode({
name: "bulletListItem",
content: "inline*",
group: "blockContent",
// This is to make sure that check list parse rules run before, since they
// both parse `li` elements but check lists are more specific.
priority: 90,
addInputRules() {
return [
// Creates an unordered list when starting with "-", "+", or "*".
new InputRule({
find: new RegExp(`^[-+*]\\s$`),
handler: ({ state, chain, range }) => {
const blockInfo = getBlockInfoFromSelection(state);
if (
!blockInfo.isBlockContainer ||
blockInfo.blockContent.node.type.spec.content !== "inline*"
) {
return;
}
chain()
.command(
updateBlockCommand(blockInfo.bnBlock.beforePos, {
type: "bulletListItem",
props: {},
}),
)
// Removes the "-", "+", or "*" character used to set the list.
.deleteRange({ from: range.from, to: range.to });
},
}),
];
},
addKeyboardShortcuts() {
return {
Enter: () => handleEnter(this.options.editor),
"Mod-Shift-8": () => {
const blockInfo = getBlockInfoFromSelection(this.editor.state);
if (
!blockInfo.isBlockContainer ||
blockInfo.blockContent.node.type.spec.content !== "inline*"
) {
return true;
}
return this.editor.commands.command(
updateBlockCommand(blockInfo.bnBlock.beforePos, {
type: "bulletListItem",
props: {},
}),
);
},
};
},
parseHTML() {
return [
// Parse from internal HTML.
{
tag: "div[data-content-type=" + this.name + "]",
contentElement: ".bn-inline-content",
},
// Parse from external HTML.
{
tag: "li",
getAttrs: (element) => {
if (typeof element === "string") {
return false;
}
const parent = element.parentElement;
if (parent === null) {
return false;
}
if (
parent.tagName === "UL" ||
(parent.tagName === "DIV" && parent.parentElement?.tagName === "UL")
) {
return {};
}
return false;
},
// As `li` elements can contain multiple paragraphs, we need to merge their contents
// into a single one so that ProseMirror can parse everything correctly.
getContent: (node, schema) =>
getListItemContent(node, schema, this.name),
node: "bulletListItem",
},
];
},
renderHTML({ HTMLAttributes }) {
return createDefaultBlockDOMOutputSpec(
this.name,
// We use a <p> tag, because for <li> tags we'd need a <ul> element to put
// them in to be semantically correct, which we can't have due to the
// schema.
"p",
{
...(this.options.domAttributes?.blockContent || {}),
...HTMLAttributes,
},
this.options.domAttributes?.inlineContent || {},
);
},
});
export const BulletListItem = createBlockSpecFromStronglyTypedTiptapNode(
BulletListItemBlockContent,
bulletListItemPropSchema,
);